001 // Copyright 2011 The Apache Software Foundation
002 //
003 // Licensed under the Apache License, Version 2.0 (the "License");
004 // you may not use this file except in compliance with the License.
005 // You may obtain a copy of the License at
006 //
007 // http://www.apache.org/licenses/LICENSE-2.0
008 //
009 // Unless required by applicable law or agreed to in writing, software
010 // distributed under the License is distributed on an "AS IS" BASIS,
011 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
012 // See the License for the specific language governing permissions and
013 // limitations under the License.
014
015 package org.apache.tapestry5.corelib.components;
016
017 import org.apache.tapestry5.BindingConstants;
018 import org.apache.tapestry5.Block;
019 import org.apache.tapestry5.ComponentResources;
020 import org.apache.tapestry5.annotations.Parameter;
021 import org.apache.tapestry5.annotations.SupportsInformalParameters;
022 import org.apache.tapestry5.ioc.annotations.Inject;
023 import org.apache.tapestry5.ioc.util.AvailableValues;
024 import org.apache.tapestry5.ioc.util.UnknownValueException;
025 import org.apache.tapestry5.runtime.RenderCommand;
026 import org.apache.tapestry5.services.dynamic.DynamicDelegate;
027 import org.apache.tapestry5.services.dynamic.DynamicTemplate;
028 import org.apache.tapestry5.services.dynamic.DynamicTemplateParser;
029
030 /**
031 * The Dynamic component allows a component to render itself differently at different times, by making use of
032 * an external template file.
033 * <p>
034 * The content of the template file replaces the Dynamic component entirely with one exception: certain elements will be
035 * replaced with {@linkplain Block}s passed to the Dynamic component as informal parameters; this is triggered by
036 * <strong>id</strong> of the element. When the id attribute has the prefix {@code param:}, the remainder is the name of
037 * a Block parameter. There are no limitations on what can appear inside such a Block: text, components, forms, even the
038 * {@literal <t:body/>} directive.
039 * <p>
040 * Dynamic templates emulate how expansions work in standard Tapestry templates: Expansions (the <code>${ ... }</code>
041 * syntax) can appear in attribute values or interspersed in element text. This allows container properties, messages,
042 * assets, and so forth to be referenced within the external template. It should be noted that such access is quite a
043 * bit less efficient than putting such expansions inside a referenced Block, but this should not be a big concern
044 * outside of some kind of tight rendering loop.
045 *
046 * @since 5.3
047 * @see DynamicTemplate
048 * @see DynamicTemplateParser
049 * @tapestrydoc
050 */
051 @SupportsInformalParameters
052 public class Dynamic
053 {
054 /** The dynamic template containing the content to be rendered by the component. */
055 @Parameter(required = true, allowNull = false, defaultPrefix = BindingConstants.ASSET)
056 private DynamicTemplate template;
057
058 @Inject
059 private ComponentResources resources;
060
061 private final DynamicDelegate delegate = new DynamicDelegate()
062 {
063 public ComponentResources getComponentResources()
064 {
065 return resources;
066 }
067
068 public Block getBlock(String name)
069 {
070 Block result = resources.getBlockParameter(name);
071
072 if (result != null)
073 return result;
074
075 throw new UnknownValueException(String.format(
076 "Component %s does not have an informal Block parameter with id '%s'.", resources.getCompleteId(),
077 name), null, null, new AvailableValues("Available Blocks", resources.getInformalParameterNames()));
078 }
079 };
080
081 RenderCommand beginRender()
082 {
083 // Probably some room for caching here as well. It shouldn't be necessary to re-create the outermost
084 // RenderCommand every time, unless the template has changed from the previous render.
085 return template.createRenderCommand(delegate);
086 }
087 }