| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 | = AssetManager:revnumber: 2.0:revdate: 2020/07/13By assets we mean multi-media files, such as 3D models, materials, textures, scenes, custom shaders, music and sound files, and custom fonts. JME3 has an integrated asset manager that helps you keep your project assets organized. Think of the asset manager as the filesystem of your game, independent of the actual deployment platform. By default, store your assets in the `MyGame/assets/ ` directory of your project.Advantages of the AssetManager:*  The paths stay the same, no matter whether the game runs on Windows, Mac, Linux, etc!*  The AssetManager automatically caches and optimizes the handling of OpenGL objects. +For example, the same textures are not uploaded to the graphics card multiple times when multiple models use them.*  The xref:sdk:default_build_script.adoc[default build script] automatically bundles the contents of the `assets` directory into the executable.Advanced users can write a custom build and packaging script, and can register custom paths to the AssetManager, but this is up to you then.== Context[source]----jMonkeyProjects/MyGame/assets/    # You store assets in subfolders here! <------jMonkeyProjects/MyGame/build/     # SDK generates built classes here (*)jMonkeyProjects/MyGame/build.xml  # You customize Ant build script herejMonkeyProjects/MyGame/nbproject/ # SDK stores default build.xml and meta data (*)jMonkeyProjects/MyGame/dist/      # SDK generates executable distribution here (*)jMonkeyProjects/MyGame/src/       # You store Java sources herejMonkeyProjects/MyGame/test/      # You store test classes here (optional)(*) Managed by jMonkeyEngine SDK -- don't edit!----See also xref:tutorials:intermediate/best_practices.adoc[Best Practices].=== UsageThe `assetManager` object is an com.jme3.asset.AssetManager instance that every com.jme3.app.Application can access. It maintains a root that also includes your project's classpath by default, so you can load any asset that's on the classpath, that is, the top level of your project directory.You can use the inherited `assetManager` object directly, or use the accessor `app.getAssetManager()`.Here is an example how you load assets using the AssetManager. This lines loads a default Material from the built-in `Common/` directory:[source,java]----Material mat = (Material) assetManager.loadAsset(    new AssetKey("Common/Materials/RedColor.j3m"));----This Material is "`somewhere`" in the jME3 JAR; the default Asset Manager is configured to handle a `Common/…` path correctly, so you don't have to specify the whole path when referring to built-in assets (such as default Materials).Additionally, you can configure the Asset Manager and add any path to its root. This means, you can load assets from any project directory you specify. The next example shows how you load assets from your project's assets directory.=== Asset DirectoryBy default, jME3 searches for models in a directory named `assets`.[IMPORTANT]====In Java projects created with the jMonkeyEngine SDK, an `assets` folder is created by default in your project directory. If you are using any other IDE, or the command line, you simply create an `assets` directory manually (see the Codeless Project tip below).====This is our recommended directory structure for storing assets:[source]----jMonkeyProjects/MyGame/src/...           # Packages, .java source code.jMonkeyProjects/MyGame/assets/...        # The assets directory:jMonkeyProjects/MyGame/assets/Interface/   # .font, .jpg, .png, .xmljMonkeyProjects/MyGame/assets/MatDefs/     # .j3mdjMonkeyProjects/MyGame/assets/Materials/   # .j3mjMonkeyProjects/MyGame/assets/Models/      # .j3ojMonkeyProjects/MyGame/assets/Scenes/      # .j3ojMonkeyProjects/MyGame/assets/Shaders/     # .j3f, .vert, .fragjMonkeyProjects/MyGame/assets/Sounds/      # .ogg, .wavjMonkeyProjects/MyGame/assets/Textures/    # .jpg, .png; also .mesh.xml+.material, .mtl+.obj, .blend, .gltf (!)----These subdirectories are just the most common examples.[IMPORTANT]====You can rename/delete/add (sub)directories inside the `assets` directory in any way you like. Note however that there is no automatic refactoring for asset paths in the SDK, so if you modify them late in the development process, you have to refactor all paths manually.====*Examples:* You can rename `assets/Sounds` to `assets/Audio`, you can delete `assets/MatDefs` if you don't use it, you can create `assets/AIscripts`, etc. You can rename/move the `assets/Textures` directory or its subdirectories, but then you have to re-export all models, and re-convert them all to .j3o, so plan ahead![IMPORTANT]====Store textures in `assets/Textures/` before you work with them in a mesh editor! Export and save 3D model files (.mesh.xml+.material, .mtl+.obj, .gltf) into the `assets/Textures/` (!) before you convert the model to binary format (.j3o)! This ensures that texture paths correctly point to the `assets/Textures` directory. +After the conversion, you move the .j3o file into the `assets/Models/` or `assets/Scenes/` directories. This way, you can reuse textures, your binaries consistently link the correct textures, and the `assets/Models` and `assets/Scenes` directories don't become cluttered.======= Example Code: Loading AssetsCreating a material instance with the definition "`Unshaded.j3md`":[source,java]----Material mat_brick = new Material(    assetManager, "Common/MatDefs/Misc/Unshaded.j3md");----Applying a texture to the material:[source,java]----mat_brick.setTexture("ColorMap",    assetManager.loadTexture("Textures/Terrain/BrickWall/BrickWall.jpg"));----Loading a font:[source,java]----guiFont = assetManager.loadFont("Interface/Fonts/Default.fnt");----Loading a model:[source,java]----Spatial ninja = assetManager.loadModel("Models/Ninja/Ninja.mesh.xml");----Loading a scene from an Ogre3D dotScene file stored inside a zip:[source,java]----assetManager.registerLocator("town.zip", ZipLocator.class);Spatial scene = assetManager.loadModel("main.scene");rootNode.attachChild(scene);----Alternatively to ZipLocator, there is also a HttpZipLocator that can stream models from a zip file online:[source,java]----assetManager.registerLocator("https://storage.googleapis.com/"    + "google-code-archive-downloads/v2/code.google.com/"    + "jmonkeyengine/wildhouse.zip", HttpZipLocator.class);Spatial scene = assetManager.loadModel("main.scene");rootNode.attachChild(scene);----jME3 also offers a ClasspathLocator, ZipLocator, FileLocator, HttpZipLocator, and UrlLocator (see `com.jme3.asset.plugins`).[IMPORTANT]====The custom build script does not automatically include all ZIP files in the executable build. See "`Cannot Locate Resource`" solution below.======= Common AssetManager Tasks[cols="15,85", options="header"]|===a| Task?a| Solution!a| Load a model with materialsa| Use the asset manager's `loadModel()` method and attach the Spatial to the rootNode.[source,java]----Spatial elephant = assetManager.loadModel("Models/Elephant/Elephant.mesh.xml");rootNode.attachChild(elephant);----[source,java]----Spatial elephant = assetManager.loadModel("Models/Elephant/Elephant.j3o");rootNode.attachChild(elephant);----a| Load a model without materialsa| If you have a model without materials, you have to add a default material to make it visible.[source,java]----Spatial teapot = assetManager.loadModel("Models/Teapot/Teapot.obj");Material mat = new Material(assetManager, "Common/MatDefs/Misc/ShowNormals.j3md");teapot.setMaterial(mat);rootNode.attachChild(teapot);----a| Load a scenea| You load scenes just like you load models:[source,java]----Spatial scene = assetManager.loadModel("Scenes/house/main.scene");rootNode.attachChild(scene);----|====== NullPointerException: Cannot locate resource?*Problem:*My game runs fine when I run it right from the jMonkeyEngine SDK. But when I run the stand-alone executables (.jar, .jnlp .exe, .app), a DesktopAssetManager error message occurs in the console, and it quits?[source]----com.jme3.asset.DesktopAssetManager loadAssetWARNING: Cannot locate resource: Scenes/town/main.scenecom.jme3.app.Application handleErrorSEVERE: Uncaught exception thrown in Thread[LWJGL Renderer Thread,5,main]java.lang.NullPointerException----*Reason:*If you use the default build script, *original models and scenes (.mesh.xml, .obj, gltf, .zip), are excluded* from the distribution automatically. A stand-alone executable includes converted *.j3o files* (models and scenes) only. The default build script makes sure to bundle existing .j3o files in the distribution, but you need to remember to convert the models (from mesh.xmlicon:long-arrow-right[].j3o, or .objicon:long-arrow-right[].j3o, etc) yourself.*Solution*Before building the executable, you must use the jMonkeyEngine SDK's context menu action to xref:sdk:model_loader_and_viewer.adoc[convert 3D models to .j3o binary format]..  Save your original models (.mesh.xml, .scene, .gltf, or .obj files, plus textures) into `assets/Textures/`. (!).  Open the jME3 project in the jMonkeyEngine SDK..  Browse to the `assets` directory in the Projects window..  Right-click an original model in `assets/Textures/`, and choose "`Convert to JME3 binary`"..  The converted file appears in the same directory as the original file. It has the same name and a `.j3o` suffix..  Move the .j3o file into the `assets/Models/` or `assets/Scenes/` directory..  Use the assetManager's `load()` method to load the `.j3o` file.This ensures that the model's Texture paths keep working between your 3D mesh editor and JME3.[IMPORTANT]====If you must load custom assets from a non-.j3o ZIP file, you must manually ammend the xref:sdk:default_build_script.adoc[default build script] to copy ZIP files into your distribution. ZIPs are skipped by default.======= Asset Handling For Other IDEs: Codeless Projects*Problem:*I use another IDE than jMonkeyEngine SDK for coding (Eclipse, IntelliJ, text editor). Where is my `asset` folder and .j3o converter?*Solution:*You can code in any IDE, but you must create a so-called codeless project in the jMonkeyEngine SDK to maintain assets. *A code-less jMonkeyEngine project does not meddle with your sources or custom build scripts.* You merely use it to convert models to .j3o binaries..  Create your (Eclipse or whatever) project as you like..  Create a directory in your project folder and name it, for example, `assets`. +Store your assets there as described above..  Download and install the jMonkeyEngine SDK..  In the SDK, go to `menu:File[Import Projects>External Project Assets]`..  Select your (Eclipse or whatever) project and your assets folder in the Import Wizard..  You can now open this (Eclipse or whatever) project in the jMonkeyEngine SDK. +Convert assets as described above.[IMPORTANT]====If you don't use the SDK for some reason, you can still convert models to j3o format: Load any model in Ogre3D or Wavefront format with the AssetManager.loadModel() as a spatial. Then save the spatial as j3o file using xref:jme3/advanced/save_and_load.adoc[BinaryExporter].====[TIP]====Use file version control and let team members check out the project. Your developers open the project in Eclipse (etc) as they are used to. Additionally to their graphic tools, ask your graphic designers to install the jMonkeyEngine SDK, and to check out the codeless project that you just prepared. This makes it easy for non-coding team member to browse and preview game assets, to arrange scenes, and to convert files. At the same time, non-coders don't accidentally mess with code, and developers don't accidentally mess with assets. :)====
 |