All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.codename1.cloud.CloudObject Maven / Gradle / Ivy

/*
 * Copyright (c) 2012, Codename One and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Codename One designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *  
 * This code 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 General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 * 
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 * 
 * Please contact Codename One through http://www.codenameone.com/ if you 
 * need additional information or have any questions.
 */
package com.codename1.cloud;

import com.codename1.cloud.BindTarget;
import com.codename1.io.Externalizable;
import com.codename1.io.Util;
import com.codename1.ui.Component;
import com.codename1.ui.Container;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Vector;

/**
 * A cloud object can be persisted to the cloud or locally
 * it is a set of key/value pairs that can be either strings
 * or numbers. There is a 512 character limit on string length!
 * Notice: keys starting with CN1 are reserved for internal usage!
 *
 * @author Shai Almog
 * @deprecated the cloud storage API is no longer supported, we recommend switching to a solution such as parse4cn1
 */
public final class CloudObject implements Externalizable {
    /**
     * Indicates the state of the current object, a new object never persisted
     */
    public static final int STATUS_NEW = 0;

    /**
     * Indicates the state of the current object, an object that is in sync with the database as far as
     * the client code is aware (the client code doesn't check!)
     */
    public static final int STATUS_COMMITTED = 1;

    /**
     * Indicates the state of the current object, a locally modified object that wasn't committed yet
     */
    public static final int STATUS_MODIFIED = 2;

    /**
     * Indicates the state of the current object, an object in the process of committing
     */
    public static final int STATUS_COMMIT_IN_PROGRESS = 3;

    /**
     * Indicates the state of the current object, an object that is in the process of deletion
     */
    public static final int STATUS_DELETE_IN_PROGRESS = 4;

    /**
     * Indicates the state of the current object, a deleted object
     */
    public static final int STATUS_DELETED = 5;

    /**
     * Indicates the state of the current object, an object in the process of refresh
     */
    public static final int STATUS_REFRESH_IN_PROGRESS = 6;
    
    /**
     * A world visible/modifiable object!
     */
    public static final int ACCESS_PUBLIC = 1;

    /**
     * A world visible object! Can only be modified by its creator.
     */
    public static final int ACCESS_PUBLIC_READ_ONLY = 2;

    /**
     * An application visible/modifiable object!
     */
    public static final int ACCESS_APPLICATION = 3;

    /**
     * An application scope readable object! Can only be modified by its creator
     */
    public static final int ACCESS_APPLICATION_READ_ONLY = 4;
    
    /**
     * An object that can only be viewed or modified by its creator
     */
    public static final int ACCESS_PRIVATE = 5;
    
    
    /**
     * Changes to the bound property won't be reflected into the bound cloud object until commit binding is invoked
     */
    public static final int BINDING_DEFERRED = 1;

    /**
     * Changes to the bound property will be reflected instantly into the cloud object
     */
    public static final int BINDING_IMMEDIATE = 2;

    /**
     * Changes to the bound property will be reflected instantly into the cloud object and the object would
     * be saved immediately (not committed!).
     */
    public static final int BINDING_AUTO_SAVE = 3;

    private static Hashtable custom = new Hashtable();
    private Hashtable values = new Hashtable();
    private Hashtable deferedValues;
    
    private String cloudId;
    private long lastModified;
    private int status;
    private boolean owner = true;
    private int accessPermissions = ACCESS_PRIVATE;
    
    /**
     * Default constructor for externalization purposes only!
     */
    public CloudObject() {}
    
    /**
     * Constructor
     * @param type the type of the object
     */
    public CloudObject(String type) {
        values.put(CloudStorage.TYPE_FIELD, type);
    }
    
    /**
     * Create an object with different permissions settings
     * 
     * @param type the type of the object
     * @param permissions one of the ACCESS_* values
     */
    public CloudObject(String type, int permissions) {
        accessPermissions = permissions;
        values.put(CloudStorage.TYPE_FIELD, type);
    }
    
    CloudObject(int permissions) {
        accessPermissions = permissions;
    }
    
    
    /**
     * Returns one of the status constants in this class
     * @return the status of the object against the cloud
     */
    public int getStatus() {
        return status;
    }
    
