/*************************************************************************
 *
 *  OpenOffice.org - a multi-platform office productivity suite
 *
 *  $RCSfile: GenericModelTest.java,v $
 *
 *  $Revision: 1.5 $
 *
 *  last change: $Author: ihi $ $Date: 2007-06-04 13:33:07 $
 *
 *  The Contents of this file are made available subject to
 *  the terms of GNU Lesser General Public License Version 2.1.
 *
 *
 *    GNU Lesser General Public License Version 2.1
 *    =============================================
 *    Copyright 2005 by Sun Microsystems, Inc.
 *    901 San Antonio Road, Palo Alto, CA 94303, USA
 *
 *    This library is free software; you can redistribute it and/or
 *    modify it under the terms of the GNU Lesser General Public
 *    License version 2.1, as published by the Free Software Foundation.
 *
 *    This library is distributed in the hope that it will be useful,
 *    but WITHOUT ANY WARRANTY; without even the implied warranty of
 *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *    Lesser General Public License for more details.
 *
 *    You should have received a copy of the GNU Lesser General Public
 *    License along with this library; if not, write to the Free Software
 *    Foundation, Inc., 59 Temple Place, Suite 330, Boston,
 *    MA  02111-1307  USA
 *
 ************************************************************************/
package mod._forms;
import com.sun.star.beans.NamedValue;
import com.sun.star.beans.PropertyValue;
import java.io.PrintWriter;
import lib.StatusException;
import lib.TestCase;
import lib.TestEnvironment;
import lib.TestParameters;
import util.DBTools;
import util.FormTools;
import util.WriterTools;
import com.sun.star.beans.XPropertySet;
import com.sun.star.drawing.XControlShape;
import com.sun.star.drawing.XShape;
import com.sun.star.form.XBoundComponent;
import com.sun.star.form.XForm;
import com.sun.star.form.XLoadable;
import com.sun.star.lang.XMultiServiceFactory;
import com.sun.star.lang.XComponent;
import com.sun.star.sdbc.XConnection;
import com.sun.star.sdbc.XResultSetUpdate;
import com.sun.star.sdb.XDocumentDataSource;
import com.sun.star.text.XTextDocument;
import com.sun.star.uno.AnyConverter;
import com.sun.star.uno.Type;
import com.sun.star.uno.UnoRuntime;
import com.sun.star.uno.XInterface;
import com.sun.star.util.XCloseable;
import java.util.ArrayList;
import lib.Status;
import util.utils;
/**
* Test for object which is represented by service
* com.sun.star.form.component.DateField. 
* Object implements the following interfaces : *
com::sun::star::io::XPersistObjectcom::sun::star::awt::UnoControlDateFieldModelcom::sun::star::form::XResetcom::sun::star::form::XBoundComponentcom::sun::star::form::FormComponentcom::sun::star::form::component::DateFieldcom::sun::star::beans::XFastPropertySetcom::sun::star::beans::XMultiPropertySetcom::sun::star::form::XUpdateBroadcastercom::sun::star::form::DataAwareControlModelcom::sun::star::beans::XPropertyStatecom::sun::star::form::FormControlModelcom::sun::star::container::XNamedcom::sun::star::lang::XComponentcom::sun::star::lang::XEventListenercom::sun::star::beans::XPropertyAccesscom::sun::star::beans::XPropertyContainercom::sun::star::beans::XPropertySetcom::sun::star::form::XLoadListenercom::sun::star::container::XChild
* This object test  is NOT  designed to be run in several
* threads concurently.
* @see com.sun.star.io.XPersistObject
* @see com.sun.star.awt.UnoControlDateFieldModel
* @see com.sun.star.form.XReset
* @see com.sun.star.form.XBoundComponent
* @see com.sun.star.form.FormComponent
* @see com.sun.star.form.component.DateField
* @see com.sun.star.beans.XFastPropertySet
* @see com.sun.star.beans.XMultiPropertySet
* @see com.sun.star.form.XUpdateBroadcaster
* @see com.sun.star.form.DataAwareControlModel
* @see com.sun.star.beans.XPropertyState
* @see com.sun.star.form.FormControlModel
* @see com.sun.star.container.XNamed
* @see com.sun.star.lang.XComponent
* @see com.sun.star.lang.XEventListener
* @see com.sun.star.beans.XPropertyAccess
* @see com.sun.star.beans.XPropertyContainer
* @see com.sun.star.beans.XPropertySet
* @see com.sun.star.form.XLoadListener
* @see com.sun.star.container.XChild
* @see ifc.io._XPersistObject
* @see ifc.awt._UnoControlDateFieldModel
* @see ifc.form._XReset
* @see ifc.form._XBoundComponent
* @see ifc.form._FormComponent
* @see ifc.form.component._DateField
* @see ifc.beans._XFastPropertySet
* @see ifc.beans._XMultiPropertySet
* @see ifc.form._XUpdateBroadcaster
* @see ifc.form._DataAwareControlModel
* @see ifc.beans._XPropertyState
* @see ifc.form._FormControlModel
* @see ifc.container._XNamed
* @see ifc.lang._XComponent
* @see ifc.lang._XEventListener
* @see ifc.beans._XPropertySet
* @see ifc.form._XLoadListener
* @see ifc.container._XChild
*/
public class GenericModelTest extends TestCase {
    private static XTextDocument m_xTextDoc;
    private static Object m_dbSrc = null;
    private static DBTools.DataSourceInfo m_srcInf = null;
    /**
     * This is the name of the Data Base which the test uses: "APITestDatabase"
     */
    protected final static String m_dbSourceName = "APITestDatabase";
    protected final static String m_TestDB = "TestDB";
    private DBTools m_dbTools = null;
    private static boolean m_ConnectionColsed = false;
    /**
     * descibes the kind of the shape which should be created.
     * Example: m_kindOfshape=DateFiled
     */
    public static String m_kindOfControl = null;
    /**
     * If your object needs some special propery values you can specify them with this
     * ArrayList. You have to add a NamedValue to this list.
     * Example:
     * NamedValue myProp = new NamedValue();
     * myProp.Name = "Test";
     * myProp.Value = "My special Value";
     * m_propertiesToSet.add(myProp);
     */
    public static ArrayList m_propertiesToSet = new ArrayList();
    /**
     * This variable contains the name of the property which should be changed while
     * interface com::sun::star::form::XUpdateBroadcaster is tested. The
     * interface test needs the ObjectRelation
     * "XUpdateBroadcaster.Checker" which is a ifc.form._XUpdateBroadcaster.UpdateChecker.
     * @see ifc.form._XUpdateBroadcaster.UpdateChecker
     * @see ifc.form._XUpdateBroadcaster
     */
    public static String m_ChangePropertyName = null;
    /**
     * This variable contains the value the property should be set while
     * interface com::sun::star::form::XUpdateBroadcaster is tested.
     * The interface test needs the ObjectRelation
     * "XUpdateBroadcaster.Checker" which is a ifc.form._XUpdateBroadcaster.UpdateChecker.
     * Normaly the Checker uses util.ValueChanger to change
     * the value of the property. If the current of this property is NULL the
     * ValueChanger is unable to change the value. In this case the value
     * of this variable was used.
     */
    public static Object m_ChangePropertyValue = null;
    /**
     * This variable contains the implelemtation name of the object.
     */
    public static String m_ObjectName = null;
    /**
     * For local implementaions of Checker this variable contains the
     * FormLoader
     */
    protected static XLoadable m_XFormLoader = null;
    /**
     * For local implementaions of Checker this variable contains the
     * XPropertySet
     */
    protected static XPropertySet m_XPS = null;
    /**
     * For local implementaions of Checker this variable contains the
     * Control
     */
    protected static XInterface m_XCtrl = null;
    /**
     * The insterface test of ifc.form._DataWareControlModel expects an
     * object relation 'LC'. This is a XControlModel of a shape.
     * This variable contains the kind of shape to create for the interface test,
     * f.e. "FixedText"
     * @see ifc.form._DataAwareControlModel
     */
    protected static String m_LCShape_Type = null;
    protected static String m_XPropertyAccess_propertyToChange = "HelpText";
    protected static String m_XPropertyContainer_propertyNotRemovable = "HelpText";
    /**
     * If this variable is true some more debug info was logged. It was setted by the parameter variable
     * debug_is_active
     */
    protected static boolean debug = false;
    /**
     * Creates Writer document where controls are placed.
     * @param tParam the test paremter
     * @param log the log writer
     */
    protected void initialize(TestParameters tParam, PrintWriter log) {
        log.println("creating a textdocument");
        m_xTextDoc = WriterTools.createTextDoc(((XMultiServiceFactory) tParam.getMSF()));
        m_ConnectionColsed = false;
        debug = tParam.getBool(util.PropertyName.DEBUG_IS_ACTIVE);
        m_propertiesToSet.clear();
    }
    /**
     * close the connection
     * close the data source
     * close the document
     * revoke the data source
     * @param tParam the test parameter
     * @param log the log writer
     */
    protected void cleanup(TestParameters tParam, PrintWriter log) {
        log.println("closing connection...");
        // some interface tests call cleanup to reset the environment. If such
        // a test is the last one cleanup was called twice. The second call
        // causes then nasty exceptions...
        if (m_ConnectionColsed) return;
        try {
            XForm myForm = (XForm) AnyConverter.toObject(new Type(XForm.class),
                                                 (FormTools.getForms(
                                                         WriterTools.getDrawPage(
                                                                 m_xTextDoc)))
                                                     .getByName("Standard"));
            if (debug){
                if (myForm == null){
                    log.println("ERROR: could not get 'Standard' from drawpage!");
                }
                log.println("the draw page contains folowing elemtens:");
                String[] elements = FormTools.getForms(WriterTools.getDrawPage(m_xTextDoc)).getElementNames();
                for (int i = 0; i< elements.length; i++){
                    log.println("Element[" + i + "] :" + elements[i]);
                }
            }
            XPropertySet xSetProp = (XPropertySet) UnoRuntime.queryInterface(
                                            XPropertySet.class, myForm);
            XConnection connection = (XConnection) AnyConverter.toObject(
                                 new Type(XConnection.class),
                                 xSetProp.getPropertyValue("ActiveConnection"));
            if (debug && connection == null){
                log.println("ERROR: could not get property 'ActiveConnection' from the XForm");
            }
            connection.close();
        } catch (Exception e) {
            log.println("ERROR: Can't close the connection: " + e.toString());
        }
        log.println("closing data source...");
        try {
            XCloseable closer = (XCloseable) UnoRuntime.queryInterface(
                                        XCloseable.class, m_dbSrc);
            if ( closer == null )
            {
                XDocumentDataSource dataSource = (XDocumentDataSource)UnoRuntime.queryInterface(
                    XDocumentDataSource.class, m_dbSrc);
                if ( dataSource != null )
                    closer = (XCloseable) UnoRuntime.queryInterface(
                        XCloseable.class, dataSource.getDatabaseDocument() );
            }
            if (debug && closer==null){
                log.println("ERROR: couldn't get 'XCloseable' from DataSource");
            }
            closer.close(true);
        } catch (com.sun.star.util.CloseVetoException e) {
            log.println("ERROR: couldn't close data source: " + e.toString());
        } catch (com.sun.star.lang.DisposedException e) {
            log.println("ERROR: couldn't close data source: " + e.toString());
        } catch (Exception e) {
            log.println("ERROR: couldn't close data source: " + e.toString());
        }
        log.println("disposing data source...");
        try {
            XComponent dataSourceComp = (XComponent)UnoRuntime.queryInterface(
                XComponent.class, m_dbSrc);
            dataSourceComp.dispose();
        }
        catch (Exception e) {
            log.println("couldn't dispose the data source");
        }
        log.println("closing document...");
        try {
            XCloseable closer = (XCloseable) UnoRuntime.queryInterface(
                                        XCloseable.class, m_xTextDoc);
            closer.close(true);
        } catch (com.sun.star.util.CloseVetoException e) {
            log.println("ERROR: couldn't close document: " + e.toString());
        } catch (com.sun.star.lang.DisposedException e) {
            log.println("ERROR: couldn't close document: " + e.toString());
        } catch (Exception e) {
            log.println("ERROR: couldn't close document: " + e.toString());
        }
        log.println("revoking data source...");
        try {
            m_dbTools.revokeDB(m_dbSourceName);
        } catch (com.sun.star.container.NoSuchElementException e){
        } catch (com.sun.star.uno.Exception e) {
            log.println("ERROR: Error while object test cleaning up: " + e.toString());
        }
        m_ConnectionColsed = true;
    }
    /**
     * Creating a Testenvironment for the interfaces to be tested.
     * First TestDB database is registered.
     * Creates DateField in the Form, then binds it to TestDB
     * database and returns Field's control. 
* Object relations created : *
'OBJNAME' for
     *      {@link ifc.io._XPersistObject} : name of service which is
     *    represented by this object. 'LC' for {@link ifc.form._DataAwareControlModel}.
     *    Specifies the value for LabelControl property. It is
     *    FixedText component added to the document.'FL' for
     *      {@link ifc.form._DataAwareControlModel} interface.
     *    Specifies XLoadable implementation which connects form to
     *    the data source.'XUpdateBroadcaster.Checker' : 
     *    _XUpdateBroadcaster.UpdateChecker interface implementation
     *    which can update, commit data and check if the data was successfully
     *    commited.'DataAwareControlModel.NewFieldName' : for
     *    com.sun.star.form.DataAwareControlModel service
     *    which contains new name of the field ('_DATE') to bind control to.
     *  'XFastPropertySet.ExcludeProps' : for
     *    com.sun.star.beans.XFastPropertySet interface
     *    the property FormatKey can have only restricted set of values.
     *