dw_htmlchm.inc 16 KB

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