    void setStatus(int s) {
        status = s;
    }
    
    void setValues(Hashtable values) {
        this.values = values;
    }
    
    Hashtable getValues() {
        return values;
    }
     
    /**
     * Set the type of the object
     * @param type the type of the field
     */
    public void setType(String type) {
        setString(CloudStorage.TYPE_FIELD, type);
    }

    /**
     * Returns the type of the object
     * @return the type of the object
     */
    public String getType() {
        return getString(CloudStorage.TYPE_FIELD);
    }
    
    /**
     * Only indexed values can be queried and sorted
     * @param index the index which must be a value between 1 and 10.
     * @param value the value for the given index
     */
    public void setIndexString(int index, String value) {
        if(index > 10 || index < 1) {
            throw new IllegalArgumentException("Invalid index: " + index);
        }
        setString(CloudStorage.INDEX_FIELD + index, value);
    }
    
    /**
     * Returns the index value for the given index number
     * 
     * @param index the index number
     * @return the value of this entry for that index as a String
     */
    public String getIndexString(int index) {
        return getString(CloudStorage.INDEX_FIELD + index);
    }
    
    /**
     * Only indexed values can be queried and sorted
     * @param index the index which must be a value between 1 and 10.
     * @param value the value for the given index
     */
    public void setIndexLong(int index, long value) {
        if(index > 10 || index < 1) {
            throw new IllegalArgumentException("Invalid index: " + index);
        }
        setLong(CloudStorage.INDEX_FIELD + index, value);
    }
    
    /**
     * Returns the index value for the given index number
     * 
     * @param index the index number
     * @return the value of this entry for that index as a Long value
     */
    public Long getIndexLong(int index) {
        return getLong(CloudStorage.INDEX_FIELD + index);
    }

    
    /**
     * Only indexed values can be queried and sorted
     * @param index the index which must be a value between 1 and 10.
     * @param value the value for the given index
     */
    public void setIndexDouble(int index, double value) {
        if(index > 10 || index < 1) {
            throw new IllegalArgumentException("Invalid index: " + index);
        }
        setDouble(CloudStorage.INDEX_FIELD + index, value);
    }
    
    /**
     * Returns the index value for the given index number
     * 
     * @param index the index number
     * @return the value of this entry for that index as a Double value
     */
    public Double getIndexDouble(int index) {
        return getDouble(CloudStorage.INDEX_FIELD + index);
    }
    
    /**
     * Returns true if this object is owned by me or is world writeable
     * @return ownership status
     */
    public boolean isOwner() {
        return owner;
    }
    
    /**
     * The object id is a unique key that allows you to find an object that was persisted in the 
     * store (a primary key). When it is null the object is effectively unsynchronized!
     * @return the object id or null for an object that isn't fully persisted yet
     */
    public String getCloudId() {
        return cloudId;
    }
    
    void setCloudId(String cloudId) {
        this.cloudId = cloudId;
    }
    
    /**
     * Indicates the last modification date for the object
     * @return the last modification date
     */
    public long getLastModified() {
        return lastModified;
    }
    
    void setLastModified(long lastModified) {
        this.lastModified = lastModified;
    }
    
    /**
     * Allows us to extract an object from the cloud object without knowing its type in advance
     * or whether it exists
     * @param key the key for the object
     * @return the value of the object
     */
    public Object getObject(String key) {
        Object o = values.get(key);
        if(o == null) {
            CustomProperty cp = custom.get(key);
            if(cp != null) {
                return cp.propertyValue(this, key);
            }
        }
        return o;
    }
    
    /**
     * Install a custom property on the given property name
     * 
     * @param key the key on which to install the custom property
     * @param cp the custom property implementation
     */
    public static void setCustomProperty(String key, CustomProperty cp) {
        if(cp == null) {
            custom.remove(key);
        } else {
            custom.put(key, cp);
        }
    }
    
    /**
     * Sets a value that can be no more than 512 characters
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setString(String key, String value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        if(value == null) {
            values.remove(key);
            return;
        }
        if(value.length() > 512) {
            throw new IllegalArgumentException("String too long!");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a string value
     */
    public String getString(String key) {
        return (String)getObject(key);
    }
    
