/* SimpleScope.java Purpose: Description: History: Sat Sep 12 13:22:02 2009, Created by tomyeh Copyright (C) 2009 Potix Corporation. All Rights Reserved. This program is distributed under LGPL Version 2.1 in the hope that it will be useful, but WITHOUT ANY WARRANTY. */ package org.zkoss.zk.ui.impl; import java.util.AbstractSet; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.Set; import org.zkoss.zk.ui.Component; import org.zkoss.zk.ui.IdSpace; import org.zkoss.zk.ui.ext.Scope; import org.zkoss.zk.ui.ext.ScopeListener; import org.zkoss.zk.ui.util.ComponentCloneListener; /** * A simple implementation of {@link Scope}. * It supports {@link ScopeListener}, but it doesn't support * the concept of parent scope. * Thus, the deriving class can override * {@link #getAttribute(String,boolean)}, * {@link #hasAttribute(String,boolean)}, * and invoke {@link #notifyParentChanged} if the parent is changed. * * <p>Not thread safe. * * @author tomyeh * @since 5.0.0 */ public class SimpleScope implements Scope { private final Scope _owner; private Map<String, Object> _attrs; // make it lazy init private ScopeListeners _listeners; /** Constructor. * @param owner the real scope that an user can access. * If this object is the scope that an user accesses directly, pass <code>null</code> */ public SimpleScope(Scope owner) { _owner = owner != null ? owner : this; } private ScopeListeners initListeners() { if (_listeners == null) _listeners = new ScopeListeners(_owner); return _listeners; } /*package*/ Map<String, Object> newInitMap() { return new Attrs(2); } //Scope// public Map<String, Object> getAttributes() { if (_attrs == null) _attrs = newInitMap(); return _attrs; } public Object getAttribute(String name) { return _attrs != null ? _attrs.get(name) : null; } /** The same as getAttribute(name). */ public Object getAttribute(String name, boolean recurse) { return getAttribute(name); } public boolean hasAttribute(String name) { return _attrs != null && _attrs.containsKey(name); } /** The same as hasAttribute(name). */ public boolean hasAttribute(String name, boolean recurse) { return hasAttribute(name); } public Object setAttribute(String name, Object value) { if (_attrs == null) _attrs = newInitMap(); return _attrs.put(name, value); } /** The same as setAttribute(name, value). */ public Object setAttribute(String name, Object value, boolean recurse) { return setAttribute(name, value); } public Object removeAttribute(String name) { return _attrs != null ? _attrs.remove(name) : null; } /** The same as removeAttribute(name). */ public Object removeAttribute(String name, boolean recurse) { return removeAttribute(name); } public boolean addScopeListener(ScopeListener listener) { return initListeners().addScopeListener(listener); } public boolean removeScopeListener(ScopeListener listener) { return initListeners().removeScopeListener(listener); } /** Invokes {@link ScopeListener#parentChanged} for registered * listeners. * * @see #addScopeListener */ public void notifyParentChanged(Scope newparent) { initListeners().notifyParentChanged(newparent); } /** Invokes {@link ScopeListener#idSpaceChanged} for registered * listeners. * * @see #addScopeListener * @since 5.0.1 */ public void notifyIdSpaceChanged(IdSpace newIdSpace) { initListeners().notifyIdSpaceChanged(newIdSpace); } /** Returns a ist of all scope listners (never null). */ public List<ScopeListener> getListeners() { return initListeners().getListeners(); } //clone// /** Clones this scope. * @param owner the owner of the cloned scope. */ public SimpleScope clone(Scope owner) { final SimpleScope clone = new SimpleScope(owner); if (_attrs != null) { clone._attrs = new HashMap<String, Object>(); for (Map.Entry<String, Object> me : _attrs.entrySet()) { Object val = me.getValue(); if (val instanceof ComponentCloneListener && owner instanceof Component) { val = ((ComponentCloneListener) val).willClone((Component) owner); if (val == null) continue; //don't use it in clone } clone._attrs.put(me.getKey(), val); } } for (ScopeListener val : initListeners().getListeners()) { if (val instanceof ComponentCloneListener && owner instanceof Component) { val = (ScopeListener) ((ComponentCloneListener) val).willClone((Component) owner); if (val == null) continue; //don't use it in clone } clone.initListeners().addScopeListener(val); } return clone; } //Object// public String toString() { return _attrs != null ? _attrs.toString() : "{}"; } //Helper Class// private class Attrs extends HashMap<String, Object> { public Attrs(int initialCapacity) { super(initialCapacity); } public Object remove(Object key) { final Object o = super.remove(key); if (o != null && _listeners != null) _listeners.notifyRemoved((String) key); return o; } public Object put(String key, Object val) { final Object o = super.put(key, val); if (_listeners != null) { if (o != null) _listeners.notifyReplaced(key, val); else _listeners.notifyAdded(key, val); } return o; } @SuppressWarnings("unchecked") public Set<Map.Entry<String, Object>> entrySet() { return new AttrSet(super.entrySet(), true); } @SuppressWarnings("unchecked") public Set<String> keySet() { return new AttrSet(super.keySet(), false); } private class AttrSet extends AbstractSet { private final Set _set; private final boolean _entry; private AttrSet(Set set, boolean entry) { _set = set; _entry = entry; } public Iterator iterator() { return new AttrIter(_set.iterator()); } public int size() { return _set.size(); } @SuppressWarnings("unchecked") public boolean add(Object o) { if (_set.add(o)) { if (_listeners != null) { if (_entry) { final Map.Entry me = (Map.Entry) o; _listeners.notifyAdded((String) me.getKey(), me.getValue()); } else _listeners.notifyAdded((String) o, null); } return true; } else { if (_listeners != null) { if (_entry) { final Map.Entry me = (Map.Entry) o; _listeners.notifyReplaced((String) me.getKey(), me.getValue()); } else _listeners.notifyReplaced((String) o, null); } return false; } } public boolean remove(Object o) { if (_set.remove(o)) { if (_listeners != null) _listeners.notifyRemoved((String) (_entry ? ((Map.Entry) o).getKey() : o)); return true; } return false; } public boolean contains(Object o) { return _set.contains(o); } private class AttrIter implements Iterator { private final Iterator _it; private Object _last; private AttrIter(Iterator it) { _it = it; } public boolean hasNext() { return _it.hasNext(); } public Object next() { return _last = _it.next(); } public void remove() { _it.remove(); if (_listeners != null) _listeners.notifyRemoved((String) (_entry ? ((Map.Entry) _last).getKey() : _last)); } } } } }