/* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * 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 . */ package ifc.beans; import java.util.ArrayList; import lib.MultiMethodTest; import util.ValueChanger; import util.utils; import com.sun.star.beans.Property; import com.sun.star.beans.PropertyAttribute; import com.sun.star.beans.PropertyChangeEvent; import com.sun.star.beans.XPropertyChangeListener; import com.sun.star.beans.XPropertySet; import com.sun.star.beans.XPropertySetInfo; import com.sun.star.beans.XVetoableChangeListener; import com.sun.star.lang.EventObject; /** * Testing com.sun.star.beans.XPropertySet * interface methods : * * @see com.sun.star.beans.XPropertySet */ public class _XPropertySet extends MultiMethodTest { public XPropertySet oObj = null; /** * Flag that indicates change listener was called. */ private boolean propertyChanged = false; /** * Listener that must be called on bound property changing. */ public class MyChangeListener implements XPropertyChangeListener { /** * Just set propertyChanged flag to true. */ public void propertyChange(PropertyChangeEvent e) { propertyChanged = true; } public void disposing (EventObject obj) {} } private final XPropertyChangeListener PClistener = new MyChangeListener(); /** * Flag that indicates veto listener was called. */ private boolean vetoableChanged = false; /** * Listener that must be called on constrained property changing. */ public class MyVetoListener implements XVetoableChangeListener { /** * Just set vetoableChanged flag to true. */ public void vetoableChange(PropertyChangeEvent e) { vetoableChanged = true; } public void disposing (EventObject obj) {} } private final XVetoableChangeListener VClistener = new MyVetoListener(); /** * Structure that collects the properties of different types to test : * Constrained, Bound and Normal. */ private static final class PropsToTest { ArrayList< String > constrained = new ArrayList< String >(); ArrayList< String > bound = new ArrayList< String >(); ArrayList< String > normal = new ArrayList< String >(); } private final PropsToTest PTT = new PropsToTest(); /** * Tests method getPropertySetInfo. After test completed * call {@link #getPropsToTest} method to retrieve different kinds * of properties to test then.

* Has OK status if not null XPropertySetInfo * object returned.

* Since getPropertySetInfo is optional, it may return null, * if it is not implemented. This method uses then an object relation * PTT (Properties To Test) to determine available properties. * All tests for services without getPropertySetInfo must * provide this object relation. */ public void _getPropertySetInfo() { XPropertySetInfo propertySetInfo = oObj.getPropertySetInfo(); if (propertySetInfo == null) { log.println("getPropertySetInfo() method returned null"); tRes.tested("getPropertySetInfo()", true) ; String[] ptt = (String[]) tEnv.getObjRelation("PTT"); PTT.normal.clear(); PTT.bound.clear(); PTT.constrained.clear(); PTT.normal.add( ptt[0] ); PTT.bound.add( ptt[1] ); PTT.constrained.add( ptt[2] ); } else { tRes.tested("getPropertySetInfo()", true ); getPropsToTest(propertySetInfo); } } // end of getPropertySetInfo() /** * Tests change listener which added for bound properties. * Adds listener to bound property (if it exists), then changes * its value and check if listener was called.

* Method tests to be successfully completed before : *

* Has OK status if NO bound properties exist or if listener * was successfully called. */ public void _addPropertyChangeListener() { requiredMethod("getPropertySetInfo()"); int count = PTT.bound.size(); if ( count==0 || PTT.bound.get(0).equals("none") ) { log.println("*** No bound properties found ***"); tRes.tested("addPropertyChangeListener()", true) ; } else { boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.bound.get(i); propertyChanged = false; try { oObj.addPropertyChangeListener(propertyName,PClistener); Object gValue = oObj.getPropertyValue(propertyName); log.println("Check bound property: " + propertyName ); oObj.setPropertyValue(propertyName, ValueChanger.changePValue(gValue)); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } // end of try-catch error = error || !propertyChanged; if (!propertyChanged) { log.println("propertyChangeListener wasn't called for '"+ propertyName+"'"); } } tRes.tested("addPropertyChangeListener()", !error); } } // end of addPropertyChangeListener() /** * Tests vetoable listener which added for constrained properties. * Adds listener to constrained property (if it exists), then changes * its value and check if listener was called.

* Method tests to be successfully completed before : *

* Has OK status if NO constrained properties exist or if listener * was successfully called. */ public void _addVetoableChangeListener() { requiredMethod("getPropertySetInfo()"); int count = PTT.constrained.size(); if ( count==0 || PTT.constrained.get(0).equals("none") ) { log.println("*** No constrained properties found ***"); tRes.tested("addVetoableChangeListener()", true) ; } else { boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.constrained.get(i); vetoableChanged = false; try { oObj.addVetoableChangeListener(propertyName,VClistener); Object gValue = oObj.getPropertyValue(propertyName); oObj.setPropertyValue(propertyName, ValueChanger.changePValue(gValue)); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } // end of try-catch error = error || !vetoableChanged; if (!vetoableChanged) { log.println("vetoableChangeListener wasn't called for '"+ propertyName+"'"); } } tRes.tested("addVetoableChangeListener()",!error); } } // end of addVetoableChangeListener() /** * Tests setPropertyValue method. * Stores value before call, and compares it with value after * call.

* Method tests to be successfully completed before : *

* Has OK status if NO normal properties exist or if value before * method call is not equal to value after. */ public void _setPropertyValue() { requiredMethod("getPropertySetInfo()"); Object gValue = null; Object sValue = null; int count = PTT.normal.size(); if ( count==0 || PTT.normal.get(0).equals("none") ) { log.println("*** No changeable properties found ***"); tRes.tested("setPropertyValue()", true) ; } else { boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.normal.get(i); try { log.println("try to change value of property '" + propertyName + "'" ); gValue = oObj.getPropertyValue(propertyName); sValue = ValueChanger.changePValue(gValue, propertyName); oObj.setPropertyValue(propertyName, sValue); sValue = oObj.getPropertyValue(propertyName); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } // end of try-catch /* this is stupid: we can't set properties whose semantics we * don't know to random values in an arbitrary order and * expect that to actually work. if( gValue.equals(sValue) ) { log.println("setting property '"+ propertyName+"' failed"); error = true; } */ } tRes.tested("setPropertyValue()",!error); } //endif } // end of setPropertyValue() /** * Tests getPropertyValue method for the given property. * Returns true if no exceptions occurred */ private boolean getSinglePropertyValue( String propertyName ) { boolean runOk = false; try { oObj.getPropertyValue(propertyName); runOk = true; } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to get property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to get property '"+ propertyName+"'"); e.printStackTrace(log); } return runOk; } /** * Tests getPropertyValue method. * Just call this method and checks for no exceptions

* Method tests to be successfully completed before : *

* Has OK status if NO normal properties exist or if no * exceptions were thrown. */ public void _getPropertyValue() { requiredMethod("getPropertySetInfo()"); int count = PTT.normal.size(); if ( count==0 || PTT.normal.get(0).equals("none") ) { Property[] properties = oObj.getPropertySetInfo().getProperties(); if( properties.length > 0 ) { String propertyName = properties[0].Name; log.println("All properties are Read Only"); log.println("Using: "+propertyName); tRes.tested("getPropertyValue()", getSinglePropertyValue( propertyName ) ); } else { log.println("*** No properties found ***"); tRes.tested("getPropertyValue()", true) ; } } else { boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.normal.get(i); boolean runOk = getSinglePropertyValue( propertyName ); if( !runOk ) { error = true; log.println("getPropertyValue() failed for property '"+propertyName+"'"); } } tRes.tested("getPropertyValue()", !error) ; } } /** * Tests removePropertyChangeListener method. * Removes change listener, then changes bound property value * and checks if the listener was NOT called. * Method tests to be successfully completed before : *

* Has OK status if NO bound properties exist or if listener * was not called and no exceptions arose. */ public void _removePropertyChangeListener() { requiredMethod("addPropertyChangeListener()"); int count = PTT.bound.size(); if ( count==0 || PTT.bound.get(0).equals("none") ) { log.println("*** No bound properties found ***"); tRes.tested("removePropertyChangeListener()", true) ; } else { //remove all listeners first for (int i = 0; i < count; i++) { String propertyName = PTT.bound.get(i); try { oObj.removePropertyChangeListener(propertyName,PClistener); } catch (Exception e) { log.println("Exception occurred while removing change listener from"+ "property '"+ propertyName+"'"); e.printStackTrace(log); } } boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.bound.get(i); try { propertyChanged = false; oObj.addPropertyChangeListener(propertyName,PClistener); oObj.removePropertyChangeListener(propertyName,PClistener); Object gValue = oObj.getPropertyValue(propertyName); oObj.setPropertyValue(propertyName, ValueChanger.changePValue(gValue)); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } // end of try-catch error = error || propertyChanged; if (propertyChanged) { log.println("propertyChangeListener was called after removing"+ " for '"+propertyName+"'"); } } tRes.tested("removePropertyChangeListener()",!error); } } // end of removePropertyChangeListener() /** * Tests removeVetoableChangeListener method. * Removes vetoable listener, then changes constrained property value * and checks if the listener was NOT called. * Method tests to be successfully completed before : *

* Has OK status if NO constrained properties exist or if listener * was NOT called and no exceptions arose. */ public void _removeVetoableChangeListener() { requiredMethod("addVetoableChangeListener()"); int count = PTT.constrained.size(); if ( count==0 || PTT.constrained.get(0).equals("none") ) { log.println("*** No constrained properties found ***"); tRes.tested("removeVetoableChangeListener()", true) ; } else { //remove all listeners first for (int i = 0; i < count; i++) { String propertyName = PTT.constrained.get(i); try { oObj.removeVetoableChangeListener(propertyName,VClistener); } catch (Exception e) { log.println("Exception occurred while removing veto listener from"+ "property '"+ propertyName+"'"); e.printStackTrace(log); } } boolean error = false; for (int i = 0; i < count; i++) { String propertyName = PTT.constrained.get(i); vetoableChanged = false; try { oObj.addVetoableChangeListener(propertyName,VClistener); oObj.removeVetoableChangeListener(propertyName,VClistener); Object gValue = oObj.getPropertyValue(propertyName); oObj.setPropertyValue(propertyName, ValueChanger.changePValue(gValue)); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("Exception occurred while trying to change "+ "property '"+ propertyName+"'"); e.printStackTrace(log); } // end of try-catch error = error || vetoableChanged; if (vetoableChanged) { log.println("vetoableChangeListener was called after "+ "removing for '"+propertyName+"'"); } } tRes.tested("removeVetoableChangeListener()",!error); } } // end of removeVetoableChangeListener() /** * Gets the properties being tested. Searches and stores by one * property of each kind (Bound, Vetoable, Normal). */ public void getPropsToTest(XPropertySetInfo xPSI) { Property[] properties = xPSI.getProperties(); // some properties should not be changed in a unspecific way String[] skip = {"PrinterName", "CharRelief", "IsLayerMode"}; for (int i = 0; i < properties.length; i++) { Property property = properties[i]; String name = property.Name; boolean cont = false; for (int j = 0; j < skip.length; j++) { if (name.equals(skip[j])){ log.println("skipping '" + name + "'"); cont = true; } } if (cont) continue; boolean isWritable = ((property.Attributes & PropertyAttribute.READONLY) == 0); boolean isNotNull = ((property.Attributes & PropertyAttribute.MAYBEVOID) == 0); boolean isBound = ((property.Attributes & PropertyAttribute.BOUND) != 0); boolean isConstr = ((property.Attributes & PropertyAttribute.CONSTRAINED) != 0); boolean canChange = false; if ( !isWritable ) log.println("Property '"+name+"' is READONLY"); if (name.endsWith("URL")) isWritable = false; if (name.startsWith("Fill")) isWritable = false; if (name.startsWith("Font")) isWritable = false; if (name.startsWith("IsNumbering")) isWritable = false; if (name.startsWith("LayerName")) isWritable = false; if (name.startsWith("Line")) isWritable = false; if (name.startsWith("TextWriting")) isWritable = false; if ( isWritable && isNotNull ) canChange = isChangeable(name); if ( isWritable && isNotNull && isBound && canChange) { PTT.bound.add(name); } if ( isWritable && isNotNull && isConstr && canChange) { PTT.constrained.add(name); } if ( isWritable && isNotNull && canChange) { PTT.normal.add(name); } } // endfor } public boolean isChangeable(String name) { boolean hasChanged = false; try { Object getProp = oObj.getPropertyValue(name); log.println("Getting: "+getProp); if (name.equals("xinterfaceA")) { System.out.println("drin"); } Object setValue = null; if (getProp != null) { if (!utils.isVoid(getProp)) setValue = ValueChanger.changePValue(getProp); else log.println("Property '"+name+ "' is void but MAYBEVOID isn't set"); } else log.println("Property '"+name+"'is null and can't be changed"); if (name.equals("LineStyle")) setValue = null; if (setValue != null) { log.println("Setting to :"+setValue); oObj.setPropertyValue(name, setValue); hasChanged = (! getProp.equals(oObj.getPropertyValue(name))); } else log.println("Couldn't change Property '"+name+"'"); } catch (com.sun.star.beans.PropertyVetoException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } catch (com.sun.star.lang.IllegalArgumentException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } catch (com.sun.star.beans.UnknownPropertyException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } catch (com.sun.star.lang.WrappedTargetException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } catch (com.sun.star.uno.RuntimeException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } catch (java.lang.ArrayIndexOutOfBoundsException e) { log.println("'" + name + "' throws exception '" + e + "'"); e.printStackTrace(log); } return hasChanged; } /** * Forces environment recreation. */ @Override protected void after() { disposeEnvironment(); } } // finish class _XPropertySet