All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.xwiki.rendering.util.ParserUtils Maven / Gradle / Ivy

/*
 * See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 * This is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * This software 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.xwiki.rendering.util;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import org.xwiki.rendering.block.Block;
import org.xwiki.rendering.block.CompositeBlock;
import org.xwiki.rendering.block.MacroBlock;
import org.xwiki.rendering.block.ParagraphBlock;
import org.xwiki.rendering.block.XDOM;

/**
 * Methods for helping in parsing.
 *
 * @version $Id: 02e6db52fcd0fb0d62be352e3ea1751e289edee7 $
 * @since 1.7M1
 */
public class ParserUtils
{
    /**
     * Removes any top level paragraph since for example for the following use case we don't want an extra paragraph
     * block: = hello {{velocity}}world{{/velocity}}.
     *
     * @param blocks the blocks to check and convert
     */
    public void removeTopLevelParagraph(List blocks)
    {
        // Remove any top level paragraph so that the result of a macro can be used inline for example.
        // We only remove the paragraph if there's only one top level element and if it's a paragraph.
        if ((blocks.size() == 1) && blocks.get(0) instanceof ParagraphBlock) {
            Block paragraphBlock = blocks.remove(0);
            blocks.addAll(0, paragraphBlock.getChildren());

            // Remove parent block
            for (Block block : blocks) {
                block.setParent(null);
            }
        }
    }

    /**
     * Best effort to convert a passed block to its inline version. Sometimes, it's simply impossible to
     * convert to an inline block, in which case it will just be returned as is.
     * Note: this method may mutate its parameter.
     *
     * @param rootBlock the block to convert
     * @param preserveXDOM if true and rootBlock is an XDOM, rootBlock will be returned with its children
     *                     converted inline.
     *                     Otherwise, the converted children will be returned directly (in a CompositeBlock
     *                     if there are several after the conversion)
     * @return the inline version of the passed block
     * @since 14.0RC1
     */
    public Block convertToInline(Block rootBlock, boolean preserveXDOM)
    {
        List blocks;
        if (rootBlock instanceof XDOM || rootBlock instanceof CompositeBlock) {
            // We can't modify directly the block's children list
            blocks = new ArrayList<>(rootBlock.getChildren());
        } else {
            blocks = Arrays.asList(rootBlock);
        }

        convertToInline(blocks);

        // Preserve source metadata if any
        if (preserveXDOM && rootBlock instanceof XDOM) {
            rootBlock.setChildren(blocks);

            return rootBlock;
        }

        return blocks.size() == 1 ? blocks.get(0) : new CompositeBlock(blocks);
    }

    /**
     * Best effort to convert passed blocks to their inline versions. Sometimes, it's simply impossible to convert an
     * to an inline block, in which case it will just be returned as is.
     * 
     * @param blocks the blocks to convert
     * @since 14.0RC1
     */
    // TODO: improve the implementation to really convert to inline everything that can be converted
    public void convertToInline(List blocks)
    {
        if (!blocks.isEmpty()) {
            // Clean top level paragraph
            removeTopLevelParagraph(blocks);

            // Synchronize macro
            if (!blocks.isEmpty()) {
                handleMacroInline(blocks);
            }
        }
    }

    /**
     * Make sure included macro is inline when script macro itself is inline.
     *
     * @param blocks the blocks to align
     */
    private void handleMacroInline(List blocks)
    {
        Block block = blocks.get(0);
        if (block instanceof MacroBlock) {
            MacroBlock macro = (MacroBlock) block;
            if (!macro.isInline()) {
                blocks.set(0, new MacroBlock(macro.getId(), macro.getParameters(), macro.getContent(), true));
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy