159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta/*
259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * Copyright (c) 2009-2010 jMonkeyEngine
359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * All rights reserved.
459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * Redistribution and use in source and binary forms, with or without
659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * modification, are permitted provided that the following conditions are
759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * met:
859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * * Redistributions of source code must retain the above copyright
1059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *   notice, this list of conditions and the following disclaimer.
1159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
1259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * * Redistributions in binary form must reproduce the above copyright
1359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *   notice, this list of conditions and the following disclaimer in the
1459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *   documentation and/or other materials provided with the distribution.
1559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
1659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * * Neither the name of 'jMonkeyEngine' nor the names of its contributors
1759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *   may be used to endorse or promote products derived from this software
1859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *   without specific prior written permission.
1959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
2059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
2159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
2259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
2359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
2459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
2559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
2659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
2759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
2859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
2959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
3059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
3159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta */
3259b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartapackage com.jme3.material;
3359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
3459b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.asset.TextureKey;
3559b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.export.*;
3659b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.math.*;
3759b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.renderer.GL1Renderer;
3859b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.renderer.Renderer;
3959b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.shader.VarType;
4059b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.texture.Texture;
4159b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport com.jme3.texture.Texture.WrapMode;
4259b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartaimport java.io.IOException;
4359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
4459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta/**
4559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * Describes a material parameter. This is used for both defining a name and type
4659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * as well as a material parameter value.
4759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta *
4859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta * @author Kirill Vainer
4959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta */
5059b2e6871c65f58fdad78cd7229c292f6a177578Scott Bartapublic class MatParam implements Savable, Cloneable {
5159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
5259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    protected VarType type;
5359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    protected String name;
5459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    protected String prefixedName;
5559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    protected Object value;
5659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    protected FixedFuncBinding ffBinding;
5759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
5859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
5959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Create a new material parameter. For internal use only.
6059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
6159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public MatParam(VarType type, String name, Object value, FixedFuncBinding ffBinding) {
6259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.type = type;
6359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.name = name;
6459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.prefixedName = "m_" + name;
6559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.value = value;
6659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.ffBinding = ffBinding;
6759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
6859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
6959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
7059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Serialization only. Do not use.
7159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
7259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public MatParam() {
7359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
7459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
7559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
7659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the fixed function binding.
7759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *
7859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return the fixed function binding.
7959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
8059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public FixedFuncBinding getFixedFuncBinding() {
8159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return ffBinding;
8259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
8359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
8459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
8559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the material parameter type.
8659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *
8759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return the material parameter type.
8859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
8959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public VarType getVarType() {
9059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return type;
9159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
9259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
9359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
9459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the name of the material parameter.
9559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return the name of the material parameter.
9659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
9759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public String getName() {
9859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return name;
9959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
10059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
10159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
10259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the name with "m_" prefixed to it.
10359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *
10459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return the name with "m_" prefixed to it
10559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
10659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public String getPrefixedName() {
10759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return prefixedName;
10859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
10959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
11059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
11159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Used internally
11259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @param name
11359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
11459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    void setName(String name) {
11559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.name = name;
11659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.prefixedName = "m_" + name;
11759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
11859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
11959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
12059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the value of this material parameter.
12159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * <p>
12259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Material parameters that are used for material definitions
12359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * will not have a value, unless there's a default value declared
12459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * in the definition.
12559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *
12659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return the value of this material parameter.
12759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
12859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public Object getValue() {
12959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return value;
13059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
13159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
13259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
13359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Sets the value of this material parameter.
13459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * <p>
13559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * It is assumed the value is of the same {@link MatParam#getVarType() type}
13659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * as this material parameter.
13759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *
13859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @param value the value of this material parameter.
13959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
14059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public void setValue(Object value) {
14159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        this.value = value;
14259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
14359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
14459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    void apply(Renderer r, Technique technique) {
14559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        TechniqueDef techDef = technique.getDef();
14659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        if (techDef.isUsingShaders()) {
14759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            technique.updateUniformParam(getPrefixedName(), getVarType(), getValue(), true);
14859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
14959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        if (ffBinding != null && r instanceof GL1Renderer) {
15059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            ((GL1Renderer) r).setFixedFuncBinding(ffBinding, getValue());
15159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
15259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
15359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
15459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    /**
15559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Returns the material parameter value as it would appear in a J3M
15659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * file. E.g.<br/>
15759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * <code>
15859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * MaterialParameters {<br/>
15959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     *     ABC : 1 2 3 4<br/>
16059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * }<br/>
16159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * </code>
16259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * Assuming "ABC" is a Vector4 parameter, then the value
16359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * "1 2 3 4" would be returned by this method.
16459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * <br/><br/>
16559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     * @return material parameter value as it would appear in a J3M file.
16659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta     */
16759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public String getValueAsString() {
16859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        switch (type) {
16959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Boolean:
17059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Float:
17159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Int:
17259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return value.toString();
17359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector2:
17459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                Vector2f v2 = (Vector2f) value;
17559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return v2.getX() + " " + v2.getY();
17659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta/*
17759b2e6871c65f58fdad78cd7229c292f6a177578Scott BartaThis may get used at a later point of time
17859b2e6871c65f58fdad78cd7229c292f6a177578Scott BartaWhen arrays can be inserted in J3M files
17959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
18059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector2Array:
18159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                Vector2f[] v2Arr = (Vector2f[]) value;
18259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                String v2str = "";
18359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                for (int i = 0; i < v2Arr.length ; i++) {
18459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    v2str += v2Arr[i].getX() + " " + v2Arr[i].getY() + "\n";
18559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
18659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return v2str;
18759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta*/
18859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector3:
18959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                Vector3f v3 = (Vector3f) value;
19059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return v3.getX() + " " + v3.getY() + " " + v3.getZ();
19159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta/*
19259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector3Array:
19359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                Vector3f[] v3Arr = (Vector3f[]) value;
19459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                String v3str = "";
19559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                for (int i = 0; i < v3Arr.length ; i++) {
19659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    v3str += v3Arr[i].getX() + " "
19759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                            + v3Arr[i].getY() + " "
19859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                            + v3Arr[i].getZ() + "\n";
19959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
20059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return v3str;
20159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector4Array:
20259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                // can be either ColorRGBA, Vector4f or Quaternion
20359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                if (value instanceof Vector4f) {
20459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    Vector4f[] v4arr = (Vector4f[]) value;
20559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    String v4str = "";
20659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    for (int i = 0; i < v4arr.length ; i++) {
20759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                        v4str += v4arr[i].getX() + " "
20859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + v4arr[i].getY() + " "
20959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + v4arr[i].getZ() + " "
21059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + v4arr[i].getW() + "\n";
21159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    }
21259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return v4str;
21359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else if (value instanceof ColorRGBA) {
21459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    ColorRGBA[] colorArr = (ColorRGBA[]) value;
21559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    String colStr = "";
21659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    for (int i = 0; i < colorArr.length ; i++) {
21759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                        colStr += colorArr[i].getRed() + " "
21859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + colorArr[i].getGreen() + " "
21959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + colorArr[i].getBlue() + " "
22059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + colorArr[i].getAlpha() + "\n";
22159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    }
22259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return colStr;
22359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else if (value instanceof Quaternion) {
22459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    Quaternion[] quatArr = (Quaternion[]) value;
22559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    String quatStr = "";
22659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    for (int i = 0; i < quatArr.length ; i++) {
22759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                        quatStr += quatArr[i].getX() + " "
22859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + quatArr[i].getY() + " "
22959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + quatArr[i].getZ() + " "
23059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                                + quatArr[i].getW() + "\n";
23159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    }
23259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return quatStr;
23359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else {
23459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    throw new UnsupportedOperationException("Unexpected Vector4Array type: " + value);
23559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
23659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta*/
23759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Vector4:
23859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                // can be either ColorRGBA, Vector4f or Quaternion
23959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                if (value instanceof Vector4f) {
24059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    Vector4f v4 = (Vector4f) value;
24159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return v4.getX() + " " + v4.getY() + " "
24259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                            + v4.getZ() + " " + v4.getW();
24359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else if (value instanceof ColorRGBA) {
24459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    ColorRGBA color = (ColorRGBA) value;
24559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return color.getRed() + " " + color.getGreen() + " "
24659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                            + color.getBlue() + " " + color.getAlpha();
24759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else if (value instanceof Quaternion) {
24859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    Quaternion quat = (Quaternion) value;
24959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    return quat.getX() + " " + quat.getY() + " "
25059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                            + quat.getZ() + " " + quat.getW();
25159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                } else {
25259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    throw new UnsupportedOperationException("Unexpected Vector4 type: " + value);
25359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
25459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Texture2D:
25559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Texture3D:
25659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case TextureArray:
25759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case TextureBuffer:
25859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case TextureCubeMap:
25959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                Texture texVal = (Texture) value;
26059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                TextureKey texKey = (TextureKey) texVal.getKey();
26159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                if (texKey == null){
26259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    throw new UnsupportedOperationException("The specified MatParam cannot be represented in J3M");
26359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
26459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
26559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                String ret = "";
26659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                if (texKey.isFlipY()) {
26759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    ret += "Flip ";
26859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
26959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                if (texVal.getWrap(Texture.WrapAxis.S) == WrapMode.Repeat) {
27059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                    ret += "Repeat ";
27159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                }
27259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
27359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return ret + texKey.getName();
27459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            default:
27559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                return null; // parameter type not supported in J3M
27659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
27759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
27859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
27959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    @Override
28059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public MatParam clone() {
28159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        try {
28259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            MatParam param = (MatParam) super.clone();
28359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            return param;
28459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        } catch (CloneNotSupportedException ex) {
28559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            throw new AssertionError();
28659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
28759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
28859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
28959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public void write(JmeExporter ex) throws IOException {
29059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        OutputCapsule oc = ex.getCapsule(this);
29159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        oc.write(type, "varType", null);
29259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        oc.write(name, "name", null);
29359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        oc.write(ffBinding, "ff_binding", null);
29459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        if (value instanceof Savable) {
29559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            Savable s = (Savable) value;
29659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            oc.write(s, "value_savable", null);
29759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        } else if (value instanceof Float) {
29859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            Float f = (Float) value;
29959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            oc.write(f.floatValue(), "value_float", 0f);
30059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        } else if (value instanceof Integer) {
30159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            Integer i = (Integer) value;
30259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            oc.write(i.intValue(), "value_int", 0);
30359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        } else if (value instanceof Boolean) {
30459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            Boolean b = (Boolean) value;
30559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            oc.write(b.booleanValue(), "value_bool", false);
30659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
30759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
30859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
30959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public void read(JmeImporter im) throws IOException {
31059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        InputCapsule ic = im.getCapsule(this);
31159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        type = ic.readEnum("varType", VarType.class, null);
31259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        name = ic.readString("name", null);
31359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        ffBinding = ic.readEnum("ff_binding", FixedFuncBinding.class, null);
31459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        switch (getVarType()) {
31559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Boolean:
31659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                value = ic.readBoolean("value_bool", false);
31759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                break;
31859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Float:
31959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                value = ic.readFloat("value_float", 0f);
32059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                break;
32159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            case Int:
32259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                value = ic.readInt("value_int", 0);
32359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                break;
32459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            default:
32559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                value = ic.readSavable("value_savable", null);
32659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                break;
32759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
32859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
32959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
33059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    @Override
33159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public boolean equals(Object other) {
33259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        if (!(other instanceof MatParam)) {
33359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta            return false;
33459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        }
33559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
33659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        MatParam otherParam = (MatParam) other;
33759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return otherParam.type == type
33859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta                && otherParam.name.equals(name);
33959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
34059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
34159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    @Override
34259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public int hashCode() {
34359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        int hash = 5;
34459b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        hash = 17 * hash + (this.type != null ? this.type.hashCode() : 0);
34559b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        hash = 17 * hash + (this.name != null ? this.name.hashCode() : 0);
34659b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return hash;
34759b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
34859b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta
34959b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    @Override
35059b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    public String toString() {
35159b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta        return type.name() + " " + name + " : " + getValueAsString();
35259b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta    }
35359b2e6871c65f58fdad78cd7229c292f6a177578Scott Barta}
354