(* History++ plugin for Miranda IM: the free IM client for Microsoft* Windows* Copyright (C) 2006-2009 theMIROn, 2003-2006 Art Fedorov. History+ parts (C) 2001 Christian Kastner This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA *) unit hpp_externalgrid; interface uses Windows, Classes, Controls, Forms, Graphics, Messages, SysUtils, Dialogs, m_api, hpp_global, HistoryGrid, RichEdit, Menus, ShellAPI; type PExtCustomItem = ^TExtCustomItem; TExtCustomItem = record Nick: String; Text: String; Sent: Boolean; Time: DWord; HEvent: TMEVENT; end; TExtItem = record hDBEvent: THandle; hContact: THandle; Codepage: THandle; RTLMode: TRTLMode; Custom: Boolean; CustomEvent: TExtCustomItem; end; TOnDestroyWindow = procedure(Sender: TObject; Handle: HWND) of object; TExtHistoryGrid = class(THistoryGrid) private FCachedHandle: HWND; FControlID: Cardinal; FSavedKeyMessage: TWMKey; FOnDestroyWindow: TOnDestroyWindow; procedure WMKeyDown(var Message: TWMKeyDown); message WM_KEYDOWN; procedure WMKeyUp(var Message: TWMKeyUp); message WM_KEYUP; procedure WMSysKeyUp(var Message: TWMSysKeyUp); message WM_SYSKEYUP; procedure WMChar(var Message: TWMChar); message WM_CHAR; procedure WMDestroy(var Message: TWMNCDestroy); message WM_DESTROY; procedure WMNCDestroy(var Message: TWMNCDestroy); message WM_NCDESTROY; protected function GetCachedHandle: HWND; function SendMsgFilterMessage(var Message: TMessage): Integer; public constructor Create(AOwner: TComponent); override; property CachedHandle: HWND read GetCachedHandle; property ControlID: Cardinal read FControlID write FControlID; property OnDestroyWindow: TOnDestroyWindow read FOnDestroyWindow write FOnDestroyWindow; end; TExternalGrid = class(TObject) private Items: array of TExtItem; Grid: TExtHistoryGrid; FParentWindow: HWND; FSelection: Pointer; SavedLinkUrl: String; SavedFileDir: String; pmGrid: TPopupMenu; pmLink: TPopupMenu; miEventsFilter: TMenuItem; WasKeyPressed: Boolean; FUseHistoryRTLMode: Boolean; FExternalRTLMode: TRTLMode; FUseHistoryCodepage: Boolean; FExternalCodepage: Cardinal; FGridState: TGridState; SaveDialog: TSaveDialog; RecentFormat: TSaveFormat; FSubContact: TMCONTACT; FSubProtocol: AnsiString; function GetGridHandle: HWND; procedure SetUseHistoryRTLMode(const Value: Boolean); procedure SetUseHistoryCodepage(const Value: Boolean); procedure SetGroupLinked(const Value: Boolean); procedure SetShowHeaders(const Value: Boolean); procedure SetShowBookmarks(const Value: Boolean); procedure CreateEventsFilterMenu; procedure SetEventFilter(FilterIndex: Integer = -1); function IsFileEvent(Index: Integer): Boolean; protected procedure GridItemData(Sender: TObject; Index: Integer; var Item: THistoryItem); procedure GridTranslateTime(Sender: TObject; Time: DWord; var Text: String); procedure GridNameData(Sender: TObject; Index: Integer; var Name: String); procedure GridProcessRichText(Sender: TObject; Handle: THandle; Item: Integer); procedure GridUrlClick(Sender: TObject; Item: Integer; URLText: String; Button: TMouseButton); procedure GridBookmarkClick(Sender: TObject; Item: Integer); procedure GridSelectRequest(Sender: TObject); procedure GridDblClick(Sender: TObject); procedure GridKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState); procedure GridKeyUp(Sender: TObject; var Key: Word; Shift: TShiftState); procedure GridPopup(Sender: TObject); procedure GridInlineKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState); procedure GridItemDelete(Sender: TObject; Index: Integer); procedure GridXMLData(Sender: TObject; Index: Integer; var Item: TXMLItem); procedure GridMCData(Sender: TObject; Index: Integer; var Item: TMCItem; Stage: TSaveStage); procedure OnCopyClick(Sender: TObject); procedure OnCopyTextClick(Sender: TObject); procedure OnSelectAllClick(Sender: TObject); procedure OnTextFormattingClick(Sender: TObject); procedure OnReplyQuotedClick(Sender: TObject); procedure OnBookmarkClick(Sender: TObject); procedure OnOpenClick(Sender: TObject); procedure OnOpenLinkClick(Sender: TObject); procedure OnOpenLinkNWClick(Sender: TObject); procedure OnCopyLinkClick(Sender: TObject); procedure OnDeleteClick(Sender: TObject); procedure OnBidiModeLogClick(Sender: TObject); procedure OnBidiModeHistoryClick(Sender: TObject); procedure OnCodepageLogClick(Sender: TObject); procedure OnCodepageHistoryClick(Sender: TObject); procedure OnSaveSelectedClick(Sender: TObject); procedure OnEventsFilterItemClick(Sender: TObject); procedure OnBrowseReceivedFilesClick(Sender: TObject); procedure OnOpenFileFolderClick(Sender: TObject); public constructor Create(AParentWindow: HWND; ControlID: Cardinal = 0); destructor Destroy; override; procedure AddEvent(hContact:TMCONTACT; hDBEvent: THandle; Codepage: Integer; RTL: Boolean; DoScroll: Boolean); procedure AddCustomEvent(hContact: THandle; const CustomItem: TExtCustomItem; Codepage: Integer; RTL: Boolean; DoScroll: Boolean); procedure SetPosition(x, y, cx, cy: Integer); procedure ScrollToBottom; function GetSelection(): PAnsiChar; procedure SaveSelected; procedure Clear; property ParentWindow: HWND read FParentWindow; property GridHandle: HWND read GetGridHandle; property UseHistoryRTLMode: Boolean read FUseHistoryRTLMode write SetUseHistoryRTLMode; property UseHistoryCodepage: Boolean read FUseHistoryCodepage write SetUseHistoryCodepage; function Perform(Msg: Cardinal; WParam:WPARAM; LParam: LPARAM): LRESULT; procedure HMBookmarkChanged(var M: TMessage); message HM_NOTF_BOOKMARKCHANGED; // procedure HMIcons2Changed(var M: TMessage); message HM_NOTF_ICONS2CHANGED; procedure HMFiltersChanged(var M: TMessage); message HM_NOTF_FILTERSCHANGED; procedure HMNickChanged(var M: TMessage); message HM_NOTF_NICKCHANGED; procedure HMEventDeleted(var M: TMessage); message HM_MIEV_EVENTDELETED; procedure HMMetaDefaultChanged(var M: TMessage); message HM_MIEV_METADEFCHANGED; procedure BeginUpdate; procedure EndUpdate; property ShowHeaders: Boolean write SetShowHeaders; property GroupLinked: Boolean write SetGroupLinked; property ShowBookmarks: Boolean write SetShowBookmarks; end; implementation uses hpp_richedit, hpp_database, hpp_contacts, hpp_eventfilters, hpp_itemprocess, hpp_events, hpp_services, hpp_forms, hpp_bookmarks, hpp_messages, hpp_options, hpp_sessionsthread; { TExtHistoryGrid } constructor TExtHistoryGrid.Create(AOwner: TComponent); begin FCachedHandle := 0; FControlID := 0; inherited; end; function TExtHistoryGrid.GetCachedHandle: HWND; begin if (FCachedHandle = 0) or HandleAllocated then Result := Handle else Result := FCachedHandle; end; function TExtHistoryGrid.SendMsgFilterMessage(var Message: TMessage): Integer; var mf: TMsgFilter; begin Result := 0; if FControlID <> 0 then begin mf.nmhdr.hwndFrom := WindowHandle; mf.nmhdr.idFrom := FControlID; mf.nmhdr.code := EN_MSGFILTER; mf.Msg := Message.Msg; mf.wParam := Message.wParam; mf.lParam := Message.lParam; Result := SendMessage(ParentWindow, WM_NOTIFY, FControlID, LParam(@mf)); end; end; procedure TExtHistoryGrid.WMKeyDown(var Message: TWMKeyDown); begin inherited; FSavedKeyMessage := Message; if Message.CharCode <> 0 then SendMsgFilterMessage(TMessage(Message)) end; procedure TExtHistoryGrid.WMKeyUp(var Message: TWMKeyUp); begin inherited; if FSavedKeyMessage.CharCode = 0 then exit; if Message.CharCode <> 0 then SendMsgFilterMessage(TMessage(Message)) end; procedure TExtHistoryGrid.WMSysKeyUp(var Message: TWMSysKeyUp); begin inherited; if FSavedKeyMessage.CharCode = 0 then exit; if Message.CharCode <> 0 then SendMsgFilterMessage(TMessage(Message)) end; procedure TExtHistoryGrid.WMChar(var Message: TWMChar); begin inherited; if FSavedKeyMessage.CharCode = 0 then exit; if Message.CharCode <> 0 then SendMsgFilterMessage(TMessage(Message)) end; procedure TExtHistoryGrid.WMDestroy(var Message: TWMDestroy); begin if not(csDestroyingHandle in ControlState) then FCachedHandle := Handle; inherited; end; procedure TExtHistoryGrid.WMNCDestroy(var Message: TWMNCDestroy); begin inherited; if not(csDestroyingHandle in ControlState) then if Assigned(FOnDestroyWindow) then FOnDestroyWindow(Self, FCachedHandle); end; { TExternalGrid } function TExternalGrid.Perform(Msg: Cardinal; WParam:WPARAM; LParam: LPARAM): LRESULT; var M: TMessage; begin M.Msg := Msg; M.WParam := WParam; M.LParam := LParam; Dispatch(M); Result := M.Result; end; procedure TExternalGrid.AddEvent(hContact:TMCONTACT; hDBEvent: THandle; Codepage: Integer; RTL: Boolean; DoScroll: Boolean); var RTLMode: TRTLMode; begin SetLength(Items, Length(Items) + 1); Items[High(Items)].hDBEvent := hDBEvent; Items[High(Items)].hContact := hContact; Items[High(Items)].Codepage := Codepage; Items[High(Items)].Custom := False; if RTL then RTLMode := hppRTLEnable else RTLMode := hppRTLDefault; Items[High(Items)].RTLMode := RTLMode; if THandle(Grid.Contact) <> hContact then begin Grid.Contact := hContact; Grid.Protocol := GetContactProto(hContact, FSubContact, FSubProtocol); FExternalRTLMode := RTLMode; UseHistoryRTLMode := GetDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryRTLMode', FUseHistoryRTLMode); FExternalCodepage := Codepage; UseHistoryRTLMode := GetDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryCodepage', FUseHistoryCodepage); end; // comment or we'll get rerendering the whole grid // if Grid.Codepage <> Codepage then Grid.Codepage := Codepage; Grid.Allocate(Length(Items), DoScroll and (Grid.State <> gsInline)); end; procedure TExternalGrid.AddCustomEvent(hContact: THandle; const CustomItem: TExtCustomItem; Codepage: Integer; RTL: Boolean; DoScroll: Boolean); var RTLMode: TRTLMode; begin SetLength(Items, Length(Items) + 1); Items[High(Items)].hDBEvent := CustomItem.hEvent; Items[High(Items)].hContact := hContact; Items[High(Items)].Codepage := Codepage; Items[High(Items)].Custom := True; Items[High(Items)].CustomEvent.Nick := CustomItem.Nick; Items[High(Items)].CustomEvent.Text := CustomItem.Text; Items[High(Items)].CustomEvent.Sent := CustomItem.Sent; Items[High(Items)].CustomEvent.Time := CustomItem.Time; if RTL then RTLMode := hppRTLEnable else RTLMode := hppRTLDefault; Items[High(Items)].RTLMode := RTLMode; if THandle(Grid.Contact) <> hContact then begin Grid.Contact := hContact; Grid.Protocol := GetContactProto(hContact, FSubContact, FSubProtocol); FExternalRTLMode := RTLMode; UseHistoryRTLMode := GetDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryRTLMode', FUseHistoryRTLMode); FExternalCodepage := Codepage; UseHistoryRTLMode := GetDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryCodepage', FUseHistoryCodepage); end; // comment or we'll get rerendering the whole grid // if Grid.Codepage <> Codepage then Grid.Codepage := Codepage; Grid.Allocate(Length(Items), DoScroll and (Grid.State <> gsInline)); end; function RadioItem(Value: Boolean; mi: TMenuItem): TMenuItem; begin Result := mi; Result.RadioItem := Value; end; constructor TExternalGrid.Create(AParentWindow: HWND; ControlID: Cardinal = 0); begin FParentWindow := AParentWindow; WasKeyPressed := False; FUseHistoryRTLMode := False; FExternalRTLMode := hppRTLDefault; FUseHistoryCodepage := False; FExternalCodepage := CP_ACP; FSelection := nil; FGridState := gsIdle; RecentFormat := sfHtml; Grid := TExtHistoryGrid.CreateParented(ParentWindow); Grid.Reversed := False; Grid.ShowHeaders := True; Grid.ReversedHeader := True; Grid.ExpandHeaders := GetDBBool(hppDBName, 'ExpandLogHeaders', False); Grid.HideSelection := True; Grid.ControlID := ControlID; Grid.ParentCtl3D := False; Grid.Ctl3D := True; Grid.ParentColor := False; Grid.Color := clBtnFace; Grid.BevelEdges := [beLeft, beTop, beRight, beBottom]; Grid.BevelKind := bkNone; Grid.BevelInner := bvNone; Grid.BevelOuter := bvNone; Grid.BevelWidth := 1; if GetDBBool(hppDBName, 'NoLogBorder', False) then Grid.BorderStyle := bsNone else Grid.BorderStyle := bsSingle; Grid.BorderWidth := 0; Grid.HideScrollBar := GetDBBool(hppDBName, 'NoLogScrollBar', False); Grid.OnItemData := GridItemData; Grid.OnTranslateTime := GridTranslateTime; Grid.OnNameData := GridNameData; Grid.OnProcessRichText := GridProcessRichText; Grid.OnUrlClick := GridUrlClick; Grid.OnBookmarkClick := GridBookmarkClick; Grid.OnSelectRequest := GridSelectRequest; Grid.OnDblClick := GridDblClick; Grid.OnKeyDown := GridKeyDown; Grid.OnKeyUp := GridKeyUp; Grid.OnPopup := GridPopup; Grid.OnInlinePopup := GridPopup; Grid.OnInlineKeyDown := GridInlineKeyDown; Grid.OnItemDelete := GridItemDelete; Grid.OnXMLData := GridXMLData; Grid.OnMCData := GridMCData; Grid.TxtFullLog := TranslateUnicodeString(Grid.TxtFullLog { TRANSLATE-IGNORE } ); Grid.TxtGenHist1 := TranslateUnicodeString(Grid.TxtGenHist1 { TRANSLATE-IGNORE } ); Grid.TxtGenHist2 := TranslateUnicodeString(Grid.TxtGenHist2 { TRANSLATE-IGNORE } ); Grid.TxtHistExport := TranslateUnicodeString(Grid.TxtHistExport { TRANSLATE-IGNORE } ); Grid.TxtNoItems := ''; Grid.TxtNoSuch := TranslateUnicodeString(Grid.TxtNoSuch { TRANSLATE-IGNORE } ); Grid.TxtPartLog := TranslateUnicodeString(Grid.TxtPartLog { TRANSLATE-IGNORE } ); Grid.TxtStartUp := TranslateUnicodeString(Grid.TxtStartUp { TRANSLATE-IGNORE } ); Grid.TxtSessions := TranslateUnicodeString(Grid.TxtSessions { TRANSLATE-IGNORE } ); Grid.Options := GridOptions; Grid.GroupLinked := GetDBBool(hppDBName, 'GroupLogItems', False); pmGrid := TPopupMenu.Create(Grid); pmGrid.ParentBiDiMode := False; pmGrid.Items.Add(NewItem('Sh&ow in history', 0, False, True, OnOpenClick, 0, 'pmOpen')); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN1')); pmGrid.Items.Add(NewItem('&Copy', TextToShortCut('Ctrl+C'), False, True, OnCopyClick, 0, 'pmCopy')); pmGrid.Items.Add(NewItem('Copy &Text', TextToShortCut('Ctrl+T'), False, True, OnCopyTextClick, 0, 'pmCopyText')); pmGrid.Items.Add(NewItem('Select &All', TextToShortCut('Ctrl+A'), False, True, OnSelectAllClick, 0, 'pmSelectAll')); pmGrid.Items.Add(NewItem('&Delete', TextToShortCut('Del'), False, True, OnDeleteClick, 0, 'pmDelete')); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN2')); pmGrid.Items.Add(NewItem('Text Formatting', TextToShortCut('Ctrl+P'), False, True, OnTextFormattingClick, 0, 'pmTextFormatting')); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN3')); pmGrid.Items.Add(NewItem('&Reply Quoted', TextToShortCut('Ctrl+R'), False, True, OnReplyQuotedClick, 0, 'pmReplyQuoted')); pmGrid.Items.Add(NewItem('Set &Bookmark', TextToShortCut('Ctrl+B'), False, True, OnBookmarkClick, 0, 'pmBookmark')); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN4')); pmGrid.Items.Add(NewItem('&Save Selected...', TextToShortCut('Ctrl+S'), False, True, OnSaveSelectedClick, 0, 'pmSaveSelected')); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN5')); pmGrid.Items.Add(NewSubMenu('&File Actions', 0, 'pmFileActions', [NewItem('&Browse Received Files', 0, False, True, OnBrowseReceivedFilesClick, 0,'pmBrowseReceivedFiles'), NewItem('&Open file folder', 0, False, True, OnOpenFileFolderClick, 0, 'pmOpenFileFolder'), NewItem('-', 0, False, True, nil, 0, 'pmN7'), NewItem('&Copy Filename', 0, False, True, OnCopyLinkClick, 0, 'pmCopyLink')], True)); pmGrid.Items.Add(NewSubMenu('Text direction', 0, 'pmBidiMode', [RadioItem(True, NewItem('Log default', 0, True, True, OnBidiModeLogClick, 0, 'pmBidiModeLog')), RadioItem(True, NewItem('History default', 0, False, True, OnBidiModeHistoryClick, 0, 'pmBidiModeHistory'))], True)); pmGrid.Items.Add(NewSubMenu('ANSI Encoding', 0, 'pmCodepage', [RadioItem(True, NewItem('Log default', 0, True, True, OnCodepageLogClick, 0, 'pmCodepageLog')), RadioItem(True, NewItem('History default', 0, False, True, OnCodepageHistoryClick, 0, 'pmCodepageHistory'))], True)); pmGrid.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN6')); miEventsFilter := TMenuItem.Create(pmGrid); miEventsFilter.Caption := 'Events filter'; pmGrid.Items.Add(miEventsFilter); pmLink := TPopupMenu.Create(Grid); pmLink.ParentBiDiMode := False; pmLink.Items.Add(NewItem('Open &Link', 0, False, True, OnOpenLinkClick, 0, 'pmOpenLink')); pmLink.Items.Add(NewItem('Open Link in New &Window', 0, False, True, OnOpenLinkNWClick, 0, 'pmOpenLinkNW')); pmLink.Items.Add(NewItem('-', 0, False, True, nil, 0, 'pmN4')); pmLink.Items.Add(NewItem('&Copy Link', 0, False, True, OnCopyLinkClick, 0, 'pmCopyLink')); TranslateMenu(pmGrid.Items); TranslateMenu(pmLink.Items); CreateEventsFilterMenu; SetEventFilter(GetShowAllEventsIndex); end; destructor TExternalGrid.Destroy; begin WriteDBBool(hppDBName, 'ExpandLogHeaders', Grid.ExpandHeaders); if FSelection <> nil then FreeMem(FSelection); Grid.Free; Finalize(Items); inherited; end; function TExternalGrid.GetGridHandle: HWND; begin Result := Grid.CachedHandle; end; procedure TExternalGrid.BeginUpdate; begin Grid.BeginUpdate; end; procedure TExternalGrid.EndUpdate; begin Grid.EndUpdate; end; procedure TExternalGrid.GridItemData(Sender: TObject; Index: Integer; var Item: THistoryItem); const Direction: array [False .. True] of TMessageTypes = ([mtIncoming], [mtOutgoing]); var PrevTimestamp: DWord; Codepage: Cardinal; begin if FUseHistoryCodepage then Codepage := Grid.Codepage else Codepage := Items[Index].Codepage; if Items[Index].Custom then begin Item.Height := -1; Item.Time := Items[Index].CustomEvent.Time; Item.MessageType := [mtOther] + Direction[Items[Index].CustomEvent.Sent]; Item.Text := Items[Index].CustomEvent.Text; Item.IsRead := True; end else begin Item := ReadEvent(Items[Index].hDBEvent, Codepage); Item.Bookmarked := BookmarkServer[Items[Index].hContact].Bookmarked[Items[Index].hDBEvent]; end; Item.Proto := Grid.Protocol; if Index = 0 then Item.HasHeader := IsEventInSession(Item.EventType) else begin if Items[Index].Custom then PrevTimestamp := Items[Index - 1].CustomEvent.Time else PrevTimestamp := GetEventTimestamp(Items[Index - 1].hDBEvent); if IsEventInSession(Item.EventType) then Item.HasHeader := ((DWord(Item.Time) - PrevTimestamp) > SESSION_TIMEDIFF); if (not Item.Bookmarked) and (Item.MessageType = Grid.Items[Index - 1].MessageType) then Item.LinkedToPrev := ((DWord(Item.Time) - PrevTimestamp) < 60); end; if (not FUseHistoryRTLMode) and (Item.RTLMode <> hppRTLEnable) then Item.RTLMode := Items[Index].RTLMode; // tabSRMM still doesn't marks events read in case of hpp log is in use... // if (FGridMode = gmIEView) and if (mtIncoming in Item.MessageType) and (MessageTypesToDWord(Item.MessageType) and MessageTypesToDWord([mtMessage, mtUrl]) > 0) then begin if (not Item.IsRead) then db_event_markRead(Items[Index].hContact, Items[Index].hDBEvent); Clist_RemoveEvent(Items[Index].hContact, Items[Index].hDBEvent); end else if (not Item.IsRead) and (MessageTypesToDWord(Item.MessageType) and MessageTypesToDWord([mtStatus, mtNickChange, mtAvatarChange]) > 0) then begin db_event_markRead(Items[Index].hContact, Items[Index].hDBEvent); end; end; procedure TExternalGrid.GridTranslateTime(Sender: TObject; Time: DWord; var Text: String); begin Text := TimestampToString(Time); end; procedure TExternalGrid.GridNameData(Sender: TObject; Index: Integer; var Name: String); begin if Name = '' then begin if Grid.Protocol = '' then begin if Items[Index].hContact = 0 then begin Grid.Protocol := 'ICQ'; FSubProtocol := Grid.Protocol; end else Grid.Protocol := GetContactProto(Items[Index].hContact, FSubContact, FSubProtocol); end; if Items[Index].Custom then Name := Items[Index].CustomEvent.Nick else if mtIncoming in Grid.Items[Index].MessageType then begin Grid.ContactName := GetContactDisplayName(Items[Index].hContact, Grid.Protocol, True); Name := Grid.ContactName; end else begin Grid.ProfileName := GetContactDisplayName(0, FSubProtocol); Name := Grid.ProfileName; end; end; end; procedure TExternalGrid.GridProcessRichText(Sender: TObject; Handle: THandle; Item: Integer); var ItemRenderDetails: TItemRenderDetails; begin ZeroMemory(@ItemRenderDetails, SizeOf(ItemRenderDetails)); ItemRenderDetails.cbSize := SizeOf(ItemRenderDetails); // use meta's subcontact info, if available // ItemRenderDetails.hContact := Items[Item].hContact; ItemRenderDetails.hContact := FSubContact; ItemRenderDetails.hDBEvent := Items[Item].hDBEvent; // use meta's subcontact info, if available // ItemRenderDetails.pProto := PAnsiChar(Grid.Items[Item].Proto); ItemRenderDetails.pProto := PAnsiChar(FSubProtocol); ItemRenderDetails.pModule := PAnsiChar(Grid.Items[Item].Module); ItemRenderDetails.pText := nil; ItemRenderDetails.pExtended := PAnsiChar(Grid.Items[Item].Extended); ItemRenderDetails.dwEventTime := Grid.Items[Item].Time; ItemRenderDetails.wEventType := Grid.Items[Item].EventType; ItemRenderDetails.IsEventSent := (mtOutgoing in Grid.Items[Item].MessageType); if Handle = Grid.InlineRichEdit.Handle then ItemRenderDetails.dwFlags := ItemRenderDetails.dwFlags or IRDF_INLINE; if Grid.IsSelected(Item) then ItemRenderDetails.dwFlags := ItemRenderDetails.dwFlags or IRDF_SELECTED; ItemRenderDetails.bHistoryWindow := IRDHW_EXTERNALGRID; NotifyEventHooks(hHppRichEditItemProcess, WParam(Handle), LParam(@ItemRenderDetails)); end; procedure TExternalGrid.ScrollToBottom; begin if Grid.State <> gsInline then begin Grid.ScrollToBottom; Grid.Invalidate; end; end; procedure TExternalGrid.SetPosition(x, y, cx, cy: Integer); begin Grid.Left := x; Grid.Top := y; Grid.Width := cx; Grid.Height := cy; if Grid.HandleAllocated then SetWindowPos(Grid.Handle, 0, x, y, cx, cy, SWP_SHOWWINDOW); end; function TExternalGrid.GetSelection(): PAnsiChar; var TextW: WideString; Source: Pointer; Size: Integer; begin TextW := Grid.SelectionString; if Length(TextW) > 0 then begin TextW := TextW + #0; Source := @TextW[1]; Size := Length(TextW) * SizeOf(WideChar); ReallocMem(FSelection, Size); Move(Source^, FSelection^, Size); Result := FSelection; end else Result := nil; end; procedure TExternalGrid.Clear; begin Finalize(Items); Grid.Allocate(0); // Grid.Repaint; end; procedure TExternalGrid.GridUrlClick(Sender: TObject; Item: Integer; URLText: String; Button: TMouseButton); begin if URLText = '' then exit; if (Button = mbLeft) or (Button = mbMiddle) then OpenUrl(URLText, True) else if Button = mbRight then begin SavedLinkUrl := URLText; pmLink.Popup(Mouse.CursorPos.x, Mouse.CursorPos.y); end; end; procedure TExternalGrid.GridBookmarkClick(Sender: TObject; Item: Integer); var val: Boolean; hContact, hDBEvent: THandle; begin if Items[Item].Custom then exit; hContact := Items[Item].hContact; hDBEvent := Items[Item].hDBEvent; val := not BookmarkServer[hContact].Bookmarked[hDBEvent]; BookmarkServer[hContact].Bookmarked[hDBEvent] := val; end; procedure TExternalGrid.HMBookmarkChanged(var M: TMessage); var i: Integer; begin if M.WParam <> Grid.Contact then exit; for i := 0 to Grid.Count - 1 do if Items[i].hDBEvent = THandle(M.LParam) then begin Grid.Bookmarked[i] := BookmarkServer[M.WParam].Bookmarked[M.LParam]; Grid.ResetItem(i); Grid.Invalidate; exit; end; end; procedure TExternalGrid.GridSelectRequest(Sender: TObject); begin if (Grid.Selected <> -1) and Grid.IsVisible(Grid.Selected) then exit; if Grid.Count > 0 then Grid.Selected := Grid.BottomItem; end; procedure TExternalGrid.GridDblClick(Sender: TObject); begin if Grid.Selected = -1 then exit; Grid.EditInline(Grid.Selected); end; procedure TExternalGrid.GridKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState); begin if (Shift = [ssCtrl]) and (Key = VK_INSERT) then Key := Ord('C'); if IsFormShortCut([pmGrid], Key, Shift) then begin Key := 0; exit; end; WasKeyPressed := (Key in [VK_RETURN, VK_ESCAPE]); end; procedure TExternalGrid.GridKeyUp(Sender: TObject; var Key: Word; Shift: TShiftState); begin if not WasKeyPressed then exit; WasKeyPressed := False; if (Key = VK_RETURN) and (Shift = []) then begin GridDblClick(Grid); Key := 0; end; if (Key = VK_RETURN) and (Shift = [ssCtrl]) then begin OnOpenClick(Grid); Key := 0; end; if (Key = VK_ESCAPE) and (Shift = []) then begin PostMessage(FParentWindow, WM_CLOSE, 0, 0); Key := 0; end; end; function TExternalGrid.IsFileEvent(Index: Integer): Boolean; begin Result := (Index <> -1) and (mtFile in Grid.Items[Index].MessageType); if Result then begin // Auto CP_ACP usage SavedLinkUrl := ExtractFileName(String(Grid.Items[Index].Extended)); SavedFileDir := ExtractFileDir(String(Grid.Items[Index].Extended)); end; end; procedure TExternalGrid.GridPopup(Sender: TObject); var GridSelected: Boolean; begin GridSelected := (Grid.Selected <> -1); pmGrid.Items[0].Visible := GridSelected and (Grid.State = gsIdle) and not Items[Grid.Selected].Custom; pmGrid.Items[2].Visible := GridSelected; pmGrid.Items[3].Visible := GridSelected; pmGrid.Items[4].Visible := GridSelected and (Grid.State = gsInline); // works even if not in pseudo-edit pmGrid.Items[5].Visible := GridSelected; pmGrid.Items[7].Visible := GridSelected and (Grid.State = gsInline); pmGrid.Items[8].Visible := GridSelected; if GridSelected then begin pmGrid.Items[7].Checked := GridOptions.TextFormatting; if Grid.State = gsInline then pmGrid.Items[2].Enabled := Grid.InlineRichEdit.SelLength > 0 else pmGrid.Items[2].Enabled := True; pmGrid.Items[8].Enabled := pmGrid.Items[2].Enabled; end; pmGrid.Items[9].Visible := GridSelected and not Items[Grid.Selected].Custom; pmGrid.Items[10].Visible := GridSelected; if GridSelected then begin if Items[Grid.Selected].Custom then pmGrid.Items[10].Visible := False else if Grid.Items[Grid.Selected].Bookmarked then TMenuItem(pmGrid.Items[10]).Caption := TranslateW('Remove &Bookmark') else TMenuItem(pmGrid.Items[10]).Caption := TranslateW('Set &Bookmark'); end; pmGrid.Items[12].Visible := (Grid.SelCount > 1); pmGrid.Items[14].Visible := GridSelected and IsFileEvent(Grid.Selected); if pmGrid.Items[14].Visible then pmGrid.Items[14].Items[1].Visible := (SavedFileDir <> ''); pmGrid.Items[15].Visible := (Grid.State = gsIdle); pmGrid.Items[15].Items[0].Checked := not FUseHistoryRTLMode; pmGrid.Items[15].Items[1].Checked := FUseHistoryRTLMode; pmGrid.Items[16].Visible := (Grid.State = gsIdle); pmGrid.Items[16].Items[0].Checked := not FUseHistoryCodepage; pmGrid.Items[16].Items[1].Checked := FUseHistoryCodepage; pmGrid.Items[18].Visible := (Grid.State = gsIdle); pmGrid.Popup(Mouse.CursorPos.x, Mouse.CursorPos.y); end; procedure TExternalGrid.OnCopyClick(Sender: TObject); begin if Grid.Selected = -1 then exit; if Grid.State = gsInline then begin if Grid.InlineRichEdit.SelLength = 0 then exit; Grid.InlineRichEdit.CopyToClipboard; end else begin CopyToClip(Grid.FormatSelected(GridOptions.ClipCopyFormat), Grid.Handle, Items[Grid.Selected].Codepage); end; end; procedure TExternalGrid.OnCopyTextClick(Sender: TObject); var cr: TCharRange; begin if Grid.Selected = -1 then exit; if Grid.State = gsInline then begin Grid.InlineRichEdit.Lines.BeginUpdate; Grid.InlineRichEdit.Perform(EM_EXGETSEL, 0, LParam(@cr)); Grid.InlineRichEdit.SelectAll; Grid.InlineRichEdit.CopyToClipboard; Grid.InlineRichEdit.Perform(EM_EXSETSEL, 0, LParam(@cr)); Grid.InlineRichEdit.Lines.EndUpdate; end else CopyToClip(Grid.FormatSelected(GridOptions.ClipCopyTextFormat), Grid.Handle, Items[Grid.Selected].Codepage); end; procedure TExternalGrid.OnSelectAllClick(Sender: TObject); begin if Grid.State = gsInline then begin if Grid.Selected = -1 then exit; Grid.InlineRichEdit.SelectAll; end else begin Grid.SelectAll; end; end; procedure TExternalGrid.OnDeleteClick(Sender: TObject); begin if Grid.SelCount = 0 then exit; if Grid.SelCount > 1 then begin if HppMessageBox(FParentWindow, WideFormat(TranslateW('Do you really want to delete selected items (%.0f)?'), [Grid.SelCount / 1]), TranslateW('Delete Selected'), MB_YESNOCANCEL or MB_DEFBUTTON1 or MB_ICONQUESTION) <> IDYES then exit; end else begin if HppMessageBox(FParentWindow, TranslateW('Do you really want to delete selected item?'), TranslateW('Delete'), MB_YESNOCANCEL or MB_DEFBUTTON1 or MB_ICONQUESTION) <> IDYES then exit; end; SetSafetyMode(False); try FGridState := gsDelete; Grid.DeleteSelected; finally FGridState := gsIdle; SetSafetyMode(True); end; end; procedure TExternalGrid.OnTextFormattingClick(Sender: TObject); begin if (Grid.Selected = -1) or (Grid.State <> gsInline) then exit; GridOptions.TextFormatting := not GridOptions.TextFormatting; end; procedure TExternalGrid.OnReplyQuotedClick(Sender: TObject); begin if Grid.Selected = -1 then exit; if Grid.State = gsInline then begin if Grid.InlineRichEdit.SelLength = 0 then exit; SendMessageTo(Items[Grid.Selected].hContact, Grid.FormatSelected(GridOptions.ReplyQuotedTextFormat)); end else begin // if (hContact = 0) or (hg.SelCount = 0) then exit; SendMessageTo(Items[Grid.Selected].hContact, Grid.FormatSelected(GridOptions.ReplyQuotedFormat)); end; end; procedure TExternalGrid.OnBookmarkClick(Sender: TObject); var val: Boolean; hContact, hDBEvent: THandle; begin if Grid.Selected = -1 then exit; if Items[Grid.Selected].Custom then exit; hContact := Items[Grid.Selected].hContact; hDBEvent := Items[Grid.Selected].hDBEvent; val := not BookmarkServer[hContact].Bookmarked[hDBEvent]; BookmarkServer[hContact].Bookmarked[hDBEvent] := val; end; procedure TExternalGrid.OnOpenClick(Sender: TObject); var oep: TOpenEventParams; begin if Grid.Selected = -1 then exit; if Items[Grid.Selected].Custom then exit; oep.cbSize := SizeOf(oep); oep.hContact := Items[Grid.Selected].hContact; oep.hDBEvent := Items[Grid.Selected].hDBEvent; CallService(MS_HPP_OPENHISTORYEVENT, WParam(@oep), 0); end; procedure TExternalGrid.GridInlineKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState); begin if IsFormShortCut([pmGrid], Key, Shift) then begin Key := 0; exit; end; end; procedure TExternalGrid.OnOpenLinkClick(Sender: TObject); begin if SavedLinkUrl = '' then exit; OpenUrl(SavedLinkUrl, False); SavedLinkUrl := ''; end; procedure TExternalGrid.GridItemDelete(Sender: TObject; Index: Integer); begin if (FGridState = gsDelete) and (Items[Index].hDBEvent <> 0) then db_event_delete(Items[Index].hDBEvent); if Index <> High(Items) then begin Finalize(Items[Index]); Move(Items[Index + 1], Items[Index], (Length(Items) - Index - 1) * SizeOf(Items[0])); ZeroMemory(@Items[High(Items)], SizeOf(Items[0])); // reset has_header and linked_to_pervous_messages fields Grid.ResetItem(Index); end; SetLength(Items, Length(Items) - 1); // Application.ProcessMessages; end; procedure TExternalGrid.OnOpenLinkNWClick(Sender: TObject); begin if SavedLinkUrl = '' then exit; OpenUrl(SavedLinkUrl, True); SavedLinkUrl := ''; end; procedure TExternalGrid.OnCopyLinkClick(Sender: TObject); begin if SavedLinkUrl = '' then exit; CopyToClip(SavedLinkUrl, Grid.Handle, CP_ACP); SavedLinkUrl := ''; end; procedure TExternalGrid.OnBidiModeLogClick(Sender: TObject); begin UseHistoryRTLMode := False; WriteDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryRTLMode', UseHistoryRTLMode); end; procedure TExternalGrid.OnBidiModeHistoryClick(Sender: TObject); begin UseHistoryRTLMode := True; WriteDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryRTLMode', UseHistoryRTLMode); end; procedure TExternalGrid.SetUseHistoryRTLMode(const Value: Boolean); begin if FUseHistoryRTLMode = Value then exit; FUseHistoryRTLMode := Value; if FUseHistoryRTLMode then Grid.RTLMode := GetContactRTLModeTRTL(Grid.Contact, Grid.Protocol) else Grid.RTLMode := FExternalRTLMode; end; procedure TExternalGrid.OnCodepageLogClick(Sender: TObject); begin UseHistoryCodepage := False; WriteDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryCodepage', UseHistoryCodepage); end; procedure TExternalGrid.OnCodepageHistoryClick(Sender: TObject); begin UseHistoryCodepage := True; WriteDBBool(Grid.Contact, Grid.Protocol, 'UseHistoryCodepage', UseHistoryCodepage); end; procedure TExternalGrid.SetUseHistoryCodepage(const Value: Boolean); begin if FUseHistoryCodepage = Value then exit; FUseHistoryCodepage := Value; if FUseHistoryCodepage then Grid.Codepage := GetContactCodePage(Grid.Contact, Grid.Protocol) else Grid.Codepage := FExternalCodepage; end; procedure TExternalGrid.SetGroupLinked(const Value: Boolean); begin if Grid.GroupLinked = Value then exit; Grid.GroupLinked := Value; end; procedure TExternalGrid.SetShowHeaders(const Value: Boolean); begin if Grid.ShowHeaders = Value then exit; Grid.ShowHeaders := Value; end; procedure TExternalGrid.SetShowBookmarks(const Value: Boolean); begin if Grid.ShowBookmarks = Value then exit; Grid.ShowBookmarks := Value; end; procedure TExternalGrid.HMEventDeleted(var M: TMessage); var i: Integer; begin if Grid.State = gsDelete then exit; if Grid.Contact <> M.WParam then exit; for i := 0 to Grid.Count - 1 do begin if (Items[i].hDBEvent = uint_ptr(M.LParam)) then begin Grid.Delete(i); exit; end; end; end; procedure TExternalGrid.HMNickChanged(var M: TMessage); begin if FSubProtocol = '' then exit; Grid.BeginUpdate; if M.WParam = 0 then Grid.ProfileName := GetContactDisplayName(0, FSubProtocol) else if Grid.Contact = M.WParam then begin Grid.ProfileName := GetContactDisplayName(0, FSubProtocol); Grid.ContactName := GetContactDisplayName(Grid.Contact, Grid.Protocol, True) end; Grid.EndUpdate; Grid.Invalidate; end; procedure TExternalGrid.HMMetaDefaultChanged(var M: TMessage); var newSubContact: TMCONTACT; newSubProtocol: AnsiString; begin if Grid.Contact <> M.WParam then exit; GetContactProto(Grid.Contact, newSubContact, newSubProtocol); if (FSubContact <> newSubContact) or (FSubProtocol <> newSubProtocol) then begin Grid.BeginUpdate; FSubContact := newSubContact; FSubProtocol := newSubProtocol; Grid.ProfileName := GetContactDisplayName(0, FSubProtocol); Grid.ContactName := GetContactDisplayName(Grid.Contact, Grid.Protocol, True); Grid.GridUpdate([guOptions]); Grid.EndUpdate; // Grid.Invalidate; end; end; procedure TExternalGrid.OnSaveSelectedClick(Sender: TObject); var t: String; SaveFormat: TSaveFormat; begin if Grid.Selected = -1 then exit; RecentFormat := TSaveFormat(GetDBInt(hppDBName, 'ExportFormat', 0)); SaveFormat := RecentFormat; if not Assigned(SaveDialog) then begin SaveDialog := TSaveDialog.Create(Grid); SaveDialog.Title := TranslateW('Save History'); SaveDialog.Options := [ofOverwritePrompt, ofHideReadOnly, ofPathMustExist, ofShareAware, ofEnableSizing]; end; PrepareSaveDialog(SaveDialog, SaveFormat, True); t := TranslateW('Partial History [%s] - [%s]'); t := Format(t, [Grid.ProfileName, Grid.ContactName]); t := MakeFileName(t); SaveDialog.FileName := t; if not SaveDialog.Execute then exit; for SaveFormat := High(SaveFormats) downto Low(SaveFormats) do if SaveDialog.FilterIndex = SaveFormats[SaveFormat].Index then break; if SaveFormat <> sfAll then RecentFormat := SaveFormat; Grid.SaveSelected(SaveDialog.Files[0], SaveFormat); WriteDBInt(hppDBName, 'ExportFormat', Integer(RecentFormat)); end; procedure TExternalGrid.SaveSelected; begin OnSaveSelectedClick(Self); end; procedure TExternalGrid.GridXMLData(Sender: TObject; Index: Integer; var Item: TXMLItem); var tmp: Utf8String; dt: TDateTime; mes: String; begin dt := TimestampToDateTime(Grid.Items[Index].Time); Item.Time := MakeTextXMLedA(FormatDateTime('hh:mm:ss', dt)); Item.Date := MakeTextXMLedA(FormatDateTime('yyyy-mm-dd', dt)); Item.Contact := MakeTextXMLedW(Grid.ContactName); if mtIncoming in Grid.Items[Index].MessageType then Item.From := Item.Contact else Item.From := '&ME;'; Item.EventType := '&' + GetEventRecord(Grid.Items[Index]).XML + ';'; mes := Grid.Items[Index].Text; if GridOptions.RawRTFEnabled and IsRTF(mes) then begin Grid.ApplyItemToRich(Index); mes := GetRichString(Grid.RichEdit.Handle, False); end; if GridOptions.BBCodesEnabled then mes := DoStripBBCodes(mes); Item.mes := MakeTextXMLedW(mes); if mtFile in Grid.Items[Index].MessageType then begin tmp := Grid.Items[Index].Extended; if tmp = '' then Item.FileName := '&UNK;' else Item.FileName := MakeTextXMLedA(tmp); end else if mtAvatarChange in Grid.Items[Index].MessageType then begin tmp := Grid.Items[Index].Extended; if tmp = '' then Item.FileName := '&UNK;' else Item.FileName := MakeTextXMLedA(tmp); end; { 2.8.2004 OXY: Change protocol guessing order. Now first use protocol name, then, if missing, use module } Item.Protocol := Grid.Items[Index].Proto; if Item.Protocol = '' then Item.Protocol := MakeTextXMLedA(Grid.Items[Index].Module); if Item.Protocol = '' then Item.Protocol := '&UNK;'; if mtIncoming in Grid.Items[Index].MessageType then Item.ID := GetContactID(Grid.Contact, Grid.Protocol, True) else Item.ID := GetContactID(0, Grid.Protocol); if Item.ID = '' then Item.ID := '&UNK;' else Item.ID := MakeTextXMLedA(Item.ID); end; procedure TExternalGrid.GridMCData(Sender: TObject; Index: Integer; var Item: TMCItem; Stage: TSaveStage); var DBEventInfo: TOldDBEventInfo; hDBEvent: THandle; DataOffset: PAnsiChar; TextUTF: Utf8String; begin if Stage = ssInit then begin Item.Size := 0; if Items[Index].Custom then begin ZeroMemory(@DBEventInfo, SizeOf(DBEventInfo)); DBEventInfo.cbSize := SizeOf(DBEventInfo); DBEventInfo.timestamp := Items[Index].CustomEvent.Time; DBEventInfo.flags := DBEF_READ or DBEF_UTF; if Items[Index].CustomEvent.Sent then DBEventInfo.flags := DBEventInfo.flags or DBEF_SENT; DBEventInfo.EventType := EVENTTYPE_MESSAGE; TextUTF := Items[Index].CustomEvent.Text + #0; DBEventInfo.cbBlob := Length(TextUTF) + 1; DBEventInfo.pBlob := Pointer(TextUTF); Item.Size := Cardinal(DBEventInfo.cbSize) + Cardinal(DBEventInfo.cbBlob); end else begin hDBEvent := Items[Index].hDBEvent; if hDBEvent <> 0 then begin DBEventInfo.cbSize := SizeOf(DBEventInfo); DBEventInfo := GetOldEventInfo(hDBEvent); DBEventInfo.szModule := nil; Item.Size := Cardinal(DBEventInfo.cbSize) + Cardinal(DBEventInfo.cbBlob); end; end; if Item.Size > 0 then begin GetMem(Item.Buffer, Item.Size); DataOffset := PAnsiChar(Item.Buffer) + DBEventInfo.cbSize; Move(DBEventInfo, Item.Buffer^, DBEventInfo.cbSize); Move(DBEventInfo.pBlob^, DataOffset^, DBEventInfo.cbBlob); end; end else if Stage = ssDone then begin if Item.Size > 0 then FreeMem(Item.Buffer, Item.Size); end; end; procedure TExternalGrid.SetEventFilter(FilterIndex: Integer = -1); var i, fi: Integer; ShowAllEventsIndex: Integer; begin ShowAllEventsIndex := GetShowAllEventsIndex; if FilterIndex = -1 then begin fi := miEventsFilter.Tag + 1; if fi > High(hppEventFilters) then fi := 0; end else begin fi := FilterIndex; if fi > High(hppEventFilters) then fi := ShowAllEventsIndex; end; miEventsFilter.Tag := fi; for i := 0 to miEventsFilter.Count - 1 do miEventsFilter[i].Checked := (miEventsFilter[i].Tag = fi); if fi = ShowAllEventsIndex then Grid.TxtNoSuch := TranslateW('No such items') else Grid.TxtNoSuch := WideFormat(TranslateW('No "%s" items'), [hppEventFilters[fi].Name]); // Grid.ShowHeaders := mtMessage in hppEventFilters[fi].Events; Grid.Filter := hppEventFilters[fi].Events; end; procedure TExternalGrid.HMFiltersChanged(var M: TMessage); begin CreateEventsFilterMenu; SetEventFilter(GetShowAllEventsIndex); // WriteDBInt(hppDBName,'RecentLogFilter',miEventsFilter.Tag); end; procedure TExternalGrid.OnEventsFilterItemClick(Sender: TObject); begin SetEventFilter(TMenuItem(Sender).Tag); // WriteDBInt(hppDBName,'RecentLogFilter',miEventsFilter.Tag); end; procedure TExternalGrid.CreateEventsFilterMenu; var i: Integer; mi: TMenuItem; ShowAllEventsIndex: Integer; begin ShowAllEventsIndex := GetShowAllEventsIndex; miEventsFilter.Clear; for i := 0 to Length(hppEventFilters) - 1 do begin mi := TMenuItem.Create(pmGrid); mi.Caption := StringReplace(hppEventFilters[i].Name, '&', '&&', [rfReplaceAll]); mi.GroupIndex := 1; mi.RadioItem := True; mi.Tag := i; mi.OnClick := OnEventsFilterItemClick; if i = ShowAllEventsIndex then mi.Default := True; miEventsFilter.Insert(i, mi); end; end; procedure TExternalGrid.OnOpenFileFolderClick(Sender: TObject); begin if SavedFileDir = '' then exit; ShellExecuteW(0, 'open', PWideChar(SavedFileDir), nil, nil, SW_SHOW); SavedFileDir := ''; end; procedure TExternalGrid.OnBrowseReceivedFilesClick(Sender: TObject); var Path: Array [0 .. MAX_PATH] of AnsiChar; begin if Grid.Selected = -1 then exit; CallService(MS_FILE_GETRECEIVEDFILESFOLDER, Items[Grid.Selected].hContact,LParam(@Path)); ShellExecuteA(0, 'open', Path, nil, nil, SW_SHOW); end; end.