You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

FO2StructureTreeConverter.java 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. /*
  2. * Licensed to the Apache Software Foundation (ASF) under one or more
  3. * contributor license agreements. See the NOTICE file distributed with
  4. * this work for additional information regarding copyright ownership.
  5. * The ASF licenses this file to You under the Apache License, Version 2.0
  6. * (the "License"); you may not use this file except in compliance with
  7. * the License. You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. /* $Id$ */
  18. package org.apache.fop.accessibility.fo;
  19. import java.util.HashMap;
  20. import java.util.Map;
  21. import java.util.Stack;
  22. import org.xml.sax.SAXException;
  23. import org.apache.fop.accessibility.Accessibility;
  24. import org.apache.fop.accessibility.StructureTreeEventHandler;
  25. import org.apache.fop.fo.DelegatingFOEventHandler;
  26. import org.apache.fop.fo.FOEventHandler;
  27. import org.apache.fop.fo.FOText;
  28. import org.apache.fop.fo.extensions.ExternalDocument;
  29. import org.apache.fop.fo.flow.AbstractRetrieveMarker;
  30. import org.apache.fop.fo.flow.BasicLink;
  31. import org.apache.fop.fo.flow.Block;
  32. import org.apache.fop.fo.flow.BlockContainer;
  33. import org.apache.fop.fo.flow.Character;
  34. import org.apache.fop.fo.flow.ExternalGraphic;
  35. import org.apache.fop.fo.flow.Footnote;
  36. import org.apache.fop.fo.flow.FootnoteBody;
  37. import org.apache.fop.fo.flow.Inline;
  38. import org.apache.fop.fo.flow.InstreamForeignObject;
  39. import org.apache.fop.fo.flow.Leader;
  40. import org.apache.fop.fo.flow.ListBlock;
  41. import org.apache.fop.fo.flow.ListItem;
  42. import org.apache.fop.fo.flow.ListItemBody;
  43. import org.apache.fop.fo.flow.ListItemLabel;
  44. import org.apache.fop.fo.flow.PageNumber;
  45. import org.apache.fop.fo.flow.PageNumberCitation;
  46. import org.apache.fop.fo.flow.PageNumberCitationLast;
  47. import org.apache.fop.fo.flow.RetrieveMarker;
  48. import org.apache.fop.fo.flow.RetrieveTableMarker;
  49. import org.apache.fop.fo.flow.Wrapper;
  50. import org.apache.fop.fo.flow.table.Table;
  51. import org.apache.fop.fo.flow.table.TableBody;
  52. import org.apache.fop.fo.flow.table.TableCell;
  53. import org.apache.fop.fo.flow.table.TableColumn;
  54. import org.apache.fop.fo.flow.table.TableFooter;
  55. import org.apache.fop.fo.flow.table.TableHeader;
  56. import org.apache.fop.fo.flow.table.TableRow;
  57. import org.apache.fop.fo.pagination.Flow;
  58. import org.apache.fop.fo.pagination.PageSequence;
  59. import org.apache.fop.fo.pagination.Root;
  60. import org.apache.fop.fo.pagination.StaticContent;
  61. import org.apache.fop.fo.properties.CommonAccessibility;
  62. import org.apache.fop.fo.properties.CommonAccessibilityHolder;
  63. /**
  64. * Allows to create the structure tree of an FO document, by converting FO
  65. * events into appropriate structure tree events.
  66. */
  67. public class FO2StructureTreeConverter extends DelegatingFOEventHandler {
  68. /** The top of the {@link converters} stack. */
  69. private FOEventHandler converter;
  70. private Stack<FOEventHandler> converters = new Stack<FOEventHandler>();
  71. private final StructureTreeEventTrigger structureTreeEventTrigger;
  72. /** The descendants of some elements like fo:leader must be ignored. */
  73. private final FOEventHandler eventSwallower = new FOEventHandler() {
  74. };
  75. private final Map<AbstractRetrieveMarker, State> states = new HashMap<AbstractRetrieveMarker, State>();
  76. private static final class State {
  77. private final FOEventHandler converter;
  78. private final Stack<FOEventHandler> converters;
  79. @SuppressWarnings("unchecked")
  80. State(FO2StructureTreeConverter o) {
  81. this.converter = o.converter;
  82. this.converters = (Stack<FOEventHandler>) o.converters.clone();
  83. }
  84. }
  85. /**
  86. * Creates a new instance.
  87. *
  88. * @param structureTreeEventHandler the object that will hold the structure tree
  89. * @param delegate the FO event handler that must be wrapped by this instance
  90. */
  91. public FO2StructureTreeConverter(StructureTreeEventHandler structureTreeEventHandler,
  92. FOEventHandler delegate) {
  93. super(delegate);
  94. this.structureTreeEventTrigger = new StructureTreeEventTrigger(structureTreeEventHandler);
  95. this.converter = structureTreeEventTrigger;
  96. }
  97. @Override
  98. public void startDocument() throws SAXException {
  99. converter.startDocument();
  100. super.startDocument();
  101. }
  102. @Override
  103. public void endDocument() throws SAXException {
  104. converter.endDocument();
  105. super.endDocument();
  106. }
  107. @Override
  108. public void startRoot(Root root) {
  109. converter.startRoot(root);
  110. super.startRoot(root);
  111. }
  112. @Override
  113. public void endRoot(Root root) {
  114. converter.endRoot(root);
  115. super.endRoot(root);
  116. }
  117. @Override
  118. public void startPageSequence(PageSequence pageSeq) {
  119. converter.startPageSequence(pageSeq);
  120. super.startPageSequence(pageSeq);
  121. }
  122. @Override
  123. public void endPageSequence(PageSequence pageSeq) {
  124. converter.endPageSequence(pageSeq);
  125. super.endPageSequence(pageSeq);
  126. }
  127. @Override
  128. public void startPageNumber(PageNumber pagenum) {
  129. converter.startPageNumber(pagenum);
  130. super.startPageNumber(pagenum);
  131. }
  132. @Override
  133. public void endPageNumber(PageNumber pagenum) {
  134. converter.endPageNumber(pagenum);
  135. super.endPageNumber(pagenum);
  136. }
  137. @Override
  138. public void startPageNumberCitation(PageNumberCitation pageCite) {
  139. converter.startPageNumberCitation(pageCite);
  140. super.startPageNumberCitation(pageCite);
  141. }
  142. @Override
  143. public void endPageNumberCitation(PageNumberCitation pageCite) {
  144. converter.endPageNumberCitation(pageCite);
  145. super.endPageNumberCitation(pageCite);
  146. }
  147. @Override
  148. public void startPageNumberCitationLast(PageNumberCitationLast pageLast) {
  149. converter.startPageNumberCitationLast(pageLast);
  150. super.startPageNumberCitationLast(pageLast);
  151. }
  152. @Override
  153. public void endPageNumberCitationLast(PageNumberCitationLast pageLast) {
  154. converter.endPageNumberCitationLast(pageLast);
  155. super.endPageNumberCitationLast(pageLast);
  156. }
  157. @Override
  158. public void startStatic(StaticContent staticContent) {
  159. handleStartArtifact(staticContent);
  160. converter.startStatic(staticContent);
  161. super.startStatic(staticContent);
  162. }
  163. @Override
  164. public void endStatic(StaticContent staticContent) {
  165. converter.endStatic(staticContent);
  166. handleEndArtifact(staticContent);
  167. super.endStatic(staticContent);
  168. }
  169. @Override
  170. public void startFlow(Flow fl) {
  171. converter.startFlow(fl);
  172. super.startFlow(fl);
  173. }
  174. @Override
  175. public void endFlow(Flow fl) {
  176. converter.endFlow(fl);
  177. super.endFlow(fl);
  178. }
  179. @Override
  180. public void startBlock(Block bl) {
  181. converter.startBlock(bl);
  182. super.startBlock(bl);
  183. }
  184. @Override
  185. public void endBlock(Block bl) {
  186. converter.endBlock(bl);
  187. super.endBlock(bl);
  188. }
  189. @Override
  190. public void startBlockContainer(BlockContainer blc) {
  191. converter.startBlockContainer(blc);
  192. super.startBlockContainer(blc);
  193. }
  194. @Override
  195. public void endBlockContainer(BlockContainer blc) {
  196. converter.endBlockContainer(blc);
  197. super.endBlockContainer(blc);
  198. }
  199. @Override
  200. public void startInline(Inline inl) {
  201. converter.startInline(inl);
  202. super.startInline(inl);
  203. }
  204. @Override
  205. public void endInline(Inline inl) {
  206. converter.endInline(inl);
  207. super.endInline(inl);
  208. }
  209. @Override
  210. public void startTable(Table tbl) {
  211. converter.startTable(tbl);
  212. super.startTable(tbl);
  213. }
  214. @Override
  215. public void endTable(Table tbl) {
  216. converter.endTable(tbl);
  217. super.endTable(tbl);
  218. }
  219. @Override
  220. public void startColumn(TableColumn tc) {
  221. converter.startColumn(tc);
  222. super.startColumn(tc);
  223. }
  224. @Override
  225. public void endColumn(TableColumn tc) {
  226. converter.endColumn(tc);
  227. super.endColumn(tc);
  228. }
  229. @Override
  230. public void startHeader(TableHeader header) {
  231. converter.startHeader(header);
  232. super.startHeader(header);
  233. }
  234. @Override
  235. public void endHeader(TableHeader header) {
  236. converter.endHeader(header);
  237. super.endHeader(header);
  238. }
  239. @Override
  240. public void startFooter(TableFooter footer) {
  241. converter.startFooter(footer);
  242. super.startFooter(footer);
  243. }
  244. @Override
  245. public void endFooter(TableFooter footer) {
  246. converter.endFooter(footer);
  247. super.endFooter(footer);
  248. }
  249. @Override
  250. public void startBody(TableBody body) {
  251. converter.startBody(body);
  252. super.startBody(body);
  253. }
  254. @Override
  255. public void endBody(TableBody body) {
  256. converter.endBody(body);
  257. super.endBody(body);
  258. }
  259. @Override
  260. public void startRow(TableRow tr) {
  261. converter.startRow(tr);
  262. super.startRow(tr);
  263. }
  264. @Override
  265. public void endRow(TableRow tr) {
  266. converter.endRow(tr);
  267. super.endRow(tr);
  268. }
  269. @Override
  270. public void startCell(TableCell tc) {
  271. converter.startCell(tc);
  272. super.startCell(tc);
  273. }
  274. @Override
  275. public void endCell(TableCell tc) {
  276. converter.endCell(tc);
  277. super.endCell(tc);
  278. }
  279. @Override
  280. public void startList(ListBlock lb) {
  281. converter.startList(lb);
  282. super.startList(lb);
  283. }
  284. @Override
  285. public void endList(ListBlock lb) {
  286. converter.endList(lb);
  287. super.endList(lb);
  288. }
  289. @Override
  290. public void startListItem(ListItem li) {
  291. converter.startListItem(li);
  292. super.startListItem(li);
  293. }
  294. @Override
  295. public void endListItem(ListItem li) {
  296. converter.endListItem(li);
  297. super.endListItem(li);
  298. }
  299. @Override
  300. public void startListLabel(ListItemLabel listItemLabel) {
  301. converter.startListLabel(listItemLabel);
  302. super.startListLabel(listItemLabel);
  303. }
  304. @Override
  305. public void endListLabel(ListItemLabel listItemLabel) {
  306. converter.endListLabel(listItemLabel);
  307. super.endListLabel(listItemLabel);
  308. }
  309. @Override
  310. public void startListBody(ListItemBody listItemBody) {
  311. converter.startListBody(listItemBody);
  312. super.startListBody(listItemBody);
  313. }
  314. @Override
  315. public void endListBody(ListItemBody listItemBody) {
  316. converter.endListBody(listItemBody);
  317. super.endListBody(listItemBody);
  318. }
  319. @Override
  320. public void startMarkup() {
  321. converter.startMarkup();
  322. super.startMarkup();
  323. }
  324. @Override
  325. public void endMarkup() {
  326. converter.endMarkup();
  327. super.endMarkup();
  328. }
  329. @Override
  330. public void startLink(BasicLink basicLink) {
  331. converter.startLink(basicLink);
  332. super.startLink(basicLink);
  333. }
  334. @Override
  335. public void endLink(BasicLink basicLink) {
  336. converter.endLink(basicLink);
  337. super.endLink(basicLink);
  338. }
  339. @Override
  340. public void image(ExternalGraphic eg) {
  341. converter.image(eg);
  342. super.image(eg);
  343. }
  344. @Override
  345. public void pageRef() {
  346. converter.pageRef();
  347. super.pageRef();
  348. }
  349. @Override
  350. public void startInstreamForeignObject(InstreamForeignObject ifo) {
  351. converter.startInstreamForeignObject(ifo);
  352. super.startInstreamForeignObject(ifo);
  353. }
  354. @Override
  355. public void endInstreamForeignObject(InstreamForeignObject ifo) {
  356. converter.endInstreamForeignObject(ifo);
  357. super.endInstreamForeignObject(ifo);
  358. }
  359. @Override
  360. public void startFootnote(Footnote footnote) {
  361. converter.startFootnote(footnote);
  362. super.startFootnote(footnote);
  363. }
  364. @Override
  365. public void endFootnote(Footnote footnote) {
  366. converter.endFootnote(footnote);
  367. super.endFootnote(footnote);
  368. }
  369. @Override
  370. public void startFootnoteBody(FootnoteBody body) {
  371. converter.startFootnoteBody(body);
  372. super.startFootnoteBody(body);
  373. }
  374. @Override
  375. public void endFootnoteBody(FootnoteBody body) {
  376. converter.endFootnoteBody(body);
  377. super.endFootnoteBody(body);
  378. }
  379. @Override
  380. public void startLeader(Leader l) {
  381. converters.push(converter);
  382. converter = eventSwallower;
  383. converter.startLeader(l);
  384. super.startLeader(l);
  385. }
  386. @Override
  387. public void endLeader(Leader l) {
  388. converter.endLeader(l);
  389. converter = converters.pop();
  390. super.endLeader(l);
  391. }
  392. @Override
  393. public void startWrapper(Wrapper wrapper) {
  394. handleStartArtifact(wrapper);
  395. converter.startWrapper(wrapper);
  396. super.startWrapper(wrapper);
  397. }
  398. @Override
  399. public void endWrapper(Wrapper wrapper) {
  400. converter.endWrapper(wrapper);
  401. handleEndArtifact(wrapper);
  402. super.endWrapper(wrapper);
  403. }
  404. @Override
  405. public void startRetrieveMarker(RetrieveMarker retrieveMarker) {
  406. converter.startRetrieveMarker(retrieveMarker);
  407. saveState(retrieveMarker);
  408. super.startRetrieveMarker(retrieveMarker);
  409. }
  410. private void saveState(AbstractRetrieveMarker retrieveMarker) {
  411. states.put(retrieveMarker, new State(this));
  412. }
  413. @Override
  414. public void endRetrieveMarker(RetrieveMarker retrieveMarker) {
  415. converter.endRetrieveMarker(retrieveMarker);
  416. super.endRetrieveMarker(retrieveMarker);
  417. }
  418. @Override
  419. public void restoreState(RetrieveMarker retrieveMarker) {
  420. restoreRetrieveMarkerState(retrieveMarker);
  421. converter.restoreState(retrieveMarker);
  422. super.restoreState(retrieveMarker);
  423. }
  424. @SuppressWarnings("unchecked")
  425. private void restoreRetrieveMarkerState(AbstractRetrieveMarker retrieveMarker) {
  426. State state = states.get(retrieveMarker);
  427. this.converter = state.converter;
  428. this.converters = (Stack<FOEventHandler>) state.converters.clone();
  429. }
  430. @Override
  431. public void startRetrieveTableMarker(RetrieveTableMarker retrieveTableMarker) {
  432. converter.startRetrieveTableMarker(retrieveTableMarker);
  433. saveState(retrieveTableMarker);
  434. super.startRetrieveTableMarker(retrieveTableMarker);
  435. }
  436. @Override
  437. public void endRetrieveTableMarker(RetrieveTableMarker retrieveTableMarker) {
  438. converter.endRetrieveTableMarker(retrieveTableMarker);
  439. super.endRetrieveTableMarker(retrieveTableMarker);
  440. }
  441. @Override
  442. public void restoreState(RetrieveTableMarker retrieveTableMarker) {
  443. restoreRetrieveMarkerState(retrieveTableMarker);
  444. converter.restoreState(retrieveTableMarker);
  445. super.restoreState(retrieveTableMarker);
  446. }
  447. @Override
  448. public void character(Character c) {
  449. converter.character(c);
  450. super.character(c);
  451. }
  452. @Override
  453. public void characters(FOText foText) {
  454. converter.characters(foText);
  455. super.characters(foText);
  456. }
  457. @Override
  458. public void startExternalDocument(ExternalDocument document) {
  459. converter.startExternalDocument(document);
  460. super.startExternalDocument(document);
  461. }
  462. @Override
  463. public void endExternalDocument(ExternalDocument document) {
  464. converter.endExternalDocument(document);
  465. super.endExternalDocument(document);
  466. }
  467. private void handleStartArtifact(CommonAccessibilityHolder fobj) {
  468. if (isArtifact(fobj)) {
  469. converters.push(converter);
  470. converter = eventSwallower;
  471. }
  472. }
  473. private void handleEndArtifact(CommonAccessibilityHolder fobj) {
  474. if (isArtifact(fobj)) {
  475. converter = converters.pop();
  476. }
  477. }
  478. private boolean isArtifact(CommonAccessibilityHolder fobj) {
  479. CommonAccessibility accessibility = fobj.getCommonAccessibility();
  480. return Accessibility.ROLE_ARTIFACT.equalsIgnoreCase(accessibility.getRole());
  481. }
  482. }