Merge branch 'master' of https://github.com/jMonkeyEngine/jmonkeyengine.git
commit
fa3ea41a8d
@ -0,0 +1,81 @@ |
||||
/* |
||||
* Copyright (c) 2016 jMonkeyEngine |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* |
||||
* * Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in the |
||||
* documentation and/or other materials provided with the distribution. |
||||
* |
||||
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||
* may be used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
*/ |
||||
|
||||
package com.jme3.util.clone; |
||||
|
||||
|
||||
/** |
||||
* Provides custom cloning for a particular object type. Once |
||||
* registered with the Cloner, this function object will be called twice |
||||
* for any cloned object that matches the class for which it was registered. |
||||
* It will first call cloneObject() to shallow clone the object and then call |
||||
* cloneFields() to deep clone the object's values. |
||||
* |
||||
* <p>This two step process is important because this is what allows |
||||
* circular references in the cloned object graph.</p> |
||||
* |
||||
* @author Paul Speed |
||||
*/ |
||||
public interface CloneFunction<T> { |
||||
|
||||
/** |
||||
* Performs a shallow clone of the specified object. This is similar |
||||
* to the JmeCloneable.clone() method in semantics and is the first part |
||||
* of a two part cloning process. Once the shallow clone is created, it |
||||
* is cached and CloneFunction.cloneFields() is called. In this way, |
||||
* the CloneFunction interface can completely take over the JmeCloneable |
||||
* style cloning for an object that doesn't otherwise implement that interface. |
||||
* |
||||
* @param cloner The cloner performing the cloning operation. |
||||
* @param original The original object that needs to be cloned. |
||||
*/ |
||||
public T cloneObject( Cloner cloner, T original ); |
||||
|
||||
|
||||
/** |
||||
* Performs a deep clone of the specified clone's fields. This is similar |
||||
* to the JmeCloneable.cloneFields() method in semantics and is the second part |
||||
* of a two part cloning process. Once the shallow clone is created, it |
||||
* is cached and CloneFunction.cloneFields() is called. In this way, |
||||
* the CloneFunction interface can completely take over the JmeCloneable |
||||
* style cloning for an object that doesn't otherwise implement that interface. |
||||
* |
||||
* @param cloner The cloner performing the cloning operation. |
||||
* @param clone The clone previously returned from cloneObject(). |
||||
* @param original The original object that was cloned. This is provided for |
||||
* the very special case where field cloning needs to refer to |
||||
* the original object. Mostly the necessary fields should already |
||||
* be on the clone. |
||||
*/ |
||||
public void cloneFields( Cloner cloner, T clone, T original ); |
||||
|
||||
} |
@ -0,0 +1,348 @@ |
||||
/* |
||||
* Copyright (c) 2016 jMonkeyEngine |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* |
||||
* * Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in the |
||||
* documentation and/or other materials provided with the distribution. |
||||
* |
||||
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||
* may be used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
*/ |
||||
|
||||
package com.jme3.util.clone; |
||||
|
||||
import java.lang.reflect.Array; |
||||
import java.lang.reflect.InvocationTargetException; |
||||
import java.lang.reflect.Method; |
||||
import java.util.HashMap; |
||||
import java.util.IdentityHashMap; |
||||
import java.util.Map; |
||||
import java.util.concurrent.ConcurrentHashMap; |
||||
|
||||
/** |
||||
* A deep clone utility that provides similar object-graph-preserving |
||||
* qualities to typical serialization schemes. An internal registry |
||||
* of cloned objects is kept to be used by other objects in the deep |
||||
* clone process that implement JmeCloneable. |
||||
* |
||||
* <p>By default, objects that do not implement JmeCloneable will |
||||
* be treated like normal Java Cloneable objects. If the object does |
||||
* not implement the JmeCloneable or the regular JDK Cloneable interfaces |
||||
* AND has no special handling defined then an IllegalArgumentException |
||||
* will be thrown.</p> |
||||
* |
||||
* <p>Enhanced object cloning is done in a two step process. First, |
||||
* the object is cloned using the normal Java clone() method and stored |
||||
* in the clone registry. After that, if it implements JmeCloneable then |
||||
* its cloneFields() method is called to deep clone any of the fields. |
||||
* This two step process has a few benefits. First, it means that objects |
||||
* can easily have a regular shallow clone implementation just like any |
||||
* normal Java objects. Second, the deep cloning of fields happens after |
||||
* creation wich means that the clone is available to future field cloning |
||||
* to resolve circular references.</p> |
||||
* |
||||
* <p>Similar to Java serialization, the handling of specific object |
||||
* types can be customized. This allows certain objects to be cloned gracefully |
||||
* even if they aren't normally Cloneable. This can also be used as a |
||||
* sort of filter to keep certain types of objects from being cloned. |
||||
* (For example, adding the IdentityCloneFunction for Mesh.class would cause |
||||
* all mesh instances to be shared with the original object graph.)</p> |
||||
* |
||||
* <p>By default, the Cloner registers serveral default clone functions |
||||
* as follows:</p> |
||||
* <ul> |
||||
* <li>java.util.ArrayList: ListCloneFunction |
||||
* <li>java.util.LinkedList: ListCloneFunction |
||||
* <li>java.util.concurrent.CopyOnWriteArrayList: ListCloneFunction |
||||
* <li>java.util.Vector: ListCloneFunction |
||||
* <li>java.util.Stack: ListCloneFunction |
||||
* <li>com.jme3.util.SafeArrayList: ListCloneFunction |
||||
* </ul> |
||||
* |
||||
* <p>Usage:</p> |
||||
* <pre> |
||||
* // Example 1: using an instantiated, reusable cloner.
|
||||
* Cloner cloner = new Cloner(); |
||||
* Foo fooClone = cloner.clone(foo); |
||||
* cloner.clearIndex(); // prepare it for reuse
|
||||
* Foo fooClone2 = cloner.clone(foo); |
||||
* |
||||
* // Example 2: using the utility method that self-instantiates a temporary cloner.
|
||||
* Foo fooClone = Cloner.deepClone(foo); |
||||
* |
||||
* </pre> |
||||
* |
||||
* @author Paul Speed |
||||
*/ |
||||
public class Cloner { |
||||
|
||||
/** |
||||
* Keeps track of the objects that have been cloned so far. |
||||
*/ |
||||
private IdentityHashMap<Object, Object> index = new IdentityHashMap<Object, Object>(); |
||||
|
||||
/** |
||||
* Custom functions for cloning objects. |
||||
*/ |
||||
private Map<Class, CloneFunction> functions = new HashMap<Class, CloneFunction>(); |
||||
|
||||
/** |
||||
* Cache the clone methods once for all cloners. |
||||
*/ |
||||
private static final Map<Class, Method> methodCache = new ConcurrentHashMap<>(); |
||||
|
||||
/** |
||||
* Creates a new cloner with only default clone functions and an empty |
||||
* object index. |
||||
*/ |
||||
public Cloner() { |
||||
// Register some standard types
|
||||
ListCloneFunction listFunction = new ListCloneFunction(); |
||||
functions.put(java.util.ArrayList.class, listFunction); |
||||
functions.put(java.util.LinkedList.class, listFunction); |
||||
functions.put(java.util.concurrent.CopyOnWriteArrayList.class, listFunction); |
||||
functions.put(java.util.Vector.class, listFunction); |
||||
functions.put(java.util.Stack.class, listFunction); |
||||
functions.put(com.jme3.util.SafeArrayList.class, listFunction); |
||||
} |
||||
|
||||
/** |
||||
* Convenience utility function that creates a new Cloner, uses it to |
||||
* deep clone the object, and then returns the result. |
||||
*/ |
||||
public static <T> T deepClone( T object ) { |
||||
return new Cloner().clone(object); |
||||
} |
||||
|
||||
/** |
||||
* Deeps clones the specified object, reusing previous clones when possible. |
||||
* |
||||
* <p>Object cloning priority works as follows:</p> |
||||
* <ul> |
||||
* <li>If the object has already been cloned then its clone is returned. |
||||
* <li>If there is a custom CloneFunction then it is called to clone the object. |
||||
* <li>If the object implements Cloneable then its clone() method is called, arrays are |
||||
* deep cloned with entries passing through clone(). |
||||
* <li>If the object implements JmeCloneable then its cloneFields() method is called on the |
||||
* clone. |
||||
* <li>Else an IllegalArgumentException is thrown. |
||||
* </ul> |
||||
* |
||||
* Note: objects returned by this method may not have yet had their cloneField() |
||||
* method called. |
||||
*/ |
||||
public <T> T clone( T object ) { |
||||
return clone(object, true); |
||||
} |
||||
|
||||
/** |
||||
* Internal method to work around a Java generics typing issue by |
||||
* isolating the 'bad' case into a method with suppressed warnings. |
||||
*/ |
||||
@SuppressWarnings("unchecked") |
||||
private <T> Class<T> objectClass( T object ) { |
||||
// This should be 100% allowed without a cast but Java generics
|
||||
// is not that smart sometimes.
|
||||
// Wrapping it in a method at least isolates the warning suppression
|
||||
return (Class<T>)object.getClass(); |
||||
} |
||||
|
||||
/** |
||||
* Deeps clones the specified object, reusing previous clones when possible. |
||||
* |
||||
* <p>Object cloning priority works as follows:</p> |
||||
* <ul> |
||||
* <li>If the object has already been cloned then its clone is returned. |
||||
* <li>If useFunctions is true and there is a custom CloneFunction then it is |
||||
* called to clone the object. |
||||
* <li>If the object implements Cloneable then its clone() method is called, arrays are |
||||
* deep cloned with entries passing through clone(). |
||||
* <li>If the object implements JmeCloneable then its cloneFields() method is called on the |
||||
* clone. |
||||
* <li>Else an IllegalArgumentException is thrown. |
||||
* </ul> |
||||
* |
||||
* <p>The abililty to selectively use clone functions is useful when |
||||
* being called from a clone function.</p> |
||||
* |
||||
* Note: objects returned by this method may not have yet had their cloneField() |
||||
* method called. |
||||
*/ |
||||
public <T> T clone( T object, boolean useFunctions ) { |
||||
if( object == null ) { |
||||
return null; |
||||
} |
||||
Class<T> type = objectClass(object); |
||||
|
||||
// Check the index to see if we already have it
|
||||
Object clone = index.get(object); |
||||
if( clone != null ) { |
||||
return type.cast(clone); |
||||
} |
||||
|
||||
// See if there is a custom function... that trumps everything.
|
||||
CloneFunction<T> f = getCloneFunction(type); |
||||
if( f != null ) { |
||||
T result = f.cloneObject(this, object); |
||||
|
||||
// Store the object in the identity map so that any circular references
|
||||
// are resolvable.
|
||||
index.put(object, result); |
||||
|
||||
// Now call the function again to deep clone the fields
|
||||
f.cloneFields(this, result, object); |
||||
|
||||
return result; |
||||
} |
||||
|
||||
if( object.getClass().isArray() ) { |
||||
// Perform an array clone
|
||||
clone = arrayClone(object); |
||||
|
||||
// Array clone already indexes the clone
|
||||
} else if( object instanceof JmeCloneable ) { |
||||
// Use the two-step cloning semantics
|
||||
clone = ((JmeCloneable)object).jmeClone(); |
||||
|
||||
// Store the object in the identity map so that any circular references
|
||||
// are resolvable
|
||||
index.put(object, clone); |
||||
|
||||
((JmeCloneable)clone).cloneFields(this, object); |
||||
} else if( object instanceof Cloneable ) { |
||||
|
||||
// Perform a regular Java shallow clone
|
||||
try { |
||||
clone = javaClone(object); |
||||
} catch( CloneNotSupportedException e ) { |
||||
throw new IllegalArgumentException("Object is not cloneable, type:" + type, e); |
||||
} |
||||
|
||||
// Store the object in the identity map so that any circular references
|
||||
// are resolvable
|
||||
index.put(object, clone); |
||||
} else { |
||||
throw new IllegalArgumentException("Object is not cloneable, type:" + type); |
||||
} |
||||
|
||||
return type.cast(clone); |
||||
} |
||||
|
||||
/** |
||||
* Sets a custom CloneFunction for the exact Java type. Note: no inheritence |
||||
* checks are performed so a function must be registered for each specific type |
||||
* that it handles. By default ListCloneFunction is registered for |
||||
* ArrayList, LinkedList, CopyOnWriteArrayList, Vector, Stack, and JME's SafeArrayList. |
||||
*/ |
||||
public <T> void setCloneFunction( Class<T> type, CloneFunction<T> function ) { |
||||
if( function == null ) { |
||||
functions.remove(type); |
||||
} else { |
||||
functions.put(type, function); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Returns a previously registered clone function for the specified type or null |
||||
* if there is no custom clone function for the type. |
||||
*/ |
||||
@SuppressWarnings("unchecked") |
||||
public <T> CloneFunction<T> getCloneFunction( Class<T> type ) { |
||||
return (CloneFunction<T>)functions.get(type); |
||||
} |
||||
|
||||
/** |
||||
* Clears the object index allowing the cloner to be reused for a brand new |
||||
* cloning operation. |
||||
*/ |
||||
public void clearIndex() { |
||||
index.clear(); |
||||
} |
||||
|
||||
/** |
||||
* Performs a raw shallow Java clone using reflection. This call does NOT |
||||
* check against the clone index and so will return new objects every time |
||||
* it is called. That's because these are shallow clones and have not (and may |
||||
* not ever, depending on the caller) get resolved. |
||||
* |
||||
* <p>This method is provided as a convenient way for CloneFunctions to call |
||||
* clone() and objects without necessarily knowing their real type.</p> |
||||
*/ |
||||
public <T> T javaClone( T object ) throws CloneNotSupportedException { |
||||
Method m = methodCache.get(object.getClass()); |
||||
if( m == null ) { |
||||
try { |
||||
// Lookup the method and cache it
|
||||
m = object.getClass().getMethod("clone"); |
||||
} catch( NoSuchMethodException e ) { |
||||
throw new CloneNotSupportedException("No public clone method found for:" + object.getClass()); |
||||
} |
||||
methodCache.put(object.getClass(), m); |
||||
|
||||
// Note: yes we might cache the method twice... but so what?
|
||||
} |
||||
|
||||
try { |
||||
Class<? extends T> type = objectClass(object); |
||||
return type.cast(m.invoke(object)); |
||||
} catch( IllegalAccessException | InvocationTargetException e ) { |
||||
throw new RuntimeException("Error cloning object of type:" + object.getClass(), e); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Clones a primitive array by coping it and clones an object |
||||
* array by coping it and then running each of its values through |
||||
* Cloner.clone(). |
||||
*/ |
||||
protected <T> T arrayClone( T object ) { |
||||
|
||||
// Java doesn't support the cloning of arrays through reflection unless
|
||||
// you open access to Object's protected clone array... which requires
|
||||
// elevated privileges. So we will do a work-around that is slightly less
|
||||
// elegant.
|
||||
// This should be 100% allowed without a case but Java generics
|
||||
// is not that smart
|
||||
Class<T> type = objectClass(object); |
||||
Class elementType = type.getComponentType(); |
||||
int size = Array.getLength(object); |
||||
Object clone = Array.newInstance(elementType, size); |
||||
|
||||
// Store the clone for later lookups
|
||||
index.put(object, clone); |
||||
|
||||
if( elementType.isPrimitive() ) { |
||||
// Then our job is a bit easier
|
||||
System.arraycopy(object, 0, clone, 0, size); |
||||
} else { |
||||
// Else it's an object array so we'll clone it and its children
|
||||
for( int i = 0; i < size; i++ ) { |
||||
Object element = clone(Array.get(object, i)); |
||||
Array.set(clone, i, element); |
||||
} |
||||
} |
||||
|
||||
return type.cast(clone); |
||||
} |
||||
} |
@ -0,0 +1,99 @@ |
||||
/* |
||||
* Copyright (c) 2016 jMonkeyEngine |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* |
||||
* * Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in the |
||||
* documentation and/or other materials provided with the distribution. |
||||
* |
||||
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||
* may be used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
*/ |
||||
|
||||
package com.jme3.util.clone; |
||||
|
||||
|
||||
/** |
||||
* Indicates an object that wishes to more actively participate in the |
||||
* two-part deep copying process provided by the Cloner. Objects implementing |
||||
* this interface can access the already cloned object graph to resolve |
||||
* their local dependencies in a way that will be equivalent to the |
||||
* original object graph. In other words, if two objects in the graph |
||||
* share the same target reference then the cloned version will share |
||||
* the cloned reference. |
||||
* |
||||
* <p>For example, if an object wishes to deep clone one of its fields |
||||
* then it will call cloner.clone(object) instead of object.clone(). |
||||
* The cloner will keep track of any clones already created for 'object' |
||||
* and return that instead of a new clone.</p> |
||||
* |
||||
* <p>Cloning of a JmeCloneable object is done in two parts. First, |
||||
* the standard Java clone() method is called to create a shallow clone |
||||
* of the object. Second, the cloner wil lcall the cloneFields() method |
||||
* to let the object deep clone any of its fields that should be cloned.</p> |
||||
* |
||||
* <p>This two part process is necessary to facilitate circular references. |
||||
* When an object calls cloner.clone() during its cloneFields() method, it |
||||
* may get only a shallow copy that will be filled in later.</p> |
||||
* |
||||
* @author Paul Speed |
||||
*/ |
||||
public interface JmeCloneable extends Cloneable { |
||||
|
||||
/** |
||||
* Performs a regular shallow clone of the object. Some fields |
||||
* may also be cloned but generally only if they will never be |
||||
* shared with other objects. (For example, local Vector3fs and so on.) |
||||
* |
||||
* <p>This method is separate from the regular clone() method |
||||
* so that objects might still maintain their own regular java clone() |
||||
* semantics (perhaps even using Cloner for those methods). However, |
||||
* because Java's clone() has specific features in the sense of Object's |
||||
* clone() implementation, it's usually best to have some path for |
||||
* subclasses to bypass the public clone() method that might be cloning |
||||
* fields and instead get at the superclass protected clone() methods. |
||||
* For example, through super.jmeClone() or another protected clone |
||||
* method that some base class eventually calls super.clone() in.</p> |
||||
*/ |
||||
public Object jmeClone(); |
||||
|
||||
/** |
||||
* Implemented to perform deep cloning for this object, resolving |
||||
* local cloned references using the specified cloner. The object |
||||
* can call cloner.clone(fieldValue) to deep clone any of its fields. |
||||
* |
||||
* <p>Note: during normal clone operations the original object |
||||
* will not be needed as the clone has already had all of the fields |
||||
* shallow copied.</p> |
||||
* |
||||
* @param cloner The cloner that is performing the cloning operation. The |
||||
* cloneFields method can call back into the cloner to make |
||||
* clones if its subordinate fields. |
||||
* @param original The original object from which this object was cloned. |
||||
* This is provided for the very rare case that this object needs |
||||
* to refer to its original for some reason. In general, all of |
||||
* the relevant values should have been transferred during the |
||||
* shallow clone and this object need merely clone what it wants. |
||||
*/ |
||||
public void cloneFields( Cloner cloner, Object original ); |
||||
} |
@ -0,0 +1,97 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.util.mikktspace; |
||||
|
||||
/** |
||||
* |
||||
* @author Nehon |
||||
*/ |
||||
public interface MikkTSpaceContext { |
||||
|
||||
/** |
||||
* Returns the number of faces (triangles/quads) on the mesh to be |
||||
* processed. |
||||
* |
||||
* @return |
||||
*/ |
||||
public int getNumFaces(); |
||||
|
||||
/** |
||||
* Returns the number of vertices on face number iFace iFace is a number in |
||||
* the range {0, 1, ..., getNumFaces()-1} |
||||
* |
||||
* @param face |
||||
* @return |
||||
*/ |
||||
public int getNumVerticesOfFace(int face); |
||||
|
||||
/** |
||||
* returns the position/normal/texcoord of the referenced face of vertex |
||||
* number iVert. iVert is in the range {0,1,2} for triangles and {0,1,2,3} |
||||
* for quads. |
||||
* |
||||
* @param posOut |
||||
* @param face |
||||
* @param vert |
||||
*/ |
||||
public void getPosition(float posOut[], int face, int vert); |
||||
|
||||
public void getNormal(float normOut[], int face, int vert); |
||||
|
||||
public void getTexCoord(float texOut[], int face, int vert); |
||||
|
||||
/** |
||||
* The call-backsetTSpaceBasic() is sufficient for basic normal mapping. |
||||
* This function is used to return the tangent and sign to the application. |
||||
* tangent is a unit length vector. For normal maps it is sufficient to use |
||||
* the following simplified version of the bitangent which is generated at |
||||
* pixel/vertex level. |
||||
* |
||||
* bitangent = fSign * cross(vN, tangent); |
||||
* |
||||
* Note that the results are returned unindexed. It is possible to generate |
||||
* a new index list But averaging/overwriting tangent spaces by using an |
||||
* already existing index list WILL produce INCRORRECT results. DO NOT! use |
||||
* an already existing index list. |
||||
* |
||||
* @param tangent |
||||
* @param sign |
||||
* @param face |
||||
* @param vert |
||||
*/ |
||||
public void setTSpaceBasic(float tangent[], float sign, int face, int vert); |
||||
|
||||
/** |
||||
* This function is used to return tangent space results to the application. |
||||
* tangent and biTangent are unit length vectors and fMagS and fMagT are |
||||
* their true magnitudes which can be used for relief mapping effects. |
||||
* |
||||
* biTangent is the "real" bitangent and thus may not be perpendicular to |
||||
* tangent. However, both are perpendicular to the vertex normal. For normal |
||||
* maps it is sufficient to use the following simplified version of the |
||||
* bitangent which is generated at pixel/vertex level. |
||||
* |
||||
* <pre> |
||||
* fSign = bIsOrientationPreserving ? 1.0f : (-1.0f); |
||||
* bitangent = fSign * cross(vN, tangent); |
||||
* </pre> |
||||
* |
||||
* Note that the results are returned unindexed. It is possible to generate |
||||
* a new index list. But averaging/overwriting tangent spaces by using an |
||||
* already existing index list WILL produce INCRORRECT results. DO NOT! use |
||||
* an already existing index list. |
||||
* |
||||
* @param tangent |
||||
* @param biTangent |
||||
* @param magS |
||||
* @param magT |
||||
* @param isOrientationPreserving |
||||
* @param face |
||||
* @param vert |
||||
*/ |
||||
void setTSpace(float tangent[], float biTangent[], float magS, float magT, |
||||
boolean isOrientationPreserving, int face, int vert); |
||||
} |
@ -0,0 +1,100 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.util.mikktspace; |
||||
|
||||
import com.jme3.scene.Mesh; |
||||
import com.jme3.scene.VertexBuffer; |
||||
import com.jme3.scene.mesh.IndexBuffer; |
||||
import com.jme3.util.BufferUtils; |
||||
import java.nio.FloatBuffer; |
||||
|
||||
/** |
||||
* |
||||
* @author Nehon |
||||
*/ |
||||
public class MikkTSpaceImpl implements MikkTSpaceContext { |
||||
|
||||
Mesh mesh; |
||||
|
||||
public MikkTSpaceImpl(Mesh mesh) { |
||||
this.mesh = mesh; |
||||
VertexBuffer tangentBuffer = mesh.getBuffer(VertexBuffer.Type.Tangent); |
||||
if(tangentBuffer == null){ |
||||
FloatBuffer fb = BufferUtils.createFloatBuffer(mesh.getVertexCount() * 4); |
||||
mesh.setBuffer(VertexBuffer.Type.Tangent, 4, fb); |
||||
} |
||||
|
||||
//TODO ensure the Tangent buffer exists, else create one.
|
||||
} |
||||
|
||||
@Override |
||||
public int getNumFaces() { |
||||
return mesh.getTriangleCount(); |
||||
} |
||||
|
||||
@Override |
||||
public int getNumVerticesOfFace(int face) { |
||||
return 3; |
||||
} |
||||
|
||||
@Override |
||||
public void getPosition(float[] posOut, int face, int vert) { |
||||
int vertIndex = getIndex(face, vert); |
||||
VertexBuffer position = mesh.getBuffer(VertexBuffer.Type.Position); |
||||
FloatBuffer pos = (FloatBuffer) position.getData(); |
||||
pos.position(vertIndex * 3); |
||||
posOut[0] = pos.get(); |
||||
posOut[1] = pos.get(); |
||||
posOut[2] = pos.get(); |
||||
} |
||||
|
||||
@Override |
||||
public void getNormal(float[] normOut, int face, int vert) { |
||||
int vertIndex = getIndex(face, vert); |
||||
VertexBuffer normal = mesh.getBuffer(VertexBuffer.Type.Normal); |
||||
FloatBuffer norm = (FloatBuffer) normal.getData(); |
||||
norm.position(vertIndex * 3); |
||||
normOut[0] = norm.get(); |
||||
normOut[1] = norm.get(); |
||||
normOut[2] = norm.get(); |
||||
} |
||||
|
||||
@Override |
||||
public void getTexCoord(float[] texOut, int face, int vert) { |
||||
int vertIndex = getIndex(face, vert); |
||||
VertexBuffer texCoord = mesh.getBuffer(VertexBuffer.Type.TexCoord); |
||||
FloatBuffer tex = (FloatBuffer) texCoord.getData(); |
||||
tex.position(vertIndex * 2); |
||||
texOut[0] = tex.get(); |
||||
texOut[1] = tex.get(); |
||||
} |
||||
|
||||
@Override |
||||
public void setTSpaceBasic(float[] tangent, float sign, int face, int vert) { |
||||
int vertIndex = getIndex(face, vert); |
||||
VertexBuffer tangentBuffer = mesh.getBuffer(VertexBuffer.Type.Tangent); |
||||
FloatBuffer tan = (FloatBuffer) tangentBuffer.getData(); |
||||
|
||||
tan.position(vertIndex * 4); |
||||
tan.put(tangent); |
||||
tan.put(sign); |
||||
|
||||
tan.rewind(); |
||||
tangentBuffer.setUpdateNeeded(); |
||||
} |
||||
|
||||
@Override |
||||
public void setTSpace(float[] tangent, float[] biTangent, float magS, float magT, boolean isOrientationPreserving, int face, int vert) { |
||||
//Do nothing
|
||||
} |
||||
|
||||
private int getIndex(int face, int vert) { |
||||
IndexBuffer index = mesh.getIndexBuffer(); |
||||
int vertIndex = index.get(face * 3 + vert); |
||||
return vertIndex; |
||||
} |
||||
|
||||
} |
File diff suppressed because it is too large
Load Diff
@ -1,80 +0,0 @@ |
||||
#import "Common/ShaderLib/Shadows15.glsllib" |
||||
|
||||
out vec4 outFragColor; |
||||
|
||||
#if defined(PSSM) || defined(FADE) |
||||
in float shadowPosition; |
||||
#endif |
||||
|
||||
in vec4 projCoord0; |
||||
in vec4 projCoord1; |
||||
in vec4 projCoord2; |
||||
in vec4 projCoord3; |
||||
|
||||
#ifdef POINTLIGHT |
||||
in vec4 projCoord4; |
||||
in vec4 projCoord5; |
||||
in vec4 worldPos; |
||||
uniform vec3 m_LightPos; |
||||
#else |
||||
#ifndef PSSM |
||||
in float lightDot; |
||||
#endif |
||||
#endif |
||||
|
||||
#ifdef DISCARD_ALPHA |
||||
#ifdef COLOR_MAP |
||||
uniform sampler2D m_ColorMap; |
||||
#else |
||||
uniform sampler2D m_DiffuseMap; |
||||
#endif |
||||
uniform float m_AlphaDiscardThreshold; |
||||
varying vec2 texCoord; |
||||
#endif |
||||
|
||||
#ifdef FADE |
||||
uniform vec2 m_FadeInfo; |
||||
#endif |
||||
|
||||
void main(){ |
||||
|
||||
#ifdef DISCARD_ALPHA |
||||
#ifdef COLOR_MAP |
||||
float alpha = texture2D(m_ColorMap,texCoord).a; |
||||
#else |
||||
float alpha = texture2D(m_DiffuseMap,texCoord).a; |
||||
#endif |
||||
|
||||
if(alpha < m_AlphaDiscardThreshold){ |
||||
discard; |
||||
} |
||||
#endif |
||||
|
||||
float shadow = 1.0; |
||||
#ifdef POINTLIGHT |
||||
shadow = getPointLightShadows(worldPos, m_LightPos, |
||||
m_ShadowMap0,m_ShadowMap1,m_ShadowMap2,m_ShadowMap3,m_ShadowMap4,m_ShadowMap5, |
||||
projCoord0, projCoord1, projCoord2, projCoord3, projCoord4, projCoord5); |
||||
#else |
||||
#ifdef PSSM |
||||
shadow = getDirectionalLightShadows(m_Splits, shadowPosition, |
||||
m_ShadowMap0,m_ShadowMap1,m_ShadowMap2,m_ShadowMap3, |
||||
projCoord0, projCoord1, projCoord2, projCoord3); |
||||
#else |
||||
//spotlight |
||||
if(lightDot < 0){ |
||||
outFragColor = vec4(1.0); |
||||
return; |
||||
} |
||||
shadow = getSpotLightShadows(m_ShadowMap0,projCoord0); |
||||
#endif |
||||
#endif |
||||
|
||||
#ifdef FADE |
||||
shadow = max(0.0,mix(shadow,1.0,(shadowPosition - m_FadeInfo.x) * m_FadeInfo.y)); |
||||
#endif |
||||
|
||||
shadow = shadow * m_ShadowIntensity + (1.0 - m_ShadowIntensity); |
||||
outFragColor = vec4(shadow, shadow, shadow, 1.0); |
||||
} |
||||
|
@ -1,82 +0,0 @@ |
||||
#import "Common/ShaderLib/Instancing.glsllib" |
||||
#import "Common/ShaderLib/Skinning.glsllib" |
||||
uniform mat4 m_LightViewProjectionMatrix0; |
||||
uniform mat4 m_LightViewProjectionMatrix1; |
||||
uniform mat4 m_LightViewProjectionMatrix2; |
||||
uniform mat4 m_LightViewProjectionMatrix3; |
||||
|
||||
|
||||
out vec4 projCoord0; |
||||
out vec4 projCoord1; |
||||
out vec4 projCoord2; |
||||
out vec4 projCoord3; |
||||
|
||||
#ifdef POINTLIGHT |
||||
uniform mat4 m_LightViewProjectionMatrix4; |
||||
uniform mat4 m_LightViewProjectionMatrix5; |
||||
out vec4 projCoord4; |
||||
out vec4 projCoord5; |
||||
out vec4 worldPos; |
||||
#else |
||||
#ifndef PSSM |
||||
uniform vec3 m_LightPos; |
||||
uniform vec3 m_LightDir; |
||||
out float lightDot; |
||||
#endif |
||||
#endif |
||||
|
||||
#if defined(PSSM) || defined(FADE) |
||||
out float shadowPosition; |
||||
#endif |
||||
out vec3 lightVec; |
||||
|
||||
out vec2 texCoord; |
||||
|
||||
in vec3 inPosition; |
||||
|
||||
#ifdef DISCARD_ALPHA |
||||
in vec2 inTexCoord; |
||||
#endif |
||||
|
||||
const mat4 biasMat = mat4(0.5, 0.0, 0.0, 0.0, |
||||
0.0, 0.5, 0.0, 0.0, |
||||
0.0, 0.0, 0.5, 0.0, |
||||
0.5, 0.5, 0.5, 1.0); |
||||
|
||||
|
||||
void main(){ |
||||
vec4 modelSpacePos = vec4(inPosition, 1.0); |
||||
|
||||
#ifdef NUM_BONES |
||||
Skinning_Compute(modelSpacePos); |
||||
#endif |
||||
gl_Position = TransformWorldViewProjection(modelSpacePos); |
||||
|
||||
#if defined(PSSM) || defined(FADE) |
||||
shadowPosition = gl_Position.z; |
||||
#endif |
||||
|
||||
#ifndef POINTLIGHT |
||||
vec4 worldPos=vec4(0.0); |
||||
#endif |
||||
// get the vertex in world space |
||||
worldPos = TransformWorld(modelSpacePos); |
||||
|
||||
#ifdef DISCARD_ALPHA |
||||
texCoord = inTexCoord; |
||||
#endif |
||||
// populate the light view matrices array and convert vertex to light viewProj space |
||||
projCoord0 = biasMat * m_LightViewProjectionMatrix0 * worldPos; |
||||
projCoord1 = biasMat * m_LightViewProjectionMatrix1 * worldPos; |
||||
projCoord2 = biasMat * m_LightViewProjectionMatrix2 * worldPos; |
||||
projCoord3 = biasMat * m_LightViewProjectionMatrix3 * worldPos; |
||||
#ifdef POINTLIGHT |
||||
projCoord4 = biasMat * m_LightViewProjectionMatrix4 * worldPos; |
||||
projCoord5 = biasMat * m_LightViewProjectionMatrix5 * worldPos; |
||||
#else |
||||
#ifndef PSSM |
||||
vec3 lightDir = worldPos.xyz - m_LightPos; |
||||
lightDot = dot(m_LightDir,lightDir); |
||||
#endif |
||||
#endif |
||||
} |
@ -1,242 +0,0 @@ |
||||
// Because gpu_shader5 is actually where those |
||||
// gather functions are declared to work on shadowmaps |
||||
#extension GL_ARB_gpu_shader5 : enable |
||||
|
||||
#ifdef HARDWARE_SHADOWS |
||||
#define SHADOWMAP sampler2DShadow |
||||
#define SHADOWCOMPAREOFFSET(tex,coord,offset) textureProjOffset(tex, coord, offset) |
||||
#define SHADOWCOMPARE(tex,coord) textureProj(tex, coord) |
||||
#define SHADOWGATHER(tex,coord) textureGather(tex, coord.xy, coord.z) |
||||
#else |
||||
#define SHADOWMAP sampler2D |
||||
#define SHADOWCOMPAREOFFSET(tex,coord,offset) step(coord.z, textureProjOffset(tex, coord, offset).r) |
||||
#define SHADOWCOMPARE(tex,coord) step(coord.z, textureProj(tex, coord).r) |
||||
#define SHADOWGATHER(tex,coord) step(coord.z, textureGather(tex, coord.xy)) |
||||
#endif |
||||
|
||||
|
||||
#if FILTER_MODE == 0 |
||||
#define GETSHADOW Shadow_Nearest |
||||
#define KERNEL 1.0 |
||||
#elif FILTER_MODE == 1 |
||||
#ifdef HARDWARE_SHADOWS |
||||
#define GETSHADOW Shadow_Nearest |
||||
#else |
||||
#define GETSHADOW Shadow_DoBilinear_2x2 |
||||
#endif |
||||
#define KERNEL 1.0 |
||||
#elif FILTER_MODE == 2 |
||||
#define GETSHADOW Shadow_DoDither_2x2 |
||||
#define KERNEL 1.0 |
||||
#elif FILTER_MODE == 3 |
||||
#define GETSHADOW Shadow_DoPCF |
||||
#define KERNEL 4.0 |
||||
#elif FILTER_MODE == 4 |
||||
#define GETSHADOW Shadow_DoPCFPoisson |
||||
#define KERNEL 4.0 |
||||
#elif FILTER_MODE == 5 |
||||
#define GETSHADOW Shadow_DoPCF |
||||
#define KERNEL 8.0 |
||||
#endif |
||||
|
||||
|
||||
|
||||
uniform SHADOWMAP m_ShadowMap0; |
||||
uniform SHADOWMAP m_ShadowMap1; |
||||
uniform SHADOWMAP m_ShadowMap2; |
||||
uniform SHADOWMAP m_ShadowMap3; |
||||
#ifdef POINTLIGHT |
||||
uniform SHADOWMAP m_ShadowMap4; |
||||
uniform SHADOWMAP m_ShadowMap5; |
||||
#endif |
||||
|
||||
#ifdef PSSM |
||||
uniform vec4 m_Splits; |
||||
#endif |
||||
uniform float m_ShadowIntensity; |
||||
|
||||
const vec2 pixSize2 = vec2(1.0 / SHADOWMAP_SIZE); |
||||
float shadowBorderScale = 1.0; |
||||
|
||||
float Shadow_BorderCheck(in vec2 coord){ |
||||
// Fastest, "hack" method (uses 4-5 instructions) |
||||
vec4 t = vec4(coord.xy, 0.0, 1.0); |
||||
t = step(t.wwxy, t.xyzz); |
||||
return dot(t,t); |
||||
} |
||||
|
||||
float Shadow_Nearest(in SHADOWMAP tex, in vec4 projCoord){ |
||||
float border = Shadow_BorderCheck(projCoord.xy); |
||||
if (border > 0.0){ |
||||
return 1.0; |
||||
} |
||||
return SHADOWCOMPARE(tex,projCoord); |
||||
} |
||||
|
||||
float Shadow_DoDither_2x2(in SHADOWMAP tex, in vec4 projCoord){ |
||||
float border = Shadow_BorderCheck(projCoord.xy); |
||||
if (border > 0.0) |
||||
return 1.0; |
||||
|
||||
vec2 pixSize = pixSize2 * shadowBorderScale; |
||||
|
||||
float shadow = 0.0; |
||||
ivec2 o = ivec2(mod(floor(gl_FragCoord.xy), 2.0)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy+pixSize*(vec2(-1.5, 1.5)+o), projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy+pixSize*(vec2( 0.5, 1.5)+o), projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy+pixSize*(vec2(-1.5, -0.5)+o), projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy+pixSize*(vec2( 0.5, -0.5)+o), projCoord.zw)); |
||||
shadow *= 0.25; |
||||
return shadow; |
||||
} |
||||
|
||||
float Shadow_DoBilinear_2x2(in SHADOWMAP tex, in vec4 projCoord){ |
||||
float border = Shadow_BorderCheck(projCoord.xy); |
||||
if (border > 0.0) |
||||
return 1.0; |
||||
|
||||
#ifdef GL_ARB_gpu_shader5 |
||||
vec4 coord = vec4(projCoord.xyz / projCoord.www,0.0); |
||||
vec4 gather = SHADOWGATHER(tex, coord); |
||||
#else |
||||
vec4 gather = vec4(0.0); |
||||
gather.x = SHADOWCOMPAREOFFSET(tex, projCoord, ivec2(0, 1)); |
||||
gather.y = SHADOWCOMPAREOFFSET(tex, projCoord, ivec2(1, 1)); |
||||
gather.z = SHADOWCOMPAREOFFSET(tex, projCoord, ivec2(1, 0)); |
||||
gather.w = SHADOWCOMPAREOFFSET(tex, projCoord, ivec2(0, 0)); |
||||
#endif |
||||
|
||||
vec2 f = fract( projCoord.xy * SHADOWMAP_SIZE ); |
||||
vec2 mx = mix( gather.wx, gather.zy, f.x ); |
||||
return mix( mx.x, mx.y, f.y ); |
||||
} |
||||
|
||||
float Shadow_DoPCF(in SHADOWMAP tex, in vec4 projCoord){ |
||||
|
||||
vec2 pixSize = pixSize2 * shadowBorderScale; |
||||
float shadow = 0.0; |
||||
float border = Shadow_BorderCheck(projCoord.xy); |
||||
if (border > 0.0) |
||||
return 1.0; |
||||
|
||||
float bound = KERNEL * 0.5 - 0.5; |
||||
bound *= PCFEDGE; |
||||
for (float y = -bound; y <= bound; y += PCFEDGE){ |
||||
for (float x = -bound; x <= bound; x += PCFEDGE){ |
||||
vec4 coord = vec4(projCoord.xy + vec2(x,y) * pixSize, projCoord.zw); |
||||
shadow += SHADOWCOMPARE(tex, coord); |
||||
} |
||||
} |
||||
|
||||
shadow = shadow / (KERNEL * KERNEL); |
||||
return shadow; |
||||
} |
||||
|
||||
|
||||
//12 tap poisson disk |
||||
const vec2 poissonDisk0 = vec2(-0.1711046, -0.425016); |
||||
const vec2 poissonDisk1 = vec2(-0.7829809, 0.2162201); |
||||
const vec2 poissonDisk2 = vec2(-0.2380269, -0.8835521); |
||||
const vec2 poissonDisk3 = vec2(0.4198045, 0.1687819); |
||||
const vec2 poissonDisk4 = vec2(-0.684418, -0.3186957); |
||||
const vec2 poissonDisk5 = vec2(0.6026866, -0.2587841); |
||||
const vec2 poissonDisk6 = vec2(-0.2412762, 0.3913516); |
||||
const vec2 poissonDisk7 = vec2(0.4720655, -0.7664126); |
||||
const vec2 poissonDisk8 = vec2(0.9571564, 0.2680693); |
||||
const vec2 poissonDisk9 = vec2(-0.5238616, 0.802707); |
||||
const vec2 poissonDisk10 = vec2(0.5653144, 0.60262); |
||||
const vec2 poissonDisk11 = vec2(0.0123658, 0.8627419); |
||||
|
||||
|
||||
float Shadow_DoPCFPoisson(in SHADOWMAP tex, in vec4 projCoord){ |
||||
|
||||
float shadow = 0.0; |
||||
float border = Shadow_BorderCheck(projCoord.xy); |
||||
if (border > 0.0){ |
||||
return 1.0; |
||||
} |
||||
|
||||
vec2 texelSize = pixSize2 * 4.0 * PCFEDGE * shadowBorderScale; |
||||
|
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk0 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk1 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk2 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk3 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk4 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk5 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk6 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk7 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk8 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk9 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk10 * texelSize, projCoord.zw)); |
||||
shadow += SHADOWCOMPARE(tex, vec4(projCoord.xy + poissonDisk11 * texelSize, projCoord.zw)); |
||||
|
||||
//this is divided by 12 |
||||
return shadow * 0.08333333333; |
||||
} |
||||
|
||||
#ifdef POINTLIGHT |
||||
float getPointLightShadows(in vec4 worldPos,in vec3 lightPos, |
||||
in SHADOWMAP shadowMap0,in SHADOWMAP shadowMap1,in SHADOWMAP shadowMap2,in SHADOWMAP shadowMap3,in SHADOWMAP shadowMap4,in SHADOWMAP shadowMap5, |
||||
in vec4 projCoord0,in vec4 projCoord1,in vec4 projCoord2,in vec4 projCoord3,in vec4 projCoord4,in vec4 projCoord5){ |
||||
float shadow = 1.0; |
||||
vec3 vect = worldPos.xyz - lightPos; |
||||
vec3 absv= abs(vect); |
||||
float maxComp = max(absv.x,max(absv.y,absv.z)); |
||||
if(maxComp == absv.y){ |
||||
if(vect.y < 0.0){ |
||||
shadow = GETSHADOW(shadowMap0, projCoord0 / projCoord0.w); |
||||
}else{ |
||||
shadow = GETSHADOW(shadowMap1, projCoord1 / projCoord1.w); |
||||
} |
||||
}else if(maxComp == absv.z){ |
||||
if(vect.z < 0.0){ |
||||
shadow = GETSHADOW(shadowMap2, projCoord2 / projCoord2.w); |
||||
}else{ |
||||
shadow = GETSHADOW(shadowMap3, projCoord3 / projCoord3.w); |
||||
} |
||||
}else if(maxComp == absv.x){ |
||||
if(vect.x < 0.0){ |
||||
shadow = GETSHADOW(shadowMap4, projCoord4 / projCoord4.w); |
||||
}else{ |
||||
shadow = GETSHADOW(shadowMap5, projCoord5 / projCoord5.w); |
||||
} |
||||
} |
||||
return shadow; |
||||
} |
||||
#else |
||||
#ifdef PSSM |
||||
float getDirectionalLightShadows(in vec4 splits,in float shadowPosition, |
||||
in SHADOWMAP shadowMap0,in SHADOWMAP shadowMap1,in SHADOWMAP shadowMap2,in SHADOWMAP shadowMap3, |
||||
in vec4 projCoord0,in vec4 projCoord1,in vec4 projCoord2,in vec4 projCoord3){ |
||||
float shadow = 1.0; |
||||
if(shadowPosition < splits.x){ |
||||
shadow = GETSHADOW(shadowMap0, projCoord0 ); |
||||
}else if( shadowPosition < splits.y){ |
||||
shadowBorderScale = 0.5; |
||||
shadow = GETSHADOW(shadowMap1, projCoord1); |
||||
}else if( shadowPosition < splits.z){ |
||||
shadowBorderScale = 0.25; |
||||
shadow = GETSHADOW(shadowMap2, projCoord2); |
||||
}else if( shadowPosition < splits.w){ |
||||
shadowBorderScale = 0.125; |
||||
shadow = GETSHADOW(shadowMap3, projCoord3); |
||||
} |
||||
return shadow; |
||||
} |
||||
#else |
||||
float getSpotLightShadows(in SHADOWMAP shadowMap,in vec4 projCoord){ |
||||
float shadow = 1.0; |
||||
projCoord /= projCoord.w; |
||||
shadow = GETSHADOW(shadowMap,projCoord); |
||||
|
||||
//a small falloff to make the shadow blend nicely into the not lighten |
||||
//we translate the texture coordinate value to a -1,1 range so the length |
||||
//of the texture coordinate vector is actually the radius of the lighten area on the ground |
||||
projCoord = projCoord * 2.0 - 1.0; |
||||
float fallOff = ( length(projCoord.xy) - 0.9 ) / 0.1; |
||||
return mix(shadow,1.0,clamp(fallOff,0.0,1.0)); |
||||
|
||||
} |
||||
#endif |
||||
#endif |
@ -0,0 +1,367 @@ |
||||
/* |
||||
* Copyright (c) 2016 jMonkeyEngine |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* |
||||
* * Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in the |
||||
* documentation and/or other materials provided with the distribution. |
||||
* |
||||
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||
* may be used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
*/ |
||||
|
||||
package jme3test.app; |
||||
|
||||
import java.util.*; |
||||
|
||||
import com.jme3.util.clone.*; |
||||
|
||||
/** |
||||
* |
||||
* |
||||
* @author Paul Speed |
||||
*/ |
||||
public class TestCloner { |
||||
|
||||
public static void main( String... args ) { |
||||
|
||||
System.out.println("Clone test:"); |
||||
|
||||
Cloner cloner = new Cloner(); |
||||
|
||||
RegularObject ro = new RegularObject(42); |
||||
System.out.println("Regular Object:" + ro); |
||||
RegularObject roCloneLegacy = ro.clone(); |
||||
System.out.println("Regular Object Clone:" + roCloneLegacy); |
||||
RegularObject roClone = cloner.clone(ro); |
||||
System.out.println("cloner: Regular Object Clone:" + roClone); |
||||
|
||||
System.out.println("------------------------------------"); |
||||
System.out.println(); |
||||
|
||||
cloner = new Cloner(); |
||||
RegularSubclass rsc = new RegularSubclass(69, "test"); |
||||
System.out.println("Regular subclass:" + rsc); |
||||
RegularSubclass rscCloneLegacy = (RegularSubclass)rsc.clone(); |
||||
System.out.println("Regular subclass Clone:" + rscCloneLegacy); |
||||
RegularSubclass rscClone = cloner.clone(rsc); |
||||
System.out.println("cloner: Regular subclass Clone:" + rscClone); |
||||
|
||||
System.out.println("------------------------------------"); |
||||
System.out.println(); |
||||
|
||||
cloner = new Cloner(); |
||||
Parent parent = new Parent("Foo", 34); |
||||
System.out.println("Parent:" + parent); |
||||
Parent parentCloneLegacy = parent.clone(); |
||||
System.out.println("Parent Clone:" + parentCloneLegacy); |
||||
Parent parentClone = cloner.clone(parent); |
||||
System.out.println("cloner: Parent Clone:" + parentClone); |
||||
|
||||
System.out.println("------------------------------------"); |
||||
System.out.println(); |
||||
|
||||
cloner = new Cloner(); |
||||
GraphNode root = new GraphNode("root"); |
||||
GraphNode child1 = root.addLink("child1"); |
||||
GraphNode child2 = root.addLink("child2"); |
||||
GraphNode shared = child1.addLink("shared"); |
||||
child2.addLink(shared); |
||||
|
||||
// Add a circular reference to get fancy
|
||||
shared.addLink(root); |
||||
|
||||
System.out.println("Simple graph:"); |
||||
root.dump(" "); |
||||
|
||||
GraphNode rootClone = cloner.clone(root); |
||||
System.out.println("clone:"); |
||||
rootClone.dump(" "); |
||||
|
||||
System.out.println("original:"); |
||||
root.dump(" "); |
||||
|
||||
GraphNode reclone = Cloner.deepClone(root); |
||||
System.out.println("reclone:"); |
||||
reclone.dump(" "); |
||||
|
||||
System.out.println("------------------------------------"); |
||||
System.out.println(); |
||||
cloner = new Cloner(); |
||||
|
||||
ArrayHolder arrays = new ArrayHolder(5, 3, 7, 3, 7, 2, 1, 4); |
||||
System.out.println("Array holder:" + arrays); |
||||
ArrayHolder arraysClone = cloner.clone(arrays); |
||||
System.out.println("Array holder clone:" + arraysClone); |
||||
|
||||
|
||||
|
||||
} |
||||
|
||||
public static class RegularObject implements Cloneable { |
||||
protected int i; |
||||
|
||||
public RegularObject( int i ) { |
||||
this.i = i; |
||||
} |
||||
|
||||
public RegularObject clone() { |
||||
try { |
||||
return (RegularObject)super.clone(); |
||||
} catch( CloneNotSupportedException e ) { |
||||
throw new RuntimeException(e); |
||||
} |
||||
} |
||||
|
||||
public String toString() { |
||||
return getClass().getSimpleName() + "@" + System.identityHashCode(this) |
||||
+ "[i=" + i + "]"; |
||||
} |
||||
} |
||||
|
||||
public static class RegularSubclass extends RegularObject { |
||||
protected String name; |
||||
|
||||
public RegularSubclass( int i, String name ) { |
||||
super(i); |
||||
this.name = name; |
||||
} |
||||
|
||||
public String toString() { |
||||
return getClass().getSimpleName() + "@" + System.identityHashCode(this) |
||||
+ "[i=" + i + ", name=" + name + "]"; |
||||
} |
||||
} |
||||
|
||||
public static class Parent implements Cloneable, JmeCloneable { |
||||
|
||||
private RegularObject ro; |
||||
private RegularSubclass rsc; |
||||
|
||||
public Parent( String name, int age ) { |
||||
this.ro = new RegularObject(age); |
||||
this.rsc = new RegularSubclass(age, name); |
||||
} |
||||
|
||||
public Parent clone() { |
||||
try { |
||||
return (Parent)super.clone(); |
||||
} catch( CloneNotSupportedException e ) { |
||||
throw new RuntimeException(e); |
||||
} |
||||
} |
||||
|
||||
public Parent jmeClone() { |
||||
// Ok to delegate to clone() in this case because no deep
|
||||
// cloning is done there.
|
||||
return clone(); |
||||
} |
||||
|
||||
public void cloneFields( Cloner cloner, Object original ) { |
||||
this.ro = cloner.clone(ro); |
||||
this.rsc = cloner.clone(rsc); |
||||
} |
||||
|
||||
public String toString() { |
||||
return getClass().getSimpleName() + "@" + System.identityHashCode(this) |
||||
+ "[ro=" + ro + ", rsc=" + rsc + "]"; |
||||
} |
||||
} |
||||
|
||||
public static class GraphNode implements Cloneable, JmeCloneable { |
||||
|
||||
private String name; |
||||
private List<GraphNode> links = new ArrayList<>(); |
||||
|
||||
public GraphNode( String name ) { |
||||
this.name = name; |
||||
} |
||||
|
||||
public void dump( String indent ) { |
||||
dump(indent, new HashSet<GraphNode>()); |
||||
} |
||||
|
||||
private void dump( String indent, Set<GraphNode> visited ) { |
||||
if( visited.contains(this) ) { |
||||
// already been here
|
||||
System.out.println(indent + this + " ** circular."); |
||||
return; |
||||
} |
||||
System.out.println(indent + this); |
||||
visited.add(this); |
||||
for( GraphNode n : links ) { |
||||
n.dump(indent + " ", visited); |
||||
} |
||||
visited.remove(this); |
||||
} |
||||
|
||||
public GraphNode addLink( String name ) { |
||||
GraphNode node = new GraphNode(name); |
||||
links.add(node); |
||||
return node; |
||||
} |
||||
|
||||
public GraphNode addLink( GraphNode node ) { |
||||
links.add(node); |
||||
return node; |
||||
} |
||||
|
||||
public List<GraphNode> getLinks() { |
||||
return links; |
||||
} |
||||
|
||||
public GraphNode jmeClone() { |
||||
try { |
||||
return (GraphNode)super.clone(); |
||||
} catch( CloneNotSupportedException e ) { |
||||
throw new RuntimeException(e); |
||||
} |
||||
} |
||||
|
||||
public void cloneFields( Cloner cloner, Object original ) { |
||||
this.links = cloner.clone(links); |
||||
} |
||||
|
||||
public String toString() { |
||||
return getClass().getSimpleName() + "@" + System.identityHashCode(this) |
||||
+ "[name=" + name + "]"; |
||||
} |
||||
} |
||||
|
||||
public static class ArrayHolder implements JmeCloneable { |
||||
|
||||
private int[] intArray; |
||||
private int[][] intArray2D; |
||||
private Object[] objects; |
||||
private RegularObject[] regularObjects; |
||||
private String[] strings; |
||||
|
||||
public ArrayHolder( int... values ) { |
||||
this.intArray = values; |
||||
this.intArray2D = new int[values.length][2]; |
||||
for( int i = 0; i < values.length; i++ ) { |
||||
intArray2D[i][0] = values[i] + 1; |
||||
intArray2D[i][1] = values[i] * 2; |
||||
} |
||||
this.objects = new Object[values.length]; |
||||
this.regularObjects = new RegularObject[values.length]; |
||||
this.strings = new String[values.length]; |
||||
for( int i = 0; i < values.length; i++ ) { |
||||
objects[i] = values[i]; |
||||
regularObjects[i] = new RegularObject(values[i]); |
||||
strings[i] = String.valueOf(values[i]); |
||||
} |
||||
} |
||||
|
||||
public ArrayHolder jmeClone() { |
||||
try { |
||||
return (ArrayHolder)super.clone(); |
||||
} catch( CloneNotSupportedException e ) { |
||||
throw new RuntimeException(e); |
||||
} |
||||
} |
||||
|
||||
public void cloneFields( Cloner cloner, Object original ) { |
||||
intArray = cloner.clone(intArray); |
||||
intArray2D = cloner.clone(intArray2D); |
||||
|
||||
// Boxed types are not cloneable so this will fail
|
||||
//objects = cloner.clone(objects);
|
||||
|
||||
regularObjects = cloner.clone(regularObjects); |
||||
|
||||
// Strings are also not cloneable
|
||||
//strings = cloner.clone(strings);
|
||||
} |
||||
|
||||
public String toString() { |
||||
StringBuilder sb = new StringBuilder(); |
||||
sb.append("intArray=" + intArray); |
||||
for( int i = 0; i < intArray.length; i++ ) { |
||||
if( i == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append(intArray[i]); |
||||
} |
||||
sb.append("], "); |
||||
|
||||
sb.append("intArray2D=" + intArray2D); |
||||
for( int i = 0; i < intArray2D.length; i++ ) { |
||||
if( i == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append("intArray2D[" + i + "]=" + intArray2D[i]); |
||||
for( int j = 0; j < 2; j++ ) { |
||||
if( j == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append(intArray2D[i][j]); |
||||
} |
||||
sb.append("], "); |
||||
} |
||||
sb.append("], "); |
||||
|
||||
sb.append("objectArray=" + objects); |
||||
for( int i = 0; i < objects.length; i++ ) { |
||||
if( i == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append(objects[i]); |
||||
} |
||||
sb.append("], "); |
||||
|
||||
sb.append("objectArray=" + regularObjects); |
||||
for( int i = 0; i < regularObjects.length; i++ ) { |
||||
if( i == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append(regularObjects[i]); |
||||
} |
||||
sb.append("], "); |
||||
|
||||
sb.append("stringArray=" + strings); |
||||
for( int i = 0; i < strings.length; i++ ) { |
||||
if( i == 0 ) { |
||||
sb.append("["); |
||||
} else { |
||||
sb.append(", "); |
||||
} |
||||
sb.append(strings[i]); |
||||
} |
||||
sb.append("]"); |
||||
|
||||
return getClass().getSimpleName() + "@" + System.identityHashCode(this) |
||||
+ "[" + sb + "]"; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,78 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.material.plugin.export.material; |
||||
|
||||
import com.jme3.export.JmeExporter; |
||||
import com.jme3.export.OutputCapsule; |
||||
import com.jme3.export.Savable; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.material.MaterialDef; |
||||
|
||||
import java.io.File; |
||||
import java.io.FileOutputStream; |
||||
import java.io.IOException; |
||||
import java.io.OutputStream; |
||||
import java.io.OutputStreamWriter; |
||||
import java.nio.charset.Charset; |
||||
|
||||
/** |
||||
* Saves a Material to a j3m file with proper formatting. |
||||
* |
||||
* usage is : |
||||
* <pre> |
||||
* J3MExporter exporter = new J3MExporter(); |
||||
* exporter.save(material, myFile); |
||||
* //or
|
||||
* exporter.save(material, myOutputStream); |
||||
* </pre> |
||||
* |
||||
* @author tsr |
||||
* @author nehon (documentation and safety check) |
||||
*/ |
||||
public class J3MExporter implements JmeExporter { |
||||
|
||||
private final J3MRootOutputCapsule rootCapsule; |
||||
|
||||
/** |
||||
* Create a J3MExporter |
||||
*/ |
||||
public J3MExporter() { |
||||
rootCapsule = new J3MRootOutputCapsule(this); |
||||
} |
||||
|
||||
@Override |
||||
public void save(Savable object, OutputStream f) throws IOException { |
||||
|
||||
if (!(object instanceof Material)) { |
||||
throw new IllegalArgumentException("J3MExporter can only save com.jme3.material.Material class"); |
||||
} |
||||
|
||||
OutputStreamWriter out = new OutputStreamWriter(f, Charset.forName("UTF-8")); |
||||
|
||||
rootCapsule.clear(); |
||||
object.write(this); |
||||
rootCapsule.writeToStream(out); |
||||
|
||||
out.flush(); |
||||
} |
||||
|
||||
@Override |
||||
public void save(Savable object, File f) throws IOException { |
||||
try (FileOutputStream fos = new FileOutputStream(f)) { |
||||
save(object, fos); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public OutputCapsule getCapsule(Savable object) { |
||||
if ((object instanceof Material) || (object instanceof MaterialDef)) { |
||||
return rootCapsule; |
||||
} |
||||
|
||||
return rootCapsule.getCapsule(object); |
||||
} |
||||
|
||||
} |
@ -0,0 +1,383 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.material.plugin.export.material; |
||||
|
||||
import com.jme3.asset.TextureKey; |
||||
import com.jme3.export.OutputCapsule; |
||||
import com.jme3.export.Savable; |
||||
import com.jme3.material.MatParam; |
||||
import com.jme3.material.MatParamTexture; |
||||
import com.jme3.math.*; |
||||
import com.jme3.shader.VarType; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture.WrapMode; |
||||
import com.jme3.util.IntMap; |
||||
import java.io.IOException; |
||||
import java.io.OutputStreamWriter; |
||||
import java.nio.ByteBuffer; |
||||
import java.nio.FloatBuffer; |
||||
import java.nio.IntBuffer; |
||||
import java.nio.ShortBuffer; |
||||
import java.util.ArrayList; |
||||
import java.util.BitSet; |
||||
import java.util.HashMap; |
||||
import java.util.Map; |
||||
|
||||
/** |
||||
* |
||||
* @author tsr |
||||
*/ |
||||
public class J3MOutputCapsule implements OutputCapsule { |
||||
|
||||
private final HashMap<String, String> parameters; |
||||
protected final J3MExporter exporter; |
||||
|
||||
public J3MOutputCapsule(J3MExporter exporter) { |
||||
this.exporter = exporter; |
||||
parameters = new HashMap<>(); |
||||
} |
||||
|
||||
public void writeToStream(OutputStreamWriter out) throws IOException { |
||||
for (String key : parameters.keySet()) { |
||||
out.write(" "); |
||||
writeParameter(out, key, parameters.get(key)); |
||||
out.write("\n"); |
||||
} |
||||
} |
||||
|
||||
protected void writeParameter(OutputStreamWriter out, String name, String value) throws IOException { |
||||
out.write(name); |
||||
out.write(" : "); |
||||
out.write(value); |
||||
} |
||||
|
||||
public void clear() { |
||||
parameters.clear(); |
||||
} |
||||
|
||||
protected void putParameter(String name, String value) { |
||||
parameters.put(name, value); |
||||
} |
||||
|
||||
@Override |
||||
public void write(boolean value, String name, boolean defVal) throws IOException { |
||||
if (value == defVal) { |
||||
return; |
||||
} |
||||
|
||||
putParameter(name, ((value) ? "On" : "Off")); |
||||
} |
||||
|
||||
@Override |
||||
public void writeStringSavableMap(Map<String, ? extends Savable> map, String name, Map<String, ? extends Savable> defVal) throws IOException { |
||||
for (String key : map.keySet()) { |
||||
Savable value = map.get(key); |
||||
if (defVal == null || !defVal.containsKey(key) || !defVal.get(key).equals(value)) { |
||||
putParameter(key, format(value)); |
||||
} |
||||
} |
||||
} |
||||
|
||||
protected String format(Savable value) { |
||||
if (value instanceof MatParamTexture) { |
||||
return formatMatParamTexture((MatParamTexture) value); |
||||
} |
||||
if (value instanceof MatParam) { |
||||
return formatMatParam((MatParam) value); |
||||
} |
||||
|
||||
throw new UnsupportedOperationException(value.getClass() + ": Not supported yet."); |
||||
} |
||||
|
||||
private String formatMatParam(MatParam param){ |
||||
VarType type = param.getVarType(); |
||||
Object val = param.getValue(); |
||||
switch (type) { |
||||
case Boolean: |
||||
case Float: |
||||
case Int: |
||||
return val.toString(); |
||||
case Vector2: |
||||
Vector2f v2 = (Vector2f) val; |
||||
return v2.getX() + " " + v2.getY(); |
||||
case Vector3: |
||||
Vector3f v3 = (Vector3f) val; |
||||
return v3.getX() + " " + v3.getY() + " " + v3.getZ(); |
||||
case Vector4: |
||||
// can be either ColorRGBA, Vector4f or Quaternion
|
||||
if (val instanceof Vector4f) { |
||||
Vector4f v4 = (Vector4f) val; |
||||
return v4.getX() + " " + v4.getY() + " " |
||||
+ v4.getZ() + " " + v4.getW(); |
||||
} else if (val instanceof ColorRGBA) { |
||||
ColorRGBA color = (ColorRGBA) val; |
||||
return color.getRed() + " " + color.getGreen() + " " |
||||
+ color.getBlue() + " " + color.getAlpha(); |
||||
} else if (val instanceof Quaternion) { |
||||
Quaternion quat = (Quaternion) val; |
||||
return quat.getX() + " " + quat.getY() + " " |
||||
+ quat.getZ() + " " + quat.getW(); |
||||
} else { |
||||
throw new UnsupportedOperationException("Unexpected Vector4 type: " + val); |
||||
} |
||||
|
||||
default: |
||||
return null; // parameter type not supported in J3M
|
||||
} |
||||
} |
||||
|
||||
protected static String formatMatParamTexture(MatParamTexture param) { |
||||
StringBuilder ret = new StringBuilder(); |
||||
Texture tex = (Texture) param.getValue(); |
||||
TextureKey key; |
||||
if (tex != null) { |
||||
key = (TextureKey) tex.getKey(); |
||||
|
||||
if (key != null && key.isFlipY()) { |
||||
ret.append("Flip "); |
||||
} |
||||
|
||||
ret.append(formatWrapMode(tex, Texture.WrapAxis.S)); |
||||
ret.append(formatWrapMode(tex, Texture.WrapAxis.T)); |
||||
ret.append(formatWrapMode(tex, Texture.WrapAxis.R)); |
||||
|
||||
//Min and Mag filter
|
||||
Texture.MinFilter def = Texture.MinFilter.BilinearNoMipMaps; |
||||
if (tex.getImage().hasMipmaps() || (key != null && key.isGenerateMips())) { |
||||
def = Texture.MinFilter.Trilinear; |
||||
} |
||||
if (tex.getMinFilter() != def) { |
||||
ret.append("Min").append(tex.getMinFilter().name()).append(" "); |
||||
} |
||||
|
||||
if (tex.getMagFilter() != Texture.MagFilter.Bilinear) { |
||||
ret.append("Mag").append(tex.getMagFilter().name()).append(" "); |
||||
} |
||||
|
||||
ret.append("\"").append(key.getName()).append("\""); |
||||
} |
||||
|
||||
return ret.toString(); |
||||
} |
||||
|
||||
protected static String formatWrapMode(Texture texVal, Texture.WrapAxis axis) { |
||||
WrapMode mode; |
||||
try { |
||||
mode = texVal.getWrap(axis); |
||||
} catch (IllegalArgumentException e) { |
||||
//this axis doesn't exist on the texture
|
||||
return ""; |
||||
} |
||||
if (mode != WrapMode.EdgeClamp) { |
||||
return "Wrap" + mode.name() + "_" + axis.name() + " "; |
||||
} |
||||
return ""; |
||||
} |
||||
|
||||
@Override |
||||
public void write(Enum value, String name, Enum defVal) throws IOException { |
||||
if (value == defVal) { |
||||
return; |
||||
} |
||||
|
||||
putParameter(name, value.toString()); |
||||
} |
||||
|
||||
@Override |
||||
public void write(float value, String name, float defVal) throws IOException { |
||||
if (value == defVal) { |
||||
return; |
||||
} |
||||
|
||||
putParameter(name, Float.toString(value)); |
||||
} |
||||
|
||||
@Override |
||||
public void write(float[] value, String name, float[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(float[][] value, String name, float[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(double value, String name, double defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(double[] value, String name, double[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(double[][] value, String name, double[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(long value, String name, long defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(long[] value, String name, long[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(long[][] value, String name, long[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(short value, String name, short defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(short[] value, String name, short[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(short[][] value, String name, short[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(boolean[] value, String name, boolean[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(boolean[][] value, String name, boolean[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(String value, String name, String defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(String[] value, String name, String[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(String[][] value, String name, String[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(BitSet value, String name, BitSet defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(Savable object, String name, Savable defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(Savable[] objects, String name, Savable[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(Savable[][] objects, String name, Savable[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeSavableArrayList(ArrayList array, String name, ArrayList defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeSavableArrayListArray(ArrayList[] array, String name, ArrayList[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeSavableArrayListArray2D(ArrayList[][] array, String name, ArrayList[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeFloatBufferArrayList(ArrayList<FloatBuffer> array, String name, ArrayList<FloatBuffer> defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeByteBufferArrayList(ArrayList<ByteBuffer> array, String name, ArrayList<ByteBuffer> defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeSavableMap(Map<? extends Savable, ? extends Savable> map, String name, Map<? extends Savable, ? extends Savable> defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void writeIntSavableMap(IntMap<? extends Savable> map, String name, IntMap<? extends Savable> defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(FloatBuffer value, String name, FloatBuffer defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(IntBuffer value, String name, IntBuffer defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(ByteBuffer value, String name, ByteBuffer defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(ShortBuffer value, String name, ShortBuffer defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(byte value, String name, byte defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(byte[] value, String name, byte[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(byte[][] value, String name, byte[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(int value, String name, int defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(int[] value, String name, int[] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void write(int[][] value, String name, int[][] defVal) throws IOException { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
} |
@ -0,0 +1,81 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.material.plugin.export.material; |
||||
|
||||
import com.jme3.export.OutputCapsule; |
||||
import com.jme3.export.Savable; |
||||
import java.io.IOException; |
||||
import java.io.OutputStreamWriter; |
||||
import java.util.HashMap; |
||||
|
||||
/** |
||||
* |
||||
* @author tsr |
||||
*/ |
||||
public class J3MRenderStateOutputCapsule extends J3MOutputCapsule { |
||||
protected final static HashMap<String, String> NAME_MAP; |
||||
protected String offsetUnit; |
||||
|
||||
static { |
||||
NAME_MAP = new HashMap<>(); |
||||
NAME_MAP.put( "wireframe", "Wireframe"); |
||||
NAME_MAP.put( "cullMode", "FaceCull"); |
||||
NAME_MAP.put( "depthWrite", "DepthWrite"); |
||||
NAME_MAP.put( "depthTest", "DepthTest"); |
||||
NAME_MAP.put( "blendMode", "Blend"); |
||||
NAME_MAP.put( "alphaFallOff", "AlphaTestFalloff"); |
||||
NAME_MAP.put( "offsetFactor", "PolyOffset"); |
||||
NAME_MAP.put( "colorWrite", "ColorWrite"); |
||||
NAME_MAP.put( "pointSprite", "PointSprite"); |
||||
NAME_MAP.put( "depthFunc", "DepthFunc"); |
||||
NAME_MAP.put( "alphaFunc", "AlphaFunc"); |
||||
} |
||||
public J3MRenderStateOutputCapsule(J3MExporter exporter) { |
||||
super(exporter); |
||||
} |
||||
|
||||
public OutputCapsule getCapsule(Savable object) { |
||||
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||
} |
||||
|
||||
@Override |
||||
public void clear() { |
||||
super.clear(); |
||||
offsetUnit = ""; |
||||
} |
||||
|
||||
@Override |
||||
public void writeToStream(OutputStreamWriter out) throws IOException { |
||||
out.write(" AdditionalRenderState {\n"); |
||||
super.writeToStream(out); |
||||
out.write(" }\n"); |
||||
} |
||||
|
||||
@Override |
||||
protected void writeParameter(OutputStreamWriter out, String name, String value) throws IOException { |
||||
out.write(name); |
||||
out.write(" "); |
||||
out.write(value); |
||||
|
||||
if( "PolyOffset".equals(name) ) { |
||||
out.write(" "); |
||||
out.write(offsetUnit); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
protected void putParameter(String name, String value ) { |
||||
if( "offsetUnits".equals(name) ) { |
||||
offsetUnit = value; |
||||
return; |
||||
} |
||||
|
||||
if( !NAME_MAP.containsKey(name) ) |
||||
return; |
||||
|
||||
super.putParameter(NAME_MAP.get(name), value); |
||||
} |
||||
} |
@ -0,0 +1,99 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.material.plugin.export.material; |
||||
|
||||
import com.jme3.export.OutputCapsule; |
||||
import com.jme3.export.Savable; |
||||
|
||||
import java.io.IOException; |
||||
import java.io.OutputStreamWriter; |
||||
import java.util.HashMap; |
||||
|
||||
/** |
||||
* @author tsr |
||||
*/ |
||||
public class J3MRootOutputCapsule extends J3MOutputCapsule { |
||||
|
||||
private final HashMap<Savable, J3MOutputCapsule> outCapsules; |
||||
private String name; |
||||
private String materialDefinition; |
||||
private Boolean isTransparent; |
||||
|
||||
public J3MRootOutputCapsule(J3MExporter exporter) { |
||||
super(exporter); |
||||
outCapsules = new HashMap<>(); |
||||
} |
||||
|
||||
@Override |
||||
public void clear() { |
||||
super.clear(); |
||||
isTransparent = null; |
||||
name = ""; |
||||
materialDefinition = ""; |
||||
outCapsules.clear(); |
||||
|
||||
} |
||||
|
||||
public OutputCapsule getCapsule(Savable object) { |
||||
if (!outCapsules.containsKey(object)) { |
||||
outCapsules.put(object, new J3MRenderStateOutputCapsule(exporter)); |
||||
} |
||||
|
||||
return outCapsules.get(object); |
||||
} |
||||
|
||||
@Override |
||||
public void writeToStream(OutputStreamWriter out) throws IOException { |
||||
out.write("Material " + name + " : " + materialDefinition + " {\n\n"); |
||||
if (isTransparent != null) |
||||
out.write(" Transparent " + ((isTransparent) ? "On" : "Off") + "\n\n"); |
||||
|
||||
out.write(" MaterialParameters {\n"); |
||||
super.writeToStream(out); |
||||
out.write(" }\n\n"); |
||||
|
||||
for (J3MOutputCapsule c : outCapsules.values()) { |
||||
c.writeToStream(out); |
||||
} |
||||
out.write("}\n"); |
||||
} |
||||
|
||||
@Override |
||||
public void write(String value, String name, String defVal) throws IOException { |
||||
switch (name) { |
||||
case "material_def": |
||||
materialDefinition = value; |
||||
break; |
||||
case "name": |
||||
this.name = value; |
||||
break; |
||||
default: |
||||
throw new UnsupportedOperationException(name + " string material parameter not supported yet"); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void write(boolean value, String name, boolean defVal) throws IOException { |
||||
if( value == defVal) |
||||
return; |
||||
|
||||
switch (name) { |
||||
case "is_transparent": |
||||
isTransparent = value; |
||||
break; |
||||
default: |
||||
throw new UnsupportedOperationException(name + " boolean material parameter not supported yet"); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void write(Savable object, String name, Savable defVal) throws IOException { |
||||
if(object != null && !object.equals(defVal)) { |
||||
object.write(exporter); |
||||
} |
||||
} |
||||
|
||||
} |
@ -0,0 +1,115 @@ |
||||
/* |
||||
* Copyright (c) 2009-2016 jMonkeyEngine |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* |
||||
* * Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in the |
||||
* documentation and/or other materials provided with the distribution. |
||||
* |
||||
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||
* may be used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
*/ |
||||
package com.jme3.material.plugin; |
||||
|
||||
import com.jme3.asset.AssetInfo; |
||||
import com.jme3.asset.AssetKey; |
||||
import com.jme3.asset.AssetManager; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.material.RenderState; |
||||
import com.jme3.material.plugin.export.material.J3MExporter; |
||||
import com.jme3.material.plugins.J3MLoader; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.system.JmeSystem; |
||||
import static org.junit.Assert.*; |
||||
|
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import org.junit.Before; |
||||
import org.junit.Test; |
||||
|
||||
import java.io.ByteArrayInputStream; |
||||
import java.io.ByteArrayOutputStream; |
||||
import java.io.IOException; |
||||
import java.io.InputStream; |
||||
import java.util.Scanner; |
||||
|
||||
public class TestMaterialWrite { |
||||
|
||||
private AssetManager assetManager; |
||||
|
||||
@Before |
||||
public void init() { |
||||
assetManager = JmeSystem.newAssetManager( |
||||
TestMaterialWrite.class.getResource("/com/jme3/asset/Desktop.cfg")); |
||||
|
||||
|
||||
} |
||||
|
||||
|
||||
@Test |
||||
public void testWriteMat() throws Exception { |
||||
|
||||
Material mat = new Material(assetManager,"Common/MatDefs/Light/Lighting.j3md"); |
||||
|
||||
mat.setBoolean("UseMaterialColors", true); |
||||
mat.setColor("Diffuse", ColorRGBA.White); |
||||
mat.setColor("Ambient", ColorRGBA.DarkGray); |
||||
mat.setFloat("AlphaDiscardThreshold", 0.5f); |
||||
|
||||
mat.setFloat("Shininess", 2.5f); |
||||
|
||||
Texture tex = assetManager.loadTexture("Common/Textures/MissingTexture.png"); |
||||
tex.setMagFilter(Texture.MagFilter.Nearest); |
||||
tex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
tex.setWrap(Texture.WrapAxis.S, Texture.WrapMode.Repeat); |
||||
tex.setWrap(Texture.WrapAxis.T, Texture.WrapMode.MirroredRepeat); |
||||
|
||||
mat.setTexture("DiffuseMap", tex); |
||||
mat.getAdditionalRenderState().setDepthWrite(false); |
||||
mat.getAdditionalRenderState().setDepthTest(false); |
||||
mat.getAdditionalRenderState().setBlendMode(RenderState.BlendMode.Alpha); |
||||
|
||||
final ByteArrayOutputStream stream = new ByteArrayOutputStream(); |
||||
|
||||
J3MExporter exporter = new J3MExporter(); |
||||
try { |
||||
exporter.save(mat, stream); |
||||
} catch (IOException e) { |
||||
e.printStackTrace(); |
||||
} |
||||
|
||||
System.err.println(stream.toString()); |
||||
|
||||
J3MLoader loader = new J3MLoader(); |
||||
AssetInfo info = new AssetInfo(assetManager, new AssetKey("test")) { |
||||
@Override |
||||
public InputStream openStream() { |
||||
return new ByteArrayInputStream(stream.toByteArray()); |
||||
} |
||||
}; |
||||
Material mat2 = (Material)loader.load(info); |
||||
|
||||
assertTrue(mat.contentEquals(mat2)); |
||||
} |
||||
|
||||
} |
@ -1,11 +1,11 @@ |
||||
Material Signpost : Common/MatDefs/Light/Lighting.j3md { |
||||
MaterialParameters { |
||||
Shininess: 4.0 |
||||
DiffuseMap: Models/Sign Post/Sign Post.jpg |
||||
NormalMap: Models/Sign Post/Sign Post_normal.jpg |
||||
SpecularMap: Models/Sign Post/Sign Post_specular.jpg |
||||
DiffuseMap: "Models/Sign Post/Sign Post.jpg" |
||||
NormalMap: "Models/Sign Post/Sign Post_normal.jpg" |
||||
SpecularMap: "Models/Sign Post/Sign Post_specular.jpg" |
||||
UseMaterialColors : true |
||||
Ambient : 0.5 0.5 0.5 1.0 |
||||
Ambient : 1.0 1.0 1.0 1.0 |
||||
Diffuse : 1.0 1.0 1.0 1.0 |
||||
Specular : 1.0 1.0 1.0 1.0 |
||||
} |
||||
|
Binary file not shown.
@ -1 +0,0 @@ |
||||
X-Comment: Created with jMonkeyPlatform |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue