diff --git a/assets/templates/code/tools.d.ts b/assets/templates/code/tools.d.ts index 3e50745da..ad0dd149a 100644 --- a/assets/templates/code/tools.d.ts +++ b/assets/templates/code/tools.d.ts @@ -74,6 +74,11 @@ interface BehaviorCodeTools { * @param name the name of the prefab to instantiate */ instantiatePrefab (name: string): T; + /** + * Instantiates a particle system set identified by the given name + * @param name the name of the particle system set to instantiate + */ + instantiateParticleSystemSet (name: string): BABYLON.ParticleSystemSet; /** * Calls the given method with the given parameters on the given object which has scripts providing the given method * @param object the object reference where to send the message by calling the given method name diff --git a/assets/templates/particles-creator/class.js b/assets/templates/particles-creator/class.js deleted file mode 100644 index e69de29bb..000000000 diff --git a/assets/templates/particles-creator/class.ts b/assets/templates/particles-creator/class.ts deleted file mode 100644 index fab8514b7..000000000 --- a/assets/templates/particles-creator/class.ts +++ /dev/null @@ -1,79 +0,0 @@ -class CustomParticles { - // Public members - public finalSize: number = 2; - - /** - * Constructor - */ - constructor () { - // Scope - const scope = this; - - // Custom update function - particleSystem.updateFunction = function (particles: BABYLON.Particle[]) { - for (var index = 0; index < particles.length; index++) { - var particle = particles[index]; - particle.age += this._scaledUpdateSpeed; - - if (particle.age < particle.lifeTime * .35) { - particle.size = scope.finalSize * particle.age / (particle.lifeTime * 0.35); - } - - if (particle.age >= particle.lifeTime) { // Recycle by swapping with last particle - this.recycleParticle(particle); - index--; - continue; - } - else { - var speed = this._scaledUpdateSpeed * 2; - if (particle.age >= particle.lifeTime / 2) { - speed = -speed; - } - - particle.colorStep.scaleToRef(speed, this._scaledColorStep); - particle.color.addInPlace(this._scaledColorStep); - - if (particle.color.a < 0) - particle.color.a = 0; - - particle.angle += particle.angularSpeed * this._scaledUpdateSpeed; - particle.direction.scaleToRef(this._scaledUpdateSpeed, this._scaledDirection); - particle.position.addInPlace(this._scaledDirection); - - this.gravity.scaleToRef(this._scaledUpdateSpeed, this._scaledGravity); - particle.direction.addInPlace(this._scaledGravity); - } - } - } - } - - /** - * Set the uniforms and samplers of the shader - * @param uniforms the shader's uniforms - * @param samplers the shader's samplers - */ - public setUniforms (uniforms: string[], samplers: string[]): void { - - } - - /** - * Set the defines of the shader - * @param defines the defines for the shader - */ - public setDefines (defines: string[]): void { - defines.push('#define CUSTOM_DEFINE'); - } - - /** - * On bind the particle system shader - * @param effect the effect for the particles - */ - public onBind (effect: BABYLON.Effect): void { - - } -} - -return { - ctor: CustomParticles, - finalSize: 2 -}; diff --git a/assets/templates/particles-creator/default-set.json b/assets/templates/particles-creator/default-set.json new file mode 100644 index 000000000..f8200f3c3 --- /dev/null +++ b/assets/templates/particles-creator/default-set.json @@ -0,0 +1,107 @@ +{ + "systems": [ + { + "name": "New Particle System", + "id": "1e0f653a-ef4b-4fae-b5ea-1917e0b4d9f4", + "capacity": 10000, + "emitterId": "aadd662b-8127-454e-905c-2e01b6f9f56c", + "particleEmitterType": { + "type": "BoxParticleEmitter", + "direction1": [ + -1, + 1, + -1 + ], + "direction2": [ + 1, + 1, + 1 + ], + "minEmitBox": [ + 0, + 0, + 0 + ], + "maxEmitBox": [ + 0, + 0, + 0 + ] + }, + "textureName": "", + "invertY": true, + "animations": [], + "beginAnimationOnStart": false, + "beginAnimationFrom": 0, + "beginAnimationTo": 60, + "beginAnimationLoop": false, + "startDelay": 0, + "renderingGroupId": 0, + "isBillboardBased": true, + "billboardMode": 7, + "minAngularSpeed": -0.5, + "maxAngularSpeed": 0.5, + "minSize": 0.1, + "maxSize": 0.5, + "minScaleX": 1, + "maxScaleX": 1, + "minScaleY": 1, + "maxScaleY": 1, + "minEmitPower": 0.5, + "maxEmitPower": 4, + "minLifeTime": 0.5, + "maxLifeTime": 2, + "emitRate": 400, + "gravity": [ + 0, + -2, + 0 + ], + "noiseStrength": [ + 10, + 10, + 10 + ], + "color1": [ + 1, + 0, + 0, + 1 + ], + "color2": [ + 0, + 1, + 1, + 1 + ], + "colorDead": [ + 0, + 0, + 0, + 1 + ], + "updateSpeed": 0.01, + "targetStopDuration": 0, + "blendMode": 0, + "preWarmCycles": 0, + "preWarmStepOffset": 1, + "minInitialRotation": 0, + "maxInitialRotation": 0, + "startSpriteCellID": 0, + "endSpriteCellID": 0, + "spriteCellChangeSpeed": 1, + "spriteCellWidth": 0, + "spriteCellHeight": 0, + "spriteRandomStartCell": false, + "isAnimationSheetEnabled": false, + "textureMask": [ + 1, + 1, + 1, + 1 + ], + "customShader": null, + "preventAutoStart": false + } + ] +} \ No newline at end of file diff --git a/assets/templates/particles-creator/shader.fragment.fx b/assets/templates/particles-creator/shader.fragment.fx deleted file mode 100644 index a750ebfb6..000000000 --- a/assets/templates/particles-creator/shader.fragment.fx +++ /dev/null @@ -1,22 +0,0 @@ -// Varying -varying vec2 vUV; -varying vec4 vColor; -#ifdef CLIPPLANE -varying float fClipDistance; -#endif - -// Uniforms -uniform sampler2D diffuseSampler; -uniform vec4 textureMask; - -// Main -void main(void) -{ - #ifdef CLIPPLANE - if (fClipDistance > 0.0) - discard; - #endif - - vec4 baseColor = texture2D(diffuseSampler, vUV); - gl_FragColor = (baseColor * textureMask + (vec4(1., 1., 1., 1.) - textureMask)) * vColor; -} \ No newline at end of file diff --git a/assets/templates/particles-creator/shader.vertex.fx b/assets/templates/particles-creator/shader.vertex.fx deleted file mode 100644 index fdd43242f..000000000 --- a/assets/templates/particles-creator/shader.vertex.fx +++ /dev/null @@ -1,55 +0,0 @@ -// Attributes -attribute vec3 position; -attribute vec4 color; -attribute vec3 options; -attribute vec2 size; -attribute float cellIndex; - -// Varyings -varying vec2 vUV; -varying vec4 vColor; - -// Uniforms -uniform mat4 view; -uniform mat4 projection; -uniform vec3 particlesInfos; - -#ifdef CLIPPLANE -uniform vec4 vClipPlane; -uniform mat4 invView; -varying float fClipDistance; -#endif - -// Main -void main(void) -{ - vec3 viewPos = (view * vec4(position, 1.0)).xyz; - vec2 cornerPos; - float angle = options.x; - vec2 offset = options.yz; - cornerPos = vec2(offset.x - 0.5, offset.y - 0.5) * size; - - vec3 rotatedCorner; - rotatedCorner.x = cornerPos.x * cos(angle) - cornerPos.y * sin(angle); - rotatedCorner.y = cornerPos.x * sin(angle) + cornerPos.y * cos(angle); - rotatedCorner.z = 0.; - - viewPos += rotatedCorner; - gl_Position = projection * vec4(viewPos, 1.0); - vColor = color; - -#ifdef ANIMATESHEET - float rowOffset = floor(cellIndex / particlesInfos.z); - float columnOffset = cellIndex - rowOffset * particlesInfos.z; - vec2 uvScale = particlesInfos.xy; - vec2 uvOffset = vec2(offset.x, 1.0 - offset.y); - vUV = (uvOffset + vec2(columnOffset, rowOffset)) * uvScale; -#else - vUV = offset; -#endif - -#ifdef CLIPPLANE - vec4 worldPos = invView * vec4(viewPos, 1.0); - fClipDistance = dot(worldPos, vClipPlane); -#endif -} \ No newline at end of file diff --git a/babylonjs-editor.d.ts b/babylonjs-editor.d.ts index b9b14bb57..1d0c39fc8 100644 --- a/babylonjs-editor.d.ts +++ b/babylonjs-editor.d.ts @@ -13,6 +13,7 @@ declare module 'babylonjs-editor' { import Request from 'babylonjs-editor/editor/tools/request'; import UndoRedo from 'babylonjs-editor/editor/tools/undo-redo'; import ThemeSwitcher, { ThemeType } from 'babylonjs-editor/editor/tools/theme'; + import GraphicsTools from 'babylonjs-editor/editor/tools/graphics-tools'; import Layout from 'babylonjs-editor/editor/gui/layout'; import Toolbar from 'babylonjs-editor/editor/gui/toolbar'; import List from 'babylonjs-editor/editor/gui/list'; @@ -37,9 +38,11 @@ declare module 'babylonjs-editor' { import ScenePreview from 'babylonjs-editor/editor/scene/scene-preview'; import PrefabAssetComponent from 'babylonjs-editor/editor/prefabs/asset-component'; import { Prefab, PrefabNodeType } from 'babylonjs-editor/editor/prefabs/prefab'; + import ParticlesCreatorExtension, { ParticlesCreatorMetadata } from 'babylonjs-editor/editor/particles/asset-component'; + import Storage from 'babylonjs-editor/editor/storage/storage'; import VSCodeSocket from 'babylonjs-editor/editor/vscode/vscode-socket'; export default Editor; - export { Editor, Tools, Request, UndoRedo, ThemeSwitcher, ThemeType, IStringDictionary, INumberDictionary, IDisposable, EditorPlugin, Layout, Toolbar, List, Grid, GridRow, Picker, Graph, GraphNode, Window, CodeEditor, Form, Edition, Tree, TreeContextMenuItem, TreeNode, Dialog, ContextMenu, ContextMenuItem, ResizableLayout, ComponentConfig, ItemConfigType, AbstractEditionTool, ProjectRoot, CodeProjectEditorFactory, SceneManager, SceneFactory, ScenePreview, PrefabAssetComponent, Prefab, PrefabNodeType, VSCodeSocket }; + export { Editor, Tools, Request, UndoRedo, ThemeSwitcher, ThemeType, GraphicsTools, IStringDictionary, INumberDictionary, IDisposable, EditorPlugin, Layout, Toolbar, List, Grid, GridRow, Picker, Graph, GraphNode, Window, CodeEditor, Form, Edition, Tree, TreeContextMenuItem, TreeNode, Dialog, ContextMenu, ContextMenuItem, ResizableLayout, ComponentConfig, ItemConfigType, AbstractEditionTool, ProjectRoot, CodeProjectEditorFactory, SceneManager, SceneFactory, ScenePreview, PrefabAssetComponent, Prefab, PrefabNodeType, ParticlesCreatorExtension, ParticlesCreatorMetadata, Storage, VSCodeSocket }; } declare module 'babylonjs-editor/editor/editor' { @@ -276,6 +279,11 @@ declare module 'babylonjs-editor/editor/tools/tools' { * @param sources One or more source objects from which to copy properties */ static Assign(target: Object, ...sources: Object[]): T; + /** + * Deep clones the given object. Take care of cycling objects! + * @param data the data of the object to clone + */ + static Clone(data: T): T; /** * Reads the given file * @param file the file to read @@ -400,6 +408,22 @@ declare module 'babylonjs-editor/editor/tools/theme' { } } +declare module 'babylonjs-editor/editor/tools/graphics-tools' { + import { BaseTexture } from 'babylonjs'; + export default class GraphicsTools { + /** + * Configures the given texture to retrieve its pixels and create a new file (blob) + * @param tex the texture to transform to a blob + */ + static TextureToFile(tex: BaseTexture): Promise; + /** + * Converts the given canvas data to blob + * @param canvas the canvas to take its data and convert to a blob + */ + static CanvasToBlob(canvas: HTMLCanvasElement): Promise; + } +} + declare module 'babylonjs-editor/editor/gui/layout' { export default class Layout { element: W2UI.W2Layout; @@ -921,7 +945,7 @@ declare module 'babylonjs-editor/editor/gui/form' { } declare module 'babylonjs-editor/editor/gui/edition' { - import { Color3, Color4, Vector2, Vector3, Vector4, BaseTexture } from 'babylonjs'; + import { Color3, Color4, Vector2, Vector3, Vector4, BaseTexture, Scene } from 'babylonjs'; import * as dat from 'dat-gui'; import Editor from 'babylonjs-editor/editor/editor'; export default class Edition { @@ -1002,7 +1026,7 @@ declare module 'babylonjs-editor/editor/gui/edition' { * @param object the object which has a texture * @param callback: called when changed texture */ - addTexture(parent: dat.GUI, editor: Editor, property: string, object: any, allowCubes?: boolean, onlyCubes?: boolean, callback?: (texture: BaseTexture) => void): dat.GUIController; + addTexture(parent: dat.GUI, editor: Editor, scene: Scene, property: string, object: any, allowCubes?: boolean, onlyCubes?: boolean, callback?: (texture: BaseTexture) => void): dat.GUIController; } } @@ -1893,6 +1917,152 @@ declare module 'babylonjs-editor/editor/prefabs/prefab' { } } +declare module 'babylonjs-editor/editor/particles/asset-component' { + import { AbstractMesh, PickingInfo } from 'babylonjs'; + import { IAssetComponent, AssetElement } from 'babylonjs-editor/extensions/typings/asset'; + import Editor from 'babylonjs-editor/editor/editor'; + export interface ParticlesCreatorMetadata { + name: string; + psData: any; + } + export default class ParticlesAssetComponent implements IAssetComponent { + editor: Editor; + id: string; + assetsCaption: string; + datas: AssetElement[]; + /** + * Constructor + * @param scene: the babylonjs scene + */ + constructor(editor: Editor); + /** + * On the user renames the asset + * @param asset the asset being renamed + * @param name the new name of the asset + */ + onRenameAsset(asset: AssetElement, name: string): void; + /** + * On the user wants to remove the asset + * @param asset the asset to remove + */ + onRemoveAsset(asset: AssetElement): void; + /** + * On the user adds an asset + * @param asset the asset to add + */ + onAddAsset(asset: AssetElement): void; + /** + * Creates a new particle systems set asset + */ + onCreateAsset(name: string): Promise>; + /** + * On get all the assets to be drawn in the assets component + */ + onGetAssets(): AssetElement[]; + /** + * On the user double clicks on asset + * @param asset the asset being double-clicked by the user + */ + onDoubleClickAsset(asset: AssetElement): void; + /** + * On the user drops an asset in the scene + * @param targetMesh the mesh under the pointer + * @param asset the asset being dropped + * @param pickInfo the pick info once the user dropped the asset + */ + onDragAndDropAsset(targetMesh: AbstractMesh, asset: AssetElement, pickInfo: PickingInfo): void; + /** + * Called by the editor when serializing the scene + */ + onSerializeAssets(): AssetElement[]; + /** + * On the user loads the editor project + * @param data the previously saved data + */ + onParseAssets(data: AssetElement[]): void; + } +} + +declare module 'babylonjs-editor/editor/storage/storage' { + import Editor from 'babylonjs-editor/editor/editor'; + import Picker from 'babylonjs-editor/editor/gui/picker'; + export type FileType = string | Uint8Array | ArrayBuffer; + export interface CreateFiles { + name: string; + data?: FileType | Promise; + file?: File; + folder?: CreateFiles[]; + doNotOverride?: boolean; + } + export interface GetFiles { + name: string; + folder: any; + } + export default abstract class Storage { + editor: Editor; + picker: Picker; + onCreateFiles: (folder: string) => void; + protected filesCount: number; + protected _uploadedCount: number; + /** + * Returns the appropriate storage (OneDrive, Electron, etc.) + * @param editor the editor reference + */ + static GetStorage(editor: Editor): Promise; + /** + * Constructor + * @param editor: the editor reference + */ + constructor(editor: Editor); + /** + * Opens the folder picker + * @param title the title of the picker + * @param filesToWrite the array of files to write on the HDD + * @param folder the current working directory to browse + * @param overrideFilename if the file browser should override the filename + */ + openPicker(title: string, filesToWrite: CreateFiles[], folder?: string, overrideFilename?: boolean): Promise; + /** + * Uploads the files + * @param folder the target folder + * @param filesToWrite the files to upload + */ + protected uploadFiles(folder: string, filesToWrite: CreateFiles[]): Promise; + /** + * Recursively creates the given files (uncluding folders) + * @param folder: the parent folder of the files + * @param files files to create + */ + protected recursivelyCreateFiles(folder: any, files: CreateFiles[]): Promise; + /** + * Returns the number of files to upload + * @param files the files to count + */ + protected recursivelyGetFilesToUploadCount(files: CreateFiles[]): number; + /** + * Returns the number of uploaded files + */ + protected uploadedCount: number; + /** + * Creates the given folders + * @param folder the parent folder + * @param names the folders names + */ + abstract createFolders(folder: any, names: string[]): Promise; + /** + * Creates the given files + * @param folder the parent folder + * @param files the files to write + */ + abstract createFiles(folder: any, files: CreateFiles[]): Promise; + /** + * Returns the files available in the given folder + * @param folder the parent folder + */ + abstract getFiles(folder?: any): Promise; + } +} + declare module 'babylonjs-editor/editor/vscode/vscode-socket' { import Editor from 'babylonjs-editor/editor/editor'; export default class VSCodeSocket { @@ -1973,13 +2143,8 @@ declare module 'babylonjs-editor/editor/core' { onResize: Observable<{}>; onAddObject: Observable<{}>; onRemoveObject: Observable<{}>; - onGlobalPropertyChange: Observable<{ - baseObject?: any; - object: any; - property: string; - value: any; - initialValue: any; - }>; + onModifyingObject: Observable<{}>; + onModifiedObject: Observable<{}>; onDropFiles: Observable<{ target: HTMLElement; files: FileList; @@ -2300,6 +2465,7 @@ declare module 'babylonjs-editor/editor/components/assets' { import Toolbar from 'babylonjs-editor/editor/gui/toolbar'; import { IAssetComponent, AssetElement } from 'babylonjs-editor/extensions/typings/asset'; import PrefabAssetComponent from 'babylonjs-editor/editor/prefabs/asset-component'; + import ParticlesAssetComponent from 'babylonjs-editor/editor/particles/asset-component'; import { IStringDictionary } from 'babylonjs-editor/editor/typings/typings'; export interface AssetPreviewData { asset: AssetElement; @@ -2314,6 +2480,7 @@ declare module 'babylonjs-editor/editor/components/assets' { toolbar: Toolbar; components: IAssetComponent[]; prefabs: PrefabAssetComponent; + particles: ParticlesAssetComponent; assetPreviewDatas: AssetPreviewData[]; protected currentComponent: IAssetComponent; protected emptyTextNode: HTMLHeadElement; @@ -2542,10 +2709,6 @@ declare module 'babylonjs-editor/editor/scene/scene-icons' { * @param editor: the editor instance */ constructor(editor: Editor); - /** - * On before render the scene - */ - onPreUpdate(): void; /** * On post update the scenes */ diff --git a/css/editor.css b/css/editor.css index 686817360..49232d944 100644 --- a/css/editor.css +++ b/css/editor.css @@ -138,6 +138,10 @@ html, body { z-index: 0; } +.menu { + overflow: hidden !important; +} + /* JSTREE */ .vakata-context li > a { font-size: 12px; diff --git a/src/editor/components/assets.ts b/src/editor/components/assets.ts index 0090093c5..f909feb23 100644 --- a/src/editor/components/assets.ts +++ b/src/editor/components/assets.ts @@ -12,6 +12,7 @@ import Toolbar from '../gui/toolbar'; import { IAssetComponent, AssetElement } from '../../extensions/typings/asset'; import PrefabAssetComponent from '../prefabs/asset-component'; +import ParticlesAssetComponent from '../particles/asset-component'; import { Dialog } from 'babylonjs-editor'; import VSCodeSocket from '../vscode/vscode-socket'; @@ -33,6 +34,8 @@ export default class EditorAssets { public components: IAssetComponent[] = []; public prefabs: PrefabAssetComponent; + public particles: ParticlesAssetComponent; + public assetPreviewDatas: AssetPreviewData[] = []; // Protected members @@ -84,6 +87,7 @@ export default class EditorAssets { // Create components this.prefabs = new PrefabAssetComponent(editor); + this.particles = new ParticlesAssetComponent(editor); // Add components tabs this.addDefaultComponents(); @@ -119,6 +123,7 @@ export default class EditorAssets { */ public addDefaultComponents (): void { this.addTab(this.prefabs); + this.addTab(this.particles); } /** diff --git a/src/editor/components/inspector.ts b/src/editor/components/inspector.ts index 2e834389e..ad0a2c24a 100644 --- a/src/editor/components/inspector.ts +++ b/src/editor/components/inspector.ts @@ -9,7 +9,7 @@ import NodeTool from '../edition-tools/node-tool'; import LightTool from '../edition-tools/light-tool'; import PhysicsTool from '../edition-tools/physics-tool'; import RenderTargetTool from '../edition-tools/render-target-tool'; -import ParticleSystemTool from '../edition-tools/particle-system-tool'; +import ParticleSystemTool from '../edition-tools/particles/particle-system-tool'; import SoundTool from '../edition-tools/sound-tool'; import AnimationTool from '../edition-tools/animation-tool'; @@ -210,12 +210,18 @@ export default class EditorInspector { if (t.divId === this.lastTabName) lastTool = t; + // On change + t.tool.onChange(t.tool.element, (property, result, object, initialValue) => { + this.editor.core.onModifyingObject.notifyObservers(this.currentObject); + }); + // Manage undo / redo t.tool.onFinishChange(t.tool.element, (property, result, object, initialValue) => { UndoRedo.Push({ baseObject: t.object, property: property, to: result, from: initialValue, object: object }); Tags.AddTagsTo(t.object, 'modified'); this.editor.graph.updateObjectMark(t.object); t.onModified && t.onModified(); + this.editor.core.onModifiedObject.notifyObservers(this.currentObject); }); this.currentTools.push(t); diff --git a/src/editor/components/toolbar.ts b/src/editor/components/toolbar.ts index 2a8dfe3a0..38c9ecbfe 100644 --- a/src/editor/components/toolbar.ts +++ b/src/editor/components/toolbar.ts @@ -86,7 +86,7 @@ export default class EditorToolbar { { type: 'break' }, { id: 'material-editor', img: 'icon-shaders', text: 'Material Editor...' }, { id: 'post-process-editor', img: 'icon-shaders', text: 'Post-Process Editor...' }, - // { id: 'particles-creator', img: 'icon-particles', text: 'Particles Creator' }, + { id: 'particles-creator', img: 'icon-particles', text: 'Particles Creator' }, { type: 'break' }, { id: 'path-finder', img: 'icon-graph', text: 'Path Finder...' }, { type: 'break' }, diff --git a/src/editor/core.ts b/src/editor/core.ts index d7db0f238..6442f8a4e 100644 --- a/src/editor/core.ts +++ b/src/editor/core.ts @@ -33,7 +33,8 @@ export default class Core { public onResize: Observable<{ }> = new Observable<{ }>(); public onAddObject: Observable<{ }> = new Observable<{ }>(); public onRemoveObject: Observable<{ }> = new Observable<{ }>(); - public onGlobalPropertyChange = new Observable<{ baseObject?: any; object: any; property: string; value: any; initialValue: any; }>(); + public onModifyingObject: Observable<{ }> = new Observable<{ }>(); + public onModifiedObject: Observable<{ }> = new Observable<{ }>(); public onDropFiles = new Observable<{ target: HTMLElement; files: FileList }>(); public onSceneLoaded = new Observable<{ scene: Scene, file: File, project?: ProjectRoot }>(); diff --git a/src/editor/edition-tools/gui/image.ts b/src/editor/edition-tools/gui/image.ts index f9fe8c1fd..d3e57613b 100644 --- a/src/editor/edition-tools/gui/image.ts +++ b/src/editor/edition-tools/gui/image.ts @@ -54,7 +54,7 @@ export default class GuiImageTool extends AbstractEditionTool { const texture = this.tool.addFolder('Texture'); texture.open(); - this.tool.addTexture(texture, this.editor, '_texture', this, false, false, (tex) => { + this.tool.addTexture(texture, this.editor, this.editor.core.scene, '_texture', this, false, false, (tex) => { let blobURL = ''; try { blobURL = URL.createObjectURL(FilesInputStore.FilesToLoad[tex['url']]); diff --git a/src/editor/edition-tools/materials/cell-tool.ts b/src/editor/edition-tools/materials/cell-tool.ts index 4a64fd7c9..e3e35b3d9 100644 --- a/src/editor/edition-tools/materials/cell-tool.ts +++ b/src/editor/edition-tools/materials/cell-tool.ts @@ -27,7 +27,7 @@ export default class CellMaterialTool extends MaterialTool { diffuse.open(); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor).open(); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture', this.object, false).name('Texture'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture', this.object, false).name('Texture'); // Cell const cell = this.tool.addFolder('Fire'); diff --git a/src/editor/edition-tools/materials/custom-tool.ts b/src/editor/edition-tools/materials/custom-tool.ts index 03a251ec8..202e2ed97 100644 --- a/src/editor/edition-tools/materials/custom-tool.ts +++ b/src/editor/edition-tools/materials/custom-tool.ts @@ -1,75 +1,75 @@ -import { Material, Vector2, Vector3 } from 'babylonjs'; - -import CustomEditorMaterial from '../../../extensions/material-editor/material'; - -import MaterialTool from './material-tool'; -import Tools from '../../tools/tools'; - -export default class CustomMaterialTool extends MaterialTool { - // Public members - public divId: string = 'CUSTOM-MATERIAL-TOOL'; - public tabName: string = 'Custom Material'; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported(object: any): boolean { - return super.isSupported(object) && this.object.getClassName && this.object.getClassName() === 'CustomMaterial'; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update(object: any): void { - super.update(object); - this.setTabName('Custom Material'); - - // Get current config of the post-process - const config = this.object.config; - - // Base Color - this.tool.addColor(this.tool.element, 'Base Color', this.object.baseColor).open(); - - // Floats - const floats = this.tool.addFolder('Floats'); - floats.open(); - - config.floats.forEach(f => { - if (this.object.userConfig[f] === undefined) - this.object.userConfig[f] = 1; - - floats.add(this.object.userConfig, f).step(0.01).name(f).onChange(() => this.object.markAsDirty(Material.MiscDirtyFlag)); - }); - - // Vectors - const vectors = this.tool.addFolder('Vectors'); - vectors.open(); - - config.vectors2.forEach(v => { - if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector2)) - this.object.userConfig[v] = Vector2.Zero(); - - this.tool.addVector(vectors, v, this.object.userConfig[v], () => this.object.markAsDirty(Material.MiscDirtyFlag)).open(); - }); - - config.vectors3.forEach(v => { - if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector3)) - this.object.userConfig[v] = Vector3.Zero(); - - this.tool.addVector(vectors, v, this.object.userConfig[v], () => this.object.markAsDirty(Material.MiscDirtyFlag)).open(); - }); - - // Samplers - const samplers = this.tool.addFolder('Samplers'); - samplers.open(); - - config.textures.forEach(t => { - this.tool.addTexture(samplers, this.editor, t.name, this.object.userConfig, false, false, () => this.object.markAsDirty(Material.TextureDirtyFlag)).name(t.name); - }); - - // Options - super.addOptions(); - } -} +import { Material, Vector2, Vector3 } from 'babylonjs'; + +import CustomEditorMaterial from '../../../extensions/material-editor/material'; + +import MaterialTool from './material-tool'; +import Tools from '../../tools/tools'; + +export default class CustomMaterialTool extends MaterialTool { + // Public members + public divId: string = 'CUSTOM-MATERIAL-TOOL'; + public tabName: string = 'Custom Material'; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported(object: any): boolean { + return super.isSupported(object) && this.object.getClassName && this.object.getClassName() === 'CustomMaterial'; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update(object: any): void { + super.update(object); + this.setTabName('Custom Material'); + + // Get current config of the post-process + const config = this.object.config; + + // Base Color + this.tool.addColor(this.tool.element, 'Base Color', this.object.baseColor).open(); + + // Floats + const floats = this.tool.addFolder('Floats'); + floats.open(); + + config.floats.forEach(f => { + if (this.object.userConfig[f] === undefined) + this.object.userConfig[f] = 1; + + floats.add(this.object.userConfig, f).step(0.01).name(f).onChange(() => this.object.markAsDirty(Material.MiscDirtyFlag)); + }); + + // Vectors + const vectors = this.tool.addFolder('Vectors'); + vectors.open(); + + config.vectors2.forEach(v => { + if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector2)) + this.object.userConfig[v] = Vector2.Zero(); + + this.tool.addVector(vectors, v, this.object.userConfig[v], () => this.object.markAsDirty(Material.MiscDirtyFlag)).open(); + }); + + config.vectors3.forEach(v => { + if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector3)) + this.object.userConfig[v] = Vector3.Zero(); + + this.tool.addVector(vectors, v, this.object.userConfig[v], () => this.object.markAsDirty(Material.MiscDirtyFlag)).open(); + }); + + // Samplers + const samplers = this.tool.addFolder('Samplers'); + samplers.open(); + + config.textures.forEach(t => { + this.tool.addTexture(samplers, this.editor, this.editor.core.scene, t.name, this.object.userConfig, false, false, () => this.object.markAsDirty(Material.TextureDirtyFlag)).name(t.name); + }); + + // Options + super.addOptions(); + } +} diff --git a/src/editor/edition-tools/materials/fire-tool.ts b/src/editor/edition-tools/materials/fire-tool.ts index c9f7c6dd4..b3f18ba6b 100644 --- a/src/editor/edition-tools/materials/fire-tool.ts +++ b/src/editor/edition-tools/materials/fire-tool.ts @@ -28,15 +28,15 @@ export default class FireMaterialTool extends MaterialTool { diffuse.open(); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor).open(); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture', this.object, false).name('Texture'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture', this.object, false).name('Texture'); // Fire const fire = this.tool.addFolder('Fire'); fire.open(); fire.add(this.object, 'speed').min(0).step(0.01).name('Speed'); - this.tool.addTexture(fire, this.editor, 'distortionTexture', this.object, false).name('Distortion') - this.tool.addTexture(fire, this.editor, 'opacityTexture', this.object, false).name('Opacity'); + this.tool.addTexture(fire, this.editor, this.editor.core.scene, 'distortionTexture', this.object, false).name('Distortion') + this.tool.addTexture(fire, this.editor, this.editor.core.scene, 'opacityTexture', this.object, false).name('Opacity'); // Options super.addOptions(); diff --git a/src/editor/edition-tools/materials/fur-tool.ts b/src/editor/edition-tools/materials/fur-tool.ts index f3a649c38..77ce8ee75 100644 --- a/src/editor/edition-tools/materials/fur-tool.ts +++ b/src/editor/edition-tools/materials/fur-tool.ts @@ -28,7 +28,7 @@ export default class FurMaterialTool extends MaterialTool { diffuse.open(); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor, () => this.object.updateFur()).open(); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture', this.object, false, false, () => this.object.updateFur()).name('Texture'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture', this.object, false, false, () => this.object.updateFur()).name('Texture'); // Fur const fur = this.tool.addFolder('Fur'); diff --git a/src/editor/edition-tools/materials/lava-tool.ts b/src/editor/edition-tools/materials/lava-tool.ts index 4882ee8f6..7a1c37bdc 100644 --- a/src/editor/edition-tools/materials/lava-tool.ts +++ b/src/editor/edition-tools/materials/lava-tool.ts @@ -28,13 +28,13 @@ export default class LavaMaterialTool extends MaterialTool { diffuse.open(); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor).open(); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture', this.object, false).name('Texture'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture', this.object, false).name('Texture'); // Lava const lava = this.tool.addFolder('Lava'); lava.open(); - this.tool.addTexture(lava, this.editor, 'noiseTexture', this.object, false).name('Noise'); + this.tool.addTexture(lava, this.editor, this.editor.core.scene, 'noiseTexture', this.object, false).name('Noise'); lava.add(this.object, 'movingSpeed').min(0).name('Moving Speed'); lava.add(this.object, 'lowFrequencySpeed').min(0).name('Low Frequency Speed'); diff --git a/src/editor/edition-tools/materials/mix-tool.ts b/src/editor/edition-tools/materials/mix-tool.ts index 3fa4c7aa0..9668baad7 100644 --- a/src/editor/edition-tools/materials/mix-tool.ts +++ b/src/editor/edition-tools/materials/mix-tool.ts @@ -26,21 +26,21 @@ export default class MixMaterialTool extends MaterialTool { const mixmap1 = this.tool.addFolder('Mix Map 1'); mixmap1.open(); - this.tool.addTexture(mixmap1, this.editor, 'mixTexture1', this.object, false).name('Mix Texture 1'); - this.tool.addTexture(mixmap1, this.editor, 'diffuseTexture1', this.object, false).name('Diffuse Texture 1'); - this.tool.addTexture(mixmap1, this.editor, 'diffuseTexture2', this.object, false).name('Diffuse Texture 2'); - this.tool.addTexture(mixmap1, this.editor, 'diffuseTexture3', this.object, false).name('Diffuse Texture 3'); - this.tool.addTexture(mixmap1, this.editor, 'diffuseTexture4', this.object, false).name('Diffuse Texture 4'); + this.tool.addTexture(mixmap1, this.editor, this.editor.core.scene, 'mixTexture1', this.object, false).name('Mix Texture 1'); + this.tool.addTexture(mixmap1, this.editor, this.editor.core.scene, 'diffuseTexture1', this.object, false).name('Diffuse Texture 1'); + this.tool.addTexture(mixmap1, this.editor, this.editor.core.scene, 'diffuseTexture2', this.object, false).name('Diffuse Texture 2'); + this.tool.addTexture(mixmap1, this.editor, this.editor.core.scene, 'diffuseTexture3', this.object, false).name('Diffuse Texture 3'); + this.tool.addTexture(mixmap1, this.editor, this.editor.core.scene, 'diffuseTexture4', this.object, false).name('Diffuse Texture 4'); // Mix map 2 const mixmap2 = this.tool.addFolder('Mix Map 2'); mixmap2.open(); - this.tool.addTexture(mixmap2, this.editor, 'mixTexture2', this.object, false).name('Mix Texture 2'); - this.tool.addTexture(mixmap2, this.editor, 'diffuseTexture5', this.object, false).name('Diffuse Texture 5'); - this.tool.addTexture(mixmap2, this.editor, 'diffuseTexture6', this.object, false).name('Diffuse Texture 6'); - this.tool.addTexture(mixmap2, this.editor, 'diffuseTexture7', this.object, false).name('Diffuse Texture 7'); - this.tool.addTexture(mixmap2, this.editor, 'diffuseTexture8', this.object, false).name('Diffuse Texture 8'); + this.tool.addTexture(mixmap2, this.editor, this.editor.core.scene, 'mixTexture2', this.object, false).name('Mix Texture 2'); + this.tool.addTexture(mixmap2, this.editor, this.editor.core.scene, 'diffuseTexture5', this.object, false).name('Diffuse Texture 5'); + this.tool.addTexture(mixmap2, this.editor, this.editor.core.scene, 'diffuseTexture6', this.object, false).name('Diffuse Texture 6'); + this.tool.addTexture(mixmap2, this.editor, this.editor.core.scene, 'diffuseTexture7', this.object, false).name('Diffuse Texture 7'); + this.tool.addTexture(mixmap2, this.editor, this.editor.core.scene, 'diffuseTexture8', this.object, false).name('Diffuse Texture 8'); // Options super.addOptions(); diff --git a/src/editor/edition-tools/materials/pbr-metallic-roughness-tool.ts b/src/editor/edition-tools/materials/pbr-metallic-roughness-tool.ts index cd55f09f0..75e8ded9a 100644 --- a/src/editor/edition-tools/materials/pbr-metallic-roughness-tool.ts +++ b/src/editor/edition-tools/materials/pbr-metallic-roughness-tool.ts @@ -1,73 +1,73 @@ -import { PBRMetallicRoughnessMaterial } from 'babylonjs'; - -import MaterialTool from './material-tool'; -import Tools from '../../tools/tools'; - -export default class PBRMetallicRoughnessMaterialTool extends MaterialTool { - // Public members - public divId: string = 'PBR-METALLIC-ROUGHNESS-MATERIAL-TOOL'; - public tabName: string = 'PBR Metallic Roughness'; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported(object: any): boolean { - return super.isSupported(object) && this.object instanceof PBRMetallicRoughnessMaterial; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update(object: any): void { - super.update(object); - - // Base Color - const baseColor = this.tool.addFolder('Base'); - baseColor.open(); - - this.tool.addColor(baseColor, 'Base Color', this.object.baseColor).open(); - this.tool.addTexture(baseColor, this.editor, 'baseTexture', this.object, false).name('Texture'); - - // Bump - const normal = this.tool.addFolder('Normal'); - normal.open(); - this.tool.addTexture(normal, this.editor, 'normalTexture', this.object).name('Normal Texture'); - normal.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); - normal.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); - - // Reflection - const reflection = this.tool.addFolder('Reflection'); - reflection.open(); - this.tool.addTexture(reflection, this.editor, 'environmentTexture', this.object, true, false).name('Environment Texture'); - - // Metallic Roughness - const metallic = this.tool.addFolder('Metallic Roughness'); - metallic.open(); - metallic.add(this.object, 'roughness').step(0.01).name('Roughness'); - metallic.add(this.object, 'metallic').step(0.01).name('Metallic'); - this.tool.addTexture(metallic, this.editor, 'metallicRoughnessTexture', this.object).name('Metallic Roughness Texture'); - - // Emissive - const emissive = this.tool.addFolder('Emissive'); - emissive.open(); - this.tool.addColor(emissive, 'Color', this.object.emissiveColor).open(); - this.tool.addTexture(emissive, this.editor, 'emissiveTexture', this.object).name('Emissive Texture'); - - // Lightmap - const lightmap = this.tool.addFolder('Lightmap'); - lightmap.open(); - lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); - this.tool.addTexture(lightmap, this.editor, 'lightmapTexture', this.object).name('Lightmap Texture'); - - // Occlusion - const occlusion = this.tool.addFolder('Occlusion'); - occlusion.open(); - occlusion.add(this.object, 'occlusionStrength').name('Occlusion Strength'); - this.tool.addTexture(occlusion, this.editor, 'occlusionTexture', this.object).name('Occlusion Texture'); - - // Options - super.addOptions(); - } -} +import { PBRMetallicRoughnessMaterial } from 'babylonjs'; + +import MaterialTool from './material-tool'; +import Tools from '../../tools/tools'; + +export default class PBRMetallicRoughnessMaterialTool extends MaterialTool { + // Public members + public divId: string = 'PBR-METALLIC-ROUGHNESS-MATERIAL-TOOL'; + public tabName: string = 'PBR Metallic Roughness'; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported(object: any): boolean { + return super.isSupported(object) && this.object instanceof PBRMetallicRoughnessMaterial; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update(object: any): void { + super.update(object); + + // Base Color + const baseColor = this.tool.addFolder('Base'); + baseColor.open(); + + this.tool.addColor(baseColor, 'Base Color', this.object.baseColor).open(); + this.tool.addTexture(baseColor, this.editor, this.editor.core.scene, 'baseTexture', this.object, false).name('Texture'); + + // Bump + const normal = this.tool.addFolder('Normal'); + normal.open(); + this.tool.addTexture(normal, this.editor, this.editor.core.scene, 'normalTexture', this.object).name('Normal Texture'); + normal.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); + normal.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); + + // Reflection + const reflection = this.tool.addFolder('Reflection'); + reflection.open(); + this.tool.addTexture(reflection, this.editor, this.editor.core.scene, 'environmentTexture', this.object, true, false).name('Environment Texture'); + + // Metallic Roughness + const metallic = this.tool.addFolder('Metallic Roughness'); + metallic.open(); + metallic.add(this.object, 'roughness').step(0.01).name('Roughness'); + metallic.add(this.object, 'metallic').step(0.01).name('Metallic'); + this.tool.addTexture(metallic, this.editor, this.editor.core.scene, 'metallicRoughnessTexture', this.object).name('Metallic Roughness Texture'); + + // Emissive + const emissive = this.tool.addFolder('Emissive'); + emissive.open(); + this.tool.addColor(emissive, 'Color', this.object.emissiveColor).open(); + this.tool.addTexture(emissive, this.editor, this.editor.core.scene, 'emissiveTexture', this.object).name('Emissive Texture'); + + // Lightmap + const lightmap = this.tool.addFolder('Lightmap'); + lightmap.open(); + lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); + this.tool.addTexture(lightmap, this.editor, this.editor.core.scene, 'lightmapTexture', this.object).name('Lightmap Texture'); + + // Occlusion + const occlusion = this.tool.addFolder('Occlusion'); + occlusion.open(); + occlusion.add(this.object, 'occlusionStrength').name('Occlusion Strength'); + this.tool.addTexture(occlusion, this.editor, this.editor.core.scene, 'occlusionTexture', this.object).name('Occlusion Texture'); + + // Options + super.addOptions(); + } +} diff --git a/src/editor/edition-tools/materials/pbr-specular-glossiness-tool.ts b/src/editor/edition-tools/materials/pbr-specular-glossiness-tool.ts index 7f472e83f..e67fece6f 100644 --- a/src/editor/edition-tools/materials/pbr-specular-glossiness-tool.ts +++ b/src/editor/edition-tools/materials/pbr-specular-glossiness-tool.ts @@ -1,71 +1,71 @@ -import { PBRSpecularGlossinessMaterial } from 'babylonjs'; - -import MaterialTool from './material-tool'; -import Tools from '../../tools/tools'; - -export default class PBRSpecularGlossinessMaterialTool extends MaterialTool { - // Public members - public divId: string = 'PBR-SPECULAR-GLOSSINESS-MATERIAL-TOOL'; - public tabName: string = 'PBR Specular Glossiness'; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported(object: any): boolean { - return super.isSupported(object) && this.object instanceof PBRSpecularGlossinessMaterial; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update(object: any): void { - super.update(object); - - // Diffuse Color - const diffuseColor = this.tool.addFolder('Diffuse'); - diffuseColor.open(); - this.tool.addColor(diffuseColor, 'Color', this.object.diffuseColor).open(); - this.tool.addTexture(diffuseColor, this.editor, 'diffuseTexture', this.object, false).name('Texture'); - - // Bump - const normal = this.tool.addFolder('Normal'); - normal.open(); - this.tool.addTexture(normal, this.editor, 'normalTexture', this.object).name('Normal Texture'); - normal.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); - normal.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); - - // Reflection - const reflection = this.tool.addFolder('Reflection'); - reflection.open(); - this.tool.addTexture(reflection, this.editor, 'environmentTexture', this.object, true, false).name('Environment Texture'); - - // Specular Roughness - const glossiness = this.tool.addFolder('Glossiness'); - glossiness.open(); - glossiness.add(this.object, 'glossiness').step(0.01).name('Glossiness'); - this.tool.addTexture(glossiness, this.editor, 'specularGlossinessTexture', this.object).name('Specular Glossiness Texture'); - - // Emissive - const emissive = this.tool.addFolder('Emissive'); - emissive.open(); - this.tool.addColor(emissive, 'Color', this.object.emissiveColor).open(); - this.tool.addTexture(emissive, this.editor, 'emissiveTexture', this.object).name('Emissive Texture'); - - // Lightmap - const lightmap = this.tool.addFolder('Lightmap'); - lightmap.open(); - lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); - this.tool.addTexture(lightmap, this.editor, 'lightmapTexture', this.object).name('Lightmap Texture'); - - // Occlusion - const occlusion = this.tool.addFolder('Occlusion'); - occlusion.open(); - occlusion.add(this.object, 'occlusionStrength').name('Occlusion Strength'); - this.tool.addTexture(occlusion, this.editor, 'occlusionTexture', this.object).name('Occlusion Texture'); - - // Options - super.addOptions(); - } -} +import { PBRSpecularGlossinessMaterial } from 'babylonjs'; + +import MaterialTool from './material-tool'; +import Tools from '../../tools/tools'; + +export default class PBRSpecularGlossinessMaterialTool extends MaterialTool { + // Public members + public divId: string = 'PBR-SPECULAR-GLOSSINESS-MATERIAL-TOOL'; + public tabName: string = 'PBR Specular Glossiness'; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported(object: any): boolean { + return super.isSupported(object) && this.object instanceof PBRSpecularGlossinessMaterial; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update(object: any): void { + super.update(object); + + // Diffuse Color + const diffuseColor = this.tool.addFolder('Diffuse'); + diffuseColor.open(); + this.tool.addColor(diffuseColor, 'Color', this.object.diffuseColor).open(); + this.tool.addTexture(diffuseColor, this.editor, this.editor.core.scene, 'diffuseTexture', this.object, false).name('Texture'); + + // Bump + const normal = this.tool.addFolder('Normal'); + normal.open(); + this.tool.addTexture(normal, this.editor, this.editor.core.scene, 'normalTexture', this.object).name('Normal Texture'); + normal.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); + normal.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); + + // Reflection + const reflection = this.tool.addFolder('Reflection'); + reflection.open(); + this.tool.addTexture(reflection, this.editor, this.editor.core.scene, 'environmentTexture', this.object, true, false).name('Environment Texture'); + + // Specular Roughness + const glossiness = this.tool.addFolder('Glossiness'); + glossiness.open(); + glossiness.add(this.object, 'glossiness').step(0.01).name('Glossiness'); + this.tool.addTexture(glossiness, this.editor, this.editor.core.scene, 'specularGlossinessTexture', this.object).name('Specular Glossiness Texture'); + + // Emissive + const emissive = this.tool.addFolder('Emissive'); + emissive.open(); + this.tool.addColor(emissive, 'Color', this.object.emissiveColor).open(); + this.tool.addTexture(emissive, this.editor, this.editor.core.scene, 'emissiveTexture', this.object).name('Emissive Texture'); + + // Lightmap + const lightmap = this.tool.addFolder('Lightmap'); + lightmap.open(); + lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); + this.tool.addTexture(lightmap, this.editor, this.editor.core.scene, 'lightmapTexture', this.object).name('Lightmap Texture'); + + // Occlusion + const occlusion = this.tool.addFolder('Occlusion'); + occlusion.open(); + occlusion.add(this.object, 'occlusionStrength').name('Occlusion Strength'); + this.tool.addTexture(occlusion, this.editor, this.editor.core.scene, 'occlusionTexture', this.object).name('Occlusion Texture'); + + // Options + super.addOptions(); + } +} diff --git a/src/editor/edition-tools/materials/pbr-tool.ts b/src/editor/edition-tools/materials/pbr-tool.ts index f26b3b6e9..2a4995b54 100644 --- a/src/editor/edition-tools/materials/pbr-tool.ts +++ b/src/editor/edition-tools/materials/pbr-tool.ts @@ -42,12 +42,12 @@ export default class PBRMaterialTool extends MaterialTool { // Albedo const albedo = this.tool.addFolder('Albedo'); albedo.open(); - this.tool.addTexture(albedo, this.editor, 'albedoTexture', this.object).name('Albedo Texture'); + this.tool.addTexture(albedo, this.editor, this.editor.core.scene, 'albedoTexture', this.object).name('Albedo Texture'); this.tool.addColor(albedo, 'Color', this.object.albedoColor).open(); // Bump const bump = this.tool.addFolder('Bump'); - this.tool.addTexture(bump, this.editor, 'bumpTexture', this.object).name('Bump Texture'); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture', this.object).name('Bump Texture'); bump.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); bump.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); bump.add(this.object, 'useParallax').name('Use Parallax'); @@ -57,19 +57,19 @@ export default class PBRMaterialTool extends MaterialTool { // Reflectivity const reflectivity = this.tool.addFolder('Reflectivity'); reflectivity.open(); - this.tool.addTexture(reflectivity, this.editor, 'reflectivityTexture', this.object).name('Reflectivity Texture'); + this.tool.addTexture(reflectivity, this.editor, this.editor.core.scene, 'reflectivityTexture', this.object).name('Reflectivity Texture'); this.tool.addColor(reflectivity, 'Color', this.object.reflectivityColor).open(); // Reflection const reflection = this.tool.addFolder('Reflection'); reflection.open(); - this.tool.addTexture(reflection, this.editor, 'reflectionTexture', this.object, true, false).name('Reflection Texture'); + this.tool.addTexture(reflection, this.editor, this.editor.core.scene, 'reflectionTexture', this.object, true, false).name('Reflection Texture'); this.tool.addColor(reflection, 'Color', this.object.reflectionColor).open(); reflection.add(this.object, 'environmentIntensity').step(0.01).name('Environment Intensity'); // Microsurface const micro = this.tool.addFolder('Micro Surface'); - this.tool.addTexture(micro, this.editor, 'microSurfaceTexture', this.object, false).name('Micro Surface Texture'); + this.tool.addTexture(micro, this.editor, this.editor.core.scene, 'microSurfaceTexture', this.object, false).name('Micro Surface Texture'); micro.add(this.object, 'microSurface').min(0).max(1).name('Micro Surface'); micro.add(this.object, 'useAutoMicroSurfaceFromReflectivityMap').name('Use Auto Micro Surface From Reflectivity Map'); micro.add(this.object, 'useMicroSurfaceFromReflectivityMapAlpha').name('Use Micro Surface From Reflectivity Map Alpha'); @@ -79,7 +79,7 @@ export default class PBRMaterialTool extends MaterialTool { metallic.add(this.object, 'useMetallnessFromMetallicTextureBlue').name('Metallness From Metallic Texture Blue'); metallic.add(this.object, 'useRoughnessFromMetallicTextureAlpha').name('Use Roughness From Metallic Texture Alpha'); metallic.add(this.object, 'useRoughnessFromMetallicTextureGreen').name('Use Roughness From Metallic Texture Green'); - this.tool.addTexture(metallic, this.editor, 'metallicTexture', this.object, false, false, t => this.update(this.object)).name('Metallic Texture'); + this.tool.addTexture(metallic, this.editor, this.editor.core.scene, 'metallicTexture', this.object, false, false, t => this.update(this.object)).name('Metallic Texture'); const metallicWorkflow = metallic.addFolder('Metallic Workflow'); metallicWorkflow.open(); @@ -128,7 +128,7 @@ export default class PBRMaterialTool extends MaterialTool { // Sub surface const subSurface = this.tool.addFolder('Sub Surface'); this.tool.addColor(subSurface, 'Tint Color', this.object.subSurface.tintColor).open(); - this.tool.addTexture(subSurface.addFolder('Thickness Texture'), this.editor, 'thicknessTexture', this.object.subSurface, false); + this.tool.addTexture(subSurface.addFolder('Thickness Texture'), this.editor, this.editor.core.scene, 'thicknessTexture', this.object.subSurface, false); subSurface.add(this.object.subSurface, 'useMaskFromThicknessTexture').name('Use Mask From Thickness Texture'); // Sub surface Refraction @@ -149,7 +149,7 @@ export default class PBRMaterialTool extends MaterialTool { clearCoat.add(this.object.clearCoat, 'isEnabled').name('Clear Coat Enabled'); clearCoat.add(this.object.clearCoat, 'roughness').min(0).step(0.01).name('Roughness'); clearCoat.add(this.object.clearCoat, 'indiceOfRefraction').min(0).step(0.01).name('Indice Of Refraction'); - this.tool.addTexture(clearCoat.addFolder('Bump Texture'), this.editor, 'bumpTexture', this.object.clearCoat, false, false); + this.tool.addTexture(clearCoat.addFolder('Bump Texture'), this.editor, this.editor.core.scene, 'bumpTexture', this.object.clearCoat, false, false); clearCoat.add(this.object.clearCoat, 'isTintEnabled').name('Tint Enabled'); clearCoat.add(this.object.clearCoat, 'tintColorAtDistance').min(0).step(0.01).name('Tint Color At Distance'); @@ -160,14 +160,14 @@ export default class PBRMaterialTool extends MaterialTool { anisotropy.add(this.object.anisotropy, 'isEnabled').name('Anisotropy Enabled'); anisotropy.add(this.object.anisotropy, 'intensity').min(0).step(0.01).name('Intensity'); this.tool.addVector(anisotropy, 'Direction', this.object.anisotropy.direction); - this.tool.addTexture(anisotropy.addFolder('Texture'), this.editor, 'texture', this.object.anisotropy, false, false); + this.tool.addTexture(anisotropy.addFolder('Texture'), this.editor, this.editor.core.scene, 'texture', this.object.anisotropy, false, false); // Sheen const sheen = this.tool.addFolder('Sheen'); sheen.add(this.object.sheen, 'isEnabled').name('Sheen Enabled'); sheen.add(this.object.sheen, 'intensity').min(0).step(0.01).name('Intensity'); this.tool.addColor(sheen, 'Color', this.object.sheen.color); - this.tool.addTexture(sheen.addFolder('Texture'), this.editor, 'texture', this.object.sheen, false, false); + this.tool.addTexture(sheen.addFolder('Texture'), this.editor, this.editor.core.scene, 'texture', this.object.sheen, false, false); // Opacity const opacity = this.tool.addFolder('Opacity'); @@ -178,25 +178,25 @@ export default class PBRMaterialTool extends MaterialTool { const emissive = this.tool.addFolder('Emissive'); this.tool.addColor(emissive, 'Emissive', this.object.emissiveColor).open(); emissive.add(this.object, 'emissiveIntensity').step(0.01).name('Emissive Intensity'); - this.tool.addTexture(emissive, this.editor, 'emissiveTexture', this.object).name('Emissive Texture'); + this.tool.addTexture(emissive, this.editor, this.editor.core.scene, 'emissiveTexture', this.object).name('Emissive Texture'); // Ambient const ambient = this.tool.addFolder('Ambient'); this.tool.addColor(ambient, 'Ambient', this.object.ambientColor).open(); - this.tool.addTexture(ambient, this.editor, 'ambientTexture', this.object).name('Ambient Texture'); + this.tool.addTexture(ambient, this.editor, this.editor.core.scene, 'ambientTexture', this.object).name('Ambient Texture'); ambient.add(this.object, 'ambientTextureStrength').step(0.01).name('Ambient Texture Strength'); // Light map const lightmap = this.tool.addFolder('Lightmap'); lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); - this.tool.addTexture(lightmap, this.editor, 'lightmapTexture', this.object).name('Lightmap Texture'); + this.tool.addTexture(lightmap, this.editor, this.editor.core.scene, 'lightmapTexture', this.object).name('Lightmap Texture'); // Refraction const refraction = this.tool.addFolder('Refraction (backward compatibility)'); refraction.add(this.object, 'indexOfRefraction').step(0.01).name('Index of Refraction'); refraction.add(this.object, 'invertRefractionY').name('Invert Y'); refraction.add(this.object, 'linkRefractionWithTransparency').name('Link Refraction With Transparency'); - this.tool.addTexture(refraction, this.editor, 'refractionTexture', this.object, true).name('Refraction Texture'); + this.tool.addTexture(refraction, this.editor, this.editor.core.scene, 'refractionTexture', this.object, true).name('Refraction Texture'); // Options super.addOptions(); diff --git a/src/editor/edition-tools/materials/standard-tool.ts b/src/editor/edition-tools/materials/standard-tool.ts index 70526bf71..21269c2e2 100644 --- a/src/editor/edition-tools/materials/standard-tool.ts +++ b/src/editor/edition-tools/materials/standard-tool.ts @@ -28,12 +28,12 @@ export default class StandardMaterialTool extends MaterialTool diffuse.open(); diffuse.add(this.object, 'linkEmissiveWithDiffuse').name('Link Emissive With Diffuse'); diffuse.add(this.object, 'useAlphaFromDiffuseTexture').name('Use Alpha From Diffuse Texture'); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture', this.object).name('Diffuse Texture'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture', this.object).name('Diffuse Texture'); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor).open(); // Bump const bump = this.tool.addFolder('Bump'); - this.tool.addTexture(bump, this.editor, 'bumpTexture', this.object).name('Bump Texture'); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture', this.object).name('Bump Texture'); bump.add(this.object, 'invertNormalMapX').name('Invert Normal Map X'); bump.add(this.object, 'invertNormalMapY').name('Invert Normal Map Y'); bump.add(this.object, 'useParallax').name('Use Parallax'); @@ -47,38 +47,38 @@ export default class StandardMaterialTool extends MaterialTool specular.add(this.object, 'useGlossinessFromSpecularMapAlpha').name('Use Glossiness From Specular Map Alpha'); specular.add(this.object, 'useReflectionFresnelFromSpecular').name('Use Reflection Fresnel From Specular'); specular.add(this.object, 'useSpecularOverAlpha').name('Use Specular Over Alpha'); - this.tool.addTexture(specular, this.editor, 'specularTexture', this.object).name('Specular Texture'); + this.tool.addTexture(specular, this.editor, this.editor.core.scene, 'specularTexture', this.object).name('Specular Texture'); this.tool.addColor(specular, 'Color', this.object.specularColor).open(); // Opacity const opacity = this.tool.addFolder('Opacity'); - this.tool.addTexture(opacity, this.editor, 'opacityTexture', this.object).name('Opacity Texture'); + this.tool.addTexture(opacity, this.editor, this.editor.core.scene, 'opacityTexture', this.object).name('Opacity Texture'); // Emissive const emissive = this.tool.addFolder('Emissive'); this.tool.addColor(emissive, 'Emissive', this.object.emissiveColor).open(); emissive.add(this.object, 'useEmissiveAsIllumination').name('Use Emissive As Illumination'); - this.tool.addTexture(emissive, this.editor, 'emissiveTexture', this.object).name('Emissive Texture'); + this.tool.addTexture(emissive, this.editor, this.editor.core.scene, 'emissiveTexture', this.object).name('Emissive Texture'); // Ambient const ambient = this.tool.addFolder('Ambient'); this.tool.addColor(ambient, 'Ambient', this.object.ambientColor).open(); - this.tool.addTexture(ambient, this.editor, 'ambientTexture', this.object).name('Ambient Texture'); + this.tool.addTexture(ambient, this.editor, this.editor.core.scene, 'ambientTexture', this.object).name('Ambient Texture'); // Light map const lightmap = this.tool.addFolder('Lightmap'); lightmap.add(this.object, 'useLightmapAsShadowmap').name('Use Lightmap As Shadowmap'); - this.tool.addTexture(lightmap, this.editor, 'lightmapTexture', this.object).name('Lightmap Texture'); + this.tool.addTexture(lightmap, this.editor, this.editor.core.scene, 'lightmapTexture', this.object).name('Lightmap Texture'); // Reflection const reflection = this.tool.addFolder('Reflection'); - this.tool.addTexture(reflection, this.editor, 'reflectionTexture', this.object, true).name('Reflection Texture'); + this.tool.addTexture(reflection, this.editor, this.editor.core.scene, 'reflectionTexture', this.object, true).name('Reflection Texture'); // Refraction const refraction = this.tool.addFolder('Refraction'); refraction.add(this.object, 'indexOfRefraction').name('Index of Refraction'); refraction.add(this.object, 'invertRefractionY').name('Invert Y'); - this.tool.addTexture(refraction, this.editor, 'refractionTexture', this.object, true).name('Refraction Texture'); + this.tool.addTexture(refraction, this.editor, this.editor.core.scene, 'refractionTexture', this.object, true).name('Refraction Texture'); // Options const options = super.addOptions(); diff --git a/src/editor/edition-tools/materials/terrain-tool.ts b/src/editor/edition-tools/materials/terrain-tool.ts index 68cd7d010..97cbd4a47 100644 --- a/src/editor/edition-tools/materials/terrain-tool.ts +++ b/src/editor/edition-tools/materials/terrain-tool.ts @@ -27,24 +27,24 @@ export default class TerrainMaterialTool extends MaterialTool { const terrain = this.tool.addFolder('Terrain'); terrain.open(); - this.tool.addTexture(terrain, this.editor, 'mixTexture', this.object, false).name('Mix Texture'); + this.tool.addTexture(terrain, this.editor, this.editor.core.scene, 'mixTexture', this.object, false).name('Mix Texture'); // Diffuse const diffuse = terrain.addFolder('Diffuse'); diffuse.open(); this.tool.addColor(diffuse, 'Color', this.object.diffuseColor).open(); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture1', this.object, false).name('Diffuse Texture R'); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture2', this.object, false).name('Diffuse Texture G'); - this.tool.addTexture(diffuse, this.editor, 'diffuseTexture3', this.object, false).name('Diffuse Texture B'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture1', this.object, false).name('Diffuse Texture R'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture2', this.object, false).name('Diffuse Texture G'); + this.tool.addTexture(diffuse, this.editor, this.editor.core.scene, 'diffuseTexture3', this.object, false).name('Diffuse Texture B'); // Bump const bump = terrain.addFolder('Bump'); bump.open(); - this.tool.addTexture(bump, this.editor, 'bumpTexture1', this.object, false).name('Bump Texture R'); - this.tool.addTexture(bump, this.editor, 'bumpTexture2', this.object, false).name('Bump Texture G'); - this.tool.addTexture(bump, this.editor, 'bumpTexture3', this.object, false).name('Bump Texture B'); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture1', this.object, false).name('Bump Texture R'); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture2', this.object, false).name('Bump Texture G'); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture3', this.object, false).name('Bump Texture B'); // Specular const specular = terrain.addFolder('Specular'); diff --git a/src/editor/edition-tools/materials/tri-planar-tool.ts b/src/editor/edition-tools/materials/tri-planar-tool.ts index af607e086..5ca7004e5 100644 --- a/src/editor/edition-tools/materials/tri-planar-tool.ts +++ b/src/editor/edition-tools/materials/tri-planar-tool.ts @@ -34,17 +34,17 @@ export default class TriPlanarMaterialTool extends MaterialTool { // Bump const bump = this.tool.addFolder('Bump'); bump.open(); - this.tool.addTexture(bump, this.editor, 'bumpTexture', this.object, false); + this.tool.addTexture(bump, this.editor, this.editor.core.scene, 'bumpTexture', this.object, false); bump.add(this.object, 'bumpHeight').min(0).max(10).step(0.001).name('Bump Height'); // Wind diff --git a/src/editor/edition-tools/particle-system-tool.ts b/src/editor/edition-tools/particle-system-tool.ts deleted file mode 100644 index 9acbf8326..000000000 --- a/src/editor/edition-tools/particle-system-tool.ts +++ /dev/null @@ -1,240 +0,0 @@ -import { - ParticleSystem, GPUParticleSystem, Vector3, IParticleSystem, - BoxParticleEmitter, SphereParticleEmitter, ConeParticleEmitter, - SphereDirectedParticleEmitter, ParticleHelper, - FilesInputStore, - Tools as BabylonTools -} from 'babylonjs'; - -import AbstractEditionTool from './edition-tool'; -import Tools from '../tools/tools'; - -import Dialog from '../gui/dialog'; - -export default class ParticleSystemTool extends AbstractEditionTool { - // Public members - public divId: string = 'PARTICLE-SYSTEM-TOOL'; - public tabName: string = 'Particle System'; - - // Private members - private _currentEmitter: string = ''; - private _currentBlendMode: string = ''; - private _currentEmiterType: string = ''; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported(object: any): boolean { - return object instanceof ParticleSystem; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update(ps: ParticleSystem | GPUParticleSystem): void { - super.update(ps); - - // Misc. - const scene = this.editor.core.scene; - - // Load / save - const presets = this.tool.addFolder('Presets'); - presets.open(); - - presets.add(this, '_saveSet').name('Save...'); - presets.add(this, '_loadSet').name('Load...'); - - // Particle System - if (ps instanceof ParticleSystem) { - // Emitter - const emitter = this.tool.addFolder('Emitter'); - emitter.open(); - - emitter.add(ps, 'id').name('Id'); - emitter.add(ps, 'name').name('Name'); - - if (ps.emitter instanceof Vector3) - this.tool.addVector(emitter, 'Emitter', ps.emitter); - else { - this._currentEmitter = ps.emitter.name; - const nodes = scene.meshes.map(m => m.name); - - emitter.add(this, '_currentEmitter', nodes).name('Emitter').onFinishChange(r => { - const mesh = scene.getMeshByName(r); - if (mesh) - ps.emitter = mesh; - }); - } - - // Emitter type - const emiterType = this.tool.addFolder('Emiter Type'); - emiterType.open(); - - this._currentEmiterType = this._getEmiterTypeString(ps); - const emiterTypes: string[] = [ - 'Box', - 'Sphere', - 'Sphere Directed', - 'Cone' - ]; - emiterType.add(this, '_currentEmiterType', emiterTypes).name('Emiter Type').onFinishChange(r => { - switch (r) { - case 'Box': ps.createBoxEmitter(ps.direction1, ps.direction2, ps.minEmitBox, ps.maxEmitBox); break; - case 'Sphere': ps.createSphereEmitter(10); break; - case 'Sphere Directed': ps.createDirectedSphereEmitter(10, ps.direction1, ps.direction2); break; - case 'Cone': ps.createConeEmitter(10, 0); break; - default: break; - } - - this.update(ps); - }); - - if (ps.particleEmitterType instanceof SphereParticleEmitter) { - emiterType.add(ps.particleEmitterType, 'radius').step(0.01).name('Radius'); - } - else if (ps.particleEmitterType instanceof ConeParticleEmitter) { - emiterType.add(ps.particleEmitterType, 'radius').step(0.01).name('Radius'); - emiterType.add(ps.particleEmitterType, 'angle').step(0.01).name('Angle'); - } - - if (!(ps.particleEmitterType instanceof BoxParticleEmitter)) - emiterType.add(ps.particleEmitterType, 'directionRandomizer').min(0).max(1).step(0.001).name('Direction Randomizer'); - - // Texture - const texture = this.tool.addFolder('Texture'); - texture.open(); - this.tool.addTexture(texture, this.editor, 'particleTexture', ps, false).name('Particle Texture'); - - const blendModes = ['BLENDMODE_ONEONE', 'BLENDMODE_STANDARD']; - this._currentBlendMode = blendModes[ps.blendMode]; - texture.add(this, '_currentBlendMode', blendModes).name('Blend Mode').onChange(r => ps.blendMode = ParticleSystem[r]); - - // Actions - const actions = this.tool.addFolder('Actions'); - actions.open(); - actions.add(ps, 'rebuild').name('Rebuild'); - actions.add(ps, 'start').name('Start'); - actions.add(ps, 'stop').name('Stop'); - - // Emit - const emit = this.tool.addFolder('Emit'); - emit.open(); - emit.add(ps, 'emitRate').min(0).step(0.01).name('Emit Rate'); - emit.add(ps, 'minEmitPower').min(0).step(0.01).name('Min Emit Power'); - emit.add(ps, 'maxEmitPower').min(0).step(0.01).name('Max Emit Power'); - - // Update - const update = this.tool.addFolder('Update'); - update.open(); - update.add(ps, 'updateSpeed').min(0).step(0.01).name('Update Speed'); - - // Life - const life = this.tool.addFolder('Life Time'); - life.open(); - life.add(ps, 'minLifeTime').min(0).step(0.01).name('Min Life Time'); - life.add(ps, 'maxLifeTime').min(0).step(0.01).name('Max Life Time'); - - // Size - const size = this.tool.addFolder('Size'); - size.open(); - size.add(ps, 'minSize').min(0).step(0.01).name('Min Size'); - size.add(ps, 'maxSize').min(0).step(0.01).name('Max Size'); - - // Angular Speed - const angular = this.tool.addFolder('Angular Speed'); - angular.open(); - angular.add(ps, 'minAngularSpeed').min(0).step(0.01).name('Min Angular Speed'); - angular.add(ps, 'maxAngularSpeed').min(0).step(0.01).name('Max Angular Speed'); - - // Sprite - if (ps.isAnimationSheetEnabled) { - const sprite = this.tool.addFolder('Sprite'); - sprite.open(); - sprite.add(ps, 'startSpriteCellID').min(0).step(1).name('Start Sprite Cell ID'); - sprite.add(ps, 'endSpriteCellID').min(0).step(1).name('End Sprite Cell ID'); - sprite.add(ps, 'spriteCellWidth').min(0).step(1).name('Sprite Cell Width'); - sprite.add(ps, 'spriteCellHeight').min(0).step(1).name('Sprite Cell Height'); - // sprite.add(ps, 'spriteCellLoop').name('Sprite Cell Loop').onFinishChange(r => ps.spriteCellLoop = r); - sprite.add(ps, 'spriteCellChangeSpeed').min(0).step(1).name('Sprite Cell Change Speed'); - } - - // Gravity - this.tool.addVector(this.tool.element, 'Gravity', ps.gravity).open(); - - if (ps.particleEmitterType instanceof BoxParticleEmitter || ps.particleEmitterType instanceof SphereDirectedParticleEmitter) { - // Direction1 - this.tool.addVector(this.tool.element, 'Direction 1', ps.direction1).open(); - - // Direction2 - this.tool.addVector(this.tool.element, 'Direction 2', ps.direction2).open(); - - if (ps.particleEmitterType instanceof BoxParticleEmitter) { - // Min Emit Box - this.tool.addVector(this.tool.element, 'Min Emit Box', ps.minEmitBox).open(); - - // Max Emit Box - this.tool.addVector(this.tool.element, 'Max Emit Box', ps.maxEmitBox).open(); - } - } - - // Color 1 - this.tool.addColor(this.tool.element, 'Color 1', ps.color1).open(); - - // Color 2 - this.tool.addColor(this.tool.element, 'Color 2', ps.color2).open(); - - // Color Dead - this.tool.addColor(this.tool.element, 'Color Dead', ps.colorDead).open(); - } - } - - // Returns the emiter type as a string - private _getEmiterTypeString (ps: IParticleSystem): string { - if (ps.particleEmitterType instanceof BoxParticleEmitter) return 'Box'; - if (ps.particleEmitterType instanceof SphereDirectedParticleEmitter) return 'Sphere Directed'; - if (ps.particleEmitterType instanceof SphereParticleEmitter) return 'Sphere'; - if (ps.particleEmitterType instanceof ConeParticleEmitter) return 'Cone'; - return 'None'; - } - - // Exports the current set - private async _saveSet (): Promise { - // Export - const set = ParticleHelper.ExportSet([this.object]); - const serializationObject = set.serialize(); - - // Embed? - const embed = await Dialog.Create('Embed textures?', 'Do you want to embed textures in the set?'); - if (embed) { - for (const s of serializationObject.systems) { - const file = FilesInputStore.FilesToLoad[s.textureName.toLowerCase()]; - if (!file) - continue; - s.textureName = await Tools.ReadFileAsBase64(file); - } - } - - // Save - const json = JSON.stringify(serializationObject, null, '\t'); - const file = Tools.CreateFile(Tools.ConvertStringToUInt8Array(json), this.object.name + '.json'); - BabylonTools.Download(file, file.name); - } - - // Loads the selected preset from files open dialog - private async _loadSet (): Promise { - const files = await Tools.OpenFileDialog(); - const content = JSON.parse(await Tools.ReadFileAsText(files[0])); - - if (!content.systems || content.systems.length === 0) - return; - - const savedEmitter = this.object.emitter; - - const rootUrl = content.systems[0].textureName.indexOf('data:') === 0 ? '' : 'file:'; - ParticleSystem._Parse(content.systems[0], this.object, this.editor.core.scene, rootUrl); - - this.object.emitter = savedEmitter; - } -} diff --git a/src/editor/edition-tools/particles/particle-system-tool.ts b/src/editor/edition-tools/particles/particle-system-tool.ts new file mode 100644 index 000000000..8c877f0f5 --- /dev/null +++ b/src/editor/edition-tools/particles/particle-system-tool.ts @@ -0,0 +1,308 @@ +import { + ParticleSystem, Vector3, IParticleSystem, + BoxParticleEmitter, SphereParticleEmitter, ConeParticleEmitter, + SphereDirectedParticleEmitter, ParticleHelper, + FilesInputStore, + Tools as BabylonTools, + Texture +} from 'babylonjs'; + +import AbstractEditionTool from '../edition-tool'; +import Tools from '../../tools/tools'; +import UndoRedo from '../../tools/undo-redo'; + +import Dialog from '../../gui/dialog'; + +export default class ParticleSystemTool extends AbstractEditionTool { + // Public members + public divId: string = 'PARTICLE-SYSTEM-TOOL'; + public tabName: string = 'Particle System'; + + // Private members + private _currentEmitter: string = ''; + private _currentBillboard: string = ''; + private _currentBlendMode: string = ''; + private _currentEmiterType: string = ''; + private _currentTexture: string = ''; + + private _isFromScene: boolean = true; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported(object: any): boolean { + return object instanceof ParticleSystem; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update(ps: ParticleSystem): void { + super.update(ps); + + // Misc. + const scene = this.editor.core.scene; + this._isFromScene = scene.getParticleSystemByID(ps.id) !== null; + + // Load / save + const presets = this.tool.addFolder('Presets'); + presets.open(); + + presets.add(this, '_saveSet').name('Save...'); + presets.add(this, '_loadSet').name('Load...'); + + // Emitter + const emitter = this.tool.addFolder('Emitter'); + emitter.open(); + + emitter.add(ps, 'name').name('Name'); + + if (this._isFromScene) { + emitter.add(ps, 'id').name('Id'); + + if (ps.emitter instanceof Vector3) + this.tool.addVector(emitter, 'Emitter', ps.emitter); + else { + this._currentEmitter = ps.emitter.name; + const nodes = scene.meshes.map(m => m.name); + + emitter.add(this, '_currentEmitter', nodes).name('Emitter').onFinishChange(r => { + const mesh = scene.getMeshByName(r); + if (mesh) + ps.emitter = mesh; + }); + } + } + else { + this.tool.addVector(emitter, 'Emitter', ps.emitter); + } + + // Emitter type + const emiterType = this.tool.addFolder('Emiter Type'); + emiterType.open(); + + this._currentEmiterType = this._getEmiterTypeString(ps); + const emiterTypes: string[] = [ + 'Box', + 'Sphere', + 'Sphere Directed', + 'Cone' + ]; + emiterType.add(this, '_currentEmiterType', emiterTypes).name('Emiter Type').onFinishChange(r => { + switch (r) { + case 'Box': ps.createBoxEmitter(ps.direction1, ps.direction2, ps.minEmitBox, ps.maxEmitBox); break; + case 'Sphere': ps.createSphereEmitter(10); break; + case 'Sphere Directed': ps.createDirectedSphereEmitter(10, ps.direction1, ps.direction2); break; + case 'Cone': ps.createConeEmitter(10, 0); break; + default: break; + } + + this.update(ps); + }); + + if (ps.particleEmitterType instanceof SphereParticleEmitter) { + emiterType.add(ps.particleEmitterType, 'radius').step(0.01).name('Radius'); + } + else if (ps.particleEmitterType instanceof ConeParticleEmitter) { + emiterType.add(ps.particleEmitterType, 'radius').step(0.01).name('Radius'); + emiterType.add(ps.particleEmitterType, 'angle').step(0.01).name('Angle'); + } + + if (!(ps.particleEmitterType instanceof BoxParticleEmitter)) + emiterType.add(ps.particleEmitterType, 'directionRandomizer').min(0).max(1).step(0.001).name('Direction Randomizer'); + + // Texture + const texture = this.tool.addFolder('Texture'); + texture.open(); + + if (this._isFromScene) { + this.tool.addTexture(texture, this.editor, this.editor.core.scene, 'particleTexture', ps, false, false).name('Particle Texture'); + } + else { + const exts = ['png' ,'jpg', 'jpeg', 'bmp']; + const files = ['None'].concat(Object.keys(FilesInputStore.FilesToLoad).filter(k => exts.indexOf(Tools.GetFileExtension(k)) !== -1)); + this._currentTexture = ps.particleTexture ? files[files.indexOf(ps.particleTexture.name)] || 'None' : 'None'; + texture.add(this, '_currentTexture', files).name('Particle Texture').onChange(r => { + const file = FilesInputStore.FilesToLoad[r]; + if (!file) + return; + + const texture = new Texture('file:' + r, ps.getScene()); + texture.name = texture.url = texture.name.replace('file:', ''); + ps.particleTexture = texture; + }); + } + + const blendModes = ['BLENDMODE_ONEONE', 'BLENDMODE_STANDARD']; + this._currentBlendMode = blendModes[ps.blendMode] || 'BLENDMODE_ONEONE'; + texture.add(this, '_currentBlendMode', blendModes).name('Blend Mode').onChange(r => ps.blendMode = ParticleSystem[r]); + + // Actions + const actions = this.tool.addFolder('Actions'); + actions.open(); + actions.add(ps, 'preventAutoStart').name('Prevent Auto Start'); + actions.add(ps, 'rebuild').name('Rebuild'); + actions.add(ps, 'start').name('Start'); + actions.add(ps, 'stop').name('Stop'); + + // Billboard + const modes: string[] = ['BILLBOARDMODE_ALL', 'BILLBOARDMODE_Y', 'BILLBOARDMODE_STRETCHED']; + this._currentBillboard = 'None'; + switch (ps.billboardMode) { + case ParticleSystem.BILLBOARDMODE_ALL: this._currentBillboard = 'BILLBOARDMODE_ALL'; break; + case ParticleSystem.BILLBOARDMODE_Y: this._currentBillboard = 'BILLBOARDMODE_Y'; break; + case ParticleSystem.BILLBOARDMODE_STRETCHED: this._currentBillboard = 'BILLBOARDMODE_STRETCHED'; break; + default: debugger; break; + } + + const billboard = this.tool.addFolder('Billboarding'); + billboard.open(); + billboard.add(ps, 'isBillboardBased').name('Is Billboard Based'); + billboard.add(this, '_currentBillboard', modes).name('Billboard Type').onChange(r => { + const from = ps.billboardMode; + const to = ParticleSystem[r]; + + UndoRedo.Push({ baseObject: ps, property: 'billboardMode', from: from, to: to }); + }); + + // Emit + const emit = this.tool.addFolder('Emit'); + emit.open(); + emit.add(ps, 'emitRate').min(0).step(0.01).name('Emit Rate'); + emit.add(ps, 'minEmitPower').min(0).step(0.01).name('Min Emit Power'); + emit.add(ps, 'maxEmitPower').min(0).step(0.01).name('Max Emit Power'); + + // Update + const update = this.tool.addFolder('Update'); + update.open(); + update.add(ps, 'updateSpeed').min(0).step(0.01).name('Update Speed'); + update.add(ps, 'startDelay').min(0).name('Start Delay (ms)'); + update.add(ps, 'targetStopDuration').min(0).name('Stop Duration (seconds)'); + + // Life + const life = this.tool.addFolder('Life Time'); + life.open(); + life.add(ps, 'minLifeTime').min(0).step(0.01).name('Min Life Time'); + life.add(ps, 'maxLifeTime').min(0).step(0.01).name('Max Life Time'); + + // Size + const size = this.tool.addFolder('Size'); + size.open(); + size.add(ps, 'minSize').min(0).step(0.01).name('Min Size'); + size.add(ps, 'maxSize').min(0).step(0.01).name('Max Size'); + + // Angular Speed + const angular = this.tool.addFolder('Angular Speed'); + angular.open(); + angular.add(ps, 'minAngularSpeed').min(0).step(0.01).name('Min Angular Speed'); + angular.add(ps, 'maxAngularSpeed').min(0).step(0.01).name('Max Angular Speed'); + + // Sprite + if (ps.isAnimationSheetEnabled) { + const sprite = this.tool.addFolder('Sprite'); + sprite.open(); + sprite.add(ps, 'startSpriteCellID').min(0).step(1).name('Start Sprite Cell ID'); + sprite.add(ps, 'endSpriteCellID').min(0).step(1).name('End Sprite Cell ID'); + sprite.add(ps, 'spriteCellWidth').min(0).step(1).name('Sprite Cell Width'); + sprite.add(ps, 'spriteCellHeight').min(0).step(1).name('Sprite Cell Height'); + // sprite.add(ps, 'spriteCellLoop').name('Sprite Cell Loop').onFinishChange(r => ps.spriteCellLoop = r); + sprite.add(ps, 'spriteCellChangeSpeed').min(0).step(1).name('Sprite Cell Change Speed'); + } + + // Gravity + this.tool.addVector(this.tool.element, 'Gravity', ps.gravity).open(); + + if (ps.particleEmitterType instanceof BoxParticleEmitter || ps.particleEmitterType instanceof SphereDirectedParticleEmitter) { + // Direction1 + this.tool.addVector(this.tool.element, 'Direction 1', ps.direction1).open(); + + // Direction2 + this.tool.addVector(this.tool.element, 'Direction 2', ps.direction2).open(); + + if (ps.particleEmitterType instanceof BoxParticleEmitter) { + // Min Emit Box + this.tool.addVector(this.tool.element, 'Min Emit Box', ps.minEmitBox).open(); + + // Max Emit Box + this.tool.addVector(this.tool.element, 'Max Emit Box', ps.maxEmitBox).open(); + } + } + + // Color 1 + this.tool.addColor(this.tool.element, 'Color 1', ps.color1).open(); + + // Color 2 + this.tool.addColor(this.tool.element, 'Color 2', ps.color2).open(); + + // Color Dead + this.tool.addColor(this.tool.element, 'Color Dead', ps.colorDead).open(); + + // Animations + if (!this._isFromScene) { + const animations = this.tool.addFolder('Animations'); + animations.open(); + + ps.beginAnimationOnStart = ps.beginAnimationOnStart || false; + animations.add(ps, 'beginAnimationOnStart').name('Begin Animations On Start'); + + ps.beginAnimationFrom = ps.beginAnimationFrom || 0; + animations.add(ps, 'beginAnimationFrom').min(0).step(1).name('Begin Animation From'); + + ps.beginAnimationTo = ps.beginAnimationTo || 60; + animations.add(ps, 'beginAnimationTo').min(0).step(1).name('Begin Animation To'); + + ps.beginAnimationLoop = ps.beginAnimationLoop || false; + animations.add(ps, 'beginAnimationLoop').name('Begin Animation Loop'); + } + } + + // Returns the emiter type as a string + private _getEmiterTypeString (ps: IParticleSystem): string { + if (ps.particleEmitterType instanceof BoxParticleEmitter) return 'Box'; + if (ps.particleEmitterType instanceof SphereDirectedParticleEmitter) return 'Sphere Directed'; + if (ps.particleEmitterType instanceof SphereParticleEmitter) return 'Sphere'; + if (ps.particleEmitterType instanceof ConeParticleEmitter) return 'Cone'; + return 'None'; + } + + // Exports the current set + private async _saveSet (): Promise { + // Export + const set = ParticleHelper.ExportSet([this.object]); + const serializationObject = set.serialize(); + + // Embed? + const embed = await Dialog.Create('Embed textures?', 'Do you want to embed textures in the set?'); + if (embed === 'Yes') { + for (const s of serializationObject.systems) { + const file = FilesInputStore.FilesToLoad[s.textureName.toLowerCase()]; + if (!file) + continue; + s.textureName = await Tools.ReadFileAsBase64(file); + } + } + + // Save + const json = JSON.stringify(serializationObject, null, '\t'); + const file = Tools.CreateFile(Tools.ConvertStringToUInt8Array(json), this.object.name + '.json'); + BabylonTools.Download(file, file.name); + } + + // Loads the selected preset from files open dialog + private async _loadSet (): Promise { + const files = await Tools.OpenFileDialog(); + const content = JSON.parse(await Tools.ReadFileAsText(files[0])); + + if (!content.systems || content.systems.length === 0) + return; + + const savedEmitter = this.object.emitter; + + const rootUrl = content.systems[0].textureName.indexOf('data:') === 0 ? '' : 'file:'; + ParticleSystem._Parse(content.systems[0], this.object, this.object.getScene(), rootUrl); + + this.object.emitter = savedEmitter; + } +} diff --git a/src/editor/edition-tools/post-processes/custom-tool.ts b/src/editor/edition-tools/post-processes/custom-tool.ts index 87dcbfc96..312cda7b7 100644 --- a/src/editor/edition-tools/post-processes/custom-tool.ts +++ b/src/editor/edition-tools/post-processes/custom-tool.ts @@ -1,68 +1,68 @@ -import { Vector2, Vector3 } from 'babylonjs'; - -import PostProcessEditor, { CustomPostProcessConfig } from '../../../extensions/post-process-editor/post-process'; -import AbstractEditionTool from '../edition-tool'; -import Tools from '../../tools/tools'; - -export default class CustomPostProcessTool extends AbstractEditionTool { - // Public members - public divId: string = 'CUSTOM-POST-PROCESS-TOOL'; - public tabName: string = 'Custom Post-Process'; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported(object: any): boolean { - return object.getClassName && object.getClassName() === 'PostProcessEditor'; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update(object: any): void { - super.update(object); - this.setTabName('Custom Post-Process'); - - // Get current config of the post-process - const config = this.object.config; - - // Floats - const floats = this.tool.addFolder('Floats'); - floats.open(); - - config.floats.forEach(f => { - if (this.object.userConfig[f] === undefined) - this.object.userConfig[f] = 1; - - floats.add(this.object.userConfig, f).step(0.01).name(f); - }); - - // Vectors - const vectors = this.tool.addFolder('Vectors'); - vectors.open(); - - config.vectors2.forEach(v => { - if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector2)) - this.object.userConfig[v] = Vector2.Zero(); - - this.tool.addVector(vectors, v, this.object.userConfig[v]).open(); - }); - - config.vectors3.forEach(v => { - if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector3)) - this.object.userConfig[v] = Vector3.Zero(); - - this.tool.addVector(vectors, v, this.object.userConfig[v]).open(); - }); - - // Samplers - const samplers = this.tool.addFolder('Samplers'); - samplers.open(); - - config.textures.forEach(t => { - this.tool.addTexture(samplers, this.editor, t, this.object.userConfig, false, false).name(t); - }); - } -} +import { Vector2, Vector3 } from 'babylonjs'; + +import PostProcessEditor, { CustomPostProcessConfig } from '../../../extensions/post-process-editor/post-process'; +import AbstractEditionTool from '../edition-tool'; +import Tools from '../../tools/tools'; + +export default class CustomPostProcessTool extends AbstractEditionTool { + // Public members + public divId: string = 'CUSTOM-POST-PROCESS-TOOL'; + public tabName: string = 'Custom Post-Process'; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported(object: any): boolean { + return object.getClassName && object.getClassName() === 'PostProcessEditor'; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update(object: any): void { + super.update(object); + this.setTabName('Custom Post-Process'); + + // Get current config of the post-process + const config = this.object.config; + + // Floats + const floats = this.tool.addFolder('Floats'); + floats.open(); + + config.floats.forEach(f => { + if (this.object.userConfig[f] === undefined) + this.object.userConfig[f] = 1; + + floats.add(this.object.userConfig, f).step(0.01).name(f); + }); + + // Vectors + const vectors = this.tool.addFolder('Vectors'); + vectors.open(); + + config.vectors2.forEach(v => { + if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector2)) + this.object.userConfig[v] = Vector2.Zero(); + + this.tool.addVector(vectors, v, this.object.userConfig[v]).open(); + }); + + config.vectors3.forEach(v => { + if (!this.object.userConfig[v] || !(this.object.userConfig[v] instanceof Vector3)) + this.object.userConfig[v] = Vector3.Zero(); + + this.tool.addVector(vectors, v, this.object.userConfig[v]).open(); + }); + + // Samplers + const samplers = this.tool.addFolder('Samplers'); + samplers.open(); + + config.textures.forEach(t => { + this.tool.addTexture(samplers, this.editor, this.editor.core.scene, t, this.object.userConfig, false, false).name(t); + }); + } +} diff --git a/src/editor/edition-tools/post-processes/post-processes-tool.ts b/src/editor/edition-tools/post-processes/post-processes-tool.ts index bcd830db3..212e6d8c5 100644 --- a/src/editor/edition-tools/post-processes/post-processes-tool.ts +++ b/src/editor/edition-tools/post-processes/post-processes-tool.ts @@ -97,7 +97,7 @@ export default class PostProcessesTool extends AbstractEditionTool { const lensTexture = bloom.addFolder('Lens Dirt Texture'); lensTexture.open(); - this.tool.addTexture(lensTexture, this.editor, 'lensTexture', SceneManager.StandardRenderingPipeline, false, false, texture => { + this.tool.addTexture(lensTexture, this.editor, this.editor.core.scene, 'lensTexture', SceneManager.StandardRenderingPipeline, false, false, texture => { SceneManager.StandardRenderingPipeline.lensFlareDirtTexture = texture; }).name('Texture'); diff --git a/src/editor/edition-tools/procedural-textures/normal-tool.ts b/src/editor/edition-tools/procedural-textures/normal-tool.ts index 6e0115e4d..02c92674f 100644 --- a/src/editor/edition-tools/procedural-textures/normal-tool.ts +++ b/src/editor/edition-tools/procedural-textures/normal-tool.ts @@ -1,32 +1,32 @@ -import { NormalMapProceduralTexture } from 'babylonjs-procedural-textures'; - -import AbstractEditionTool from '../edition-tool'; - -export default class NormalProceduralTool extends AbstractEditionTool { - // Public members - public divId: string = 'NORMAL-PROCEDURAL-TOOL'; - public tabName: string = 'Normal'; - - /** - * Returns if the object is supported - * @param object the object selected in the graph - */ - public isSupported (object: any): boolean { - return object instanceof NormalMapProceduralTexture; - } - - /** - * Updates the edition tool - * @param object the object selected in the graph - */ - public update (object: NormalMapProceduralTexture): void { - // Super - super.update(object); - - // Normal - const normal = this.tool.addFolder('Normal'); - normal.open(); - - this.tool.addTexture(normal, this.editor, 'baseTexture', object, false, false, () => object.updateShaderUniforms()); - } -} +import { NormalMapProceduralTexture } from 'babylonjs-procedural-textures'; + +import AbstractEditionTool from '../edition-tool'; + +export default class NormalProceduralTool extends AbstractEditionTool { + // Public members + public divId: string = 'NORMAL-PROCEDURAL-TOOL'; + public tabName: string = 'Normal'; + + /** + * Returns if the object is supported + * @param object the object selected in the graph + */ + public isSupported (object: any): boolean { + return object instanceof NormalMapProceduralTexture; + } + + /** + * Updates the edition tool + * @param object the object selected in the graph + */ + public update (object: NormalMapProceduralTexture): void { + // Super + super.update(object); + + // Normal + const normal = this.tool.addFolder('Normal'); + normal.open(); + + this.tool.addTexture(normal, this.editor, this.editor.core.scene, 'baseTexture', object, false, false, () => object.updateShaderUniforms()); + } +} diff --git a/src/editor/edition-tools/scene-tool.ts b/src/editor/edition-tools/scene-tool.ts index cdda7dec6..546860e05 100644 --- a/src/editor/edition-tools/scene-tool.ts +++ b/src/editor/edition-tools/scene-tool.ts @@ -98,7 +98,7 @@ export default class SceneTool extends AbstractEditionTool { // Environment texture const environment = this.tool.addFolder('Environment Texture'); environment.open(); - this.tool.addTexture(environment, this.editor, 'environmentTexture', scene, true, true).name('Environment Texture'); + this.tool.addTexture(environment, this.editor, this.editor.core.scene, 'environmentTexture', scene, true, true).name('Environment Texture'); // Collisions const collisions = this.tool.addFolder('Collisions'); diff --git a/src/editor/editor.ts b/src/editor/editor.ts index 5d141efac..a63d33c17 100644 --- a/src/editor/editor.ts +++ b/src/editor/editor.ts @@ -693,10 +693,7 @@ export default class Editor implements IUpdatable { document.addEventListener('contextmenu', (e) => e.preventDefault()); // Undo - UndoRedo.onUndo = (e) => { - this.core.onGlobalPropertyChange.notifyObservers({ baseObject: e.baseObject, object: e.object, property: e.property, value: e.to, initialValue: e.from }); - Tools.SetWindowTitle(this.projectFileName + ' *'); - }; + UndoRedo.onUndo = (e) => Tools.SetWindowTitle(this.projectFileName + ' *'); document.addEventListener('keyup', (ev) => { if (!CodeEditor.HasOneFocused() && ev.ctrlKey && ev.key === 'z') { UndoRedo.Undo(); @@ -707,10 +704,7 @@ export default class Editor implements IUpdatable { }); // Redo - UndoRedo.onRedo = (e) => { - this.core.onGlobalPropertyChange.notifyObservers({ baseObject: e.baseObject, object: e.object, property: e.property, value: e.to, initialValue: e.from }); - Tools.SetWindowTitle(this.projectFileName + ' *'); - }; + UndoRedo.onRedo = (e) => Tools.SetWindowTitle(this.projectFileName + ' *'); document.addEventListener('keyup', (ev) => { if (!CodeEditor.HasOneFocused() && ev.ctrlKey && ev.key === 'y') { UndoRedo.Redo(); diff --git a/src/editor/gui/edition.ts b/src/editor/gui/edition.ts index 626cc55c8..7215ef982 100644 --- a/src/editor/gui/edition.ts +++ b/src/editor/gui/edition.ts @@ -1,7 +1,8 @@ import { Color3, Color4, Vector2, Vector3, Vector4, - BaseTexture, CubeTexture + BaseTexture, CubeTexture, + Scene } from 'babylonjs'; import * as dat from 'dat-gui'; @@ -218,9 +219,7 @@ export default class Edition { * @param object the object which has a texture * @param callback: called when changed texture */ - public addTexture(parent: dat.GUI, editor: Editor, property: string, object: any, allowCubes: boolean = false, onlyCubes: boolean = false, callback?: (texture: BaseTexture) => void): dat.GUIController { - const scene = editor.core.scene; - + public addTexture(parent: dat.GUI, editor: Editor, scene: Scene, property: string, object: any, allowCubes: boolean = false, onlyCubes: boolean = false, callback?: (texture: BaseTexture) => void): dat.GUIController { const textures = ['None']; scene.textures.forEach(t => { const isCube = t instanceof CubeTexture; diff --git a/src/editor/particles/asset-component.ts b/src/editor/particles/asset-component.ts new file mode 100644 index 000000000..425c98688 --- /dev/null +++ b/src/editor/particles/asset-component.ts @@ -0,0 +1,138 @@ +import { Tools, AbstractMesh, PickingInfo, Mesh, ParticleSystem, Tags } from 'babylonjs'; +import { IAssetComponent, AssetElement } from '../../extensions/typings/asset'; + +import Editor from '../editor'; +import SceneFactory from '../scene/scene-factory'; + +export interface ParticlesCreatorMetadata { + name: string; + psData: any; +} + +export default class ParticlesAssetComponent implements IAssetComponent { + // Public members + public id: string = 'particles'; + public assetsCaption: string = 'Particles'; + + public datas: AssetElement[] = []; + + /** + * Constructor + * @param scene: the babylonjs scene + */ + constructor (public editor: Editor) + { } + + /** + * On the user renames the asset + * @param asset the asset being renamed + * @param name the new name of the asset + */ + public onRenameAsset (asset: AssetElement, name: string): void { + asset.data.name = name; + } + + /** + * On the user wants to remove the asset + * @param asset the asset to remove + */ + public onRemoveAsset (asset: AssetElement): void { + // Remove from library + const index = this.datas.indexOf(asset); + if (index !== -1) + this.datas.splice(index, 1); + + // Update assets + this.editor.assets.refresh(this.id); + } + + /** + * On the user adds an asset + * @param asset the asset to add + */ + public onAddAsset (asset: AssetElement): void { + this.datas.push(asset); + } + + /** + * Creates a new particle systems set asset + */ + public async onCreateAsset (name: string): Promise> { + const set = await new Promise((resolve) => { + Tools.LoadFile('./assets/templates/particles-creator/default-set.json', (data) => resolve( data)); + }); + + const asset = { + name: name, + data: { + name: name, + psData: JSON.parse(set) + } + }; + this.datas.push(asset); + + return asset; + } + + /** + * On get all the assets to be drawn in the assets component + */ + public onGetAssets (): AssetElement[] { + return this.datas; + } + + /** + * On the user double clicks on asset + * @param asset the asset being double-clicked by the user + */ + public onDoubleClickAsset (asset: AssetElement): void { + this.editor.addEditPanelPlugin('particles-creator', false, 'Particles System Creator...', asset.data); + } + + /** + * On the user drops an asset in the scene + * @param targetMesh the mesh under the pointer + * @param asset the asset being dropped + * @param pickInfo the pick info once the user dropped the asset + */ + public onDragAndDropAsset (targetMesh: AbstractMesh, asset: AssetElement, pickInfo: PickingInfo): void { + const m = new Mesh(asset.name, this.editor.core.scene); + m.position = pickInfo.pickedPoint.clone(); + SceneFactory.AddToGraph(this.editor, m); + + asset.data.psData.systems.forEach(s => { + const rootUrl = s.textureName ? (s.textureName.indexOf('data:') === 0 ? '' : 'file:') : ''; + const ps = ParticleSystem.Parse(s, this.editor.core.scene, rootUrl, true); + ps.id = Tools.RandomId(); + ps.emitter = m; + ps.preventAutoStart = false; + ps.start(); + + Tags.AddTagsTo(ps, 'added'); + this.editor.graph.add({ + id: ps.id, + data: ps, + img: 'icon-particles', + text: ps.name + }, m.id); + }); + } + + /** + * Called by the editor when serializing the scene + */ + public onSerializeAssets (): AssetElement[] { + return this.datas.map(d => ({ + name: d.name, + data: d.data + })); + } + + /** + * On the user loads the editor project + * @param data the previously saved data + */ + public onParseAssets (data: AssetElement[]): void { + this.datas = data; + } +} diff --git a/src/editor/project/project-exporter.ts b/src/editor/project/project-exporter.ts index 8adf4a256..3991cee55 100644 --- a/src/editor/project/project-exporter.ts +++ b/src/editor/project/project-exporter.ts @@ -234,7 +234,7 @@ export default class ProjectExporter { // Usable assets for extensions project.customMetadatas['AssetsExtension'] = { }; - const usableAssets = ['prefabs']; + const usableAssets = ['prefabs', 'particles']; usableAssets.forEach(ua => { const assets = project.assets[ua]; if (!assets) diff --git a/src/editor/scene/scene-icons.ts b/src/editor/scene/scene-icons.ts index 2a13c4389..3d8096938 100644 --- a/src/editor/scene/scene-icons.ts +++ b/src/editor/scene/scene-icons.ts @@ -72,9 +72,9 @@ export default class SceneIcons { } /** - * On before render the scene + * On post update the scenes */ - public onPreUpdate (): void { + public onPostUpdate (): void { const scene = this.editor.core.scene; // Cameras @@ -104,13 +104,6 @@ export default class SceneIcons { this._lastSoundsCount = sounds.length; this.createPlanes(this.soundsPlanes, this.soundMaterial, this._lastSoundsCount); } - } - - /** - * On post update the scenes - */ - public onPostUpdate (): void { - const scene = this.editor.core.scene; // Render helpers scene this.scene.activeCamera = this.editor.core.scene.activeCamera; @@ -148,10 +141,6 @@ export default class SceneIcons { }); // Sounds - const sounds: Sound[] = []; - if (scene.soundTracks) - scene.soundTracks.forEach(st => st.soundCollection.forEach(s => s.spatialSound && sounds.push(s))); - sounds.forEach((s, index) => { const plane = this.soundsPlanes[index]; plane.metadata.object = s; diff --git a/src/editor/scene/scene-loader.ts b/src/editor/scene/scene-loader.ts index 50424a8b2..34413b0fc 100644 --- a/src/editor/scene/scene-loader.ts +++ b/src/editor/scene/scene-loader.ts @@ -102,7 +102,8 @@ export default class SceneLoader { Tools.ImportScript('material-editor'), Tools.ImportScript('post-process-editor'), Tools.ImportScript('post-processes'), - Tools.ImportScript('path-finder') + Tools.ImportScript('path-finder'), + Tools.ImportScript('particles-creator') ]); editor.layout.unlockPanel('main'); diff --git a/src/editor/tools/gltf-tools.ts b/src/editor/tools/gltf-tools.ts index e50b52b1b..8714ea667 100644 --- a/src/editor/tools/gltf-tools.ts +++ b/src/editor/tools/gltf-tools.ts @@ -4,8 +4,9 @@ import { BaseTexture } from 'babylonjs'; -import Editor from "../editor"; -import Tools from "./tools"; +import Editor from '../editor'; +import Tools from './tools'; +import GraphicsTools from './graphics-tools'; export default class GLTFTools { /** @@ -35,37 +36,10 @@ export default class GLTFTools { // Configure now tex['url'] = tex.name = tex.name + '.png'; - // Retrieve pixels - const dimensions = tex.getBaseSize(); - const pixels = - tex.textureType === Engine.TEXTURETYPE_UNSIGNED_INT ? - tex.readPixels() as Uint8Array : - tex.readPixels() as Float32Array; - - const canvas = document.createElement('canvas'); - canvas.width = dimensions.width; - canvas.height = dimensions.height; - - const imageData = new ImageData(new Uint8ClampedArray(pixels.buffer), tex.getBaseSize().width, tex.getBaseSize().height); - - const context = canvas.getContext("2d"); - context.putImageData(imageData, 0, 0); - - if (canvas.toBlob) { - const blob = await this._ToBlob(canvas); - blob['name'] = tex.name; - Tags.AddTagsTo(blob, 'doNotExport'); - FilesInputStore.FilesToLoad[tex.name.toLowerCase()] = blob; - } - - context.restore(); - canvas.remove(); - } - - // Converts the canvas data to blob - private static async _ToBlob (canvas: HTMLCanvasElement): Promise { - return new Promise((resolve) => { - BabylonTools.ToBlob(canvas, b => resolve(b)); - }); + // Get blob + const blob = await GraphicsTools.TextureToFile(tex); + blob['name'] = tex.name; + Tags.AddTagsTo(blob, 'doNotExport'); + FilesInputStore.FilesToLoad[tex.name.toLowerCase()] = blob; } } diff --git a/src/editor/tools/graphics-tools.ts b/src/editor/tools/graphics-tools.ts new file mode 100644 index 000000000..dc8d7f9df --- /dev/null +++ b/src/editor/tools/graphics-tools.ts @@ -0,0 +1,42 @@ +import { Engine, BaseTexture, FilesInputStore, Tags, Tools as BabylonTools } from 'babylonjs'; + +export default class GraphicsTools { + /** + * Configures the given texture to retrieve its pixels and create a new file (blob) + * @param tex the texture to transform to a blob + */ + public static async TextureToFile (tex: BaseTexture): Promise { + // Retrieve pixels + const dimensions = tex.getBaseSize(); + const pixels = + tex.textureType === Engine.TEXTURETYPE_UNSIGNED_INT ? + tex.readPixels() as Uint8Array : + tex.readPixels() as Float32Array; + + const canvas = document.createElement('canvas'); + canvas.width = dimensions.width; + canvas.height = dimensions.height; + + const imageData = new ImageData(new Uint8ClampedArray(pixels.buffer), tex.getBaseSize().width, tex.getBaseSize().height); + + const context = canvas.getContext("2d"); + context.putImageData(imageData, 0, 0); + + const blob = await this.CanvasToBlob(canvas); + + context.restore(); + canvas.remove(); + + return blob; + } + + /** + * Converts the given canvas data to blob + * @param canvas the canvas to take its data and convert to a blob + */ + public static async CanvasToBlob (canvas: HTMLCanvasElement): Promise { + return new Promise((resolve) => { + BabylonTools.ToBlob(canvas, b => resolve(b)); + }); + } +} diff --git a/src/editor/tools/tools.ts b/src/editor/tools/tools.ts index 7f32c33dc..f90becff8 100644 --- a/src/editor/tools/tools.ts +++ b/src/editor/tools/tools.ts @@ -330,6 +330,14 @@ export default class Tools { return target; } + /** + * Deep clones the given object. Take care of cycling objects! + * @param data the data of the object to clone + */ + public static Clone (data: T): T { + return JSON.parse(JSON.stringify(data)); + } + /** * Reads the given file * @param file the file to read diff --git a/src/extensions/assets/assets.ts b/src/extensions/assets/assets.ts index 1b66463c6..d17e86e0e 100644 --- a/src/extensions/assets/assets.ts +++ b/src/extensions/assets/assets.ts @@ -1,4 +1,4 @@ -import { Node, Scene, Mesh } from 'babylonjs'; +import { Node, Scene, Mesh, ParticleSystemSet, Vector3, ParticleSystem } from 'babylonjs'; import Extensions from '../extensions'; import Extension from '../extension'; @@ -11,11 +11,16 @@ import { IStringDictionary } from '../typings/typings'; */ export interface ProjectAssets { prefabs: AssetElement[]; + particles: AssetElement[]; } export default class AssetsExtension extends Extension { // Public members public prefabs: AssetElement[] = []; + public particles: AssetElement[] = []; + + // Private members + private _particlesInstances: IStringDictionary = { }; /** * Constructor @@ -30,7 +35,8 @@ export default class AssetsExtension extends Extension { */ public onApply (data: ProjectAssets): void { // Prefabs - data.prefabs.forEach(p => this.prefabs.push(p)); + data.prefabs && data.prefabs.forEach(p => this.prefabs.push(p)); + data.particles && data.particles.forEach(p => this.particles.push(p)); } /** @@ -72,6 +78,39 @@ export default class AssetsExtension extends Extension { return null; } + /** + * Instantiates a particle system set identified by the given name + * @param name the name of the particle system set to instantiate + * @param position the position where to start systems + */ + public instantiateParticleSystemsSet (name: string, position?: Vector3): ParticleSystemSet { + for (const ps of this.particles) { + if (ps.name !== name) + continue; + + let set = this._particlesInstances[name]; + if (!set) { + set = new ParticleSystemSet(); + ps.data.psData.systems.forEach(s => { + const ps = ParticleSystem.Parse(s, this.scene, Extensions.RoolUrl, true); + set.systems.push(ps); + }); + + this._particlesInstances[name] = set; + } + + if (position) { + set.systems.forEach(s => s.emitter = position); + } + + this._particlesInstances[name] = set; + + return set; + } + + return null; + } + // Creates a new instance of the given node (InstancedMesh or just clone) private _createInstance (node: any): Node { if (node instanceof Mesh) diff --git a/src/extensions/index.ts b/src/extensions/index.ts index 637fb1ebd..1c15f65a5 100644 --- a/src/extensions/index.ts +++ b/src/extensions/index.ts @@ -5,7 +5,6 @@ import AssetsExtension from './assets/assets'; import CodeExtension from './behavior/code'; import GraphExtension, { LGraph, LGraphCanvas, LiteGraph, LiteGraphNode, LGraphGroup } from './behavior/graph'; import PathFinderExtension from './path-finder/index'; -import ParticlesCreatorExtension from './particles-creator/particles-creator'; import PostProcessEditorExtension from './post-process-editor/post-process-editor'; import MaterialEditorExtension from './material-editor/material-editor'; import PostProcessExtension from './post-process/post-processes'; @@ -24,7 +23,6 @@ export { MaterialEditorExtension, PathFinderExtension, PostProcessEditorExtension, - ParticlesCreatorExtension, CustomMetadatasExtension, IExtension, ExtensionConstructor diff --git a/src/extensions/particles-creator/particles-creator.ts b/src/extensions/particles-creator/particles-creator.ts deleted file mode 100644 index c195f3278..000000000 --- a/src/extensions/particles-creator/particles-creator.ts +++ /dev/null @@ -1,126 +0,0 @@ -import { Scene, Tools, ParticleSystem, Effect } from 'babylonjs'; - -import Extensions from '../extensions'; -import Extension from '../extension'; - -import { IStringDictionary } from '../typings/typings'; - -export interface ParticlesCreatorMetadata { - id: string; - apply: boolean; - code: string; - compiledCode?: string; - vertex: string; - pixel: string; -} - -const template = ` -EDITOR.ParticlesCreator.Constructors['{{name}}'] = function (scene, particleSystem) { -{{code}} -} -`; - -// Set EDITOR on Window -export module EDITOR { - export class ParticlesCreator { - public static Constructors = { }; - } -} -window['EDITOR'] = window['EDITOR'] || { }; -window['EDITOR'].ParticlesCreator = EDITOR.ParticlesCreator; - -export default class ParticlesCreatorExtension extends Extension { - // Public members - public instances: IStringDictionary = { }; - - /** - * Constructor - * @param scene: the babylonjs scene - */ - constructor (scene: Scene) { - super(scene); - this.datas = []; - } - - /** - * On apply the extension - */ - public onApply (data: ParticlesCreatorMetadata[], rootUrl?: string): void { - this.datas = data; - - // Add custom code - data.forEach(d => { - if (!d.apply) - return; - - const id = d.id + Tools.RandomId(); - - // Get particle system - const ps = this.scene.getParticleSystemByID(d.id); - if (!ps) - return; - - // Url - let url = window.location.href; - url = url.replace(Tools.GetFilename(url), '') + 'particle-systems/' + d.id.replace(/ /g, '') + '.js'; - - // Add script - Extension.AddScript(template.replace('{{name}}', id).replace('{{code}}', d.compiledCode || d.code), url); - - // Create code - const ctor = new EDITOR.ParticlesCreator.Constructors[id](this.scene, ps); - const code = new (ctor.ctor || ctor)(); - - // Create effect - const uniforms: string[] = []; - const samplers: string[] = []; - const defines: string[] = []; - code.setUniforms(uniforms, samplers); - code.setDefines(defines); - - Effect.ShadersStore[id + 'VertexShader'] = d.vertex; - Effect.ShadersStore[id + 'PixelShader'] = d.pixel; - - const effect = this.scene.getEngine().createEffectForParticles(id, uniforms, samplers, defines.join('\n')); - effect.onBind = effect => code.onBind(effect); - - ps.customShader = effect; - - // Save instance - this.instances[d.id] = code; - }); - } - - /** - * Called by the editor when serializing the scene - */ - public onSerialize (): ParticlesCreatorMetadata[] { - const datas: ParticlesCreatorMetadata[] = []; - this.scene.particleSystems.forEach(ps => { - if (ps['metadata'] && ps['metadata'].particlesCreator) { - ps['metadata'].particlesCreator.id = ps.id; - datas.push(ps['metadata'].particlesCreator); - } - }); - - return datas; - } - - /** - * On load the extension (called by the editor when - * loading a scene) - */ - public onLoad (data: ParticlesCreatorMetadata[]): void { - data.forEach(d => { - const ps = this.scene.getParticleSystemByID(d.id); - if (!ps) - return; - - ps['metadata'] = ps['metadata'] || { }; - ps['metadata'].particlesCreator = d; - }); - } -} - -// Register -Extensions.Register('ParticlesCreatorExtension', ParticlesCreatorExtension); diff --git a/src/extensions/path-finder/index.ts b/src/extensions/path-finder/index.ts index f0a832f4e..ef1803997 100644 --- a/src/extensions/path-finder/index.ts +++ b/src/extensions/path-finder/index.ts @@ -52,8 +52,7 @@ export default class PathFinderExtension extends Extension } /** - * On load the extension (called by the editor when - * loading a scene) + * On load the extension (called by the editor when loading a scene) */ public onLoad (data: PathFinderMetadata[]): void { this.datas = data; diff --git a/src/extensions/tools/tools.ts b/src/extensions/tools/tools.ts index 72f8519f7..131ad52d7 100644 --- a/src/extensions/tools/tools.ts +++ b/src/extensions/tools/tools.ts @@ -1,4 +1,4 @@ -import { Node, Scene, ParticleSystem, FilesInputStore } from 'babylonjs'; +import { Node, Scene, ParticleSystem, FilesInputStore, ParticleSystemSet, Vector3 } from 'babylonjs'; import Extensions from '../extensions'; import AssetsExtension from '../assets/assets'; @@ -107,7 +107,7 @@ export default class Tools { } /** - * Instantiate a prefab identified by the given name + * Instantiates a prefab identified by the given name * @param name the name of the prefab to instantiate */ public instantiatePrefab (name: string): T { @@ -115,6 +115,16 @@ export default class Tools { return ext.instantiatePrefab(name); } + /** + * Instantiates a particle system set identified by the given name + * @param name the name of the particle system set to instantiate + * @param position the position where to start systems + */ + public instantiateParticleSystemSet (name: string, position?: Vector3): ParticleSystemSet { + const ext = Extensions.Instances['AssetsExtension']; + return ext.instantiateParticleSystemsSet(name, position); + } + /** * Calls the given method with the given parameters on the given object which has scripts providing the given method * @param object the object reference where to send the message by calling the given method name diff --git a/src/index.ts b/src/index.ts index ea99d9e93..e92460196 100644 --- a/src/index.ts +++ b/src/index.ts @@ -4,6 +4,7 @@ import Tools from './editor/tools/tools'; import Request from './editor/tools/request'; import UndoRedo from './editor/tools/undo-redo'; import ThemeSwitcher, { ThemeType } from './editor/tools/theme'; +import GraphicsTools from './editor/tools/graphics-tools'; import Layout from './editor/gui/layout'; import Toolbar from './editor/gui/toolbar'; @@ -35,6 +36,10 @@ import ScenePreview from './editor/scene/scene-preview'; import PrefabAssetComponent from './editor/prefabs/asset-component'; import { Prefab, PrefabNodeType } from './editor/prefabs/prefab'; +import ParticlesCreatorExtension, { ParticlesCreatorMetadata } from './editor/particles/asset-component'; + +import Storage from './editor/storage/storage'; + import VSCodeSocket from './editor/vscode/vscode-socket'; export default Editor; @@ -45,6 +50,7 @@ export { Request, UndoRedo, ThemeSwitcher, ThemeType, + GraphicsTools, IStringDictionary, INumberDictionary, @@ -79,5 +85,10 @@ export { PrefabAssetComponent, Prefab, PrefabNodeType, + ParticlesCreatorExtension, + ParticlesCreatorMetadata, + + Storage, + VSCodeSocket } diff --git a/src/tools/animations/editor.ts b/src/tools/animations/editor.ts index 027acc30d..0772e5800 100644 --- a/src/tools/animations/editor.ts +++ b/src/tools/animations/editor.ts @@ -167,10 +167,14 @@ export default class AnimationEditor extends EditorPlugin { // No data text this.noDataText = this.paper.text(0, 0, 'No Animation Selected'); + this.noDataText.node.style.pointerEvents = 'none'; + this.noDataText.node.style.userSelect = 'none'; this.noDataText.attr('font-size', 64); // Value text this.valueText = this.paper.text(0, 0, '0.0'); + this.valueText.node.style.pointerEvents = 'none'; + this.valueText.node.style.userSelect = 'none'; this.valueText.attr('font-size', 10); this.valueText.hide(); @@ -440,9 +444,6 @@ export default class AnimationEditor extends EditorPlugin { * @param object: the IAnimatable object */ protected objectSelected(object: IAnimatable): void { - if (this.animatable === object) - return; - this.toolbar.element.disable('remove-animation'); // Clean @@ -618,6 +619,8 @@ export default class AnimationEditor extends EditorPlugin { // Text const text = this.paper.text(x, 30, (((x * maxFrame) / this.paper.width)).toFixed(2)); + text.node.style.pointerEvents = 'none'; + text.node.style.userSelect = 'none'; text.attr('opacity', 0.4); this.timelineTexts.push(text); } @@ -645,6 +648,8 @@ export default class AnimationEditor extends EditorPlugin { if (i > 0) { const text = this.paper.text(30, y, currentValue.toFixed(2)); text.attr('opacity', 0.4); + text.node.style.pointerEvents = 'none'; + text.node.style.userSelect = 'none'; this.timelineTexts.push(text); } @@ -856,6 +861,9 @@ export default class AnimationEditor extends EditorPlugin { this.updateGraph(animation); } }); + + // Notify + this.editor.core.onModifiedObject.notifyObservers(this.animatable); }; data.point.drag(onMove, onStart, onEnd); @@ -918,7 +926,7 @@ export default class AnimationEditor extends EditorPlugin { * On click on the timeline */ protected onClickTimeline(maxFrame: number): void { - this.timeline.click((ev: MouseEvent) => { + this.timeline.mousedown((ev: MouseEvent) => { if (this.isPlaying) return; diff --git a/src/tools/animations/property-browser.ts b/src/tools/animations/property-browser.ts index 3aeba70f4..b8d9a55e3 100644 --- a/src/tools/animations/property-browser.ts +++ b/src/tools/animations/property-browser.ts @@ -3,8 +3,8 @@ import { Vector2, Vector3, Vector4, Color3, Color4, Quaternion, - Camera, - Material + Camera, ParticleSystem, + Material, } from 'babylonjs'; import { Window, Graph, GraphNode, Tools } from 'babylonjs-editor'; @@ -150,7 +150,7 @@ export default class PropertyBrowser { continue; const id = `${rootName === '' ? '' : (rootName + '.')}${thing}`; - const allowed = this._allowedTypes.indexOf(ctor) !== -1; + const allowed = (thing === 'emitter' && root instanceof ParticleSystem) || (this._allowedTypes.indexOf(ctor) !== -1); const deep = this._deepTypes.find(dt => value instanceof dt); if (!allowed && !deep) diff --git a/src/tools/helpers.ts b/src/tools/helpers.ts index 875167ac9..8510f438d 100644 --- a/src/tools/helpers.ts +++ b/src/tools/helpers.ts @@ -1,7 +1,22 @@ +import { Engine, Scene, ArcRotateCamera, Vector3 } from 'babylonjs'; import { Editor, CodeEditor } from 'babylonjs-editor'; import Extensions from '../extensions/extensions'; -import CodeExtension from '../extensions/behavior/code'; + +import CodeExtension, { BehaviorCode } from '../extensions/behavior/code'; +import { GraphData } from '../extensions/behavior/graph'; + +export interface SceneMetadata { + behaviorScripts?: BehaviorCode[]; + behaviorGraphs?: GraphData[]; + [index: string]: any; +} + +export interface Preview { + engine: Engine; + scene: Scene; + camera: ArcRotateCamera; +} export default class Helpers { /** @@ -43,4 +58,31 @@ export default class Helpers { }); } } + + /** + * Returns the scene's metadatas. If empty, default object is created + * @param scene the scene to retrieve its metadatas + */ + public static GetSceneMetadatas (scene: Scene): SceneMetadata { + scene.metadata = scene.metadata || { }; + return scene.metadata; + } + + /** + * Creates a new preview object by creating a new engine, scene and camera + * @param canvas the canvas where to render the created scene + */ + public static CreatePreview (canvas: HTMLCanvasElement): Preview { + const engine = new Engine(canvas); + const scene = new Scene(engine); + + const camera = new ArcRotateCamera('PreviewCamera', 0, 0, 10, Vector3.Zero(), scene, true); + camera.attachControl(canvas, true, false); + + return { + engine: engine, + scene: scene, + camera: camera + }; + } } diff --git a/src/tools/particles-creator/index.ts b/src/tools/particles-creator/index.ts index c208d32c5..759fb6e34 100644 --- a/src/tools/particles-creator/index.ts +++ b/src/tools/particles-creator/index.ts @@ -1,43 +1,51 @@ -import { ParticleSystem, Effect, Observer } from 'babylonjs'; +import { + ParticleSystemSet, Observer, ParticleSystem, Tools as BabylonTools, + FilesInputStore, Vector3, ParticleHelper, FactorGradient +} from 'babylonjs'; import Editor, { - EditorPlugin, - - Layout, - CodeEditor, - Tools, - Toolbar + EditorPlugin, Tools, + Layout, Toolbar, Tree, + Dialog, UndoRedo, GraphicsTools, + Storage, ParticlesCreatorMetadata, INumberDictionary } from 'babylonjs-editor'; -import '../../extensions/particles-creator/particles-creator'; -import { ParticlesCreatorMetadata } from '../../extensions/particles-creator/particles-creator'; -import Extensions from '../../extensions/extensions'; +import Helpers, { Preview } from '../helpers'; + +import Timeline from './timeline'; export default class ParticlesCreator extends EditorPlugin { // Public members public layout: Layout = null; public toolbar: Toolbar = null; + public tree: Tree = null; + public tabs: W2UI.W2Tabs = null; - public functionsCode: CodeEditor = null; - public pixelCode: CodeEditor = null; - public vertexCode: CodeEditor = null; + public undoRedoId: string = 'particles-creator'; // Protected members protected data: ParticlesCreatorMetadata = null; - protected currentTab: string = 'PARTICLES-CREATOR-FUNCTIONS'; - protected selectedObjectEvent: Observer = null; + protected preview: Preview = null; + protected set: ParticleSystemSet = null; + + protected timeline: Timeline = null; + protected currentParticleSystem: ParticleSystem = null; + protected onSelectAssetObserver: Observer = null; + + // Private members + private _modifyingObjectObserver: Observer; + private _modifiedObjectObserver: Observer; // Static members - public static DefaultCode: string = ''; - public static DefaultVertex: string = ''; - public static DefaultPixel: string = ''; + public static DefaultSet: any = null; /** * Constructor * @param name: the name of the plugin */ - constructor(public editor: Editor) { - super('Particles Creator'); + constructor(public editor: Editor, asset: ParticlesCreatorMetadata = null) { + super('Particle Systems Creator'); + this.data = asset; } /** @@ -47,11 +55,14 @@ export default class ParticlesCreator extends EditorPlugin { this.layout.element.destroy(); this.toolbar.element.destroy(); - this.functionsCode.dispose(); - this.vertexCode.dispose(); - this.pixelCode.dispose(); + this.preview.scene.dispose(); + this.preview.engine.dispose(); + + this.timeline.dispose(); - this.editor.core.onSelectObject.remove(this.selectedObjectEvent); + this.editor.core.onSelectAsset.remove(this.onSelectAssetObserver); + this.editor.core.onModifyingObject.remove(this._modifyingObjectObserver); + this.editor.core.onModifiedObject.remove(this._modifiedObjectObserver); await super.close(); } @@ -60,88 +71,120 @@ export default class ParticlesCreator extends EditorPlugin { * Creates the plugin */ public async create(): Promise { - // Template - !ParticlesCreator.DefaultCode && (ParticlesCreator.DefaultCode = await Tools.LoadFile('./assets/templates/particles-creator/class.ts')); - !ParticlesCreator.DefaultVertex && (ParticlesCreator.DefaultVertex = await Tools.LoadFile('./assets/templates/particles-creator/shader.vertex.fx')); - !ParticlesCreator.DefaultPixel && (ParticlesCreator.DefaultPixel = await Tools.LoadFile('./assets/templates/particles-creator/shader.fragment.fx')); - + // Default + if (!ParticlesCreator.DefaultSet) { + const set = await Tools.LoadFile('./assets/templates/particles-creator/default-set.json', false); + ParticlesCreator.DefaultSet = JSON.parse(set); + } + // Layout this.layout = new Layout(this.divElement.id); this.layout.panels = [ - { type: 'top', size: 32, resizable: false, content: `
` }, + { type: 'top', size: 30, resizable: false, content: `
` }, { - type: 'main', - resizable: false, + type: 'left', + size: '50%', + resizable: false, content: ` -
- - - `, - tabs: [ - { id: 'functions', caption: 'Functions' }, - { id: 'vertex', caption: 'Vertex Shader' }, - { id: 'pixel', caption: 'Pixel Shader' } +
+
`, + tabs: [ + { id: 'tree', caption: 'List' }, + { id: 'timeline', caption: 'Timeline' } ] - } + }, + { type: 'main', size: '50%', resizable: false, content: `` }, ]; this.layout.build(this.divElement.id); + // Tabs + this.tabs = this.layout.getPanelFromType('left').tabs; + this.tabs.on('click', (ev) => this.tabChanged(ev.target)); + this.tabs.select('timeline'); + this.tabChanged('timeline'); + // Toolbar this.toolbar = new Toolbar('PARTICLES-CREATOR-TOOLBAR'); this.toolbar.items = [ - { type: 'button', id: 'apply', img: 'icon-play-game-windowed', text: 'Apply' }, + { id: 'add', text: 'Add System...', caption: 'Add System...', img: 'icon-add' }, + { id: 'reset', text: 'Reset', caption: 'Reset', img: 'icon-play-game' }, + { type: 'break' }, + { id: 'export', text: 'Export As...', caption: 'Export As...', img: 'icon-export' }, { type: 'break' }, - { type: 'button', img: 'icon-add', text: 'Import From...' } + { id: 'preset', type: 'menu', text: 'Presets', caption: 'Presets', img: 'icon-particles', items: [ + { id: 'smoke', text: 'Smoke', caption: 'Smoke' }, + { id: 'sun', text: 'Sun', caption: 'Sun' }, + // { id: 'rain', text: 'Rain', caption: 'Rain' }, + { id: 'fire', text: 'Fire', caption: 'Fire' }, + { id: 'explosion', text: 'Explosion', caption: 'Explosion' } + ] } ]; this.toolbar.onClick = id => this.toolbarClicked(id); this.toolbar.build('PARTICLES-CREATOR-TOOLBAR'); + this.toolbar.items.forEach(i => this.toolbar.enable(i.id, false)); - // Create editors - this.functionsCode = new CodeEditor('typescript', ''); - await this.functionsCode.build('PARTICLES-CREATOR-FUNCTIONS'); - this.functionsCode.onChange = value => { - if (this.data) { - this.data.code = value; - this.data.compiledCode = this.functionsCode.transpileTypeScript(value, this.data.id.replace(/ /, '')); - } - }; + // Create tree + this.tree = new Tree('PARTICLES-CREATOR-TREE'); + this.tree.wholerow = true; + this.tree.onClick = ( (id, data) => { + this.currentParticleSystem = data; + this.editor.core.onSelectObject.notifyObservers(data); + }); + this.tree.onCanDrag = () => false; + this.tree.onRename = ( (id, name, data) => { + this.currentParticleSystem.name = name; + this.resetSet(true); + return true; + }); + this.tree.onContextMenu = ( (id, data) => { + return [ + { id: 'remove', text: 'Remove', callback: () => this.removeSystemFromSet(data) } + ]; + }); + this.tree.build('PARTICLES-CREATOR-TREE'); - this.vertexCode = new CodeEditor('cpp', ''); - await this.vertexCode.build('PARTICLES-CREATOR-VERTEX'); - this.vertexCode.onChange = value => this.data && (this.data.vertex = value); + // Create timeline + this.timeline = new Timeline(this, $('#PARTICLES-CREATOR-TIMELINE')[0]); + + // Create preview + this.preview = Helpers.CreatePreview( $('#PARTICLES-CREATOR-CANVAS')[0]); + this.preview.camera.wheelPrecision = 100; + this.preview.engine.runRenderLoop(() => this.preview.scene.render()); - this.pixelCode = new CodeEditor('cpp', ''); - await this.pixelCode.build('PARTICLES-CREATOR-PIXEL'); - this.pixelCode.onChange = value => this.data && (this.data.pixel = value); - // Events - this.selectedObjectEvent = this.editor.core.onSelectObject.add(obj => this.selectObject(obj)); - - this.layout.getPanelFromType('main').tabs.on('click', (ev) => { - $('#' + this.currentTab).hide(); - this.currentTab = 'PARTICLES-CREATOR-' + ev.target.toUpperCase(); - $('#' + this.currentTab).show(); + this.onSelectAssetObserver = this.editor.core.onSelectAsset.add((a) => this.selectAsset(a)); + this._modifyingObjectObserver = this.editor.core.onModifyingObject.add((o: ParticleSystem) => { + if (!this.data) + return; + + this.timeline.onModifyingSystem(o); + this.saveSet(); }); + this._modifiedObjectObserver = this.editor.core.onModifiedObject.add((o: ParticleSystem) => { + if (!this.data) + return; - // Extension - Extensions.RequestExtension(this.editor.core.scene, 'ParticlesCreatorExtension'); + this.timeline.onModifiedSystem(o); + this.saveSet(); + }); - // Finish - this.selectObject(this.editor.core.currentSelectedObject); + // Select asset + if (this.data) + this.selectAsset(this.data); } /** * On hide the plugin (do not render scene) */ - public async onHide (): Promise { + public onHide (): void { } /** * On show the plugin (render scene) */ - public async onShow (): Promise { - + public onShow (asset?: ParticlesCreatorMetadata): void { + } /** @@ -149,56 +192,328 @@ export default class ParticlesCreator extends EditorPlugin { */ public onResize (): void { this.layout.element.resize(); + this.preview.engine.resize(); + + const size = this.layout.getPanelSize('left'); + this.timeline.resize(size.width, size.height); + } + + /** + * Called on the user clicks on an item of the toolbar + * @param id the id of the clicked item + */ + protected async toolbarClicked (id: string): Promise { + switch (id) { + // Add a new particle systems set + case 'add': + const name = await Dialog.CreateWithTextInput('Particle System name?'); + await this.addSystemToSet(ParticlesCreator.DefaultSet.systems[0], name); + this.saveSet(); + this.resetSet(true); + break; + // Reset particle systems set + case 'reset': + this.resetSet(true); + break; + + // Export + case 'export': + this.exportSet(); + break; + + // Presets + case 'preset:smoke': this.createFromPreset('smoke'); break; + case 'preset:sun': this.createFromPreset('sun'); break; + case 'preset:rain': this.createFromPreset('rain'); break; + case 'preset:fire': this.createFromPreset('fire'); break; + case 'preset:explosion': this.createFromPreset('explosion'); break; + } } /** - * On the user selects an object - * @param object the selected object + * Called on the user changes tab + * @param id the id of the new tab */ - public selectObject (object: any): void { - if (!(object instanceof ParticleSystem)) { - this.data = null; - this.functionsCode.setValue(''); - this.vertexCode.setValue(''); - this.pixelCode.setValue(''); - this.layout.lockPanel('main', 'No Particle System Selected...'); + protected tabChanged (id: string): void { + // Hide all + $('#PARTICLES-CREATOR-TREE').hide(); + $('#PARTICLES-CREATOR-TIMELINE').hide(); + + // Select which to show + switch (id) { + case 'tree': $('#PARTICLES-CREATOR-TREE').show(); break; + case 'timeline': $('#PARTICLES-CREATOR-TIMELINE').show(); break; + default: break; + } + } + + /** + * Called on the user selects an asset in the assets panel + * @param asset the asset being selected + */ + protected selectAsset (asset: ParticlesCreatorMetadata): void { + if (!asset || !asset.psData) { + // Lock toolbar + this.toolbar.items.forEach(i => this.toolbar.enable(i.id, false)); return; } - object['metadata'] = object['metadata'] || { }; - this.data = object['metadata'].particlesCreator || { - id: object.id, - apply: false, - code: ParticlesCreator.DefaultCode, - vertex: ParticlesCreator.DefaultVertex, - pixel: ParticlesCreator.DefaultPixel - }; - object['metadata'].particlesCreator = this.data; + // Unlock toolbar + this.toolbar.items.forEach(i => this.toolbar.enable(i.id, true)); + + // Misc. + this.data = asset; + + // Set + if (this.set) { + this.set.dispose(); + this.set = null; + } + + this.resetSet(true); + + // Timeline + this.timeline.setSet(this.set); - // Unlock and fill - this.layout.unlockPanel('main'); - this.functionsCode.setValue(this.data.code); - this.vertexCode.setValue(this.data.vertex); - this.pixelCode.setValue(this.data.pixel); + // Undo/Redo + UndoRedo.ClearScope(this.undoRedoId); } /** - * On the user clicks on the toolbar - * @param id the id of the clicked item + * Adds a new particle system to the current set according to the given data + * @param particleSystemData the particle system data to parse + */ + protected addSystemToSet (particleSystemData: any, name?: string): Promise { + if (!this.set) + return; + + // Replace id + particleSystemData.id = BabylonTools.RandomId(); + + return new Promise((resolve) => { + // Create system + const rootUrl = particleSystemData.textureName ? (particleSystemData.textureName.indexOf('data:') === 0 ? '' : 'file:') : ''; + + const ps = ParticleSystem.Parse(particleSystemData, this.preview.scene, rootUrl, true); + ps.emitter = ps.emitter || Vector3.Zero(); + ps.name = name || ps.name; + + this._cleanGradients(ps); + + // Add to set + this.set.systems.push(ps); + + // Resolve + if (ps.particleTexture && rootUrl === 'file:') + ps.particleTexture.onLoadObservable.add(tex => resolve(ps)); + else + resolve(ps); + }); + } + + /** + * Removes the given particle system from the current set + * @param ps the particle system to remove + */ + public removeSystemFromSet (ps: ParticleSystem): void { + // Remove from set + const index = this.set.systems.indexOf(ps); + if (index === -1) + return; + + this.set.systems.splice(index, 1); + + // Finally dispose + ps.dispose(); + + // Remove from tree + this.tree.remove(ps.id); + + // Save & reset + this.saveSet(); + this.resetSet(true); + } + + /** + * Resets the particle systems set + * @param fillTree wehter or not the list tree should be filled. */ - public toolbarClicked (id: string): void { + public async resetSet (fillTree: boolean = false): Promise { + if (!this.data) + return; + + // Const data + const data = this.set ? this.set.serialize() : this.data.psData; + + // Dispose previous set + if (this.set) { + this.set.dispose(); + this.preview.scene.particleSystems = []; + } + + // Clear tree? + if (fillTree) + this.tree.clear(); + + // Parse set + this.set = new ParticleSystemSet(); + for (const s of data.systems) { + const ps = await this.addSystemToSet(s); + if (fillTree) + this.tree.add({ id: ps.id, text: ps.name, data: ps, img: 'icon-particles' }); + } + + this.set.start(); + + if (this.currentParticleSystem) { + const ps = this.set.systems.find(ps => ps.name === this.currentParticleSystem.name); + if (ps) { + this.tree.select(ps.id); + this.editor.core.onSelectObject.notifyObservers(ps); + } + } + + // Refresh timeline + this.timeline.setSet(this.set); + } + + /** + * Saves the current particle systems set + */ + public saveSet (): void { + if (!this.data) + return; + + // Save! + this.data.psData = this.set.serialize(); + } + + /** + * Exports the current set + */ + protected async exportSet (): Promise { if (!this.data) return; - switch (id) { - // Apply - case 'apply': - const checked = this.toolbar.isChecked(id, true); - this.data.apply = checked; - this.toolbar.setChecked(id, checked); - break; - - default: break; + // Save + this.saveSet(); + + // Export + const serializationObject = this.set.serialize(); + + // Embed? + const embed = await Dialog.Create('Embed textures?', 'Do you want to embed textures in the set?'); + if (embed === 'Yes') { + for (const s of serializationObject.systems) { + const file = FilesInputStore.FilesToLoad[s.textureName.toLowerCase()]; + if (!file) + continue; + s.textureName = await Tools.ReadFileAsBase64(file); + } } + + // Save data + const json = JSON.stringify(serializationObject, null, '\t'); + const file = Tools.CreateFile(Tools.ConvertStringToUInt8Array(json), this.data.name + '.json'); + + // Embeded + if (embed === 'Yes') + return BabylonTools.Download(file, file.name); + + // Not embeded + const textureFiles: File[] = []; + for (const s of serializationObject.systems) { + const file = FilesInputStore.FilesToLoad[s.textureName.toLowerCase()]; + if (!file || textureFiles.indexOf(file) !== -1) + continue; + textureFiles.push(file); + } + + const storage = await Storage.GetStorage(this.editor); + await storage.openPicker('Choose destination folder...', [ + { name: 'systems', folder: [ + { name: file.name, file: file }, + { name: 'textures', folder: textureFiles.map(tf => ({ + name: tf.name, + file: tf + })) } + ] } + ]); + } + + /** + * Creates a new system set from the already know presets form babylonjs.com + * @param preset the preset id to create from the babylon.js presets + */ + protected async createFromPreset (preset: string): Promise { + // Ask override + const override = await Dialog.Create('Override current set?', 'Setting from a preset will override your current set configuration. Are you sure?'); + if (override === 'No') + return; + + // Lock panel + this.layout.lockPanel('top', 'Loading...', true); + + // Dispose existing set + if (this.set) + this.set.dispose(); + + // Create set! + this.set = await ParticleHelper.CreateAsync(preset, this.preview.scene); + this.set.systems.forEach(s => this._cleanGradients( s)); + + // Save textures + const promises: Promise[] = []; + for (const s of this.set.systems) { + promises.push(new Promise((resolve) => { + s.particleTexture.onLoadObservable.addOnce(async tex => { + const blob = await GraphicsTools.TextureToFile(tex); + const split = tex.name.split('/'); + const name = split[split.length - 1]; + + blob['name'] = tex.name = tex.url = name.toLowerCase(); + FilesInputStore.FilesToLoad[name.toLowerCase()] = blob; + + resolve(); + }); + })); + } + + await Promise.all(promises); + + // Save and reset + this.saveSet(); + this.resetSet(true); + + // Unlock + this.layout.unlockPanel('top'); + } + + /** + * Cleans the size gradients + * @param system the system to clean + * @todo remove this fix in future + */ + private _cleanGradients (system: ParticleSystem): void { + const sizeGradients = system['_sizeGradients']; + if (!sizeGradients) + return; + + const gradientsDict: INumberDictionary = { }; + sizeGradients.forEach(sg => { + gradientsDict[sg.gradient] = gradientsDict[sg.gradient] || []; + gradientsDict[sg.gradient].push(sg); + }); + + sizeGradients.splice(0, sizeGradients.length); + for (const k in gradientsDict) { + const g = gradientsDict[k]; + while (g.length > 1) + g.pop(); + + sizeGradients.push(g[0]) + } + + system['_sizeGradients'] = sizeGradients; } } diff --git a/src/tools/particles-creator/timeline.ts b/src/tools/particles-creator/timeline.ts new file mode 100644 index 000000000..e5acd72cb --- /dev/null +++ b/src/tools/particles-creator/timeline.ts @@ -0,0 +1,348 @@ +import * as Raphael from 'raphael'; +import { ParticleSystemSet, ParticleSystem, Observer } from 'babylonjs'; +import { ContextMenu } from 'babylonjs-editor'; + +import ParticlesCreator from './index'; + +export default class Timeline { + // Public members + public paper: RaphaelPaper; + public background: RaphaelElement; + public timeBackground: RaphaelElement; + public timeLines: RaphaelElement[] = []; + public playLine: RaphaelElement; + + public separators: RaphaelElement[] = []; + public systems: RaphaelElement[] = []; + public names: RaphaelElement[] = []; + public times: RaphaelElement[] = []; + + // Private members + private _maxX: number = 0; + private _backgroundX: number = 0; + private _set: ParticleSystemSet = null; + + // Static members + private static _Scale: number = 100; + + /** + * Constructor + * @param root the root element where to draw the timeline + */ + constructor (public creator: ParticlesCreator, root: HTMLDivElement) { + // Create paper + this.paper = Raphael(root, 0, 0); + + // Create background + this.background = this.paper.rect(0, 0, 1, 1); + this.background.attr('fill', '#aaa'); + this.background.attr('stroke', '#aaa'); + this._onMoveBackground(); + + // Create time background + this.timeBackground = this.paper.rect(0, 0, 1, 25); + this.timeBackground.attr('fill', '#777'); + this.timeBackground.attr('stroke', '#777'); + + // Play line + this.playLine = this.paper.rect(0, 0, 1, 1); + this.playLine.attr('fill', '#999'); + this.playLine.attr('stroke', '#999'); + } + + /** + * Disposes the timeline + */ + public dispose (): void { + this.paper.remove(); + } + + /** + * Resizes the timeline + * @param width the new width of the timeline + * @param height the new height of the timeline + */ + public resize (width: number, height: number): void { + this.paper.setSize(width, height); + + this.background.attr('width', width); + this.background.attr('height', height); + + this.timeBackground.attr('width', width); + + this.playLine.attr('height', height); + + this.setSet(this._set); + } + + /** + * Sets the current particle systems set + * @param set the new set to draw + */ + public setSet (set: ParticleSystemSet): void { + if (!set) + return; + + // Misc. + const shouldPlay = this._set !== set; + this._maxX = 0; + this._backgroundX = 0; + + // Save set + this._set = set; + + // Destroy all + this.systems.forEach(s => s.remove()); + this.systems = []; + + this.separators.forEach(s => s.remove()); + this.separators = []; + + this.names.forEach(n => n.remove()); + this.names = []; + + this.times.forEach(t => t.remove()); + this.times = []; + + this.timeLines.forEach(t => t.remove()); + this.timeLines = []; + + // Add systems + set.systems.forEach((s, i) => { + const index = i + 1; + + // Create new system element + const system = this.paper.rect(0, 40 * index + 1, 100, 35, 16); + system.attr('fill', '#ddd'); + system.attr('stroke-width', 0); + system.attr('x', (s.startDelay / 1000 * Timeline._Scale)); + system.data('bx', system.attr('x')); + this.systems.push(system); + + if (shouldPlay) { + const scaleFrom = Raphael.animation({ transform: 's1.25,1.25' }, 300); + const strokeFrom = Raphael.animation({ 'stroke-width': 5 }, 300); + const scaleTo = Raphael.animation({ transform: 's1,1' }, 300); + const strokeTo = Raphael.animation({ 'stroke-width': 0 }, 300); + + system.animate(scaleFrom.delay(s.startDelay)); + system.animate(strokeFrom.delay(s.startDelay)); + system.animate(scaleTo.delay(s.startDelay + 301)); + system.animate(strokeTo.delay(s.startDelay + 301)); + } + + // Name + const name = this.paper.text(0, 0, s.name); + name.attr('x', system.attr('x') + system.attr('width') / 2 - name.attr('width') / 2); + name.attr('y', system.attr('y') + system.attr('height') / 2 - name.attr('height') / 2 - 10); + name.data('bx', name.attr('x')); + name.node.style.pointerEvents = 'none'; + this.names.push(name); + + // Time + const time = this.paper.text(0, 0, s.startDelay + ' (ms)'); + time.attr('x', system.attr('x') + system.attr('width') / 2 - time.attr('width') / 2); + time.attr('y', system.attr('y') + system.attr('height') / 2 - time.attr('height') / 2 + 10); + time.data('bx', name.attr('x')); + time.node.style.pointerEvents = 'none'; + this.times.push(time); + + // Create line + const separator = this.paper.rect(0, 40 * (index + 1) - 2.5, this.paper.width, 1); + separator.attr('fill', '#666'); + separator.attr('stroke', '#666'); + this.separators.push(separator); + + // Events + this._onMoveSystem( s, system, name, time); + this._onShowSystemContextMenu( s, system); + + if (system.attr('x') > this._maxX) + this._maxX = system.attr('x') + system.attr('width'); + }); + + // Max z + if (this._maxX < 300) + this._maxX = 300; + + // Add timelines + const steps = 5; + const diff = Timeline._Scale / steps; + const end = (this._maxX / diff) * 2; + + for (let i = 0; i < end; i++) { + const isSecond = i % steps === 0; + + const line = this.paper.rect(i * diff, 0, 1, isSecond ? this.paper.height : (this.timeBackground.attr('height') - 15)); + line.attr('fill', '#999'); + line.attr('stroke-width', 0); + line.data('bx', line.attr('x')); + this.timeLines.push(line); + + if (isSecond) { + const text = this.paper.text(0, 20, ((i / steps) >> 0) + ' (s)'); + text.attr('x', i * diff + 5 + text.attr('width')); + text.data('bx', text.attr('x')); + text.node.style.pointerEvents = 'none'; + this.timeLines.push(text); + } + } + + // Play + this.playLine.transform('t0,0'); + this.playLine.attr('x', 0); + this.playLine.toFront(); + + if (shouldPlay) + this.playLine.animate({ transform: `t${this._maxX},0` }, (this._maxX * 1000) / Timeline._Scale); + + // Systems are front + this.systems.forEach(s => s.toFront()); + this.names.forEach(n => n.toFront()); + this.times.forEach(t => t.toFront()); + } + + /** + * Called on the user modifies a system + * @param system the system that is being modified + */ + public onModifyingSystem (system: ParticleSystem): void { + if (!this._set) + return; + + const index = this._set.systems.indexOf(system); + if (index !== -1) { + const s = this.systems[index]; + const n = this.names[index]; + const t = this.times[index]; + const diff = (system.startDelay - s.data('sd')) / 1000 * Timeline._Scale; + + s.transform(`t${diff},0`); + n.transform(`t${diff},0`); + t.transform(`t${diff},0`); + } + } + + /** + * Calld on the user modified a system + * @param system the system that has been modified + */ + public onModifiedSystem (system: ParticleSystem): void { + if (!this._set) + return; + + const index = this._set.systems.indexOf(system); + if (index !== -1) + this.setSet(this._set); + } + + // Performs a drag'n'drop animation for the background + private _onMoveBackground (): void { + // Drag'n'drop + let lx = 0; + let all: RaphaelElement[] = []; + + const onStart = ((x: number, y: number, ev: DragEvent) => { + all = this._getAllMovableElements(); + }); + + const onMove = ((dx: number, dy: number, x: number, y: number, ev: DragEvent) => { + lx = dx + this._backgroundX; + all.forEach(a => a.attr('x', (a.data('bx') || 0) + lx)); + }); + + const onEnd = ((ev) => { + this._backgroundX = lx; + }); + + this.background.drag( onMove, onStart, onEnd); + + // Wheel + this.background.node.addEventListener('wheel', (ev) => { + Timeline._Scale -= ev.deltaY * 0.05; + if (Timeline._Scale < 30) + Timeline._Scale = 30; + + this.setSet(this._set); + }); + } + + // Performs a drag'n'drop animation for systems + private _onMoveSystem (system: ParticleSystem, s: RaphaelElement, n: RaphaelElement, t: RaphaelElement): void { + const bx = s.attr('x'); + let ox = 0; + let lx = 0; + + const onStart = ((x: number, y: number, ev: DragEvent) => { + s.attr('opacity', 0.3); + + // Stroke width + this.systems.forEach(s => s.attr('stroke-width', 0)); + s.attr('stroke-width', 2); + + // Notify + this.creator.editor.core.onSelectObject.notifyObservers(system); + }); + + const onMove = ((dx: number, dy: number, x: number, y: number, ev: DragEvent) => { + const ms = ((bx + (dx + ox)) / Timeline._Scale * 1000) >> 0; + if (ms < 0) { + return; + } + + lx = dx + ox; + + s.transform(`t${lx},0`); + n.transform(`t${lx},0`); + t.transform(`t${lx},0`); + + t.attr('text', ms + ' (ms)'); + }); + + const onEnd = ((ev) => { + ox = lx; + system.startDelay = ((bx + ox) / Timeline._Scale * 1000) >> 0; + + // Update system + s.attr('opacity', 1); + s.data('sd', system.startDelay); + + // Save set + this.creator.saveSet(); + + // Update tools + if (this.creator.editor.edition.currentObject === system) + this.creator.editor.edition.updateDisplay(); + }); + + s.drag( onMove, onStart, onEnd); + } + + // Performs a context menu on the user right-clicks on the system + private _onShowSystemContextMenu (system: ParticleSystem, s: RaphaelElement): void { + s.node.classList.add('ctxmenu'); + s.node.addEventListener('contextmenu', (ev: MouseEvent) => { + ContextMenu.Show(ev, { + remove: { name: 'Remove', callback: () => this.creator.removeSystemFromSet(system) } + }); + }); + } + + // Returns all movable elements of the paper + private _getAllMovableElements (): RaphaelElement[] { + const result: RaphaelElement[] = []; + let bot = this.paper.bottom; + while (bot) { + if ( + bot === this.background || bot === this.timeBackground || this.separators.indexOf(bot) !== -1) { + bot = bot.next; + continue; + } + + result.push(bot); + bot = bot.next; + } + + return result; + } +} diff --git a/src/tools/play-game/index.ts b/src/tools/play-game/index.ts index 15bb69a8d..5c9ca495a 100644 --- a/src/tools/play-game/index.ts +++ b/src/tools/play-game/index.ts @@ -14,9 +14,6 @@ export default class PlayGame extends EditorPlugin { public capturer: CCapture = null; public isCapturing: boolean = false; public captureBlob: Blob = null; - - // Protected members - protected changeValueObserver: Observer = null; /** * Constructor @@ -36,9 +33,6 @@ export default class PlayGame extends EditorPlugin { // Capturer if (this.capturer) this.capturer.stop(); - - // Callbacks - this.editor.core.onGlobalPropertyChange.remove(this.changeValueObserver); await super.close(); } @@ -69,9 +63,6 @@ export default class PlayGame extends EditorPlugin { // Create iFrame await this.createIFrame(); - - // Events - this.changeValueObserver = this.editor.core.onGlobalPropertyChange.add(data => this.updateValue(data)); } /** @@ -227,49 +218,4 @@ export default class PlayGame extends EditorPlugin { protected showVideo (): void { } - - /** - * Updates the value in the preview page according to undo/redo - * @param data the data to undo-redo - */ - protected updateValue (data: { baseObject?: any; object: any; property: string; value: any; initialValue: any; }): void { - if (!data.baseObject) - return; - - // Get property - let additionalProperty: string = null; - - if (data.baseObject[data.property] === undefined) { - for (const thing in data.baseObject) { - if (data.baseObject[thing] !== data.object) - continue; - - additionalProperty = thing; - break; - } - - if (!additionalProperty) - return; - } - - const scene = this.contentWindow['effectiveScene']; - - const id = data.baseObject.id; - const obj = - scene.getMeshByID(id) || - scene.getMaterialByID(id) || - scene.getLightByID(id) || - scene.getCameraByID(id) || - scene.getParticleSystemByID(id) || - scene.getSkeletonById(id) || - scene.getLensFlareSystemByID(id); - - if (!obj) - return; - - if (additionalProperty) - obj[additionalProperty][data.property] = data.value; - else - obj[data.property] = data.value; - } } diff --git a/src/tools/post-process-editor/index.ts b/src/tools/post-process-editor/index.ts index 9b90a0ea0..6433d110f 100644 --- a/src/tools/post-process-editor/index.ts +++ b/src/tools/post-process-editor/index.ts @@ -128,7 +128,7 @@ export default class PostProcessEditor extends EditorPlugin { `, resizable: true, - tabs: [ + tabs: [ { id: 'code', caption: 'Code' }, { id: 'pixel', caption: 'Pixel' }, { id: 'config', caption: 'Config' } diff --git a/src/tools/textures/viewer.ts b/src/tools/textures/viewer.ts index 85fe6948b..25f099eb0 100644 --- a/src/tools/textures/viewer.ts +++ b/src/tools/textures/viewer.ts @@ -371,6 +371,11 @@ export default class TextureViewer extends EditorPlugin { const texture = new Texture('file:' + file.name, this.editor.core.scene); texture.name = texture.url = texture.name.replace('file:', ''); } + + // Drag'n'drop + const dropListener = this.dragEnd(originalTexture, true); + img.addEventListener('dragstart', () => this.editor.core.engine.getRenderingCanvas().addEventListener('drop', dropListener)); + img.addEventListener('dragend', () => this.editor.core.engine.getRenderingCanvas().removeEventListener('drop', dropListener)); } else { const data = await Tools.ReadFileAsBase64(file); @@ -391,6 +396,11 @@ export default class TextureViewer extends EditorPlugin { const texture = new Texture('file:' + file.name, this.editor.core.scene); texture.name = texture.url = texture.name.replace('file:', ''); } + + // Drag'n'drop + const dropListener = this.dragEnd(originalTexture, false); + img.addEventListener('dragstart', () => this.editor.core.engine.getRenderingCanvas().addEventListener('drop', dropListener)); + img.addEventListener('dragend', () => this.editor.core.engine.getRenderingCanvas().removeEventListener('drop', dropListener)); } // Add text @@ -409,6 +419,40 @@ export default class TextureViewer extends EditorPlugin { parent.appendChild(text); } + /** + * Returns an event called when the user drops a texture on the preview canvas + * @param texture: the texture to drop on a mesh/instanced-mesh + */ + protected dragEnd (texture: BaseTexture, isCube: boolean): (ev: DragEvent) => void { + return (ev: DragEvent) => { + const scene = this.editor.core.scene; + const pick = scene.pick(ev.offsetX, ev.offsetY); + + if (!pick.pickedMesh || !pick.pickedMesh.material) + return; + + // Apply + const material = pick.pickedMesh.material; + + if (isCube) { + UndoRedo.Push({ baseObject: material, object: material, property: 'reflectionTexture', from: material['reflectionTexture'], to: texture }); + material['reflectionTexture'] = texture; + } + else { + if (material instanceof PBRMaterial) { + UndoRedo.Push({ baseObject: material, object: material, property: 'albedoTexture', from: material.albedoTexture, to: texture }); + material.albedoTexture = texture; + } + else { + UndoRedo.Push({ baseObject: material, object: material, property: 'diffuseTexture', from: material['diffuseTexture'], to: texture }); + material['diffuseTexture'] = texture; + } + } + + this.editor.core.onSelectObject.notifyObservers(pick.pickedMesh); + }; + } + /** * Convets a cube texture to */