class Locator extends OutlinerElement { constructor(data, uuid) { super(data, uuid); for (var key in Locator.properties) { Locator.properties[key].reset(this); } if (data) { this.extend(data); } } get origin() { return this.position; } extend(object) { if (object.from) this.position.V3_set(object.from); for (var key in Locator.properties) { Locator.properties[key].merge(this, object) } this.sanitizeName(); Merge.boolean(this, object, 'export'); return this; } getUndoCopy() { var copy = new Locator(this) copy.uuid = this.uuid copy.type = this.type; delete copy.parent; return copy; } getSaveCopy() { let save = {}; for (var key in Locator.properties) { Locator.properties[key].copy(this, save) } save.export = this.export ? undefined : false; save.uuid = this.uuid; save.type = 'locator'; return save; } init() { if (this.parent instanceof Group == false) { this.addTo(Group.selected) } super.init(); return this; } flip(axis, center) { var offset = this.position[axis] - center this.position[axis] = center - offset; this.rotation.forEach((n, i) => { if (i != axis) this.rotation[i] = -n; }) // Name if (axis == 0 && this.name.includes('right')) { this.name = this.name.replace(/right/g, 'left').replace(/2$/, ''); } else if (axis == 0 && this.name.includes('left')) { this.name = this.name.replace(/left/g, 'right').replace(/2$/, ''); } this.createUniqueName(); this.preview_controller.updateTransform(this); return this; } getWorldCenter() { var pos = Reusable.vec1.set(0, 0, 0); var q = Reusable.quat1.set(0, 0, 0, 1); if (this.parent instanceof Group) { THREE.fastWorldPosition(this.parent.mesh, pos); this.parent.mesh.getWorldQuaternion(q); var offset2 = Reusable.vec2.fromArray(this.parent.origin).applyQuaternion(q); pos.sub(offset2); } var offset = Reusable.vec3.fromArray(this.position).applyQuaternion(q); pos.add(offset); return pos; } } Locator.prototype.title = tl('data.locator'); Locator.prototype.type = 'locator'; Locator.prototype.icon = 'fa fa-anchor'; Locator.prototype.name_regex = 'a-z0-9_' Locator.prototype.movable = true; Locator.prototype.rotatable = true; Locator.prototype.visibility = true; Locator.prototype.buttons = [ Outliner.buttons.export, Outliner.buttons.locked, Outliner.buttons.visibility, ]; Locator.prototype.needsUniqueName = true; Locator.prototype.menu = new Menu([ { id: 'ignore_inherited_scale', name: 'menu.locator.ignore_inherited_scale', icon: locator => locator.ignore_inherited_scale ? 'check_box' : 'check_box_outline_blank', click(clicked_locator) { let value = !clicked_locator.ignore_inherited_scale; let affected = Locator.selected.filter(locator => locator.ignore_inherited_scale != value); Undo.initEdit({elements: affected}); affected.forEach(locator => { locator.ignore_inherited_scale = value; }) Undo.finishEdit('Change locator ignore inherit scale option'); } }, '_', ...Outliner.control_menu_group, '_', 'copy', 'paste', 'duplicate', '_', 'rename', 'delete' ]) new Property(Locator, 'string', 'name', {default: 'locator'}) new Property(Locator, 'vector', 'position') new Property(Locator, 'vector', 'rotation') new Property(Locator, 'boolean', 'ignore_inherited_scale') new Property(Locator, 'boolean', 'visibility', {default: true}); new Property(Locator, 'boolean', 'locked'); OutlinerElement.registerType(Locator, 'locator'); (function() { const map = new THREE.TextureLoader().load( 'assets/locator.png' ); map.magFilter = map.minFilter = THREE.NearestFilter; new NodePreviewController(Locator, { setup(element) { let material = new THREE.SpriteMaterial({ map, alphaTest: 0.1, sizeAttenuation: false }); var mesh = new THREE.Sprite(material); Project.nodes_3d[element.uuid] = mesh; mesh.name = element.uuid; mesh.type = element.type; mesh.isElement = true; mesh.visible = element.visibility; mesh.rotation.order = 'ZYX'; this.updateTransform(element); this.dispatchEvent('setup', {element}); }, updateTransform(element) { NodePreviewController.prototype.updateTransform(element); this.updateWindowSize(element); }, updateSelection(element) { let {mesh} = element; mesh.material.color.set(element.selected ? gizmo_colors.outline : CustomTheme.data.colors.text); mesh.material.depthTest = !element.selected; mesh.renderOrder = element.selected ? 100 : 0; this.dispatchEvent('update_selection', {element}); }, updateWindowSize(element) { let size = 18 / Preview.selected.height; element.mesh.scale.set(size, size, size); } }) let locator_suggestion_list = $('').get(0); document.body.append(locator_suggestion_list); Locator.updateAutocompleteList = function() { locator_suggestion_list.innerHTML = ''; Locator.all.forEach(locator => { let option = document.createElement('option'); option.value = locator.name; locator_suggestion_list.append(option); }) } })() BARS.defineActions(function() { new Action('add_locator', { icon: 'fa-anchor', category: 'edit', condition: () => {return Format.locators && Modes.edit}, click: function () { var objs = [] Undo.initEdit({elements: objs, outliner: true}); var locator = new Locator().addTo(Group.selected||selected[0]).init(); locator.select().createUniqueName(); objs.push(locator); Undo.finishEdit('Add locator'); Vue.nextTick(function() { if (settings.create_rename.value) { locator.rename(); } }) } }) })