package org.apache.velocity.runtime.directive;
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import java.util.AbstractMap;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import org.apache.velocity.Template;
/**
* This handles context scoping and metadata for directives.
*
* @author Nathan Bubna
* @version $Id$
*/
public class Scope extends AbstractMap
{
private Map storage;
private Object replaced;
private Scope parent;
private Info info;
protected final Object owner;
public Scope(Object owner, Object previous)
{
this.owner = owner;
if (previous != null)
{
try
{
this.parent = (Scope)previous;
}
catch (ClassCastException cce)
{
this.replaced = previous;
}
}
}
private Map getStorage()
{
if (storage == null)
{
storage = new HashMap();
}
return storage;
}
public Set entrySet()
{
return getStorage().entrySet();
}
public Object get(Object key)
{
Object o = super.get(key);
if (o == null && parent != null && !containsKey(key))
{
return parent.get(key);
}
return o;
}
public Object put(Object key, Object value)
{
return getStorage().put(key, value);
}
/**
* Allows #stop to easily trigger the proper StopCommand for this scope.
*/
protected void stop()
{
throw new StopCommand(owner);
}
/**
* Returns the number of control arguments of this type
* that are stacked up. This is the distance between this
* instance and the topmost instance, plus one. This value
* will never be negative or zero.
*/
protected int getDepth()
{
if (parent == null)
{
return 1;
}
return parent.getDepth() + 1;
}
/**
* Returns the topmost parent control reference, retrieved
* by simple recursion on {@link #getParent}.
*/
public Scope getTopmost()
{
if (parent == null)
{
return this;
}
return parent.getTopmost();
}
/**
* Returns the parent control reference overridden by the placement
* of this instance in the context.
*/
public Scope getParent()
{
return parent;
}
/**
* Returns the user's context reference overridden by the placement
* of this instance in the context. If there was none (as is hoped),
* then this will return null. This never returns parent controls;
* those are returned by {@link #getParent}.
*/
public Object getReplaced()
{
if (replaced == null && parent != null)
{
return parent.getReplaced();
}
return replaced;
}
/**
* Returns info about the current scope for debugging purposes.
*/
public Info getInfo()
{
if (info == null)
{
info = new Info(this, owner);
}
return info;
}
/**
* Class to encapsulate and provide access to info about
* the current scope for debugging.
*/
public static class Info
{
private Scope scope;
private Directive directive;
private Template template;
public Info(Scope scope, Object owner)
{
if (owner instanceof Directive)
{
directive = (Directive)owner;
}
if (owner instanceof Template)
{
template = (Template)owner;
}
this.scope = scope;
}
public String getName()
{
if (directive != null)
{
return directive.getName();
}
if (template != null)
{
return template.getName();
}
return null;
}
public String getType()
{
if (directive != null)
{
switch (directive.getType())
{
case Directive.BLOCK:
return "block";
case Directive.LINE:
return "line";
}
}
if (template != null)
{
return template.getEncoding();
}
return null;
}
public int getDepth()
{
return scope.getDepth();
}
public String getTemplate()
{
if (directive != null)
{
return directive.getTemplateName();
}
if (template != null)
{
return template.getName();
}
return null;
}
public int getLine()
{
if (directive != null)
{
return directive.getLine();
}
return 0;
}
public int getColumn()
{
if (directive != null)
{
return directive.getColumn();
}
return 0;
}
public String toString()
{
StringBuffer sb = new StringBuffer();
if (directive != null)
{
sb.append('#');
}
sb.append(getName());
sb.append("[type:").append(getType());
int depth = getDepth();
if (depth > 1)
{
sb.append(" depth:").append(depth);
}
if (template == null)
{
String vtl = getTemplate();
sb.append(" template:");
if (vtl.indexOf(" ") < 0)
{
sb.append(vtl);
}
else
{
sb.append('"').append(vtl).append('"');
}
sb.append(" line:").append(getLine());
sb.append(" column:").append(getColumn());
}
sb.append(']');
return sb.toString();
}
}
}
|