dw_htmlchm.inc 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542
  1. {%mainunit dw_html}
  2. {$IFDEF chmInterface}
  3. type
  4. { TCHMHTMLWriter }
  5. TCHMHTMLWriter = class(THTMLWriter)
  6. private
  7. FOutChm: TStream;
  8. FChm: TChmWriter;
  9. FTempUncompressed: TStream;
  10. FTempUncompressedName: String;
  11. FTOCName,
  12. FIndexName,
  13. FDefaultPage: String;
  14. FMakeSearchable,
  15. FNoBinToc,
  16. FNoBinIndex,
  17. FAutoTOC,
  18. FAutoIndex: Boolean;
  19. FOtherFiles: String;
  20. procedure ProcessOptions;
  21. function RetrieveOtherFiles(const DataName: String; out PathInChm: String; out FileName: String; var Stream: TStream): Boolean;
  22. procedure LastFileAdded(Sender: TObject);
  23. procedure GenerateTOC;
  24. procedure GenerateIndex;
  25. public
  26. procedure WriteHTMLPages; override;
  27. function InterPretOption(const Cmd,Arg : String): boolean; override;
  28. class procedure Usage(List: TStrings); override;
  29. end;
  30. {$ELSE} // implementation
  31. { TCHMHTMLWriter }
  32. procedure TCHMHTMLWriter.ProcessOptions;
  33. var
  34. TempStream: TMemoryStream;
  35. begin
  36. if FDefaultPage = '' then
  37. FDefaultPage := 'index.html'
  38. else
  39. begin
  40. WriteLn('Note: --index-page not assigned. Using default "index.html"');
  41. end;
  42. if FCSSFile <> '' then
  43. begin
  44. if not FileExists(FCSSFile) Then
  45. begin
  46. Writeln(stderr,'Can''t find CSS file "',FCSSFILE,'"');
  47. halt(1);
  48. end;
  49. TempStream := TMemoryStream.Create;
  50. TempStream.LoadFromFile(FCSSFile);
  51. TempStream.Position := 0;
  52. FChm.AddStreamToArchive('fpdoc.css', '/', TempStream, True);
  53. TempStream.Free;
  54. end;
  55. FChm.DefaultPage := FDefaultPage;
  56. if FOtherFiles <> '' then
  57. begin
  58. FChm.FilesToCompress.LoadFromFile(FOtherFiles);
  59. end;
  60. FChm.FullTextSearch := FMakeSearchable;
  61. end;
  62. function TCHMHTMLWriter.RetrieveOtherFiles(const DataName: String; out
  63. PathInChm: String; out FileName: String; var Stream: TStream): Boolean;
  64. var
  65. Dir: String;
  66. begin
  67. if Stream <> nil then
  68. Stream.Free;
  69. Stream := TMemoryStream.Create;
  70. TMemoryStream(Stream).LoadFromFile(DataName);
  71. FileName := ExtractFileName(DataName);
  72. if ExtractFileDir(DataName) <> '' then
  73. PathInChm := ExtractRelativepath(GetCurrentDir, ExtractFileDir(DataName))
  74. else
  75. PathInChm := '/';
  76. FixHTMLpath(PathInChm);
  77. Stream.Position := 0;
  78. end;
  79. procedure TCHMHTMLWriter.LastFileAdded(Sender: TObject);
  80. var
  81. TmpStream: TMemoryStream;
  82. begin
  83. TmpStream := TMemoryStream.Create;
  84. if FAutoTOC then
  85. GenerateTOC
  86. else
  87. if FTOCName <> '' then
  88. begin
  89. TmpStream.LoadFromFile(FTOCName);
  90. TmpStream.Position := 0;
  91. FChm.AppendTOC(TmpStream);
  92. TmpStream.Size := 0;
  93. end;
  94. if FAutoIndex then
  95. GenerateIndex
  96. else
  97. if FIndexName <> '' then
  98. begin
  99. TmpStream.LoadFromFile(FIndexName);
  100. TmpStream.Position := 0;
  101. FChm.AppendIndex(TmpStream);
  102. end;
  103. TmpStream.Free;
  104. WriteLn('Finishing compressing...');
  105. end;
  106. function TOCSort(Item1, Item2: TChmSiteMapItem): Integer;
  107. begin
  108. Result := CompareText(LowerCase(Item1.Text), LowerCase(Item2.Text));
  109. end;
  110. function GetAlphaItem(AItems: TChmSiteMapItems; AName: String): TChmSiteMapItem;
  111. var
  112. x: Integer;
  113. begin
  114. Result := nil;
  115. for x := 0 to AItems.Count-1 do
  116. begin
  117. if AItems.Item[x].Text = AName then
  118. Exit(AItems.Item[x]);
  119. end;
  120. Result := AItems.NewItem;
  121. Result.Text := AName;
  122. end;
  123. procedure TCHMHTMLWriter.GenerateTOC;
  124. var
  125. TOC: TChmSiteMap;
  126. Element: TPasElement;
  127. k: Integer;
  128. j: Integer;
  129. i: Integer;
  130. AModule: TPasModule;
  131. Member: TPasElement;
  132. Stream: TMemoryStream;
  133. TmpItem: TChmSiteMapItem;
  134. ObjByUnitItem,
  135. AlphaObjItem,
  136. ObjUnitItem,
  137. RoutinesByUnitItem,
  138. RoutinesUnitItem,
  139. AlphaRoutinesItem: TChmSiteMapItem;
  140. begin
  141. WriteLn('Generating Table of contents...');
  142. if Assigned(Package) then
  143. begin
  144. Toc := TChmSiteMap.Create(stTOC);
  145. Stream := TMemoryStream.Create;
  146. ObjByUnitItem := TOC.Items.NewItem;
  147. ObjByUnitItem.Text := 'Classes and Objects, by Unit';
  148. AlphaObjItem := TOC.Items.NewItem;
  149. AlphaObjItem.Text := 'Alphabetical Classes and Objects List';
  150. RoutinesByUnitItem := TOC.Items.NewItem;
  151. RoutinesByUnitItem.Text := 'Routines, by Unit';
  152. AlphaRoutinesItem := TOC.Items.NewItem;
  153. AlphaRoutinesItem.Text := 'Alphabetical Routines List';
  154. // objects and classes
  155. for i := 0 to Package.Modules.Count - 1 do
  156. begin
  157. AModule := TPasModule(Package.Modules[i]);
  158. ObjUnitItem := ObjByUnitItem.Children.NewItem;
  159. ObjUnitItem.Text := AModule.Name;
  160. RoutinesUnitItem := RoutinesByUnitItem.Children.NewItem;
  161. RoutinesUnitItem.Text := AModule.Name;
  162. for j := 0 to AModule.InterfaceSection.Classes.Count-1 do
  163. begin
  164. Element := TPasClassType(AModule.InterfaceSection.Classes[j]);
  165. // by unit
  166. TmpItem := ObjUnitItem.Children.NewItem;
  167. TmpItem.Text := Element.Name;
  168. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(Element, 0));
  169. //alpha
  170. TmpItem := GetAlphaItem(AlphaObjItem.Children, UpperCase(Copy(Element.Name, 1, 2))).Children.NewItem;
  171. TmpItem.Text := Element.Name;
  172. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(Element, 0));
  173. end;
  174. // non object procedures and functions
  175. for j := 0 to AModule.InterfaceSection.Functions.Count-1 do
  176. begin
  177. Element := TPasFunctionType(AModule.InterfaceSection.Functions[j]);
  178. // by unit
  179. TmpItem := RoutinesUnitItem.Children.NewItem;
  180. TmpItem.Text := Element.Name;
  181. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(Element, 0));
  182. // alpha
  183. TmpItem := GetAlphaItem(AlphaRoutinesItem.Children, UpperCase(Element.Name[1])).Children.NewItem;
  184. TmpItem.Text := Element.Name;
  185. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(Element, 0));
  186. end;
  187. end;
  188. end;
  189. // cleanup
  190. for i := ObjByUnitItem.Children.Count-1 downto 0 do
  191. begin
  192. if ObjByUnitItem.Children.Item[i].Children.Count = 0 then
  193. ObjByUnitItem.Children.Delete(i);
  194. end;
  195. for i := RoutinesByUnitItem.Children.Count-1 downto 0 do
  196. begin
  197. if RoutinesByUnitItem.Children.Item[i].Children.Count = 0 then
  198. RoutinesByUnitItem.Children.Delete(i);
  199. end;
  200. for i := TOC.Items.Count-1 downto 0 do
  201. begin
  202. if TOC.Items.Item[i].Children.Count = 0 then
  203. TOC.Items.Delete(i);
  204. end;
  205. // Sort
  206. for i := 0 to TOC.Items.Count-1 do
  207. begin
  208. TOC.Items.Item[i].Children.Sort(TListSortCompare(@TOCSort));
  209. for j := 0 to TOC.Items.Item[i].Children.Count-1 do
  210. begin
  211. TOC.Items.Item[i].Children.Item[j].Children.Sort(TListSortCompare(@TOCSort));
  212. end;
  213. end;
  214. if not fnobintoc then
  215. fchm.AppendBinaryTOCFromSiteMap(Toc);
  216. TOC.SaveToStream(Stream);
  217. TOC.Free;
  218. fchm.AppendTOC(Stream);
  219. Stream.Free;
  220. end;
  221. type
  222. TClassMemberType = (cmtProcedure, cmtFunction, cmtConstructor, cmtDestructor,
  223. cmtInterface, cmtProperty, cmtVariable, cmtUnknown);
  224. function ElementType(Element: TPasElement): TClassMemberType;
  225. var
  226. ETypeName: String;
  227. begin
  228. Result := cmtUnknown;
  229. ETypeName := Element.ElementTypeName;
  230. //overloaded we don't care
  231. if ETypeName[1] = 'o' then ETypeName := Copy(ETypeName, 11, Length(ETypeName));
  232. if ETypeName[1] = 'f' then Exit(cmtFunction);
  233. if ETypeName[1] = 'c' then Exit(cmtConstructor);
  234. if ETypeName[1] = 'v' then Exit(cmtVariable);
  235. if ETypeName[1] = 'i' then Exit(cmtInterface);
  236. // the p's
  237. if ETypeName[4] = 'c' then Exit(cmtProcedure);
  238. if ETypeName[4] = 'p' then Exit(cmtProperty);
  239. end;
  240. procedure TCHMHTMLWriter.GenerateIndex;
  241. var
  242. Index: TChmSiteMap;
  243. i, j, k: Integer;
  244. TmpItem: TChmSiteMapItem;
  245. ParentItem: TChmSiteMapItem;
  246. AModule: TPasModule;
  247. TmpElement: TPasElement;
  248. ParentElement: TPasElement;
  249. MemberItem: TChmSiteMapItem;
  250. Stream: TMemoryStream;
  251. begin
  252. WriteLn('Generating Index...');
  253. if Assigned(Package) then
  254. begin
  255. try
  256. Index := TChmSiteMap.Create(stIndex);
  257. Stream := TMemoryStream.Create;
  258. for i := 0 to Package.Modules.Count - 1 do
  259. begin
  260. AModule := TPasModule(Package.Modules[i]);
  261. ParentItem := Index.Items.NewItem;
  262. ParentItem.Text := AModule.Name;
  263. ParentItem.Local := FixHTMLpath(Allocator.GetFilename(AModule, 0));
  264. // classes
  265. for j := 0 to AModule.InterfaceSection.Classes.Count-1 do
  266. begin
  267. ParentElement := TPasClassType(AModule.InterfaceSection.Classes[j]);
  268. ParentItem := Index.Items.NewItem;
  269. ParentItem.Text := ParentELement.Name;
  270. ParentItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  271. for k := 0 to TPasClassType(ParentElement).Members.Count-1 do
  272. begin
  273. TmpElement := TPasElement(TPasClassType(ParentElement).Members.Items[k]);
  274. if Engine.HidePrivate and(TmpElement.Visibility = visPrivate) then
  275. continue;
  276. if Engine.HideProtected and(TmpElement.Visibility = visProtected) then
  277. continue;
  278. TmpItem := ParentItem.Children.NewItem;
  279. case ElementType(TmpElement) of
  280. cmtProcedure : TmpItem.Text := TmpElement.Name + ' procedure';
  281. cmtFunction : TmpItem.Text := TmpElement.Name + ' function';
  282. cmtConstructor : TmpItem.Text := TmpElement.Name + ' constructor';
  283. cmtDestructor : TmpItem.Text := TmpElement.Name + ' destructor';
  284. cmtProperty : TmpItem.Text := TmpElement.Name + ' property';
  285. cmtVariable : TmpItem.Text := TmpElement.Name + ' variable';
  286. cmtInterface : TmpItem.Text := TmpElement.Name + ' interface';
  287. cmtUnknown : TmpItem.Text := TmpElement.Name;
  288. end;
  289. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(TmpElement, 0));
  290. {
  291. ParentElement = Class
  292. TmpElement = Member
  293. }
  294. MemberItem := nil;
  295. MemberItem := GetAlphaItem(Index.Items, TmpElement.Name);
  296. // ahh! if MemberItem.Local is empty MemberType is not shown!
  297. MemberItem.Local := FixHTMLpath(Allocator.GetFilename(TmpElement, 0));
  298. TmpItem := MemberItem.Children.NewItem;
  299. TmpItem.Text := ParentElement.Name;
  300. TmpITem.Local := FixHTMLpath(Allocator.GetFilename(TmpElement, 0));
  301. end;
  302. end;
  303. // routines
  304. for j := 0 to AModule.InterfaceSection.Functions.Count-1 do
  305. begin
  306. ParentElement := TPasProcedureType(AModule.InterfaceSection.Functions[j]);
  307. TmpItem := Index.Items.NewItem;
  308. TmpItem.Text := ParentElement.Name + ' ' + TPasFunction(ParentElement).ElementTypeName;
  309. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  310. end;
  311. // consts
  312. for j := 0 to AModule.InterfaceSection.Consts.Count-1 do
  313. begin
  314. ParentElement := TPasElement(AModule.InterfaceSection.Consts[j]);
  315. TmpItem := Index.Items.NewItem;
  316. TmpItem.Text := ParentElement.Name;
  317. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  318. end;
  319. // types
  320. for j := 0 to AModule.InterfaceSection.Types.Count-1 do
  321. begin
  322. ParentElement := TPasType(AModule.InterfaceSection.Types[j]);
  323. TmpItem := Index.Items.NewItem;
  324. TmpItem.Text := ParentElement.Name;
  325. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  326. // enums
  327. if ParentELement is TPasEnumType then
  328. begin
  329. ParentItem := TmpItem;
  330. for k := 0 to TPasEnumType(ParentElement).Values.Count-1 do
  331. begin
  332. TmpElement := TPasType(TPasEnumType(ParentElement).Values.Items[k]);
  333. // subitem
  334. TmpItem := ParentItem.Children.NewItem;
  335. TmpItem.Text := TmpElement.Name;
  336. TmpItem.Local := ParentItem.Local;
  337. // root level
  338. TmpItem := Index.Items.NewItem;
  339. TmpItem.Text := TmpElement.Name;
  340. TmpItem.Local := ParentItem.Local;
  341. end;
  342. end;
  343. end;
  344. // variables
  345. for j := 0 to AModule.InterfaceSection.Variables.Count-1 do
  346. begin
  347. ParentElement := TPasElement(AModule.InterfaceSection.Variables[j]);
  348. TmpItem := Index.Items.NewItem;
  349. TmpItem.Text := ParentElement.Name + ' var';
  350. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  351. end;
  352. // declarations
  353. {
  354. for j := 0 to AModule.InterfaceSection.Declarations.Count-1 do
  355. begin
  356. ParentElement := TPasElement(AModule.InterfaceSection.Declarations[j]);
  357. TmpItem := Index.Items.NewItem;
  358. TmpItem.Text := ParentElement.Name;
  359. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  360. end;
  361. // resource strings
  362. for j := 0 to AModule.InterfaceSection.ResStrings.Count-1 do
  363. begin
  364. ParentElement := TPasElement(AModule.InterfaceSection.ResStrings[j]);
  365. TmpItem := Index.Items.NewItem;
  366. TmpItem.Text := ParentElement.Name;
  367. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  368. end;
  369. }
  370. end;
  371. // Sort
  372. Index.Items.Sort(TListSortCompare(@TOCSort));
  373. for i := 0 to Index.Items.Count-1 do
  374. begin
  375. Index.Items.Item[i].Children.Sort(TListSortCompare(@TOCSort));
  376. end;
  377. // save
  378. Index.SaveToStream(Stream);
  379. if not fnobinindex then
  380. fchm.AppendBinaryindexFromSitemap(index,false);
  381. Index.Free;
  382. Stream.Position :=0 ;
  383. FChm.AppendIndex(Stream);
  384. Stream.Free;
  385. except
  386. Dump_Stack(StdOut, get_frame);
  387. Halt(1);
  388. end;
  389. end;
  390. end;
  391. procedure TCHMHTMLWriter.WriteHTMLPages;
  392. var
  393. i: Integer;
  394. PageDoc: TXMLDocument;
  395. FileStream: TMemoryStream;
  396. FileName: String;
  397. FilePath: String;
  398. begin
  399. if Engine.Output = '' then
  400. begin
  401. WriteLn('Error: no --output option used.');
  402. Exit;
  403. end;
  404. if ExtractFileExt(Engine.Output) <> '.chm' then
  405. ChangeFileExt(Engine.OutPut, '.chm');
  406. FOutChm := TFileStream.Create(Engine.Output, fmOpenReadWrite or fmCreate);
  407. FTempUncompressedName := GetTempFileName+IntToStr(GetProcessID) +'.raw';
  408. FTempUncompressed := TFileStream.Create(FTempUncompressedName, fmOpenReadWrite or fmCreate);
  409. FChm := TChmWriter.Create(FOutChm, False);
  410. FChm.Title := Copy(Package.Name, 2, Length(Package.Name));
  411. FChm.TempRawStream := FTempUncompressed;
  412. FChm.OnGetFileData := @RetrieveOtherFiles;
  413. FChm.OnLastFile := @LastFileAdded;
  414. fchm.hasbinarytoc:=not fnobintoc;;
  415. fchm.hasbinaryindex:=not fnobinindex;
  416. ProcessOptions;
  417. FileStream := TMemoryStream.Create;
  418. for i := 0 to PageInfos.Count - 1 do
  419. with TPageInfo(PageInfos[i]) do
  420. begin
  421. PageDoc := CreateHTMLPage(Element, SubpageIndex);
  422. try
  423. FileName := ExtractFileName(Allocator.GetFilename(Element, SubpageIndex));
  424. FilePath := '/'+FixHTMLpath(ExtractFilePath(Allocator.GetFilename(Element, SubpageIndex)));
  425. try
  426. WriteHTMLFile(PageDoc, FileStream);
  427. FChm.AddStreamToArchive(FileName, FilePath, FileStream, True);
  428. except
  429. on E: Exception do
  430. WriteLn(Format(SErrCouldNotCreateFile, [FileName, e.Message]));
  431. end;
  432. finally
  433. PageDoc.Free;
  434. FileStream.Size := 0;
  435. end;
  436. end;
  437. FileStream.Free;
  438. WriteLn('HTML Files written. Collecting other files and compressing...this could take some time');
  439. FChm.Execute;
  440. FChm.Free;
  441. // we don't need to free FTempUncompressed
  442. // FTempUncompressed.Free;
  443. FOutChm.Free;
  444. DeleteFile(FTempUncompressedName);
  445. end;
  446. function TCHMHTMLWriter.InterPretOption(const Cmd, Arg: String): boolean;
  447. begin
  448. Result:=True;
  449. FNoBinToc:=False;
  450. FnoBinIndex:=False;
  451. if Cmd = '--toc-file' then
  452. FTOCName := arg
  453. else if Cmd = '--index-file' then
  454. FIndexName := arg
  455. else if Cmd = '--default-page' then
  456. FDefaultPage := arg
  457. else if Cmd = '--other-files' then
  458. FOtherFiles := arg
  459. else if Cmd = '--auto-index' then
  460. FAutoIndex := True
  461. else if Cmd = '--auto-toc' then
  462. FAutoTOC := True
  463. else if Cmd = '--no-bintoc' then
  464. FNoBinToc := True
  465. else if Cmd = '--no-binindex' then
  466. FNoBinIndex := True
  467. else if Cmd = '--make-searchable' then
  468. FMakeSearchable := True
  469. else
  470. Result:=inherited InterPretOption(Cmd, Arg);
  471. end;
  472. class procedure TCHMHTMLWriter.Usage(List: TStrings);
  473. begin
  474. THTMLWriter.Usage(List);
  475. List.add('--default-page');
  476. List.Add(SCHMUsageDefPage);
  477. List.add('--toc-file');
  478. List.Add(SCHMUsageTOC);
  479. List.add('--index-file');
  480. List.Add(SCHMUsageIndex);
  481. List.add('--other-files');
  482. List.Add(SCHMUsageOtrFiles);
  483. List.add('--css-file');
  484. List.Add(SCHMUsageCSSFile);
  485. List.add('--auto-index');
  486. List.Add(SCHMUsageAutoIDX);
  487. List.add('--auto-toc');
  488. List.Add(SCHMUsageAutoTOC);
  489. List.add('--make-searchable');
  490. List.Add(SCHMUsageMakeSearch);
  491. end;
  492. {$ENDIF}