001// Copyright 2006, 2007, 2008, 2009, 2010, 2011, 2012 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
015package org.apache.tapestry5.internal.model;
016
017import org.apache.tapestry5.ioc.Location;
018import org.apache.tapestry5.ioc.Resource;
019import org.apache.tapestry5.ioc.internal.util.CollectionFactory;
020import org.apache.tapestry5.ioc.internal.util.InternalUtils;
021import org.apache.tapestry5.ioc.util.IdAllocator;
022import org.apache.tapestry5.model.*;
023import org.slf4j.Logger;
024
025import java.util.Collections;
026import java.util.List;
027import java.util.Map;
028import java.util.Set;
029
030/**
031 * Internal implementation of {@link org.apache.tapestry5.model.MutableComponentModel}.
032 */
033public final class MutableComponentModelImpl implements MutableComponentModel
034{
035    private final ComponentModel parentModel;
036
037    private final Resource baseResource;
038
039    private final String componentClassName;
040
041    private final IdAllocator persistentFieldNameAllocator = new IdAllocator();
042
043    private final Logger logger;
044
045    private final boolean pageClass;
046
047    private Map<String, ParameterModel> parameters;
048
049    private Map<String, EmbeddedComponentModel> embeddedComponents;
050
051    /**
052     * Maps from field name to strategy.
053     */
054    private Map<String, String> persistentFields;
055
056    private List<String> mixinClassNames;
057
058    private Map<String, String[]> mixinOrders;
059
060    private boolean informalParametersSupported;
061
062    private boolean mixinAfter;
063
064    private Map<String, String> metaData;
065
066    private Set<Class> handledRenderPhases;
067
068    private Map<String, Boolean> handledEvents;
069
070    private final String libraryName;
071
072    private boolean handleActivationEventContext;
073
074    public MutableComponentModelImpl(String componentClassName, Logger logger, Resource baseResource,
075                                     ComponentModel parentModel, boolean pageClass, String libraryName)
076    {
077        this.componentClassName = componentClassName;
078        this.logger = logger;
079        this.baseResource = baseResource;
080        this.parentModel = parentModel;
081        this.pageClass = pageClass;
082        this.libraryName = libraryName;
083
084        // Default to false, explicitly set during page transformation
085        this.handleActivationEventContext = false;
086
087        // Pre-allocate names from the parent, to avoid name collisions.
088
089        if (this.parentModel != null)
090        {
091            for (String name : this.parentModel.getPersistentFieldNames())
092            {
093                persistentFieldNameAllocator.allocateId(name);
094            }
095        }
096    }
097
098    public String getLibraryName()
099    {
100        return libraryName;
101    }
102
103    @Override
104    public String toString()
105    {
106        return String.format("ComponentModel[%s]", componentClassName);
107    }
108
109    public Logger getLogger()
110    {
111        return logger;
112    }
113
114    public Resource getBaseResource()
115    {
116        return baseResource;
117    }
118
119    public String getComponentClassName()
120    {
121        return componentClassName;
122    }
123
124    public void addParameter(String name, boolean required, boolean allowNull, String defaultBindingPrefix,
125                             boolean cached)
126    {
127        assert InternalUtils.isNonBlank(name);
128        assert InternalUtils.isNonBlank(defaultBindingPrefix);
129
130        if (parameters == null)
131        {
132            parameters = CollectionFactory.newCaseInsensitiveMap();
133        }
134
135        if (parameters.containsKey(name))
136        {
137            throw new IllegalArgumentException(String.format("Parameter '%s' of component class %s is already defined.", name, componentClassName));
138        }
139
140        ParameterModel existingModel = getParameterModel(name);
141
142        if (existingModel != null)
143        {
144            throw new IllegalArgumentException(String.format("Parameter '%s' of component class %s conflicts with the parameter defined by the %s base class.",
145                    name, componentClassName, existingModel.getComponentModel().getComponentClassName()));
146        }
147
148        parameters.put(name, new ParameterModelImpl(this, name, required, allowNull, defaultBindingPrefix, cached));
149    }
150
151    public void addParameter(String name, boolean required, boolean allowNull, String defaultBindingPrefix)
152    {
153        // assume /false/ for the default because:
154        // if the parameter is actually cached, the only effect will be to reduce that optimization
155        // in certain
156        // scenarios (mixin BindParameter). But if the value is NOT cached but we say it is,
157        // we'll get incorrect behavior.
158        addParameter(name, required, allowNull, defaultBindingPrefix, false);
159    }
160
161    public ParameterModel getParameterModel(String parameterName)
162    {
163        ParameterModel result = InternalUtils.get(parameters, parameterName);
164
165        if (result == null && parentModel != null)
166            result = parentModel.getParameterModel(parameterName);
167
168        return result;
169    }
170
171    public boolean isFormalParameter(String parameterName)
172    {
173        return getParameterModel(parameterName) != null;
174    }
175
176    public List<String> getParameterNames()
177    {
178        List<String> names = CollectionFactory.newList();
179
180        if (parameters != null)
181            names.addAll(parameters.keySet());
182
183        if (parentModel != null)
184            names.addAll(parentModel.getParameterNames());
185
186        Collections.sort(names);
187
188        return names;
189    }
190
191    public List<String> getDeclaredParameterNames()
192    {
193        return InternalUtils.sortedKeys(parameters);
194    }
195
196    public MutableEmbeddedComponentModel addEmbeddedComponent(String id, String type, String componentClassName,
197                                                              boolean inheritInformalParameters, Location location)
198    {
199        // TODO: Parent compent model? Or would we simply override the parent?
200
201        if (embeddedComponents == null)
202            embeddedComponents = CollectionFactory.newCaseInsensitiveMap();
203        else if (embeddedComponents.containsKey(id))
204            throw new IllegalArgumentException(String.format("Embedded component '%s' has already been defined for component class %s.", id, this.componentClassName));
205
206        MutableEmbeddedComponentModel embedded = new MutableEmbeddedComponentModelImpl(id, type, componentClassName,
207                this.componentClassName, inheritInformalParameters, location);
208
209        embeddedComponents.put(id, embedded);
210
211        return embedded; // So that parameters can be filled in
212    }
213
214    public List<String> getEmbeddedComponentIds()
215    {
216        List<String> result = CollectionFactory.newList();
217
218        if (embeddedComponents != null)
219            result.addAll(embeddedComponents.keySet());
220
221        if (parentModel != null)
222            result.addAll(parentModel.getEmbeddedComponentIds());
223
224        Collections.sort(result);
225
226        return result;
227    }
228
229    public EmbeddedComponentModel getEmbeddedComponentModel(String componentId)
230    {
231        EmbeddedComponentModel result = InternalUtils.get(embeddedComponents, componentId);
232
233        if (result == null && parentModel != null)
234            result = parentModel.getEmbeddedComponentModel(componentId);
235
236        return result;
237    }
238
239    public String getFieldPersistenceStrategy(String fieldName)
240    {
241        String result = InternalUtils.get(persistentFields, fieldName);
242
243        if (result == null && parentModel != null)
244            result = parentModel.getFieldPersistenceStrategy(fieldName);
245
246        if (result == null)
247            throw new IllegalArgumentException(String.format("No field persistence strategy has been defined for field '%s'.", fieldName));
248
249        return result;
250    }
251
252    public List<String> getPersistentFieldNames()
253    {
254        return persistentFieldNameAllocator.getAllocatedIds();
255    }
256
257    public String setFieldPersistenceStrategy(String fieldName, String strategy)
258    {
259        String logicalFieldName = persistentFieldNameAllocator.allocateId(fieldName);
260
261        if (persistentFields == null)
262            persistentFields = CollectionFactory.newMap();
263
264        persistentFields.put(logicalFieldName, strategy);
265
266        return logicalFieldName;
267    }
268
269    public boolean isRootClass()
270    {
271        return parentModel == null;
272    }
273
274    public void addMixinClassName(String mixinClassName, String... order)
275    {
276        if (mixinClassNames == null)
277            mixinClassNames = CollectionFactory.newList();
278
279        mixinClassNames.add(mixinClassName);
280        if (order != null && order.length > 0)
281        {
282            if (mixinOrders == null)
283                mixinOrders = CollectionFactory.newCaseInsensitiveMap();
284            mixinOrders.put(mixinClassName, order);
285        }
286    }
287
288    public List<String> getMixinClassNames()
289    {
290        List<String> result = CollectionFactory.newList();
291
292        if (mixinClassNames != null)
293            result.addAll(mixinClassNames);
294
295        if (parentModel != null)
296            result.addAll(parentModel.getMixinClassNames());
297
298        Collections.sort(result);
299
300        return result;
301    }
302
303    public void enableSupportsInformalParameters()
304    {
305        informalParametersSupported = true;
306    }
307
308    public boolean getSupportsInformalParameters()
309    {
310        return informalParametersSupported;
311    }
312
313    public ComponentModel getParentModel()
314    {
315        return parentModel;
316    }
317
318    public boolean isMixinAfter()
319    {
320        return mixinAfter;
321    }
322
323    public void setMixinAfter(boolean mixinAfter)
324    {
325        this.mixinAfter = mixinAfter;
326    }
327
328    public void setMeta(String key, String value)
329    {
330        assert InternalUtils.isNonBlank(key);
331        assert InternalUtils.isNonBlank(value);
332        if (metaData == null)
333            metaData = CollectionFactory.newCaseInsensitiveMap();
334
335        // TODO: Error if duplicate?
336
337        metaData.put(key, value);
338    }
339
340    public void addRenderPhase(Class renderPhase)
341    {
342        assert renderPhase != null;
343        if (handledRenderPhases == null)
344            handledRenderPhases = CollectionFactory.newSet();
345
346        handledRenderPhases.add(renderPhase);
347    }
348
349    public void addEventHandler(String eventType)
350    {
351        if (handledEvents == null)
352            handledEvents = CollectionFactory.newCaseInsensitiveMap();
353
354        handledEvents.put(eventType, true);
355    }
356
357    public String getMeta(String key)
358    {
359        String result = InternalUtils.get(metaData, key);
360
361        if (result == null && parentModel != null)
362            result = parentModel.getMeta(key);
363
364        return result;
365    }
366
367    public Set<Class> getHandledRenderPhases()
368    {
369        Set<Class> result = CollectionFactory.newSet();
370
371        if (parentModel != null)
372            result.addAll(parentModel.getHandledRenderPhases());
373
374        if (handledRenderPhases != null)
375            result.addAll(handledRenderPhases);
376
377        return result;
378    }
379
380    public boolean handlesEvent(String eventType)
381    {
382        if (InternalUtils.get(handledEvents, eventType) != null)
383            return true;
384
385        return parentModel == null ? false : parentModel.handlesEvent(eventType);
386    }
387
388    public String[] getOrderForMixin(String mixinClassName)
389    {
390        final String[] orders = InternalUtils.get(mixinOrders, mixinClassName);
391
392        if (orders == null && parentModel != null)
393            return parentModel.getOrderForMixin(mixinClassName);
394
395        return orders;
396    }
397
398    public boolean isPage()
399    {
400        return pageClass;
401    }
402
403    public void doHandleActivationEventContext()
404    {
405        this.handleActivationEventContext = true;
406    }
407
408    public boolean handleActivationEventContext()
409    {
410        return this.handleActivationEventContext;
411    }
412}