【问题标题】:Programmatically render template area in Magnolia CMS在 Magnolia CMS 中以编程方式呈现模板区域
【发布时间】:2016-01-25 14:06:38
【问题描述】:

我正在使用 Magnolia CMS 5.4,我想构建一个模块来呈现页面的某些内容并通过 REST API 公开它。任务很简单,但不知道如何处理和/或从哪里开始。

我希望我的模块为给定的引用生成部分模板或模板区域,假设是“标题”。我需要渲染标题模板/区域以获取 HTML 并将其作为对另一个系统的响应返回。

所以问题是:这是否可能以及从哪里开始?

【问题讨论】:

    标签: java templates freemarker jcr magnolia


    【解决方案1】:

    好的,在这里和 Magnolia 论坛上询问后无法得到答案,我在源代码中挖掘并找到了一种方法。

    首先,渲染工作基于不同的渲染器,可以是 JCR、纯文本或 Freemarker 渲染器。在 Magnolia 中,这些是在 RenderingEngine 和实现中决定和使用的:DefaultRenderingEngine。渲染引擎将允许您渲染整个页面节点,这更接近我想要实现的目标。那么让我们看看如何做到这一点:

    我将跳过一些步骤,但我添加了命令并通过 REST 完成了这项工作,因此我可以看到当我向端点发送请求时发生了什么。该命令扩展 BaseRepositoryCommand 以允许访问 JCR 存储库。

    @Inject
    public setDefaultRenderingEngine(
        final RendererRegistry rendererRegistry,
        final TemplateDefinitionAssignment templateDefinitionAssignment,
        final RenderableVariationResolver variationResolver,
        final Provider<RenderingContext> renderingContextProvider
    ) {
        renderingEngine = new DefaultRenderingEngine(rendererRegistry, templateDefinitionAssignment,
                variationResolver, renderingContextProvider);
    }
    

    这将创建您的渲染引擎,并且您可以从这里开始渲染节点,但会遇到一些小问题。我已经尝试直接注入渲染引擎,但这并没有奏效,因为所有内部都是空的/null,所以决定获取所有构造属性并初始化我自己的版本。

    下一步是我们要渲染一个页面节点。首先,渲染引擎的工作基于它为HttpServletResponse 渲染的想法,并且与请求/响应流非常相关,尽管我们需要将生成的标记放在一个变量中,所以我添加了一个新的实现FilteringResponseOutputProvider:

    public class AppendableFilteringResponseOutputProvider extends FilteringResponseOutputProvider {
    
        private final FilteringAppendableWrapper appendable;
    
        private OutputStream outputStream = new ByteArrayOutputStream();
    
        public AppendableFilteringResponseOutputProvider(HttpServletResponse aResponse) {
            super(aResponse);
    
            OutputStreamWriter writer = new OutputStreamWriter(outputStream);
            appendable = Components.newInstance(FilteringAppendableWrapper.class);
            appendable.setWrappedAppendable(writer);
        }
    
        @Override
        public Appendable getAppendable() throws IOException {
            return appendable;
        }
    
        @Override
        public OutputStream getOutputStream() throws IOException {
            ((Writer) appendable.getWrappedAppendable()).flush();
    
            return outputStream;
        }
    
        @Override
        public void setWriteEnabled(boolean writeEnabled) {
            super.setWriteEnabled(writeEnabled);
            appendable.setWriteEnabled(writeEnabled);
        }
    }
    

    所以这个类的想法是公开输出流并仍然保留FilteringAppendableWrapper,这将允许我们过滤我们想要写入的内容。这在一般情况下不需要,您可以坚持使用AppendableOnlyOutputProviderStringBuilder 可附加并轻松检索整个页面标记。

    // here I needed to create a fake HttpServletResponse
    OutputProvider outputProvider = new AppendableFilteringResponseOutputProvider(new FakeResponse());
    

    一旦你有了输出提供者,你就需要一个页面节点,因为你在伪造它,你需要设置 Magnolia 全局 env 以便能够检索 JCR 节点:

    // populate repository and root node as those are not set for commands
    super.setRepository(RepositoryConstants.WEBSITE);
    super.setPath(nodePath); // this can be any existing path like: "/home/page"
    Node pageNode = getJCRNode(context);
    

    现在我们有了内容提供者,接下来我们要渲染的节点实际上正在运行渲染引擎:

    renderingEngine.render(pageNode, outputProvider);
    outputProvider.getOutputStream().toString();
    

    就是这样,你应该渲染你的内容,你可以随意使用它。

    现在我们来看看我的特殊情况,我只想渲染整个页面的一个区域,在这种情况下,这是页面的 Header。这一切都由相同的renderingEngine 处理,尽管您需要添加一个覆盖写入过程的呈现侦听器。先在命令中注入:

    @Inject
    public void setAreaFilteringListener(final AreaFilteringListener aAreaFilteringListener) {
        areaFilteringListener = aAreaFilteringListener;
    }
    

    这就是奇迹发生的地方,AreaFilteringListener 将检查您当前是否正在渲染请求的区域,如果您这样做,则启用输出提供程序进行写入,否则保持锁定并跳过所有不相关的区域。您需要像这样将侦听器添加到渲染引擎:

    // add the area filtering listener that generates specific area HTML only
    LinkedList<AbstractRenderingListener> listeners = new LinkedList<>();
    listeners.add(areaFilteringListener);
    renderingEngine.setListeners(listeners);
    
    // we need to provide the exact same Response instance that the WebContext is using
    // otherwise the voters against the AreaFilteringListener will skip the execution
    renderingEngine.initListeners(outputProvider, MgnlContext.getWebContext().getResponse());
    

    我听到你问:“但是我们在哪里指定要渲染的区域?”,啊哈来了:

    // enable the area filtering listener through a global flag
    MgnlContext.setAttribute(AreaFilteringListener.MGNL_AREA_PARAMETER, areaName);
    MgnlContext.getAggregationState().setMainContentNode(pageNode);
    

    区域过滤侦听器正在检查要设置的特定 Magnolia 上下文属性:“mgnlArea”如果找到,它将读取其值并将其用作区域名称,检查该区域是否存在于节点中,然后启用写入一旦我们到达该地区。这也可以通过以下 URL 使用:https://demopublic.magnolia-cms.com/~mgnlArea=footer~.html,这将为您提供生成为 HTML 页面的页脚区域。

    这里是完整的解决方案:http://yysource.com/2016/03/programatically-render-template-area-in-magnolia-cms/

    【讨论】:

      【解决方案2】:

      只需使用该区域的路径并使用该 url 发出 http 请求,例如http://localhost:9080/magnoliaAuthor/travel/main/0.html 据我所知,没有必要像你那样以编程方式完成所有事情。 Direct component rendering

      【讨论】:

      • 这是边界线link-only answer。您应该在此处扩展您的答案以包含尽可能多的信息,并仅将链接用作参考。
      猜你喜欢
      • 2012-10-08
      • 2021-09-07
      • 2010-12-15
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多