/* $Id: ExpressionModel.java 18924 2010-12-18 12:19:08Z bobtarling $ ******************************************************************************* * Copyright (c) 2009 Contributors - see below * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Bob Tarling * Thomas Neustupny ******************************************************************************* */ // $Id: ExpressionModel.java 18924 2010-12-18 12:19:08Z bobtarling $ // Copyright (c) 1996-2009 The Regents of the University of California. All // Rights Reserved. Permission to use, copy, modify, and distribute this // software and its documentation without fee, and without a written // agreement is hereby granted, provided that the above copyright notice // and this paragraph appear in all copies. This software program and // documentation are copyrighted by The Regents of the University of // California. The software program and documentation are supplied "AS // IS", without any accompanying services from The Regents. The Regents // does not warrant that the operation of the program will be // uninterrupted or error-free. The end-user understands that the program // was developed for research purposes and is advised not to rely // exclusively on the program for any reason. IN NO EVENT SHALL THE // UNIVERSITY OF CALIFORNIA BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT, // SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, // ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF // THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF THE POSSIBILITY OF // SUCH DAMAGE. THE UNIVERSITY OF CALIFORNIA SPECIFICALLY DISCLAIMS ANY // WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE // PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF // CALIFORNIA HAS NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, // UPDATES, ENHANCEMENTS, OR MODIFICATIONS. package org.argouml.core.propertypanels.ui; import javax.swing.event.ChangeEvent; import javax.swing.event.ChangeListener; import javax.swing.event.EventListenerList; import org.apache.log4j.Logger; import org.argouml.core.propertypanels.model.GetterSetterManager; import org.argouml.model.Model; /** * The model for entities with body and language attributes (Expression in * UML1, OpaqueExpression/OpaqueBehavior in UML2). * The target is the UML element to which this Expression is attached. * * The ChangeEvent/ChangeListener handling is inspired by * javax.swing.DefaultBoundedRangeModel. * It listens to UML model changes not caused by us, * which need to trigger an update of the UI rendering. * * @author mkl, penyaskito, mvw, thn */ class ExpressionModel { private static final Logger LOG = Logger.getLogger(ExpressionModel.class); /** * The target model element * In UML1.x this is the element that owns the expression. * In UML2.x this is the expression itself * TODO: This makes behaviour different enough that separate subclasses * should be considered. */ private Object target; private String propertyName; private final GetterSetterManager getterSetterManager; private final Class<?> type; private static final String EMPTYSTRING = ""; /** The listeners waiting for model changes. */ protected EventListenerList listenerList = new EventListenerList(); /** * Only one <code>ChangeEvent</code> is needed per model instance * since the event's only (read-only) state is the expression. The source * of events generated here is always "this". */ protected transient ChangeEvent changeEvent = null; public ExpressionModel( final String propertyName, final Class<?> type, final Object umlElement, final GetterSetterManager getterSetterManager) { this.target = umlElement; LOG.info("Creating ExpressionModel with target " + target); this.propertyName = propertyName; this.getterSetterManager = getterSetterManager; this.type = type; } /** * @return the expression */ private Object getExpression() { Object expression = null; if (Model.getFacade().getUmlVersion().charAt(0) == '1') { expression = getterSetterManager.get(target, propertyName, type); LOG.info("Got the expression " + expression); } else { // in UML2, the target is already the "expression" (Opaque...) expression = target; } return expression; } /** * @param expr the expression */ public void setExpression(Object expr) { LOG.info("Setting the expression to " + expr); getterSetterManager.set(target, expr, propertyName); } /** * @return a new expression with given language and body */ public Object newExpression(String language, String body) { return getterSetterManager.create(propertyName, language, body); } /** * @return the language of the expression */ public String getLanguage() { Object expression = getExpression(); if (expression == null) { return EMPTYSTRING; } return Model.getDataTypesHelper().getLanguage(expression); } /** * @return The body text of the expression. */ public String getBody() { Object expression = getExpression(); if (expression == null) { return EMPTYSTRING; } return Model.getDataTypesHelper().getBody(expression); } /** * @param lang the language of the expression */ public void setLanguage(String lang) { Object expression = getExpression(); boolean mustChange = true; if (expression != null) { String oldValue = Model.getDataTypesHelper().getLanguage(expression); if (oldValue != null && oldValue.equals(lang)) { mustChange = false; } } if (mustChange) { String body = EMPTYSTRING; if (expression != null && Model.getDataTypesHelper().getBody(expression) != null) { body = Model.getDataTypesHelper().getBody(expression); } setExpression(lang, body); } } /** * @param body the body text of the expression */ public void setBody(String body) { Object expression = getExpression(); boolean mustChange = true; if (expression != null) { Object oldValue = Model.getDataTypesHelper().getBody(expression); if (oldValue != null && oldValue.equals(body)) { mustChange = false; } } if (mustChange) { String lang = null; if (expression != null) { lang = Model.getDataTypesHelper().getLanguage(expression); } if (lang == null) { lang = EMPTYSTRING; } setExpression(lang, body); } } /** * This is only called if we already know that the values differ. * * @param lang the language of the expression * @param body the body text of the expression */ private void setExpression(String lang, String body) { if (Model.getFacade().getUmlVersion().charAt(0) == '1') { // Expressions are DataTypes, not independent model elements // be careful not to reuse them final Object currentExpression = getExpression(); if (currentExpression != null) { LOG.info("Deleting the current expression " + currentExpression); Model.getUmlFactory().delete(currentExpression); } final Object newExpression; if (lang.length() == 0 && body.length()==0) { newExpression = null; } else { newExpression = newExpression(lang, body); } setExpression(newExpression); } else { // reuse Opaque..., just set it's attributes Object expression = getExpression(); Model.getDataTypesHelper().setBody(expression, body); Model.getDataTypesHelper().setLanguage(expression, lang); } } /** * Adds a <code>ChangeListener</code>. * The change listeners are run each * time the expression changes. * * @param l the ChangeListener to add * @see #removeChangeListener */ public void addChangeListener(ChangeListener l) { listenerList.add(ChangeListener.class, l); LOG.debug(">>Add listener"); } /** * Runs each <code>ChangeListener</code>'s * <code>stateChanged</code> method. * * @see #setRangeProperties * @see EventListenerList */ protected void fireStateChanged() { LOG.debug(">>Fire state changed to listeners."); Object[] listeners = listenerList.getListenerList(); for (int i = listeners.length - 2; i >= 0; i -=2 ) { if (listeners[i] == ChangeListener.class) { if (changeEvent == null) { changeEvent = new ChangeEvent(this); } ((ChangeListener)listeners[i+1]).stateChanged(changeEvent); } } } }