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.

LayoutManagerMapping.java 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  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.layoutmgr;
  19. import java.util.ArrayList;
  20. import java.util.HashMap;
  21. import java.util.Iterator;
  22. import java.util.List;
  23. import java.util.Map;
  24. import org.apache.commons.logging.Log;
  25. import org.apache.commons.logging.LogFactory;
  26. import org.apache.fop.apps.FOUserAgent;
  27. import org.apache.fop.area.AreaTreeHandler;
  28. import org.apache.fop.fo.FOElementMapping;
  29. import org.apache.fop.fo.FONode;
  30. import org.apache.fop.fo.FONode.FONodeIterator;
  31. import org.apache.fop.fo.FOText;
  32. import org.apache.fop.fo.FObjMixed;
  33. import org.apache.fop.fo.extensions.ExternalDocument;
  34. import org.apache.fop.fo.flow.BasicLink;
  35. import org.apache.fop.fo.flow.BidiOverride;
  36. import org.apache.fop.fo.flow.Block;
  37. import org.apache.fop.fo.flow.BlockContainer;
  38. import org.apache.fop.fo.flow.ChangeBarBegin;
  39. import org.apache.fop.fo.flow.ChangeBarEnd;
  40. import org.apache.fop.fo.flow.Character;
  41. import org.apache.fop.fo.flow.ExternalGraphic;
  42. import org.apache.fop.fo.flow.Float;
  43. import org.apache.fop.fo.flow.Footnote;
  44. import org.apache.fop.fo.flow.Inline;
  45. import org.apache.fop.fo.flow.InlineContainer;
  46. import org.apache.fop.fo.flow.InlineLevel;
  47. import org.apache.fop.fo.flow.InstreamForeignObject;
  48. import org.apache.fop.fo.flow.Leader;
  49. import org.apache.fop.fo.flow.ListBlock;
  50. import org.apache.fop.fo.flow.ListItem;
  51. import org.apache.fop.fo.flow.MultiCase;
  52. import org.apache.fop.fo.flow.MultiSwitch;
  53. import org.apache.fop.fo.flow.PageNumber;
  54. import org.apache.fop.fo.flow.PageNumberCitation;
  55. import org.apache.fop.fo.flow.PageNumberCitationLast;
  56. import org.apache.fop.fo.flow.RetrieveMarker;
  57. import org.apache.fop.fo.flow.RetrieveTableMarker;
  58. import org.apache.fop.fo.flow.Wrapper;
  59. import org.apache.fop.fo.flow.table.Table;
  60. import org.apache.fop.fo.flow.table.TableBody;
  61. import org.apache.fop.fo.flow.table.TableCell;
  62. import org.apache.fop.fo.flow.table.TableColumn;
  63. import org.apache.fop.fo.flow.table.TableFooter;
  64. import org.apache.fop.fo.flow.table.TableHeader;
  65. import org.apache.fop.fo.flow.table.TableRow;
  66. import org.apache.fop.fo.pagination.Flow;
  67. import org.apache.fop.fo.pagination.PageSequence;
  68. import org.apache.fop.fo.pagination.SideRegion;
  69. import org.apache.fop.fo.pagination.StaticContent;
  70. import org.apache.fop.fo.pagination.Title;
  71. import org.apache.fop.layoutmgr.inline.BasicLinkLayoutManager;
  72. import org.apache.fop.layoutmgr.inline.BidiLayoutManager;
  73. import org.apache.fop.layoutmgr.inline.CharacterLayoutManager;
  74. import org.apache.fop.layoutmgr.inline.ContentLayoutManager;
  75. import org.apache.fop.layoutmgr.inline.ExternalGraphicLayoutManager;
  76. import org.apache.fop.layoutmgr.inline.FloatLayoutManager;
  77. import org.apache.fop.layoutmgr.inline.FootnoteLayoutManager;
  78. import org.apache.fop.layoutmgr.inline.InlineContainerLayoutManager;
  79. import org.apache.fop.layoutmgr.inline.InlineLayoutManager;
  80. import org.apache.fop.layoutmgr.inline.InstreamForeignObjectLM;
  81. import org.apache.fop.layoutmgr.inline.LeaderLayoutManager;
  82. import org.apache.fop.layoutmgr.inline.PageNumberCitationLastLayoutManager;
  83. import org.apache.fop.layoutmgr.inline.PageNumberCitationLayoutManager;
  84. import org.apache.fop.layoutmgr.inline.PageNumberLayoutManager;
  85. import org.apache.fop.layoutmgr.inline.TextLayoutManager;
  86. import org.apache.fop.layoutmgr.inline.WrapperLayoutManager;
  87. import org.apache.fop.layoutmgr.list.ListBlockLayoutManager;
  88. import org.apache.fop.layoutmgr.list.ListItemLayoutManager;
  89. import org.apache.fop.layoutmgr.table.TableLayoutManager;
  90. import org.apache.fop.util.CharUtilities;
  91. /**
  92. * The default LayoutManager maker class
  93. */
  94. public class LayoutManagerMapping implements LayoutManagerMaker {
  95. /** logging instance */
  96. private static final Log LOG = LogFactory.getLog(LayoutManagerMapping.class);
  97. /** The map of LayoutManagerMakers */
  98. private final Map makers = new HashMap();
  99. private FOUserAgent userAgent;
  100. /** default constructor */
  101. public LayoutManagerMapping(FOUserAgent userAgent) {
  102. this.userAgent = userAgent;
  103. initialize();
  104. }
  105. /**
  106. * Initializes the set of maker objects associated with this LayoutManagerMapping
  107. */
  108. protected void initialize() {
  109. registerMaker(FOText.class, new FOTextLayoutManagerMaker());
  110. registerMaker(FObjMixed.class, new Maker());
  111. registerMaker(BidiOverride.class, new BidiOverrideLayoutManagerMaker());
  112. registerMaker(Inline.class, new InlineLayoutManagerMaker());
  113. registerMaker(Footnote.class, new FootnoteLayoutManagerMaker());
  114. registerMaker(InlineContainer.class,
  115. new InlineContainerLayoutManagerMaker());
  116. registerMaker(BasicLink.class, new BasicLinkLayoutManagerMaker());
  117. registerMaker(Block.class, new BlockLayoutManagerMaker());
  118. registerMaker(Leader.class, new LeaderLayoutManagerMaker());
  119. registerMaker(RetrieveMarker.class, new RetrieveMarkerLayoutManagerMaker());
  120. registerMaker(RetrieveTableMarker.class, new RetrieveTableMarkerLayoutManagerMaker());
  121. registerMaker(Character.class, new CharacterLayoutManagerMaker());
  122. registerMaker(ExternalGraphic.class,
  123. new ExternalGraphicLayoutManagerMaker());
  124. registerMaker(BlockContainer.class,
  125. new BlockContainerLayoutManagerMaker());
  126. registerMaker(ListItem.class, new ListItemLayoutManagerMaker());
  127. registerMaker(ListBlock.class, new ListBlockLayoutManagerMaker());
  128. registerMaker(InstreamForeignObject.class,
  129. new InstreamForeignObjectLayoutManagerMaker());
  130. registerMaker(PageNumber.class, new PageNumberLayoutManagerMaker());
  131. registerMaker(PageNumberCitation.class,
  132. new PageNumberCitationLayoutManagerMaker());
  133. registerMaker(PageNumberCitationLast.class,
  134. new PageNumberCitationLastLayoutManagerMaker());
  135. registerMaker(Table.class, new TableLayoutManagerMaker());
  136. registerMaker(TableBody.class, new Maker());
  137. registerMaker(TableColumn.class, new Maker());
  138. registerMaker(TableRow.class, new Maker());
  139. registerMaker(TableCell.class, new Maker());
  140. registerMaker(TableFooter.class, new Maker());
  141. registerMaker(TableHeader.class, new Maker());
  142. registerMaker(Wrapper.class, new WrapperLayoutManagerMaker());
  143. registerMaker(Title.class, new InlineLayoutManagerMaker());
  144. registerMaker(ChangeBarBegin.class, new Maker());
  145. registerMaker(ChangeBarEnd.class, new Maker());
  146. registerMaker(MultiCase.class, new MultiCaseLayoutManagerMaker());
  147. registerMaker(MultiSwitch.class, new MultiSwitchLayoutManagerMaker());
  148. registerMaker(Float.class, new FloatLayoutManagerMaker());
  149. }
  150. /**
  151. * Registers a Maker class for a specific formatting object.
  152. * @param clazz the formatting object class
  153. * @param maker the maker for the layout manager
  154. */
  155. protected void registerMaker(Class clazz, Maker maker) {
  156. makers.put(clazz, maker);
  157. }
  158. /** {@inheritDoc} */
  159. public void makeLayoutManagers(FONode node, List lms) {
  160. Maker maker = (Maker) makers.get(node.getClass());
  161. if (maker == null) {
  162. if (FOElementMapping.URI.equals(node.getNamespaceURI())) {
  163. LOG.error("No LayoutManager maker for class " + node.getClass());
  164. } else {
  165. if (LOG.isDebugEnabled()) {
  166. LOG.debug("Skipping the creation of a layout manager for " + node.getClass());
  167. }
  168. }
  169. } else {
  170. maker.make(node, lms, userAgent);
  171. }
  172. }
  173. /** {@inheritDoc} */
  174. public LayoutManager makeLayoutManager(FONode node) {
  175. List lms = new ArrayList();
  176. makeLayoutManagers(node, lms);
  177. if (lms.size() == 0) {
  178. throw new IllegalStateException("LayoutManager for class "
  179. + node.getClass()
  180. + " is missing.");
  181. } else if (lms.size() > 1) {
  182. throw new IllegalStateException("Duplicate LayoutManagers for class "
  183. + node.getClass()
  184. + " found, only one may be declared.");
  185. }
  186. return (LayoutManager) lms.get(0);
  187. }
  188. /** {@inheritDoc} */
  189. public PageSequenceLayoutManager makePageSequenceLayoutManager(
  190. AreaTreeHandler ath, PageSequence ps) {
  191. return new PageSequenceLayoutManager(ath, ps);
  192. }
  193. /** {@inheritDoc} */
  194. public ExternalDocumentLayoutManager makeExternalDocumentLayoutManager(
  195. AreaTreeHandler ath, ExternalDocument ed) {
  196. return new ExternalDocumentLayoutManager(ath, ed);
  197. }
  198. /** {@inheritDoc} */
  199. public FlowLayoutManager makeFlowLayoutManager(
  200. PageSequenceLayoutManager pslm, Flow flow) {
  201. return new FlowLayoutManager(pslm, flow);
  202. }
  203. /** {@inheritDoc} */
  204. public ContentLayoutManager makeContentLayoutManager(PageSequenceLayoutManager pslm,
  205. Title title) {
  206. return new ContentLayoutManager(pslm, title);
  207. }
  208. /** {@inheritDoc} */
  209. public StaticContentLayoutManager makeStaticContentLayoutManager(
  210. PageSequenceLayoutManager pslm, StaticContent sc, SideRegion reg) {
  211. return new StaticContentLayoutManager(pslm, sc, reg);
  212. }
  213. /** {@inheritDoc} */
  214. public StaticContentLayoutManager makeStaticContentLayoutManager(
  215. PageSequenceLayoutManager pslm, StaticContent sc, org.apache.fop.area.Block block) {
  216. return new StaticContentLayoutManager(pslm, sc, block);
  217. }
  218. /** a layout manager maker base class */
  219. public static class Maker {
  220. /**
  221. * Create a layout manager.
  222. * @param node the associated FO node
  223. * @param lms a list of layout managers to which new manager is to be added
  224. */
  225. public void make(FONode node, List lms, FOUserAgent userAgent) {
  226. // no layout manager
  227. }
  228. }
  229. /** a layout manager maker */
  230. public static class FOTextLayoutManagerMaker extends Maker {
  231. /** {@inheritDoc} */
  232. public void make(FONode node, List lms, FOUserAgent userAgent) {
  233. FOText foText = (FOText) node;
  234. if (foText.length() > 0) {
  235. lms.add(new TextLayoutManager(foText, userAgent));
  236. }
  237. }
  238. }
  239. /** a layout manager maker */
  240. public static class BidiOverrideLayoutManagerMaker extends Maker {
  241. /** {@inheritDoc} */
  242. public void make(FONode node, List lms, FOUserAgent userAgent) {
  243. if (node instanceof BidiOverride) {
  244. lms.add(new BidiLayoutManager((BidiOverride) node));
  245. }
  246. }
  247. }
  248. /** a layout manager maker */
  249. public static class InlineLayoutManagerMaker extends Maker {
  250. /** {@inheritDoc} */
  251. public void make(FONode node, List lms, FOUserAgent userAgent) {
  252. lms.add(new InlineLayoutManager((InlineLevel) node));
  253. }
  254. }
  255. /** a layout manager maker */
  256. public static class FootnoteLayoutManagerMaker extends Maker {
  257. /** {@inheritDoc} */
  258. public void make(FONode node, List lms, FOUserAgent userAgent) {
  259. lms.add(new FootnoteLayoutManager((Footnote) node));
  260. }
  261. }
  262. /** a layout manager maker */
  263. public static class InlineContainerLayoutManagerMaker extends Maker {
  264. /** {@inheritDoc} */
  265. public void make(FONode node, List lms, FOUserAgent userAgent) {
  266. lms.add(new InlineContainerLayoutManager((InlineContainer) node));
  267. }
  268. }
  269. /** a layout manager maker */
  270. public static class BasicLinkLayoutManagerMaker extends Maker {
  271. /** {@inheritDoc} */
  272. public void make(FONode node, List lms, FOUserAgent userAgent) {
  273. lms.add(new BasicLinkLayoutManager((BasicLink) node));
  274. }
  275. }
  276. /** a layout manager maker */
  277. public static class BlockLayoutManagerMaker extends Maker {
  278. /** {@inheritDoc} */
  279. public void make(FONode node, List lms, FOUserAgent userAgent) {
  280. lms.add(new BlockLayoutManager((Block) node));
  281. }
  282. }
  283. /** a layout manager maker */
  284. public static class LeaderLayoutManagerMaker extends Maker {
  285. /** {@inheritDoc} */
  286. public void make(FONode node, List lms, FOUserAgent userAgent) {
  287. lms.add(new LeaderLayoutManager((Leader) node));
  288. }
  289. }
  290. /** a layout manager maker */
  291. public static class CharacterLayoutManagerMaker extends Maker {
  292. /** {@inheritDoc} */
  293. public void make(FONode node, List lms, FOUserAgent userAgent) {
  294. Character foCharacter = (Character) node;
  295. if (foCharacter.getCharacter() != CharUtilities.CODE_EOT) {
  296. lms.add(new CharacterLayoutManager(foCharacter));
  297. }
  298. }
  299. }
  300. /** a layout manager maker */
  301. public static class ExternalGraphicLayoutManagerMaker extends Maker {
  302. /** {@inheritDoc} */
  303. public void make(FONode node, List lms, FOUserAgent userAgent) {
  304. ExternalGraphic eg = (ExternalGraphic) node;
  305. if (!eg.getSrc().equals("")) {
  306. lms.add(new ExternalGraphicLayoutManager(eg));
  307. }
  308. }
  309. }
  310. /** a layout manager maker */
  311. public static class BlockContainerLayoutManagerMaker extends Maker {
  312. /** {@inheritDoc} */
  313. public void make(FONode node, List lms, FOUserAgent userAgent) {
  314. lms.add(new BlockContainerLayoutManager((BlockContainer) node));
  315. }
  316. }
  317. /** a layout manager maker */
  318. public static class ListItemLayoutManagerMaker extends Maker {
  319. /** {@inheritDoc} */
  320. public void make(FONode node, List lms, FOUserAgent userAgent) {
  321. lms.add(new ListItemLayoutManager((ListItem) node));
  322. }
  323. }
  324. /** a layout manager maker */
  325. public static class ListBlockLayoutManagerMaker extends Maker {
  326. /** {@inheritDoc} */
  327. public void make(FONode node, List lms, FOUserAgent userAgent) {
  328. lms.add(new ListBlockLayoutManager((ListBlock) node));
  329. }
  330. }
  331. /** a layout manager maker */
  332. public static class InstreamForeignObjectLayoutManagerMaker extends Maker {
  333. /** {@inheritDoc} */
  334. public void make(FONode node, List lms, FOUserAgent userAgent) {
  335. lms.add(new InstreamForeignObjectLM((InstreamForeignObject) node));
  336. }
  337. }
  338. /** a layout manager maker */
  339. public static class PageNumberLayoutManagerMaker extends Maker {
  340. /** {@inheritDoc} */
  341. public void make(FONode node, List lms, FOUserAgent userAgent) {
  342. lms.add(new PageNumberLayoutManager((PageNumber) node));
  343. }
  344. }
  345. /** a layout manager maker */
  346. public static class PageNumberCitationLayoutManagerMaker extends Maker {
  347. /** {@inheritDoc} */
  348. public void make(FONode node, List lms, FOUserAgent userAgent) {
  349. lms.add(new PageNumberCitationLayoutManager((PageNumberCitation) node));
  350. }
  351. }
  352. /** a layout manager maker */
  353. public static class PageNumberCitationLastLayoutManagerMaker extends Maker {
  354. /** {@inheritDoc} */
  355. public void make(FONode node, List lms, FOUserAgent userAgent) {
  356. lms.add(new PageNumberCitationLastLayoutManager((PageNumberCitationLast) node));
  357. }
  358. }
  359. /** a layout manager maker */
  360. public static class TableLayoutManagerMaker extends Maker {
  361. /** {@inheritDoc} */
  362. public void make(FONode node, List lms, FOUserAgent userAgent) {
  363. Table table = (Table) node;
  364. TableLayoutManager tlm = new TableLayoutManager(table);
  365. lms.add(tlm);
  366. }
  367. }
  368. /** a layout manager maker */
  369. public class RetrieveMarkerLayoutManagerMaker extends Maker {
  370. /** {@inheritDoc} */
  371. public void make(FONode node, List lms, FOUserAgent userAgent) {
  372. Iterator baseIter;
  373. baseIter = node.getChildNodes();
  374. if (baseIter == null) {
  375. return;
  376. }
  377. while (baseIter.hasNext()) {
  378. FONode child = (FONode) baseIter.next();
  379. makeLayoutManagers(child, lms);
  380. }
  381. }
  382. }
  383. public class RetrieveTableMarkerLayoutManagerMaker extends Maker {
  384. public void make(FONode node, List lms, FOUserAgent userAgent) {
  385. FONodeIterator baseIter = node.getChildNodes();
  386. if (baseIter == null) {
  387. // this happens when the retrieve-table-marker cannot be resolved yet
  388. RetrieveTableMarker rtm = (RetrieveTableMarker) node;
  389. RetrieveTableMarkerLayoutManager rtmlm = new RetrieveTableMarkerLayoutManager(rtm);
  390. lms.add(rtmlm);
  391. return;
  392. }
  393. while (baseIter.hasNext()) {
  394. // this happens when the retrieve-table-marker has been resolved
  395. FONode child = baseIter.next();
  396. makeLayoutManagers(child, lms);
  397. }
  398. }
  399. }
  400. /** a layout manager maker */
  401. public class WrapperLayoutManagerMaker extends Maker {
  402. /** {@inheritDoc} */
  403. public void make(FONode node, List lms, FOUserAgent userAgent) {
  404. //We insert the wrapper LM before it's children so an ID
  405. //on the node can be registered on a page.
  406. lms.add(new WrapperLayoutManager((Wrapper)node));
  407. Iterator baseIter;
  408. baseIter = node.getChildNodes();
  409. if (baseIter == null) {
  410. return;
  411. }
  412. while (baseIter.hasNext()) {
  413. FONode child = (FONode) baseIter.next();
  414. makeLayoutManagers(child, lms);
  415. }
  416. }
  417. }
  418. public class MultiSwitchLayoutManagerMaker extends Maker {
  419. @Override
  420. public void make(FONode node, List lms, FOUserAgent userAgent) {
  421. lms.add(new MultiSwitchLayoutManager((MultiSwitch) node));
  422. }
  423. }
  424. public class MultiCaseLayoutManagerMaker extends Maker {
  425. @Override
  426. public void make(FONode node, List lms, FOUserAgent userAgent) {
  427. lms.add(new MultiCaseLayoutManager((MultiCase) node));
  428. }
  429. }
  430. public static class FloatLayoutManagerMaker extends Maker {
  431. public void make(FONode node, List lms, FOUserAgent userAgent) {
  432. lms.add(new FloatLayoutManager((Float) node));
  433. }
  434. }
  435. }