{ Copyright (C) 2005-2014 Andrew Haines, Lazarus contributors Main form for lhelp. Includes processing/data communication. This source 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 code 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. A copy of the GNU General Public License is available on the World Wide Web at . You can also obtain it by writing to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. } unit lhelpcore; {$IFDEF LNET_VISUAL} {$DEFINE USE_LNET} // you must manually add the lnetvisual.lpk package to the dependancy list {$ELSE} {$NOTE You can add http capability to lhelp by adding the lnetvisual package v0.6.3 or greater requirement to lhelp.} {$ENDIF} {$IFDEF UNIX} {$if FPC_FULLVERSION <= 20700} {$DEFINE STALE_PIPE_WORKAROUND} {$ENDIF} {$ENDIF} {$mode objfpc}{$H+} interface uses Classes, SysUtils, SimpleIPC, Laz2_XMLCfg, FileUtil, Forms, Controls, Dialogs, Buttons, LCLProc, ComCtrls, ExtCtrls, Menus, LCLType, LCLIntf, StdCtrls, BaseContentProvider, ChmContentProvider {$IFDEF USE_LNET}, HTTPContentProvider{$ENDIF}, lazlogger; type { TContentTab } TContentTab = class(TTabSheet) private fContentProvider: TBaseContentProvider; public constructor Create(AOwner: TComponent); override; destructor Destroy; override; property ContentProvider: TBaseContentProvider read fContentProvider write fContentProvider; end; { THelpForm } THelpForm = class(TForm) ApplicationProperties1: TApplicationProperties; FileMenuCloseItem: TMenuItem; FileMenuExitItem: TMenuItem; FileMenuItem: TMenuItem; FileMenuOpenItem: TMenuItem; FileSeperater: TMenuItem; ImageList1: TImageList; MainMenu1: TMainMenu; FileMenuOpenURLItem: TMenuItem; HelpMenuItem: TMenuItem; AboutItem: TMenuItem; FileMenuOpenRecentItem: TMenuItem; PageControl: TPageControl; Panel1: TPanel; ForwardBttn: TSpeedButton; BackBttn: TSpeedButton; HomeBttn: TSpeedButton; OpenDialog1: TOpenDialog; ViewMenuContents: TMenuItem; ViewMenuItem: TMenuItem; procedure AboutItemClick(Sender: TObject); procedure BackToolBtnClick(Sender: TObject); procedure FileMenuCloseItemClick(Sender: TObject); procedure FileMenuExitItemClick(Sender: TObject); procedure FileMenuOpenItemClick(Sender: TObject); procedure FileMenuOpenURLItemClick(Sender: TObject); procedure FormClose(Sender: TObject; var {%H-}CloseAction: TCloseAction); procedure FormCreate(Sender: TObject); procedure FormKeyUp(Sender: TObject; var Key: Word; {%H-}Shift: TShiftState); procedure FormShow(Sender: TObject); procedure ForwardToolBtnClick(Sender: TObject); procedure HomeToolBtnClick(Sender: TObject); procedure PageControlChange(Sender: TObject); procedure PageControlEnter(Sender: TObject); procedure ViewMenuContentsClick(Sender: TObject); private { private declarations } // SimpleIPC server name (including unique part as per help protocol) fServerName: String; // Receives commands from IDE fInputIPC: TSimpleIPCServer; // Sends responses back to IDE // only used if lhelp started with --ipcname to indicate // IPC communication method should be used fOutputIPC: TSimpleIPCClient; fInputIPCTimer: TTimer; fContext: LongInt; // used once when we are started on the command line with --context fConfig: TXMLConfig; fHasShowed: Boolean; fHide: boolean; //If yes, start with content hidden. Otherwise start normally // Keep track of whether size/position preferences were loaded and applied to form fLayoutApplied: boolean; // Applies layout (size/position/fullscreen) preferences once in lhelp lifetime // Needs LoadPreference to be run first to get fConfig object. procedure ApplyLayoutPreferencesOnce; // Load preferences. Preferences are unique for server-lhelp pairs and plain lhelp procedure LoadPreferences(AIPCName: String); // Saves preferences. Uses existing config loaded by LoadPreferences procedure SavePreferences; // Add filename to recent files (MRU) list procedure AddRecentFile(AFileName: String); procedure ContentTitleChange({%H-}sender: TObject); procedure OpenRecentItemClick(Sender: TObject); // Send response back to server (IDE) // Used to acknowledge commands from the server procedure SendResponse(Response: DWord); // Wait for message from server (IDE) and process procedure ServerMessage(Sender: TObject); // Parse any given command line options procedure ReadCommandLineOptions; // Start simple IPC server/client // ServerName can be the variable passed to --ipcname // It is used both for starting up the local server and to ID the remote server procedure StartComms(ServerName: String); // Stop simple IPC server/client procedure StopComms; // Open specified URL in viewer window function OpenURL(const AURL: String; AContext: THelpContext=-1): DWord; // Open specified URL - presumably used to queue URLs for delayed opening procedure LateOpenURL(Url: PStringItem); function ActivePage: TContentTab; // Update UI visibility procedure RefreshState; procedure ShowError(AError: String); // Set keyup handler for control (and any child controls) procedure SetKeyUp(AControl: TControl); public { public declarations } end; var HelpForm: THelpForm; // Sends messages to the IDE IPCClient: TSimpleIPCClient; // Receives messages from the IDE IPCServer: TSimpleIPCServer; const INVALID_FILE_TYPE = 1; VERSION_STAMP = '2014-10-16'; //used in displaying version in about form etc implementation {$R *.lfm} uses LHelpControl; const DigitsInPID=5; // Number of digits in the formatted PID according to the Help Protocol type TRecentMenuItem = class(TMenuItem) public URL: String; end; { THelpForm } procedure THelpForm.BackToolBtnClick(Sender: TObject); begin if Assigned(ActivePage) then ActivePage.ContentProvider.GoBack; end; procedure THelpForm.AboutItemClick(Sender: TObject); var f: TForm; l: TLabel; b: TButton; begin f := TForm.Create(Application); try f.Caption := 'About'; f.BorderStyle := bsDialog; f.Position := poMainFormCenter; f.Constraints.MinWidth := 150; f.Constraints.MaxWidth := 350; l := TLabel.Create(f); l.Parent := f;; l.Align := alTop; l.BorderSpacing.Around := 6; l.Caption := 'LHelp (CHM file viewer)' + LineEnding + 'Version ' + VERSION_STAMP + LineEnding + LineEnding + 'Copyright (C) Andrew Haines, ' + LineEnding + 'Lazarus contributors'; l.AutoSize := True; l.WordWrap := True; b := TButton.Create(f); b.Parent := f; b.BorderSpacing.Around := 6; b.Anchors := [akTop, akLeft]; b.AnchorSide[akTop].Control := l; b.AnchorSide[akTop].Side := asrBottom; b.AnchorSide[akLeft].Control := f; b.AnchorSide[akLeft].Side := asrCenter; b.Caption := 'Ok'; b.ModalResult := mrOk; f.AutoSize := False; f.AutoSize := True; f.ShowModal; finally f.free; end; end; procedure THelpForm.FileMenuCloseItemClick(Sender: TObject); begin if Assigned(ActivePage) then begin ViewMenuContentsClick(Self); ActivePage.Free; RefreshState; end; end; procedure THelpForm.FileMenuExitItemClick(Sender: TObject); begin Close; end; procedure THelpForm.FileMenuOpenItemClick(Sender: TObject); var TimerWasOn: boolean; begin // Work around bug 25529: Slow dialog boxes for Windows Vista+ with // themes enabled // Stop listening to incoming server messages while busy showing dialog if assigned(fInputIPCTimer) Then begin TimerWasOn := fInputIPCTimer.Enabled; fInputIPCTimer.Enabled := False; end; try if OpenDialog1.Execute then begin if OpenURL('file://'+OpenDialog1.FileName) = Ord(srSuccess) then AddRecentFile('file://'+OpenDialog1.FileName); RefreshState; end; finally if assigned(fInputIPCTimer) Then begin fInputIPCTimer.Enabled := TimerWasOn; end; end; end; procedure THelpForm.FileMenuOpenURLItemClick(Sender: TObject); var fRes: String; URLSAllowed: String; Protocol: TStrings; i: Integer; begin Protocol := GetContentProviderList; try URLSAllowed:=''; for i := 0 to Protocol.Count-1 do begin if i < 1 then URLSAllowed := URLSAllowed + Protocol[i] else URLSAllowed := URLSAllowed + ', ' +Protocol[i] end; finally Protocol.Free; end; URLSAllowed := Trim(URLSAllowed); fRes:=''; if InputQuery('Please enter a URL', 'Supported URL type(s): (' +URLSAllowed+ ')', fRes) then begin if OpenURL(fRes) = ord(srSuccess) then AddRecentFile(fRes); RefreshState; end; end; procedure THelpForm.FormClose(Sender: TObject; var CloseAction: TCloseAction); begin Visible := false; Application.ProcessMessages; FileMenuCloseItemClick(Sender); StopComms; SavePreferences; end; procedure THelpForm.FormCreate(Sender: TObject); begin fContext := -1; // Safe default: fHide := false; // ReadCommandLineOptions will set fHide if requested ReadCommandLineOptions; LoadPreferences(fServerName); // Only start IPC if server name passed in --ipcname if fServerName <> '' then begin StartComms(fServerName); end; // If user wants lhelp to hide, hide entire form. if fHide then WindowState := wsMinimized else RefreshState; SetKeyUp(Self); end; procedure THelpForm.FormKeyUp(Sender: TObject; var Key: Word; Shift: TShiftState); begin if Key = VK_ESCAPE then Close; // Backspace: go to previous page (as if BackBttn were clicked) if Key = VK_BACK then if Assigned(ActivePage) then ActivePage.ContentProvider.GoBack; end; procedure THelpForm.FormShow(Sender: TObject); begin if FHasShowed then Exit; FHasShowed := True; end; procedure THelpForm.ForwardToolBtnClick(Sender: TObject); begin if Assigned(ActivePage) then ActivePage.ContentProvider.GoForward; end; procedure THelpForm.HomeToolBtnClick(Sender: TObject); begin if Assigned(ActivePage) then ActivePage.ContentProvider.GoHome; end; procedure THelpForm.PageControlChange(Sender: TObject); begin RefreshState; end; procedure THelpForm.PageControlEnter(Sender: TObject); begin RefreshState; end; procedure THelpForm.ApplyLayoutPreferencesOnce; begin if not(assigned(fConfig)) then exit; if (not(fHide)) and (not(fLayoutApplied)) then begin if (fConfig.GetValue('Position/Maximized', false)=true) then begin Windowstate:=wsMaximized end else begin Left := fConfig.GetValue('Position/Left/Value', Left); Top := fConfig.GetValue('Position/Top/Value', Top); Width := fConfig.GetValue('Position/Width/Value', Width); Height := fConfig.GetValue('Position/Height/Value', Height); end; // Keep track so we do not reapply initial settings as user may have // changed size etc in the meantime. fLayoutApplied := true; end; end; procedure THelpForm.ViewMenuContentsClick(Sender: TObject); begin // TabsControl property in TChmContentProvider if Assigned(ActivePage) then begin with TChmContentProvider(ActivePage.ContentProvider) do begin TabsControl.Visible := not TabsControl.Visible; Splitter.Visible := TabsControl.Visible; Splitter.Left := TabsControl.Left + 4; //for splitter to move righter ViewMenuContents.Checked := TabsControl.Visible; end; end; end; procedure THelpForm.LoadPreferences(AIPCName: String); var PrefFile: String; RecentCount: Integer; ServerPart: String; i: Integer; begin PrefFile := GetAppConfigDirUTF8(False); ForceDirectoriesUTF8(PrefFile); // --ipcname passes a server ID that consists of a // server-dependent constant together with a process ID. // Strip out the formatted process ID to get fixed config file names for // one server ServerPart := Copy(AIPCName, 1, length(AIPCName)-DigitsInPID); PrefFile := Format('%slhelp-%s.conf',[IncludeTrailingPathDelimiter(PrefFile), ServerPart]); fConfig := TXMLConfig.Create(Self); fConfig.Filename := PrefFile; // Restore window but only if currently not being asked to hide ApplyLayoutPreferencesOnce; OpenDialog1.FileName := fConfig.GetValue('LastFileOpen/Value', OpenDialog1.FileName); RecentCount:= fConfig.GetValue('Recent/ItemCount/Value', 0); // downto since oldest are knocked off the list: for i := RecentCount-1 downto 0 do AddRecentFile(fConfig.GetValue('Recent/Item'+IntToStr(i)+'/Value','')); end; procedure THelpForm.SavePreferences; var i: Integer; begin if not(assigned(fConfig)) then exit; //silently abort if not (WindowState = wsMaximized) then begin fConfig.SetValue('Position/Maximized', false); fConfig.SetValue('Position/Left/Value', Left); fConfig.SetValue('Position/Top/Value', Top); fConfig.SetValue('Position/Width/Value', Width); fConfig.SetValue('Position/Height/Value', Height); end else begin fConfig.SetValue('Position/Maximized', true); end; fConfig.SetValue('LastFileOpen/Value', OpenDialog1.FileName); fConfig.SetValue('Recent/ItemCount/Value', FileMenuOpenRecentItem.Count); // downto since oldest are knocked off the list: for i := 0 to FileMenuOpenRecentItem.Count-1 do fConfig.SetValue('Recent/Item'+IntToStr(i)+'/Value', TRecentMenuItem(FileMenuOpenRecentItem.Items[I]).URL); fConfig.Flush; fConfig.Free; end; procedure THelpForm.AddRecentFile(AFileName: String); var Item : TRecentMenuItem; MaxHistory: longint; i: Integer; begin for i := FileMenuOpenRecentItem.Count-1 downto 0 do if TRecentMenuItem(FileMenuOpenRecentItem.Items[i]).URL = AFileName then begin FileMenuOpenRecentItem.Delete(i); end; Item := TRecentMenuItem.Create(FileMenuOpenRecentItem); Item.Caption:=ExtractFileNameOnly(AFileName); Item.URL:=AFileName; Item.OnClick:=@OpenRecentItemClick; Item.Hint:=Item.URL; FileMenuOpenRecentItem.Insert(0, Item); MaxHistory := fConfig.GetValue('Recent/HistoryCount/Value', 5); if FileMenuOpenRecentItem.Count > 0 then FileMenuOpenRecentItem.Enabled:=True; if FileMenuOpenRecentItem.Count > MaxHistory then FileMenuOpenRecentItem.Items[MaxHistory-1].Free; end; procedure THelpForm.ContentTitleChange(sender: TObject); begin if ActivePage = nil then Exit; Caption := 'LHelp - ' + ActivePage.fContentProvider.Title; end; procedure THelpForm.OpenRecentItemClick(Sender: TObject); var Item: TRecentMenuItem absolute Sender; begin OpenURL(Item.URL); AddRecentFile(Item.URL); end; procedure THelpForm.SendResponse(Response: DWord); var Stream: TMemoryStream; begin Stream := TMemoryStream.Create; try Stream.WriteDWord(Response); if assigned(fOutputIPC) and fOutputIPC.Active then fOutputIPC.SendMessage(mtUnknown, Stream); finally Stream.Free; end; end; procedure THelpForm.ServerMessage(Sender: TObject); var UrlReq: TUrlRequest; FileReq: TFileRequest; ConReq: TContextRequest; MiscReq: TMiscRequest; MustClose: boolean=false; Stream: TStream; Res: LongWord; Url: String=''; begin if fInputIPC.PeekMessage(5, True) then begin Stream := fInputIPC.MsgData; Stream.Position := 0; FillByte(FileReq{%H-},SizeOf(FileReq),0); Stream.Read(FileReq, SizeOf(FileReq)); Res := Ord(srError); //fail by default case FileReq.RequestType of rtFile: begin Url := 'file://'+FileReq.FileName; Res := OpenURL(URL); debugln('got rtfile, filename '+filereq.filename); end; rtUrl: begin Stream.Position := 0; FillByte(UrlReq{%H-},SizeOf(UrlReq),0); Stream.Read(UrlReq, SizeOf(UrlReq)); if UrlReq.FileRequest.FileName <> '' then begin Url := 'file://'+UrlReq.FileRequest.FileName; Res := OpenUrl(URL+'://'+UrlReq.Url) end else begin Url := UrlReq.Url; Res := OpenURL(Url); end; debugln('got rturl, filename '+urlreq.filerequest.filename+', url '+urlreq.url); end; rtContext: begin Stream.Position := 0; FillByte(ConReq{%H-},SizeOf(ConReq),0); Stream.Read(ConReq, SizeOf(ConReq)); Url := 'file://'+FileReq.FileName; Res := OpenURL(Url, ConReq.HelpContext); debugln('got rtcontext, filename '+filereq.filename+', context '+inttostr(ConReq.HelpContext)); end; rtMisc: begin Stream.Position := 0; FillByte(MiscReq{%H-},SizeOf(MiscReq),0); Stream.Read(MiscReq, SizeOf(MiscReq)); case MiscReq.RequestID of mrClose: begin MustClose:=true; Res:= ord(srSuccess); debugln('got rtmisc/mrClose'); end; mrShow: begin fHide := false; if WindowState = wsMinimized then WindowState := wsNormal; RefreshState; Res := ord(srSuccess); debugln('got rtmisc/mrShow'); end; mrVersion: begin // Protocol version encoded in the filename // Verify what we support if strtointdef(FileReq.FileName,0)=strtointdef(PROTOCOL_VERSION,0) then Res := ord(srSuccess) else Res := ord(srError); //version not supported debugln('got rtmisc/'); end else {Unknown request} Res := ord(srUnknown); end; end; //rtMisc end; // This may take some time which may allow receiving end to get ready for // receiving messages if (URL<>'') and (Res = Ord(srSuccess)) then AddRecentFile(Url); // Receiving end may not yet be ready (observed with an Intel Core i7), // so perhaps wait a bit? // Unfortunately, the delay time is guesswork=>Sleep(80)? SendResponse(Res); //send response again in case first wasn't picked up // Keep after SendResponse to avoid timing issues (e.g. writing to log file): debugln('Just sent TLHelpResponse code: '+inttostr(Res)); if MustClose then begin Application.ProcessMessages; Sleep(10); Application.Terminate; end; // We received mrShow: if (MustClose=false) and (fHide=false) then begin Self.SendToBack; Self.BringToFront; Self.ShowOnTop; // If lhelp was run with hidden parameter, we need to apply // layout preferences once: ApplyLayoutPreferencesOnce; end; end; end; procedure THelpForm.ReadCommandLineOptions; var X: Integer; IsHandled: array[0..50] of boolean; URL: String; StrItem: PStringItem; Filename: String; begin FillChar(IsHandled{%H-}, 51, 0); X:=1; while X<=ParamCount do begin if LowerCase(ParamStrUTF8(X)) = '--ipcname' then begin // IPC name; includes unique PID or other identifier IsHandled[X] := True; inc(X); if X <= ParamCount then begin fServerName := ParamStrUTF8(X); IsHandled[X] := True; inc(X); end; end else if LowerCase(ParamStrUTF8(X)) = '--context' then begin IsHandled[X] := True; inc(X); if (X <= ParamCount) then if TryStrToInt(ParamStrUTF8(X), fContext) then begin IsHandled[X] := True; inc(X); end; end else if LowerCase(ParamStrUTF8(X)) = '--hide' then begin IsHandled[X] := True; inc(X); fHide:=true; end else begin IsHandled[X]:=copy(ParamStrUTF8(X),1,1)='-'; // ignore other parameters inc(X); end; end; // Loop through a second time for the URL for X := 1 to ParamCount do if not IsHandled[X] then begin //DoOpenChm(ParamStrUTF8(X)); URL:=ParamStrUTF8(X); if Pos('://', URL) = 0 then URL := 'file://'+URL; Filename:=URL; if copy(Filename,1,length('file://'))='file://' then begin System.Delete(Filename,1,length('file://')); Filename:=SetDirSeparators(Filename); if not FileExistsUTF8(Filename) then begin debugln(['THelpForm.ReadCommandLineOptions file not found "',Filename,'"']); continue; end; end; StrItem := New(PStringItem); StrItem^.FString := URL; Application.QueueAsyncCall(TDataEvent(@LateOpenURL), {%H-}PtrUInt(StrItem)); Break; end; end; procedure THelpForm.StartComms(ServerName: String); // Starts IPC server and client for two-way communication with // controlling program (e.g. Lazarus IDE). // Only useful if IPC serverID is passed through the --ipcname // command. begin fInputIPC := TSimpleIPCServer.Create(nil); fInputIPC.ServerID := ServerName; fInputIPC.Global := True; fInputIPC.Active := True; IPCServer := fInputIPC; // Use timer to check for incoming messages from the IDE fInputIPCTimer := TTimer.Create(nil); fInputIPCTimer.OnTimer := @ServerMessage; fInputIPCTimer.Interval := 200; //milliseconds fInputIPCTimer.Enabled := True; ServerMessage(nil); fOutputIPC := TSimpleIPCClient.Create(nil); fOutputIPC.ServerID := ServerName+'client'; try if fOutputIPC.ServerRunning {$IFDEF STALE_PIPE_WORKAROUND} and not IPCPipeIsStale(fOutputIPC) {$ENDIF} then fOutputIPC.Active := True; except fOutputIPC.Active := False; end; IPCClient := fOutputIPC; end; procedure THelpForm.StopComms; begin if fInputIPC <> nil then begin if fInputIPC.Active then fInputIPC.Active := False; FreeAndNil(fInputIPC); IPCServer := nil; FreeAndNil(fInputIPCTimer); end; if fOutputIPC <> nil then begin if fOutputIPC.Active then fOutputIPC.Active := False; FreeAndNil(fOutputIPC); IPCClient := nil; end; end; function THelpForm.OpenURL(const AURL: String; AContext: THelpContext): DWord; function GetURLPrefix: String; var fPos: Integer; begin fPos := Pos('://', AURL); Result := Copy(AURL, 1, fPos+2); end; var fURLPrefix: String; fContentProvider: TBaseContentProviderClass; fRealContentProvider: TBaseContentProviderClass; fPage: TContentTab = nil; I: Integer; begin Result := Ord(srInvalidURL); fURLPrefix := GetURLPrefix; fContentProvider := GetContentProvider(fURLPrefix); if fContentProvider = nil then begin ShowError('Cannot handle this type of content. "' + fURLPrefix + '" for url:'+LineEnding+AURL); Result := Ord(srInvalidURL); Exit; end; fRealContentProvider := fContentProvider.GetProperContentProvider(AURL); if fRealContentProvider = nil then begin ShowError('Cannot handle this type of subcontent. "' + fURLPrefix + '" for url:'+LineEnding+AURL); Result := Ord(srInvalidURL); Exit; end; for I := 0 to PageControl.PageCount-1 do begin if fRealContentProvider.ClassName = TContentTab(PageControl.Pages[I]).ContentProvider.ClassName then begin fPage := TContentTab(PageControl.Pages[I]); if TContentTab(PageControl.Pages[I]).ContentProvider.LoadURL(AURL, AContext) then begin PageControl.ActivePage := PageControl.Pages[I]; Result := Ord(srSuccess); end else Result := Ord(srInvalidFile); Exit; end; end; if fPage = nil then begin // no existing page that can handle this content, so create one fPage := TContentTab.Create(PageControl); fPage.ContentProvider := fRealContentProvider.Create(fPage, ImageList1); fPAge.ContentProvider.OnTitleChange:=@ContentTitleChange; fPage.Parent := PageControl; SetKeyUp(fPage); fPage.ContentProvider.LoadPreferences(fConfig); end; if fPage.ContentProvider.LoadURL(AURL, AContext) then begin PageControl.ActivePage := fPage; RefreshState; Result := Ord(srSuccess); end else Result := Ord(srInvalidURL); if not fHide then ShowOnTop; end; procedure THelpForm.LateOpenURL ( Url: PStringItem ) ; begin if OpenURL(URL^.FString, fContext) = ord(srSuccess) then AddRecentFile(URL^.FString); // we reset the context because at this point the file has been loaded and the // context shown fContext := -1; Dispose(Url); RefreshState; end; function THelpForm.ActivePage: TContentTab; begin Result := TContentTab(PageControl.ActivePage); end; procedure THelpForm.RefreshState; var en: Boolean; begin if fHide then begin en := false; // Hide content page if Assigned(ActivePage) then begin with TChmContentProvider(ActivePage.ContentProvider) do begin ActivePage.Visible := false; Visible := false; TabsControl.Visible := false; Splitter.Visible := false; end; end; end else begin en := Assigned(ActivePage); // Show content page if en then begin with TChmContentProvider(ActivePage.ContentProvider) do begin ActivePage.Visible := true; Visible := true; TabsControl.Visible := true; Splitter.Visible := true; end; end; end; BackBttn.Enabled := en; ForwardBttn.Enabled := en; HomeBttn.Enabled := en; FileMenuCloseItem.Enabled := en; ViewMenuContents.Enabled := en; if en and not (csDestroying in ActivePage.ComponentState) then Caption := 'LHelp - ' + ActivePage.fContentProvider.Title else Caption := 'LHelp'; end; procedure THelpForm.ShowError(AError: String); begin ShowMessage(AError); end; procedure THelpForm.SetKeyUp(AControl: TControl); var WCont: TWinControl absolute AControl; i: Integer; begin if (AControl = nil) or not (AControl.InheritsFrom(TWinControl)) then Exit; for i := 0 to WCont.ControlCount-1 do SetKeyUp(WCont.Controls[i]); WCont.OnKeyUp:=@FormKeyUp; end; { TContentTab } constructor TContentTab.Create(AOwner: TComponent); begin inherited Create(AOwner); end; destructor TContentTab.Destroy; begin fContentProvider.Free; inherited Destroy; end; finalization if IPCServer <> nil then FreeAndNil(IPCServer); if IPCClient <> nil then FreeAndNil(IPCClient); end.