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.

StructureTreeEventTrigger.java 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569
  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.Locale;
  21. import java.util.Map;
  22. import java.util.Stack;
  23. import javax.xml.XMLConstants;
  24. import org.xml.sax.helpers.AttributesImpl;
  25. import org.apache.fop.accessibility.StructureTreeElement;
  26. import org.apache.fop.accessibility.StructureTreeEventHandler;
  27. import org.apache.fop.fo.FOEventHandler;
  28. import org.apache.fop.fo.FONode;
  29. import org.apache.fop.fo.FOText;
  30. import org.apache.fop.fo.FObj;
  31. import org.apache.fop.fo.extensions.ExtensionElementMapping;
  32. import org.apache.fop.fo.extensions.InternalElementMapping;
  33. import org.apache.fop.fo.flow.AbstractRetrieveMarker;
  34. import org.apache.fop.fo.flow.BasicLink;
  35. import org.apache.fop.fo.flow.Block;
  36. import org.apache.fop.fo.flow.BlockContainer;
  37. import org.apache.fop.fo.flow.Character;
  38. import org.apache.fop.fo.flow.ExternalGraphic;
  39. import org.apache.fop.fo.flow.Footnote;
  40. import org.apache.fop.fo.flow.FootnoteBody;
  41. import org.apache.fop.fo.flow.Inline;
  42. import org.apache.fop.fo.flow.InstreamForeignObject;
  43. import org.apache.fop.fo.flow.ListBlock;
  44. import org.apache.fop.fo.flow.ListItem;
  45. import org.apache.fop.fo.flow.ListItemBody;
  46. import org.apache.fop.fo.flow.ListItemLabel;
  47. import org.apache.fop.fo.flow.PageNumber;
  48. import org.apache.fop.fo.flow.PageNumberCitation;
  49. import org.apache.fop.fo.flow.PageNumberCitationLast;
  50. import org.apache.fop.fo.flow.RetrieveMarker;
  51. import org.apache.fop.fo.flow.RetrieveTableMarker;
  52. import org.apache.fop.fo.flow.Wrapper;
  53. import org.apache.fop.fo.flow.table.Table;
  54. import org.apache.fop.fo.flow.table.TableBody;
  55. import org.apache.fop.fo.flow.table.TableCell;
  56. import org.apache.fop.fo.flow.table.TableFooter;
  57. import org.apache.fop.fo.flow.table.TableHeader;
  58. import org.apache.fop.fo.flow.table.TableRow;
  59. import org.apache.fop.fo.pagination.Flow;
  60. import org.apache.fop.fo.pagination.LayoutMasterSet;
  61. import org.apache.fop.fo.pagination.PageSequence;
  62. import org.apache.fop.fo.pagination.Root;
  63. import org.apache.fop.fo.pagination.StaticContent;
  64. import org.apache.fop.fo.properties.CommonAccessibilityHolder;
  65. import org.apache.fop.fo.properties.CommonHyphenation;
  66. import org.apache.fop.util.LanguageTags;
  67. import org.apache.fop.util.XMLUtil;
  68. /**
  69. * A bridge between {@link FOEventHandler} and {@link StructureTreeEventHandler}.
  70. */
  71. class StructureTreeEventTrigger extends FOEventHandler {
  72. private StructureTreeEventHandler structureTreeEventHandler;
  73. private LayoutMasterSet layoutMasterSet;
  74. private Stack<Table> tables = new Stack<Table>();
  75. private Stack<Boolean> inTableHeader = new Stack<Boolean>();
  76. private Stack<Locale> locales = new Stack<Locale>();
  77. private final Map<AbstractRetrieveMarker, State> states = new HashMap<AbstractRetrieveMarker, State>();
  78. private static final class State {
  79. private final Stack<Table> tables;
  80. private final Stack<Boolean> inTableHeader;
  81. private final Stack<Locale> locales;
  82. @SuppressWarnings("unchecked")
  83. State(StructureTreeEventTrigger o) {
  84. this.tables = (Stack<Table>) o.tables.clone();
  85. this.inTableHeader = (Stack<Boolean>) o.inTableHeader.clone();
  86. this.locales = (Stack<Locale>) o.locales.clone();
  87. }
  88. }
  89. public StructureTreeEventTrigger(StructureTreeEventHandler structureTreeEventHandler) {
  90. this.structureTreeEventHandler = structureTreeEventHandler;
  91. }
  92. @Override
  93. public void startRoot(Root root) {
  94. locales.push(root.getLocale());
  95. }
  96. @Override
  97. public void endRoot(Root root) {
  98. locales.pop();
  99. }
  100. @Override
  101. public void startPageSequence(PageSequence pageSeq) {
  102. if (layoutMasterSet == null) {
  103. layoutMasterSet = pageSeq.getRoot().getLayoutMasterSet();
  104. }
  105. Locale locale = pageSeq.getLocale();
  106. if (locale != null) {
  107. locales.push(locale);
  108. } else {
  109. locales.push(locales.peek());
  110. }
  111. String role = pageSeq.getCommonAccessibility().getRole();
  112. structureTreeEventHandler.startPageSequence(locale, role);
  113. }
  114. @Override
  115. public void endPageSequence(PageSequence pageSeq) {
  116. structureTreeEventHandler.endPageSequence();
  117. locales.pop();
  118. }
  119. @Override
  120. public void startPageNumber(PageNumber pagenum) {
  121. startElementWithID(pagenum);
  122. }
  123. @Override
  124. public void endPageNumber(PageNumber pagenum) {
  125. endElement(pagenum);
  126. }
  127. @Override
  128. public void startPageNumberCitation(PageNumberCitation pageCite) {
  129. startElementWithID(pageCite);
  130. }
  131. @Override
  132. public void endPageNumberCitation(PageNumberCitation pageCite) {
  133. endElement(pageCite);
  134. }
  135. @Override
  136. public void startPageNumberCitationLast(PageNumberCitationLast pageLast) {
  137. startElementWithID(pageLast);
  138. }
  139. @Override
  140. public void endPageNumberCitationLast(PageNumberCitationLast pageLast) {
  141. endElement(pageLast);
  142. }
  143. @Override
  144. public void startStatic(StaticContent staticContent) {
  145. AttributesImpl flowName = createFlowNameAttribute(staticContent.getFlowName());
  146. startElement(staticContent, flowName);
  147. }
  148. private AttributesImpl createFlowNameAttribute(String flowName) {
  149. String regionName = layoutMasterSet.getDefaultRegionNameFor(flowName);
  150. AttributesImpl attribute = new AttributesImpl();
  151. addNoNamespaceAttribute(attribute, Flow.FLOW_NAME, regionName);
  152. return attribute;
  153. }
  154. @Override
  155. public void endStatic(StaticContent staticContent) {
  156. endElement(staticContent);
  157. }
  158. @Override
  159. public void startFlow(Flow fl) {
  160. AttributesImpl flowName = createFlowNameAttribute(fl.getFlowName());
  161. startElement(fl, flowName);
  162. }
  163. @Override
  164. public void endFlow(Flow fl) {
  165. endElement(fl);
  166. }
  167. @Override
  168. public void startBlock(Block bl) {
  169. CommonHyphenation hyphProperties = bl.getCommonHyphenation();
  170. AttributesImpl attributes = createLangAttribute(hyphProperties);
  171. startElement(bl, attributes);
  172. }
  173. private AttributesImpl createLangAttribute(CommonHyphenation hyphProperties) {
  174. Locale locale = hyphProperties.getLocale();
  175. AttributesImpl attributes = new AttributesImpl();
  176. if (locale == null || locale.equals(locales.peek())) {
  177. locales.push(locales.peek());
  178. } else {
  179. locales.push(locale);
  180. addAttribute(attributes, XMLConstants.XML_NS_URI, "lang", "xml",
  181. LanguageTags.toLanguageTag(locale));
  182. }
  183. return attributes;
  184. }
  185. @Override
  186. public void endBlock(Block bl) {
  187. endElement(bl);
  188. locales.pop();
  189. }
  190. @Override
  191. public void startBlockContainer(BlockContainer blc) {
  192. startElement(blc);
  193. }
  194. @Override
  195. public void endBlockContainer(BlockContainer blc) {
  196. endElement(blc);
  197. }
  198. @Override
  199. public void startInline(Inline inl) {
  200. startElement(inl);
  201. }
  202. @Override
  203. public void endInline(Inline inl) {
  204. endElement(inl);
  205. }
  206. @Override
  207. public void startTable(Table tbl) {
  208. tables.push(tbl);
  209. startElement(tbl);
  210. }
  211. @Override
  212. public void endTable(Table tbl) {
  213. endElement(tbl);
  214. tables.pop();
  215. }
  216. @Override
  217. public void startHeader(TableHeader header) {
  218. inTableHeader.push(Boolean.TRUE);
  219. startElement(header);
  220. }
  221. @Override
  222. public void endHeader(TableHeader header) {
  223. endElement(header);
  224. inTableHeader.pop();
  225. }
  226. @Override
  227. public void startFooter(TableFooter footer) {
  228. // TODO Shouldn't it be true?
  229. inTableHeader.push(Boolean.FALSE);
  230. startElement(footer);
  231. }
  232. @Override
  233. public void endFooter(TableFooter footer) {
  234. endElement(footer);
  235. inTableHeader.pop();
  236. }
  237. @Override
  238. public void startBody(TableBody body) {
  239. inTableHeader.push(Boolean.FALSE);
  240. startElement(body);
  241. }
  242. @Override
  243. public void endBody(TableBody body) {
  244. endElement(body);
  245. inTableHeader.pop();
  246. }
  247. @Override
  248. public void startRow(TableRow tr) {
  249. startElement(tr);
  250. }
  251. @Override
  252. public void endRow(TableRow tr) {
  253. endElement(tr);
  254. }
  255. @Override
  256. public void startCell(TableCell tc) {
  257. AttributesImpl attributes = new AttributesImpl();
  258. addSpanAttribute(attributes, "number-columns-spanned", tc.getNumberColumnsSpanned());
  259. addSpanAttribute(attributes, "number-rows-spanned", tc.getNumberRowsSpanned());
  260. boolean rowHeader = inTableHeader.peek();
  261. boolean columnHeader = tables.peek().getColumn(tc.getColumnNumber() - 1).isHeader();
  262. if (rowHeader || columnHeader) {
  263. final String th = "TH";
  264. String role = tc.getCommonAccessibility().getRole();
  265. /* Do not override a custom role */
  266. if (role == null) {
  267. role = th;
  268. addNoNamespaceAttribute(attributes, "role", th);
  269. }
  270. if (role.equals(th)) {
  271. if (columnHeader) {
  272. String scope = rowHeader ? "Both" : "Row";
  273. addAttribute(attributes, InternalElementMapping.URI, InternalElementMapping.SCOPE,
  274. InternalElementMapping.STANDARD_PREFIX, scope);
  275. }
  276. }
  277. }
  278. startElement(tc, attributes);
  279. }
  280. private void addSpanAttribute(AttributesImpl attributes, String attributeName, int span) {
  281. if (span > 1) {
  282. addNoNamespaceAttribute(attributes, attributeName, Integer.toString(span));
  283. }
  284. }
  285. @Override
  286. public void endCell(TableCell tc) {
  287. endElement(tc);
  288. }
  289. @Override
  290. public void startList(ListBlock lb) {
  291. startElement(lb);
  292. }
  293. @Override
  294. public void endList(ListBlock lb) {
  295. endElement(lb);
  296. }
  297. @Override
  298. public void startListItem(ListItem li) {
  299. startElement(li);
  300. }
  301. @Override
  302. public void endListItem(ListItem li) {
  303. endElement(li);
  304. }
  305. @Override
  306. public void startListLabel(ListItemLabel listItemLabel) {
  307. startElement(listItemLabel);
  308. }
  309. @Override
  310. public void endListLabel(ListItemLabel listItemLabel) {
  311. endElement(listItemLabel);
  312. }
  313. @Override
  314. public void startListBody(ListItemBody listItemBody) {
  315. startElement(listItemBody);
  316. }
  317. @Override
  318. public void endListBody(ListItemBody listItemBody) {
  319. endElement(listItemBody);
  320. }
  321. @Override
  322. public void startLink(BasicLink basicLink) {
  323. startElementWithIDAndAltText(basicLink, basicLink.getAltText());
  324. }
  325. @Override
  326. public void endLink(BasicLink basicLink) {
  327. endElement(basicLink);
  328. }
  329. @Override
  330. public void image(ExternalGraphic eg) {
  331. startElementWithIDAndAltText(eg, eg.getAltText());
  332. endElement(eg);
  333. }
  334. @Override
  335. public void startInstreamForeignObject(InstreamForeignObject ifo) {
  336. startElementWithIDAndAltText(ifo, ifo.getAltText());
  337. }
  338. @Override
  339. public void endInstreamForeignObject(InstreamForeignObject ifo) {
  340. endElement(ifo);
  341. }
  342. @Override
  343. public void startFootnote(Footnote footnote) {
  344. startElement(footnote);
  345. }
  346. @Override
  347. public void endFootnote(Footnote footnote) {
  348. endElement(footnote);
  349. }
  350. @Override
  351. public void startFootnoteBody(FootnoteBody body) {
  352. startElement(body);
  353. }
  354. @Override
  355. public void endFootnoteBody(FootnoteBody body) {
  356. endElement(body);
  357. }
  358. @Override
  359. public void startWrapper(Wrapper wrapper) {
  360. startElement(wrapper);
  361. }
  362. @Override
  363. public void endWrapper(Wrapper wrapper) {
  364. endElement(wrapper);
  365. }
  366. @Override
  367. public void startRetrieveMarker(RetrieveMarker retrieveMarker) {
  368. startElementWithID(retrieveMarker);
  369. saveState(retrieveMarker);
  370. }
  371. void saveState(AbstractRetrieveMarker retrieveMarker) {
  372. states.put(retrieveMarker, new State(this));
  373. }
  374. @Override
  375. public void endRetrieveMarker(RetrieveMarker retrieveMarker) {
  376. endElement(retrieveMarker);
  377. }
  378. @Override
  379. public void restoreState(RetrieveMarker retrieveMarker) {
  380. restoreRetrieveMarkerState(retrieveMarker);
  381. }
  382. @SuppressWarnings("unchecked")
  383. private void restoreRetrieveMarkerState(AbstractRetrieveMarker retrieveMarker) {
  384. State state = states.get(retrieveMarker);
  385. tables = (Stack<Table>) state.tables.clone();
  386. inTableHeader = (Stack<Boolean>) state.inTableHeader.clone();
  387. locales = (Stack<Locale>) state.locales.clone();
  388. }
  389. @Override
  390. public void startRetrieveTableMarker(RetrieveTableMarker retrieveTableMarker) {
  391. startElementWithID(retrieveTableMarker);
  392. saveState(retrieveTableMarker);
  393. }
  394. @Override
  395. public void endRetrieveTableMarker(RetrieveTableMarker retrieveTableMarker) {
  396. endElement(retrieveTableMarker);
  397. }
  398. @Override
  399. public void restoreState(RetrieveTableMarker retrieveTableMarker) {
  400. restoreRetrieveMarkerState(retrieveTableMarker);
  401. }
  402. @Override
  403. public void character(Character c) {
  404. AttributesImpl attributes = createLangAttribute(c.getCommonHyphenation());
  405. startElementWithID(c, attributes);
  406. endElement(c);
  407. locales.pop();
  408. }
  409. @Override
  410. public void characters(FOText foText) {
  411. startElementWithID(foText);
  412. endElement(foText);
  413. }
  414. private StructureTreeElement startElement(FONode node) {
  415. AttributesImpl attributes = new AttributesImpl();
  416. if (node instanceof Inline) {
  417. Inline in = (Inline)node;
  418. if (!in.getAbbreviation().equals("")) {
  419. addAttribute(attributes, ExtensionElementMapping.URI, "abbreviation",
  420. ExtensionElementMapping.STANDARD_PREFIX, in.getAbbreviation());
  421. }
  422. }
  423. return startElement(node, attributes);
  424. }
  425. private void startElementWithID(FONode node) {
  426. startElementWithID(node, new AttributesImpl());
  427. }
  428. private void startElementWithID(FONode node, AttributesImpl attributes) {
  429. String localName = node.getLocalName();
  430. if (node instanceof CommonAccessibilityHolder) {
  431. addRole((CommonAccessibilityHolder) node, attributes);
  432. }
  433. node.setStructureTreeElement(
  434. structureTreeEventHandler.startReferencedNode(localName, attributes,
  435. node.getParent().getStructureTreeElement()));
  436. }
  437. private void startElementWithIDAndAltText(FObj node, String altText) {
  438. AttributesImpl attributes = new AttributesImpl();
  439. String localName = node.getLocalName();
  440. addRole((CommonAccessibilityHolder)node, attributes);
  441. addAttribute(attributes, ExtensionElementMapping.URI, "alt-text",
  442. ExtensionElementMapping.STANDARD_PREFIX, altText);
  443. node.setStructureTreeElement(
  444. structureTreeEventHandler.startImageNode(localName, attributes,
  445. node.getParent().getStructureTreeElement()));
  446. }
  447. private StructureTreeElement startElement(FONode node, AttributesImpl attributes) {
  448. String localName = node.getLocalName();
  449. if (node instanceof CommonAccessibilityHolder) {
  450. addRole((CommonAccessibilityHolder) node, attributes);
  451. }
  452. return structureTreeEventHandler.startNode(localName, attributes,
  453. node.getParent().getStructureTreeElement());
  454. }
  455. private void addNoNamespaceAttribute(AttributesImpl attributes, String name, String value) {
  456. attributes.addAttribute("", name, name, XMLUtil.CDATA, value);
  457. }
  458. private void addAttribute(AttributesImpl attributes,
  459. String namespace, String localName, String prefix, String value) {
  460. assert namespace.length() > 0 && prefix.length() > 0;
  461. String qualifiedName = prefix + ":" + localName;
  462. attributes.addAttribute(namespace, localName, qualifiedName, XMLUtil.CDATA, value);
  463. }
  464. private void addRole(CommonAccessibilityHolder node, AttributesImpl attributes) {
  465. String role = node.getCommonAccessibility().getRole();
  466. if (role != null) {
  467. addNoNamespaceAttribute(attributes, "role", role);
  468. }
  469. }
  470. private void endElement(FONode node) {
  471. String localName = node.getLocalName();
  472. structureTreeEventHandler.endNode(localName);
  473. }
  474. }