Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

AbstractRenderer.java 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. /*
  2. * $Id$
  3. * Copyright (C) 2001 The Apache Software Foundation. All rights reserved.
  4. * For details on use and redistribution please refer to the
  5. * LICENSE file included with these sources.
  6. */
  7. package org.apache.fop.render;
  8. // FOP
  9. import org.apache.fop.apps.FOPException;
  10. import org.apache.fop.area.*;
  11. import org.apache.fop.area.Span;
  12. import org.apache.fop.area.inline.*;
  13. import org.apache.fop.area.inline.Character;
  14. import org.apache.fop.area.inline.Space;
  15. import org.apache.fop.fo.FOUserAgent;
  16. // Avalon
  17. import org.apache.avalon.framework.logger.Logger;
  18. // Java
  19. import java.awt.geom.Rectangle2D;
  20. import java.io.IOException;
  21. import java.io.OutputStream;
  22. import java.util.HashMap;
  23. import java.util.List;
  24. /**
  25. * Abstract base class for all renderers.
  26. * The Abstract renderer does all the top level processing
  27. * of the area tree and adds some abstract methods to handle
  28. * viewports. This keeps track of the current block and inline
  29. * position.
  30. */
  31. public abstract class AbstractRenderer implements Renderer {
  32. protected Logger log;
  33. protected FOUserAgent userAgent;
  34. protected HashMap options;
  35. // block progression position
  36. protected int currentBPPosition = 0;
  37. // inline progression position
  38. protected int currentIPPosition = 0;
  39. protected int currentBlockIPPosition = 0;
  40. public void setLogger(Logger logger) {
  41. log = logger;
  42. }
  43. public void setUserAgent(FOUserAgent agent) {
  44. userAgent = agent;
  45. }
  46. public void setOptions(HashMap opt) {
  47. options = opt;
  48. }
  49. /**
  50. * Check if this renderer supports out of order rendering.
  51. * If this renderer supports out of order rendering then it
  52. * means that the pages that are not ready will be prepared
  53. * and a future page will be rendered.
  54. */
  55. public boolean supportsOutOfOrder() {
  56. return false;
  57. }
  58. /**
  59. * Prepare a page for rendering.
  60. * This is called if the renderer supports out of order rendering.
  61. * The renderer should prepare the page so that a page further on
  62. * in the set of pages can be rendered. The body of the page should
  63. * not be rendered. The page will be rendered at a later time
  64. * by the call to render page.
  65. */
  66. public void preparePage(PageViewport page) {
  67. }
  68. /**
  69. * Utility method to convert a page sequence title to a string.
  70. * Some renderers may only be able to use a string title.
  71. * A title is a sequence of inline areas that this method
  72. * attempts to convert to an equivalent string.
  73. */
  74. public String convertTitleToString(Title title) {
  75. String str = "";
  76. List children = title.getInlineAreas();
  77. for (int count = 0; count < children.size(); count++) {
  78. InlineArea inline = (InlineArea) children.get(count);
  79. if (inline instanceof Character) {
  80. str += ((Character) inline).getChar();
  81. } else if (inline instanceof Word) {
  82. str += ((Word) inline).getWord();
  83. } else {
  84. str += " ";
  85. }
  86. }
  87. return str.trim();
  88. }
  89. public void startPageSequence(Title seqTitle) {
  90. }
  91. // normally this would be overriden to create a page in the
  92. // output
  93. public void renderPage(PageViewport page) throws IOException,
  94. FOPException {
  95. Page p = page.getPage();
  96. renderPageAreas(p);
  97. }
  98. protected void renderPageAreas(Page page) {
  99. RegionViewport viewport;
  100. viewport = page.getRegion(RegionReference.BEFORE);
  101. renderRegionViewport(viewport);
  102. viewport = page.getRegion(RegionReference.START);
  103. renderRegionViewport(viewport);
  104. viewport = page.getRegion(RegionReference.BODY);
  105. renderRegionViewport(viewport);
  106. viewport = page.getRegion(RegionReference.END);
  107. renderRegionViewport(viewport);
  108. viewport = page.getRegion(RegionReference.AFTER);
  109. renderRegionViewport(viewport);
  110. }
  111. // the region may clip the area and it establishes
  112. // a position from where the region is placed
  113. protected void renderRegionViewport(RegionViewport port) {
  114. if (port != null) {
  115. Rectangle2D view = port.getViewArea();
  116. // The CTM will transform coordinates relative to
  117. // this region-reference area into page coords, so
  118. // set origin for the region to 0,0.
  119. currentBPPosition = 0; // (int) (view.getY() / 1000);
  120. currentIPPosition = 0; // (int) (view.getX() / 1000);
  121. currentBlockIPPosition = currentIPPosition;
  122. RegionReference region = port.getRegion();
  123. startVParea(region.getCTM());
  124. if (region.getRegionClass() == Region.BODY) {
  125. renderBodyRegion((BodyRegion) region);
  126. } else {
  127. renderRegion(region);
  128. }
  129. endVParea();
  130. }
  131. }
  132. protected void startVParea(CTM ctm) { }
  133. protected void endVParea() { }
  134. protected void renderRegion(RegionReference region) {
  135. List blocks = region.getBlocks();
  136. renderBlocks(blocks);
  137. }
  138. protected void renderBodyRegion(BodyRegion region) {
  139. BeforeFloat bf = region.getBeforeFloat();
  140. if (bf != null) {
  141. renderBeforeFloat(bf);
  142. }
  143. MainReference mr = region.getMainReference();
  144. if (mr != null) {
  145. renderMainReference(mr);
  146. }
  147. Footnote foot = region.getFootnote();
  148. if (foot != null) {
  149. renderFootnote(foot);
  150. }
  151. }
  152. protected void renderBeforeFloat(BeforeFloat bf) {
  153. List blocks = bf.getBlocks();
  154. if (blocks != null) {
  155. renderBlocks(blocks);
  156. Block sep = bf.getSeparator();
  157. if (sep != null) {
  158. renderBlock(sep);
  159. }
  160. }
  161. }
  162. protected void renderFootnote(Footnote footnote) {
  163. List blocks = footnote.getBlocks();
  164. if (blocks != null) {
  165. Block sep = footnote.getSeparator();
  166. if (sep != null) {
  167. renderBlock(sep);
  168. }
  169. renderBlocks(blocks);
  170. }
  171. }
  172. // the main reference area contains a list of spans that are
  173. // stacked on the page
  174. // the spans contain a list of normal flow reference areas
  175. // that are positioned into columns.
  176. protected void renderMainReference(MainReference mr) {
  177. int saveIPPos = currentIPPosition;
  178. Span span = null;
  179. List spans = mr.getSpans();
  180. for (int count = 0; count < spans.size(); count++) {
  181. span = (Span) spans.get(count);
  182. int offset = (mr.getWidth() -
  183. (span.getColumnCount() - 1) * mr.getColumnGap()) /
  184. span.getColumnCount() + mr.getColumnGap();
  185. for (int c = 0; c < span.getColumnCount(); c++) {
  186. Flow flow = (Flow) span.getFlow(c);
  187. renderFlow(flow);
  188. currentIPPosition += offset;
  189. }
  190. currentIPPosition = saveIPPos;
  191. currentBPPosition += span.getHeight();
  192. }
  193. }
  194. // the normal flow reference area contains stacked blocks
  195. protected void renderFlow(Flow flow) {
  196. List blocks = flow.getBlocks();
  197. renderBlocks(blocks);
  198. }
  199. protected void renderBlock(Block block) {
  200. List children = block.getChildAreas();
  201. if (children == null) {
  202. // simply move position
  203. } else {
  204. renderBlocks(children);
  205. }
  206. }
  207. // a line area may have grouped styling for its children
  208. // such as underline, background
  209. protected void renderLineArea(LineArea line) {
  210. List children = line.getInlineAreas();
  211. for (int count = 0; count < children.size(); count++) {
  212. InlineArea inline = (InlineArea) children.get(count);
  213. inline.render(this);
  214. }
  215. }
  216. public void renderViewport(Viewport viewport) {
  217. Area content = viewport.getContent();
  218. int saveBP = currentBPPosition;
  219. currentBPPosition += viewport.getOffset();
  220. if (content instanceof Image) {
  221. renderImage((Image) content);
  222. } else if (content instanceof Container) {
  223. renderContainer((Container) content);
  224. } else if (content instanceof ForeignObject) {
  225. renderForeignObject((ForeignObject) content);
  226. }
  227. currentBlockIPPosition += viewport.getWidth();
  228. currentBPPosition = saveBP;
  229. }
  230. public void renderImage(Image image) {
  231. }
  232. public void renderContainer(Container cont) {
  233. int saveIP = currentIPPosition;
  234. currentIPPosition = currentBlockIPPosition;
  235. int saveBlockIP = currentBlockIPPosition;
  236. int saveBP = currentBPPosition;
  237. List blocks = cont.getBlocks();
  238. renderBlocks(blocks);
  239. currentIPPosition = saveIP;
  240. currentBlockIPPosition = saveBlockIP;
  241. currentBPPosition = saveBP;
  242. }
  243. public void renderForeignObject(ForeignObject fo) {
  244. }
  245. public void renderCharacter(Character ch) {
  246. currentBlockIPPosition += ch.getWidth();
  247. }
  248. // an inline space moves the inline progression position
  249. // for the current block by the width or height of the space
  250. // it may also have styling (only on this object) that needs
  251. // handling
  252. public void renderInlineSpace(Space space) {
  253. currentBlockIPPosition += space.getWidth();
  254. }
  255. public void renderLeader(Leader area) {
  256. currentBlockIPPosition += area.getWidth();
  257. }
  258. public void renderWord(Word word) {
  259. currentBlockIPPosition += word.getWidth();
  260. }
  261. protected void renderBlocks(List blocks) {
  262. for (int count = 0; count < blocks.size(); count++) {
  263. Object obj = blocks.get(count);
  264. if(obj instanceof Block) {
  265. renderBlock((Block)obj);
  266. } else {
  267. // a line area is rendered from the top left position
  268. // of the line, each inline object is offset from there
  269. LineArea line = (LineArea) obj;
  270. currentBlockIPPosition = currentIPPosition;
  271. renderLineArea(line);
  272. currentBPPosition += line.getHeight();
  273. }
  274. }
  275. }
  276. }