/* * Copyright (C) 2003-2010 eXo Platform SAS. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Affero General Public License * as published by the Free Software Foundation; either version 3 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, see<http://www.gnu.org/licenses/>. */ package org.exoplatform.wiki.rendering.macro.table; import java.io.StringReader; import java.util.Collections; import java.util.Iterator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import org.xwiki.component.annotation.Component; import org.xwiki.component.annotation.Requirement; import org.xwiki.component.manager.ComponentLookupException; import org.xwiki.component.manager.ComponentManager; import org.xwiki.rendering.block.Block; import org.xwiki.rendering.block.MacroBlock; import org.xwiki.rendering.block.TableCellBlock; import org.xwiki.rendering.block.TableHeadCellBlock; import org.xwiki.rendering.block.XDOM; import org.xwiki.rendering.macro.AbstractMacro; import org.xwiki.rendering.macro.MacroExecutionException; import org.xwiki.rendering.parser.ParseException; import org.xwiki.rendering.parser.Parser; import org.xwiki.rendering.transformation.MacroTransformationContext; /** * Created by The eXo Platform SAS * Author : viet nguyen * viet.nguyen@exoplatform.com * Sep 22, 2010 */ @Component("table-row") public class TableRowMacro extends AbstractMacro<Object> { /** * The description of the macro. */ private static final String DESCRIPTION = "Inserts a table row."; @Requirement private ComponentManager componentManager; /** * Create and initialize the descriptor of the macro. */ public TableRowMacro() { super("TableRow", DESCRIPTION); setDefaultCategory(DEFAULT_CATEGORY_FORMATTING); } @Override public List<Block> execute(Object parameters, String content, MacroTransformationContext context) throws MacroExecutionException { XDOM parsedDom; // get a parser for the desired syntax identifier Parser parser = getSyntaxParser(context.getSyntax().toIdString()); try { // parse the content of the wiki macro that has been injected by the component manager the content of the macro call itself is ignored. parsedDom = parser.parse(new StringReader(content)); } catch (ParseException e) { throw new MacroExecutionException("Failed to parse content [" + content + "] with Syntax parser [" + parser.getSyntax() + "]", e); } List<MacroBlock> potentialCells = parsedDom.getChildrenByType(MacroBlock.class, false); int count = this.countCells(potentialCells); if (count == 0) { throw new MacroExecutionException("TableRow macro expect at least one cell macro as first-level children"); } // Make the actual cells, injecting <td> tags around cell macros this.makeCells(potentialCells); return Collections.singletonList((Block) parsedDom); } @Override public boolean supportsInlineMode() { return true; } protected Parser getSyntaxParser(String syntaxId) throws MacroExecutionException { try { return (Parser) this.componentManager.lookup(Parser.class, syntaxId); } catch (ComponentLookupException e) { throw new MacroExecutionException("Failed to find source parser", e); } } private int countCells(List<MacroBlock> blocks) { int result = 0; for (MacroBlock maybeCell : blocks) { if (maybeCell.getId().equals("table-cell") || maybeCell.getId().equals("td") || maybeCell.getId().equals("th")) { result++; } } return result; } private void makeCells(List<MacroBlock> blocks) { Iterator<MacroBlock> it = blocks.iterator(); while (it.hasNext()) { MacroBlock probablyCell = it.next(); if (probablyCell.getId().equals("table-cell") || probablyCell.getId().equals("td") || probablyCell.getId().equals("th")) { Map<String, String> params = new LinkedHashMap<String, String>(); Block cellBlock; if (probablyCell.getId().equals("th")) { cellBlock = new TableHeadCellBlock(Collections.singletonList(probablyCell.clone()), params); } else { cellBlock = new TableCellBlock(Collections.singletonList(probablyCell.clone()), params); } probablyCell.getParent().replaceChild(cellBlock, probablyCell); } } } }