dw_chm.pp 24 KB

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