    /**
     * Delete an entry within the object
     * @param key the key to remove from the object
     */
    public void remove(String key) {
        values.remove(key);
    }
    

    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setLong(String key, long value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, new Long(value));
    }
    
    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setLong(String key, Long value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a long value
     */
    public Long getLong(String key) {
        Object o = getObject(key);
        if(o instanceof Integer) {
            return new Long(((Integer)o).intValue());
        }
        return (Long)o;
    }

    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setInteger(String key, int value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, new Integer(value));
    }
    
    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setInteger(String key, Integer value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a value
     */
    public Integer getInteger(String key) {
        Object o = getObject(key);
        if(o instanceof Long) {
            return new Integer((int)((Long)o).longValue());
        }
        return (Integer)o;
    }

    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setDouble(String key, double value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, new Double(value));
    }
    
    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setDouble(String key, Double value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a value
     */
    public Double getDouble(String key) {
        return (Double)getObject(key);
    }

    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setFloat(String key, float value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, new Float(value));
    }
    
    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setFloat(String key, Float value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a value
     */
    public Float getFloat(String key) {
        Object o = getObject(key);
        if(o instanceof Double) {
            return new Float(((Double)o).floatValue());
        }
        return (Float)o;
    }

    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setBoolean(String key, boolean value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, new Boolean(value));
    }
    
    /**
     * Sets a value 
     * 
     * @param key the key for the given value
     * @param value the value
     */
    public void setBoolean(String key, Boolean value) {
        if(!owner) {
            throw new RuntimeException("Read only object, you are not the owner");
        }
        status = STATUS_MODIFIED;
        values.put(key, value);
    }
    
    /**
     * Returns the value for the given key
     * @param key the key 
     * @return a value
     */
    public Boolean getBoolean(String key) {
        return (Boolean)getObject(key);
    }

    /**
     * {@inheritDoc}
     */
    public int getVersion() {
        return 1;
    }
    
    /**
     * {@inheritDoc}
     */
    public void externalize(DataOutputStream out) throws IOException {
        Util.writeUTF(cloudId, out);
        out.writeBoolean(owner);
        out.writeByte(getAccessPermissions());
        out.writeLong(lastModified);
        out.writeInt(status);
        Util.writeObject(values, out);
    }

    
    public String getObjectId() {
        return "CloudObject";
    }

    /**
     * {@inheritDoc}
     */
    public void internalize(int version, DataInputStream in) throws IOException {
        cloudId = Util.readUTF(in);
        owner = in.readBoolean();
        accessPermissions = in.readByte();
        lastModified = in.readLong();
        status = in.readInt();
        values = (Hashtable)Util.readObject(in);
    }

    /**
     * The access permissions for an object can only be changed for an object in which
     * the current user is the owner
     * @return the accessPermissions
     */
    public int getAccessPermissions() {
        return accessPermissions;
    }
    
    /**
     * {@inheritDoc}
     */
    public boolean equals(Object o) {
        if(!(o instanceof CloudObject)) {
            return false;
        }
        CloudObject cl = (CloudObject)o;
        if(cloudId == null && cl.cloudId == null) {
            return values.equals(cl.values);
        }
        if(cloudId == null || cl.cloudId == null) {
            return false;
        }
        return cloudId.equals(cl.cloudId);
    }

    /**
     * {@inheritDoc}
     */
    public int hashCode() {
        if(cloudId != null) {
            return cloudId.hashCode();
        }
        return 0;
    }
    
    /**
     * Binds a UI tree to the cloud object so its values automatically update in the cloud object
     * 
     * @param ui the component tree to bind
     * @param defer bind settings whether to defer the binding which requires developers to explicitly commit
     * the binding to perform the changes
     * @param objectLead if set to true the UI property is initialized from values in the CloudObject, if false
     * the cloud object property is initialized from the UI
     */
    public void bindTree(Container ui, int defer, boolean objectLead) {
        int componentCount = ui.getComponentCount();
        for(int iter = 0 ; iter < componentCount ; iter++) {
            Component c = ui.getComponentAt(iter);
            if(c instanceof Container) {
                bindTree((Container)c, defer, objectLead);
                continue;
            }
            
            String bind = c.getCloudBoundProperty();
            if(bind != null && bind.length() > 0) {
                String attributeName = c.getCloudDestinationProperty();            
                if(attributeName != null) {
                    bindProperty(c, bind, attributeName, defer, objectLead);
                }
            }
        }
    }
    
    /**
     * Clears the binding to this component tree
     * @param ui the container whose children might be bound
     */
    public void unbindTree(Container ui) {
        int componentCount = ui.getComponentCount();
        for(int iter = 0 ; iter < componentCount ; iter++) {
            Component c = ui.getComponentAt(iter);
            if(c instanceof Container) {
                unbindTree((Container)c);
                continue;
            }
            
            String bind = c.getCloudBoundProperty();
            if(bind != null && bind.length() > 0) {
                String attributeName = c.getCloudDestinationProperty();            
                if(attributeName != null) {
                    unbindProperty(c, bind);
                }
            }
        }
    }
    
    /**
     * Binds a property value within the given component to this cloud object, this means that
     * when the component changes the cloud object changes unless deferred. If the defer flag is 
     * false all changes are stored in a temporary location and only "committed" once commitBindings()
     * is invoked.
     * @param cmp the component to bind
     * @param propertyName the name of the property in the bound component
     * @param attributeName the key within the cloud object
     * @param defer bind settings whether to defer the binding which requires developers to explicitly commit
     * the binding to perform the changes
     * @param objectLead if set to true the UI property is initialized from values in the CloudObject, if false
     * the cloud object property is initialized from the UI
     */
    public void bindProperty(Component cmp, final String propertyName, final String attributeName, final int defer, boolean objectLead) {
        if(objectLead) {
            Object val = values.get(attributeName);
            Object cmpVal = cmp.getBoundPropertyValue(propertyName);
            if(val == null) {
                if(cmpVal != null) {
                    cmp.setBoundPropertyValue(propertyName, null);
                }
            } else {
                if(cmpVal == null || !(val.equals(cmpVal))) {
                    cmp.setBoundPropertyValue(propertyName, val);
                }
            }
        } else {
            Object val = values.get(attributeName);
            Object cmpVal = cmp.getBoundPropertyValue(propertyName);
            if(cmpVal == null) {
                if(val != null) {
                    values.remove(attributeName);
                    status = STATUS_MODIFIED;
                }
            } else {
                if(val == null || !(val.equals(cmpVal))) {
                    values.put(attributeName, cmpVal);
                    status = STATUS_MODIFIED;
                }
            }
        }
        BindTarget target = new BindTarget() {
            public void propertyChanged(Component source, String propertyName, Object oldValue, Object newValue) {
                switch(defer) {
                    case BINDING_DEFERRED:
                        if(deferedValues == null) {
                            deferedValues = new Hashtable();
                        }
                        deferedValues.put(attributeName, newValue);
                        break;
                    case BINDING_IMMEDIATE:
                        values.put(attributeName, newValue);
                        status = STATUS_MODIFIED;
                        break;
                    case BINDING_AUTO_SAVE:
                        values.put(attributeName, newValue);
                        status = STATUS_MODIFIED;
                        CloudStorage.getInstance().save(CloudObject.this);
                        break;
                }
            }
        };
        cmp.bindProperty(propertyName, target);
        cmp.putClientProperty("CN1Bind" + propertyName, target);
    }
    
    /**
     * Releases the binding for the specific property name
     * @param cmp the component
     * @param propertyName the name of the property
     */
    public void unbindProperty(Component cmp, String propertyName) {
        BindTarget t = (BindTarget)cmp.getClientProperty("CN1Bind" + propertyName);
        cmp.unbindProperty(propertyName, t);;
    }
    
    /**
     * If deferred changes exist this method applies these changes to the data
     */
    public void commitBinding() {
        if(deferedValues != null && deferedValues.size() > 0) {
            Enumeration en = deferedValues.keys();
            while(en.hasMoreElements()) {
                Object k = en.nextElement();
                values.put(k, deferedValues.get(k));
            }
            deferedValues = null;
        }
    }
    
    /**
     * If deferred changes exist this method discards such values
     */
    public void cancelBinding() {
        deferedValues = null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy