dw_chm.pp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816
  1. unit dw_chm;
  2. {$mode objfpc}
  3. {$h+}
  4. interface
  5. uses Classes, DOM,
  6. dGlobals, PasTree, dwriter, dw_html, chmwriter, chmtypes, chmsitemap;
  7. type
  8. { TCHmFileNameAllocator }
  9. TCHmFileNameAllocator = Class(TLongNameFileAllocator)
  10. // Override this, because the logic messes up the filenames for plain html files.
  11. function GetFilename(AElement: TPasElement; ASubindex: Integer): String; override;
  12. end;
  13. { TFpDocChmWriter }
  14. TFpDocChmWriter = class (TChmWriter)
  15. protected
  16. procedure FileAdded(AStream: TStream; const AEntry: TFileEntryRec); override;
  17. end;
  18. { TCHMHTMLWriter }
  19. TCHMHTMLWriter = class(THTMLWriter)
  20. private
  21. FOutChm: TStream;
  22. FChm: TFpDocChmWriter;
  23. FTempUncompressed: TStream;
  24. FTempUncompressedName: String;
  25. FChmTitle: String;
  26. FTOCName,
  27. FIndexName,
  28. FDefaultPage: String;
  29. FMakeSearchable,
  30. FNoBinToc,
  31. FNoBinIndex,
  32. FAutoTOC,
  33. FAutoIndex: Boolean;
  34. FOtherFiles: String;
  35. procedure ProcessOptions;
  36. function ResolveLinkIDAbs(const Name: String): DOMString;
  37. function RetrieveOtherFiles(const DataName: String; out PathInChm: String;
  38. out FileName: String; var Stream: TStream): Boolean;
  39. procedure LastFileAdded(Sender: TObject);
  40. function FindAlphaItem(AItems: TChmSiteMapItems; AName: String): TChmSiteMapItem;
  41. function GetAlphaItem(AItems: TChmSiteMapItems; AName: String): TChmSiteMapItem;
  42. procedure MultiAlphaItem(AItems: TChmSiteMapItems; AName: String;
  43. APasEl: TPasElement; Prefix:String);
  44. procedure GenerateTOC;
  45. procedure GenerateIndex;
  46. protected
  47. procedure DoWriteDocumentation; override;
  48. public
  49. function CreateAllocator: TFileAllocator; override;
  50. function InterPretOption(const Cmd,Arg : String): boolean; override;
  51. class procedure Usage(List: TStrings); override;
  52. Class Function FileNameExtension : String; override;
  53. Class procedure SplitImport(var AFilename, ALinkPrefix: String); override;
  54. end;
  55. implementation
  56. uses fpdocstrs, SysUtils, HTMWrite, dw_basehtml;
  57. { TCHmFileNameAllocator }
  58. function TCHmFileNameAllocator.GetFilename(AElement: TPasElement; ASubindex: Integer): String;
  59. var
  60. n,s: String;
  61. i: Integer;
  62. excl: Boolean; //search
  63. MElement: TPasElement;
  64. begin
  65. Result:='';
  66. excl := False;
  67. if AElement.ClassType = TPasPackage then
  68. begin
  69. Result := 'index';
  70. excl := True;
  71. end
  72. else if AElement.ClassType = TPasModule then
  73. begin
  74. Result := LowerCase(AElement.Name) + PathDelim + 'index';
  75. excl := True;
  76. end
  77. else
  78. begin
  79. if AElement is TPasOperator then
  80. begin
  81. if Assigned(AElement.Parent) then
  82. result:=LowerCase(AElement.Parent.PathName);
  83. With TPasOperator(aElement) do
  84. Result:= Result + 'op-'+OperatorTypeToOperatorName(OperatorType);
  85. s := '';
  86. N:=LowerCase(aElement.Name); // Should not contain any weird chars.
  87. Delete(N,1,Pos('(',N));
  88. i := 1;
  89. Repeat
  90. I:=Pos(',',N);
  91. if I=0 then
  92. I:=Pos(')',N);
  93. if I>1 then
  94. begin
  95. if (S<>'') then
  96. S:=S+'-';
  97. S:=S+Copy(N,1,I-1);
  98. end;
  99. Delete(N,1,I);
  100. until I=0;
  101. // First char is maybe :
  102. if (N<>'') and (N[1]=':') then
  103. Delete(N,1,1);
  104. Result:=Result + '-'+ s + '-' + N;
  105. end
  106. else
  107. begin
  108. Result := LowerCase(AElement.PathName);
  109. excl := (ASubindex > 0);
  110. end;
  111. // cut off Package Name
  112. MElement:= AElement.GetModule;
  113. if Assigned(MElement) then
  114. AElement:= MElement;
  115. Result := Copy(Result, Length(AElement.Parent.Name) + 2, MaxInt);
  116. // to skip dots in unit name
  117. i := Length(AElement.Name);
  118. while (i <= Length(Result)) and (Result[i] <> '.') do
  119. Inc(i);
  120. if (i <= Length(Result)) and (i > 0) then
  121. Result[i] := PathDelim;
  122. if excl or (Length(Result)=0) then
  123. begin
  124. // exclude the from full text search index
  125. s:= '.'+ExtractFileName(Result + '.');
  126. n:= ExtractFileDir(Result);
  127. Result := n + DirectorySeparator + s;
  128. Result := Copy(Result, 1, Length(Result)-1);
  129. end;
  130. end;
  131. if ASubindex > 0 then
  132. Result := Result + '-' + GetFilePostfix(ASubindex);
  133. Result := Result + Extension;
  134. end;
  135. { TFpDocChmWriter }
  136. procedure TFpDocChmWriter.FileAdded ( AStream: TStream;
  137. const AEntry: TFileEntryRec ) ;
  138. var FTsave : boolean;
  139. begin
  140. // Exclude Full text index for files starting from the dot
  141. if Pos('.', AEntry.Name) <> 1 then
  142. inherited FileAdded(AStream, AEntry)
  143. else
  144. begin
  145. FTsave:=FullTextSearch;
  146. FullTextSearch:=False;
  147. inherited FileAdded(AStream, AEntry);
  148. FullTextSearch:=FTsave;
  149. end;
  150. end;
  151. { TCHMHTMLWriter }
  152. function TCHMHTMLWriter.ResolveLinkIDAbs(const Name: String): DOMString;
  153. begin
  154. Result:=UTF8Decode(FixHTMLpath(Engine.ResolveLink(Module,Name, True)));
  155. // for global index: don't make it relative to the current document.
  156. end;
  157. procedure TCHMHTMLWriter.ProcessOptions;
  158. var
  159. TempStream: TMemoryStream;
  160. begin
  161. if FDefaultPage = '' then
  162. FDefaultPage := 'index.html'
  163. else
  164. begin
  165. DoLog('Note: --index-page not assigned. Using default "index.html"');
  166. end;
  167. if CSSFile <> '' then
  168. begin
  169. if not FileExists(CSSFile) Then
  170. Raise Exception.CreateFmt('Can''t find CSS file "%S"',[CSSFILE]);
  171. TempStream := TMemoryStream.Create;
  172. TempStream.LoadFromFile(CSSFile);
  173. TempStream.Position := 0;
  174. FChm.AddStreamToArchive('fpdoc.css', '/', TempStream, True);
  175. TempStream.Free;
  176. end;
  177. FChm.DefaultPage := FDefaultPage;
  178. if FOtherFiles <> '' then
  179. begin
  180. FChm.FilesToCompress.LoadFromFile(FOtherFiles);
  181. end;
  182. FChm.FullTextSearch := FMakeSearchable;
  183. end;
  184. function TCHMHTMLWriter.RetrieveOtherFiles(const DataName: String; out
  185. PathInChm: String; out FileName: String; var Stream: TStream): Boolean;
  186. begin
  187. Result:=True;
  188. if Stream <> nil then
  189. Stream.Free;
  190. Stream := TMemoryStream.Create;
  191. TMemoryStream(Stream).LoadFromFile(DataName);
  192. FileName := ExtractFileName(DataName);
  193. if ExtractFileDir(DataName) <> '' then
  194. PathInChm := ExtractRelativepath(GetCurrentDir, ExtractFileDir(DataName))
  195. else
  196. PathInChm := '/';
  197. FixHTMLpath(PathInChm);
  198. Stream.Position := 0;
  199. end;
  200. procedure TCHMHTMLWriter.LastFileAdded(Sender: TObject);
  201. var
  202. TmpStream: TMemoryStream;
  203. begin
  204. TmpStream := TMemoryStream.Create;
  205. if FAutoTOC then
  206. GenerateTOC
  207. else
  208. if FTOCName <> '' then
  209. begin
  210. TmpStream.LoadFromFile(FTOCName);
  211. TmpStream.Position := 0;
  212. FChm.AppendTOC(TmpStream);
  213. TmpStream.Size := 0;
  214. end;
  215. if FAutoIndex then
  216. GenerateIndex
  217. else
  218. if FIndexName <> '' then
  219. begin
  220. TmpStream.LoadFromFile(FIndexName);
  221. TmpStream.Position := 0;
  222. FChm.AppendIndex(TmpStream);
  223. end;
  224. TmpStream.Free;
  225. DoLog('Finishing compressing...');
  226. end;
  227. function TOCSort(Item1, Item2: TChmSiteMapItem): Integer;
  228. begin
  229. Result := CompareText(LowerCase(Item1.Text), LowerCase(Item2.Text));
  230. end;
  231. function TCHMHTMLWriter.FindAlphaItem(AItems: TChmSiteMapItems; AName: String
  232. ): TChmSiteMapItem;
  233. var
  234. x: Integer;
  235. begin
  236. Result := nil;
  237. for x := 0 to AItems.Count-1 do
  238. begin
  239. if AItems.Item[x].Text = AName then
  240. Exit(AItems.Item[x]);
  241. end;
  242. end;
  243. function TCHMHTMLWriter.GetAlphaItem(AItems: TChmSiteMapItems; AName: String
  244. ): TChmSiteMapItem;
  245. begin
  246. Result := FindAlphaItem(AItems, AName);
  247. if Result <> nil then Exit;
  248. Result := AItems.NewItem;
  249. Result.Text := AName;
  250. end;
  251. procedure TCHMHTMLWriter.MultiAlphaItem(AItems: TChmSiteMapItems; AName: String;
  252. APasEl: TPasElement; Prefix: String);
  253. var
  254. AChmItem, AChmChld: TChmSiteMapItem;
  255. begin
  256. AChmItem:= FindAlphaItem(AItems, AName);
  257. if AChmItem = nil then
  258. begin
  259. // add new
  260. AChmItem := AItems.NewItem;
  261. AChmItem.Text := AName;
  262. AChmItem.addLocal(FixHTMLpath(Allocator.GetFilename(APasEl, 0)));
  263. end
  264. else
  265. begin
  266. // add as child
  267. AChmChld := AChmItem.Children.NewItem;
  268. AChmChld.Text := Prefix + '.' + AName;
  269. AChmChld.addLocal(FixHTMLpath(Allocator.GetFilename(APasEl, 0)));
  270. end;
  271. end;
  272. procedure TCHMHTMLWriter.GenerateTOC;
  273. var
  274. TOC: TChmSiteMap;
  275. Element: TPasElement;
  276. j: Integer;
  277. i: Integer;
  278. AModule: TPasModule;
  279. Stream: TMemoryStream;
  280. TmpItem: TChmSiteMapItem;
  281. ObjByUnitItem,
  282. AlphaObjItem,
  283. ObjUnitItem,
  284. RoutinesByUnitItem,
  285. RoutinesUnitItem,
  286. AlphaRoutinesItem: TChmSiteMapItem;
  287. begin
  288. DoLog('Generating Table of contents...');
  289. if not Assigned(Package) then
  290. begin
  291. DoLog('Package is not assigned...');
  292. Exit;
  293. end;
  294. Toc := TChmSiteMap.Create(stTOC);
  295. Stream := TMemoryStream.Create;
  296. ObjByUnitItem := TOC.Items.NewItem;
  297. ObjByUnitItem.Text := 'Classes and Objects, by Unit';
  298. AlphaObjItem := TOC.Items.NewItem;
  299. AlphaObjItem.Text := 'Alphabetical Classes and Objects List';
  300. RoutinesByUnitItem := TOC.Items.NewItem;
  301. RoutinesByUnitItem.Text := 'Routines, by Unit';
  302. AlphaRoutinesItem := TOC.Items.NewItem;
  303. AlphaRoutinesItem.Text := 'Alphabetical Routines List';
  304. // objects and classes
  305. for i := 0 to Package.Modules.Count - 1 do
  306. begin
  307. AModule := TPasModule(Package.Modules[i]);
  308. If not assigned(AModule.InterfaceSection) Then
  309. Continue;
  310. ObjUnitItem := ObjByUnitItem.Children.NewItem;
  311. ObjUnitItem.Text := AModule.Name;
  312. ObjUnitItem.addLocal(FixHTMLpath(Allocator.GetFilename(AModule, ClassesSubindex)));
  313. RoutinesUnitItem := RoutinesByUnitItem.Children.NewItem;
  314. RoutinesUnitItem.Text := AModule.Name;
  315. RoutinesUnitItem.addLocal(FixHTMLpath(Allocator.GetFilename(AModule, ProcsSubindex)));
  316. for j := 0 to AModule.InterfaceSection.Classes.Count-1 do
  317. begin
  318. Element := TPasClassType(AModule.InterfaceSection.Classes[j]);
  319. // by unit
  320. TmpItem := ObjUnitItem.Children.NewItem;
  321. TmpItem.Text := Element.Name;
  322. TmpItem.addLocal(FixHTMLpath(Allocator.GetFilename(Element, 0)));
  323. //alpha
  324. TmpItem := GetAlphaItem(AlphaObjItem.Children, UpperCase(Copy(Element.Name, 1, 2))).Children.NewItem;
  325. TmpItem.Text := Element.Name;
  326. TmpItem.addLocal(FixHTMLpath(Allocator.GetFilename(Element, 0)));
  327. end;
  328. // non object procedures and functions
  329. for j := 0 to AModule.InterfaceSection.Functions.Count-1 do
  330. begin
  331. Element := TPasFunctionType(AModule.InterfaceSection.Functions[j]);
  332. // by unit
  333. TmpItem := RoutinesUnitItem.Children.NewItem;
  334. TmpItem.Text := Element.Name;
  335. TmpItem.addLocal(FixHTMLpath(Allocator.GetFilename(Element, 0)));
  336. // alpha
  337. TmpItem := GetAlphaItem(AlphaRoutinesItem.Children, UpperCase(Element.Name[1])).Children.NewItem;
  338. TmpItem.Text := Element.Name;
  339. TmpItem.addLocal(FixHTMLpath(Allocator.GetFilename(Element, 0)));
  340. end;
  341. end;
  342. // cleanup
  343. for i := ObjByUnitItem.Children.Count-1 downto 0 do
  344. begin
  345. if ObjByUnitItem.Children.Item[i].Children.Count = 0 then
  346. ObjByUnitItem.Children.Delete(i);
  347. end;
  348. for i := RoutinesByUnitItem.Children.Count-1 downto 0 do
  349. begin
  350. if RoutinesByUnitItem.Children.Item[i].Children.Count = 0 then
  351. RoutinesByUnitItem.Children.Delete(i);
  352. end;
  353. for i := TOC.Items.Count-1 downto 0 do
  354. begin
  355. if TOC.Items.Item[i].Children.Count = 0 then
  356. TOC.Items.Delete(i);
  357. end;
  358. // Sort
  359. for i := 0 to TOC.Items.Count-1 do
  360. begin
  361. TOC.Items.Item[i].Children.Sort(TListSortCompare(@TOCSort));
  362. for j := 0 to TOC.Items.Item[i].Children.Count-1 do
  363. begin
  364. TOC.Items.Item[i].Children.Item[j].Children.Sort(TListSortCompare(@TOCSort));
  365. end;
  366. end;
  367. if not fnobintoc then
  368. fchm.AppendBinaryTOCFromSiteMap(Toc);
  369. TOC.SaveToStream(Stream);
  370. TOC.Free;
  371. fchm.AppendTOC(Stream);
  372. Stream.Free;
  373. DoLog('Generating TOC done');
  374. end;
  375. type
  376. TClassMemberType = (cmtProcedure, cmtFunction, cmtConstructor, cmtDestructor,
  377. cmtInterface, cmtProperty, cmtVariable, cmtOperator, cmtConstant, cmtUnknown);
  378. function ElementType(Element: TPasElement): TClassMemberType;
  379. var
  380. C: TClass;
  381. begin
  382. Result := cmtUnknown;
  383. if not Assigned(Element) then Exit;
  384. C:=Element.ClassType;
  385. if (C=TPasProcedure) or (C=TPasClassProcedure) then
  386. exit(cmtProcedure)
  387. else if (C=TPasFunction) or (C=TPasClassFunction) then
  388. exit(cmtFunction)
  389. else if (C=TPasConstructor) or (C=TPasClassConstructor) then
  390. exit(cmtConstructor)
  391. else if (C=TPasDestructor) or (C=TPasClassDestructor) then
  392. exit(cmtDestructor)
  393. else if (C=TPasOperator) or (C=TPasClassOperator) then
  394. exit(cmtOperator)
  395. else if C=TPasConst then
  396. exit(cmtConstant)
  397. else if C=TPasVariable then
  398. exit(cmtVariable)
  399. else if C=TPasProperty then
  400. exit(cmtProperty)
  401. else
  402. begin
  403. // Unknown
  404. exit(cmtUnknown);
  405. // WriteLn(' Warning El name: '+ Element.Name+' path: '+Element.PathName+' TypeName: '+Element.ElementTypeName);
  406. end;
  407. end;
  408. procedure TCHMHTMLWriter.GenerateIndex;
  409. var
  410. Index: TChmSiteMap;
  411. i, j, k: Integer;
  412. TmpItem: TChmSiteMapItem;
  413. ParentItem: TChmSiteMapItem;
  414. AModule: TPasModule;
  415. TmpElement: TPasElement;
  416. ParentElement: TPasElement;
  417. MemberItem: TChmSiteMapItem;
  418. Stream: TMemoryStream;
  419. RedirectUrl,Urls,SName: String;
  420. begin
  421. DoLog('Generating Index...');
  422. if not Assigned(Package) then
  423. begin
  424. DoLog('Package is not assigned...');
  425. Exit;
  426. end;
  427. Index := TChmSiteMap.Create(stIndex);
  428. Stream := TMemoryStream.Create;
  429. for i := 0 to Package.Modules.Count - 1 do
  430. //if false then
  431. begin
  432. AModule := TPasModule(Package.Modules[i]);
  433. if not assigned(AModule.InterfaceSection) then
  434. continue;
  435. ParentItem := Index.Items.NewItem;
  436. ParentItem.Text := AModule.Name;
  437. ParentItem.addLocal(FixHTMLpath(Allocator.GetFilename(AModule, 0)));
  438. // classes
  439. for j := 0 to AModule.InterfaceSection.Classes.Count-1 do
  440. begin
  441. ParentElement := TPasClassType(AModule.InterfaceSection.Classes[j]);
  442. ParentItem := Index.Items.NewItem;
  443. ParentItem.Text := ParentELement.Name;
  444. ParentItem.addLocal(FixHTMLpath(Allocator.GetFilename(ParentElement, 0)));
  445. for k := 0 to TPasClassType(ParentElement).Members.Count-1 do
  446. begin
  447. TmpElement := TPasElement(TPasClassType(ParentElement).Members.Items[k]);
  448. if Engine.HidePrivate and(TmpElement.Visibility = visPrivate) then
  449. continue;
  450. if Engine.HideProtected and(TmpElement.Visibility = visProtected) then
  451. continue;
  452. Urls:=FixHTMLpath(Allocator.GetFilename(TmpElement, 0));
  453. RedirectUrl:='';
  454. if TmpElement is TPasEnumValue then
  455. RedirectUrl := UTF8Encode(ResolveLinkIDAbs(tmpElement.Parent.PathName))
  456. else
  457. RedirectUrl := UTF8Encode(ResolveLinkIDAbs(tmpElement.PathName));
  458. if(trim(RedirectUrl)<>'') and (RedirectUrl<>urls) then
  459. begin
  460. //writeln('Hint: Index Resolved:',urls,' to ',RedirectUrl);
  461. urls:=RedirectUrl;
  462. end;
  463. TmpItem := ParentItem.Children.NewItem;
  464. case ElementType(TmpElement) of
  465. cmtProcedure : TmpItem.Text := TmpElement.Name + ' procedure';
  466. cmtFunction : TmpItem.Text := TmpElement.Name + ' function';
  467. cmtConstructor : TmpItem.Text := TmpElement.Name + ' constructor';
  468. cmtDestructor : TmpItem.Text := TmpElement.Name + ' destructor';
  469. cmtProperty : TmpItem.Text := TmpElement.Name + ' property';
  470. cmtVariable : TmpItem.Text := TmpElement.Name + ' variable';
  471. cmtInterface : TmpItem.Text := TmpElement.Name + ' interface';
  472. cmtOperator : TmpItem.Text := TmpElement.Name + ' operator';
  473. cmtConstant : TmpItem.Text := TmpElement.Name + ' const';
  474. cmtUnknown : TmpItem.Text := TmpElement.Name;
  475. end;
  476. TmpItem.addLocal(Urls);
  477. {
  478. ParentElement = Class
  479. TmpElement = Member
  480. }
  481. MemberItem := nil;
  482. MemberItem := GetAlphaItem(Index.Items, TmpElement.Name);
  483. // ahh! if MemberItem.Local is empty MemberType is not shown!
  484. MemberItem.addLocal(Urls);
  485. TmpItem := MemberItem.Children.NewItem;
  486. TmpItem.Text := ParentElement.Name;
  487. TmpItem.AddLocal(Urls);
  488. end;
  489. end;
  490. // routines
  491. for j := 0 to AModule.InterfaceSection.Functions.Count-1 do
  492. begin
  493. // routine name
  494. ParentElement := TPasElement(AModule.InterfaceSection.Functions[j]);
  495. case ElementType(ParentElement) of
  496. cmtProcedure : SName:= ' procedure';
  497. cmtFunction : SName:= ' function';
  498. cmtOperator : SName:= ' operator';
  499. //cmtConstant : SName:= ' const';
  500. else SName:= ' unknown'
  501. end;
  502. SName:= ParentElement.Name + ' ' + SName;
  503. MultiAlphaItem(Index.Items, SName, ParentElement, AModule.Name);
  504. end;
  505. // consts
  506. for j := 0 to AModule.InterfaceSection.Consts.Count-1 do
  507. begin
  508. ParentElement := TPasElement(AModule.InterfaceSection.Consts[j]);
  509. SName:= ParentElement.Name + ' const';
  510. MultiAlphaItem(Index.Items, SName, ParentElement, AModule.Name);
  511. end;
  512. // types
  513. for j := 0 to AModule.InterfaceSection.Types.Count-1 do
  514. begin
  515. ParentElement := TPasType(AModule.InterfaceSection.Types[j]);
  516. TmpItem := Index.Items.NewItem;
  517. TmpItem.Text := ParentElement.Name;
  518. TmpItem.addLocal(FixHTMLpath(Allocator.GetFilename(ParentElement, 0)));
  519. // enums
  520. if ParentELement is TPasEnumType then
  521. begin
  522. ParentItem := TmpItem;
  523. for k := 0 to TPasEnumType(ParentElement).Values.Count-1 do
  524. begin
  525. TmpElement := TPasType(TPasEnumType(ParentElement).Values.Items[k]);
  526. // subitem
  527. TmpItem := ParentItem.Children.NewItem;
  528. TmpItem.Text := TmpElement.Name;
  529. TmpItem.addLocal(ParentItem.Local);
  530. // root level
  531. TmpItem := Index.Items.NewItem;
  532. TmpItem.Text := TmpElement.Name;
  533. TmpItem.addLocal(ParentItem.Local);
  534. end;
  535. end;
  536. end;
  537. // variables
  538. for j := 0 to AModule.InterfaceSection.Variables.Count-1 do
  539. begin
  540. ParentElement := TPasElement(AModule.InterfaceSection.Variables[j]);
  541. SName:= ParentElement.Name + ' variable';
  542. MultiAlphaItem(Index.Items, SName, ParentElement, AModule.Name);
  543. end;
  544. // declarations
  545. {
  546. for j := 0 to AModule.InterfaceSection.Declarations.Count-1 do
  547. begin
  548. ParentElement := TPasElement(AModule.InterfaceSection.Declarations[j]);
  549. TmpItem := Index.Items.NewItem;
  550. TmpItem.Text := ParentElement.Name;
  551. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  552. end;
  553. // resource strings
  554. for j := 0 to AModule.InterfaceSection.ResStrings.Count-1 do
  555. begin
  556. ParentElement := TPasElement(AModule.InterfaceSection.ResStrings[j]);
  557. TmpItem := Index.Items.NewItem;
  558. TmpItem.Text := ParentElement.Name;
  559. TmpItem.Local := FixHTMLpath(Allocator.GetFilename(ParentElement, 0));
  560. end;
  561. }
  562. end;
  563. // Sort
  564. Index.Items.Sort(TListSortCompare(@TOCSort));
  565. for i := 0 to Index.Items.Count-1 do
  566. begin
  567. Index.Items.Item[i].Children.Sort(TListSortCompare(@TOCSort));
  568. end;
  569. // save
  570. Index.SaveToStream(Stream);
  571. if not fnobinindex then
  572. fchm.AppendBinaryindexFromSitemap(index,false);
  573. Index.Free;
  574. Stream.Position :=0 ;
  575. FChm.AppendIndex(Stream);
  576. Stream.Free;
  577. DoLog('Generating Index Done');
  578. end;
  579. procedure TCHMHTMLWriter.DoWriteDocumentation;
  580. var
  581. i: Integer;
  582. PageDoc: TXMLDocument;
  583. FileStream: TMemoryStream;
  584. IFileName,FileName: String;
  585. FilePath: String;
  586. begin
  587. AllocatePages;
  588. DoLog(SWritingPages, [PageCount]);
  589. FileName := Engine.Output;
  590. if FileName = '' then
  591. Raise Exception.Create('Error: no --output option used.');
  592. if ExtractFileExt(FileName) <> FileNameExtension then
  593. FileName := ChangeFileExt(FileName, FileNameExtension);
  594. FOutChm := TFileStream.Create(FileName, fmOpenReadWrite or fmCreate);
  595. FTempUncompressedName := GetTempFileName+IntToStr(GetProcessID) +'.raw';
  596. FTempUncompressed := TFileStream.Create(FTempUncompressedName, fmOpenReadWrite or fmCreate);
  597. FChm := TFpDocChmWriter.Create(FOutChm, False);
  598. FChm.Title := FChmTitle;
  599. FChm.TempRawStream := FTempUncompressed;
  600. FChm.OnGetFileData := @RetrieveOtherFiles;
  601. FChm.OnLastFile := @LastFileAdded;
  602. FChm.hasbinarytoc:=not fnobintoc;
  603. FChm.hasbinaryindex:=not fnobinindex;
  604. //FChm.Cores:=1;
  605. ProcessOptions;
  606. FileStream := TMemoryStream.Create;
  607. for i := 0 to PageInfos.Count - 1 do
  608. with TPageInfo(PageInfos[i]) do
  609. begin
  610. PageDoc := CreateHTMLPage(Element, SubpageIndex);
  611. try
  612. FileName := ExtractFileName(Allocator.GetFilename(Element, SubpageIndex));
  613. FilePath := '/'+FixHTMLpath(ExtractFilePath(Allocator.GetFilename(Element, SubpageIndex)));
  614. try
  615. WriteHTMLFile(PageDoc, FileStream);
  616. FChm.AddStreamToArchive(FileName, FilePath, FileStream, True);
  617. except
  618. on E: Exception do
  619. DoLog(Format(SErrCouldNotCreateFile, [FileName, e.Message]));
  620. end;
  621. finally
  622. PageDoc.Free;
  623. FileStream.Size := 0;
  624. end;
  625. end;
  626. FileStream.Free;
  627. DoLog('HTML Files written. Collecting other files and compressing...this could take some time');
  628. //write any found images to CHM stream
  629. FileStream := TMemoryStream.Create;
  630. for iFilename in ImageFileList do
  631. begin
  632. {$ifdef imagetest} DoLog(' adding image: '+iFileName); {$endif}
  633. if FileExists(iFileName) then
  634. begin
  635. {$ifdef imagetest} DoLog(' - found'); {$endif}
  636. FileName := ExtractFileName(iFileName);
  637. FilePath := '/'+FixHTMLpath(ExtractFilePath(iFileName));
  638. FileStream.LoadFromFile(iFileName);
  639. FChm.AddStreamToArchive(FileName, FilePath, FileStream, True);
  640. FileStream.Size := 0;
  641. end
  642. else
  643. {$ifdef imagetest} DoLog(' - not found'){$endif};
  644. end;
  645. FileStream.Free;
  646. FChm.Execute;
  647. FChm.Free;
  648. DoLog('Collecting done');
  649. // we don't need to free FTempUncompressed it is freed into TFpDocChmWriter
  650. // FTempUncompressed.Free;
  651. FOutChm.Free;
  652. DeleteFile(FTempUncompressedName);
  653. end;
  654. function TCHMHTMLWriter.CreateAllocator: TFileAllocator;
  655. begin
  656. Result:=TCHmFileNameAllocator.Create('.html');
  657. end;
  658. function TCHMHTMLWriter.InterPretOption(const Cmd, Arg: String): boolean;
  659. begin
  660. Result:=True;
  661. FNoBinToc:=False;
  662. FnoBinIndex:=False;
  663. if Cmd = '--toc-file' then
  664. FTOCName := arg
  665. else if Cmd = '--index-file' then
  666. FIndexName := arg
  667. else if Cmd = '--default-page' then
  668. FDefaultPage := arg
  669. else if Cmd = '--other-files' then
  670. FOtherFiles := arg
  671. else if Cmd = '--auto-index' then
  672. FAutoIndex := True
  673. else if Cmd = '--auto-toc' then
  674. FAutoTOC := True
  675. else if Cmd = '--no-bintoc' then
  676. FNoBinToc := True
  677. else if Cmd = '--no-binindex' then
  678. FNoBinIndex := True
  679. else if Cmd = '--make-searchable' then
  680. FMakeSearchable := True
  681. else if Cmd = '--chm-title' then
  682. FChmTitle := arg
  683. else
  684. Result:=inherited InterPretOption(Cmd, Arg);
  685. if Length(FChmTitle) = 0 then
  686. FChmTitle := Copy(Package.Name, 2, Length(Package.Name));
  687. end;
  688. class procedure TCHMHTMLWriter.Usage(List: TStrings);
  689. begin
  690. THTMLWriter.Usage(List);
  691. List.add('--default-page');
  692. List.Add(SCHMUsageDefPage);
  693. List.add('--toc-file');
  694. List.Add(SCHMUsageTOC);
  695. List.add('--index-file');
  696. List.Add(SCHMUsageIndex);
  697. List.add('--other-files');
  698. List.Add(SCHMUsageOtrFiles);
  699. List.add('--css-file');
  700. List.Add(SCHMUsageCSSFile);
  701. List.add('--auto-index');
  702. List.Add(SCHMUsageAutoIDX);
  703. List.add('--auto-toc');
  704. List.Add(SCHMUsageAutoTOC);
  705. List.add('--make-searchable');
  706. List.Add(SCHMUsageMakeSearch);
  707. List.Add('--chm-title');
  708. List.Add(SCHMUsageChmTitle);
  709. end;
  710. class function TCHMHTMLWriter.FileNameExtension: String;
  711. begin
  712. result:='.chm';
  713. end;
  714. class procedure TCHMHTMLWriter.SplitImport(var AFilename, ALinkPrefix: String);
  715. var
  716. i: integer;
  717. begin
  718. i := Pos(',', AFilename);
  719. if i > 0 then
  720. begin //split into filename and prefix
  721. ALinkPrefix := Copy(AFilename,i+1,Length(AFilename));
  722. SetLength(AFilename, i-1);
  723. if copy(ALinkPrefix,1,2)='..' then // workaround for project files.
  724. begin
  725. ALinkPrefix := 'ms-its:' + ChangeFileExt(ExtractFileName(AFilename), '.chm') + '::/';
  726. AFilename := ChangeFileExt(AFilename, '.xct');
  727. end;
  728. end
  729. else if ALinkPrefix = '' then
  730. begin //synthesize outdir\pgk.xct, ms-its:pkg.chm::/
  731. ALinkPrefix := 'ms-its:' + ChangeFileExt(ExtractFileName(AFilename), '.chm') + '::/';
  732. AFilename := ChangeFileExt(AFilename, '.xct');
  733. end;
  734. end;
  735. initialization
  736. RegisterWriter(TCHMHTMLWriter,'chm','Compressed HTML file output using fpdoc.css stylesheet.');
  737. finalization
  738. UnRegisterWriter('chm');
  739. end.