dw_chm.pp 24 KB

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