2021-01-04 20:45:33 +08:00
|
|
|
|
2021-01-31 05:41:24 +08:00
|
|
|
class NullObject extends OutlinerElement {
|
2021-01-04 20:45:33 +08:00
|
|
|
constructor(data, uuid) {
|
|
|
|
super(data, uuid);
|
|
|
|
|
|
|
|
for (var key in NullObject.properties) {
|
|
|
|
NullObject.properties[key].reset(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
this.extend(data);
|
|
|
|
}
|
|
|
|
}
|
2021-09-18 04:32:53 +08:00
|
|
|
get origin() {
|
2022-09-23 23:47:37 +08:00
|
|
|
return this.position;
|
2021-09-18 04:32:53 +08:00
|
|
|
}
|
2021-01-04 20:45:33 +08:00
|
|
|
extend(object) {
|
2022-09-23 23:47:37 +08:00
|
|
|
if (object.from) this.position.V3_set(object.from);
|
2021-01-04 20:45:33 +08:00
|
|
|
for (var key in NullObject.properties) {
|
|
|
|
NullObject.properties[key].merge(this, object)
|
|
|
|
}
|
|
|
|
this.sanitizeName();
|
|
|
|
//Merge.boolean(this, object, 'export');
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
getUndoCopy() {
|
|
|
|
var copy = new NullObject(this)
|
|
|
|
copy.uuid = this.uuid
|
|
|
|
copy.type = this.type;
|
|
|
|
delete copy.parent;
|
|
|
|
return copy;
|
|
|
|
}
|
|
|
|
getSaveCopy() {
|
|
|
|
let save = {};
|
|
|
|
for (var key in NullObject.properties) {
|
|
|
|
NullObject.properties[key].copy(this, save)
|
|
|
|
}
|
|
|
|
//save.export = this.export ? undefined : false;
|
|
|
|
save.uuid = this.uuid;
|
|
|
|
save.type = 'null_object';
|
|
|
|
return save;
|
|
|
|
}
|
|
|
|
init() {
|
|
|
|
if (this.parent instanceof Group == false) {
|
|
|
|
this.addTo(Group.selected)
|
|
|
|
}
|
|
|
|
super.init();
|
|
|
|
return this;
|
|
|
|
}
|
2021-11-21 19:07:43 +08:00
|
|
|
select(event, isOutlinerClick) {
|
|
|
|
super.select(event, isOutlinerClick);
|
|
|
|
if (Animator.open && Animation.selected) {
|
|
|
|
Animation.selected.getBoneAnimator(this).select(true);
|
|
|
|
}
|
2021-12-10 01:01:56 +08:00
|
|
|
return this;
|
2021-11-21 19:07:43 +08:00
|
|
|
}
|
2021-12-23 17:40:05 +08:00
|
|
|
unselect(...args) {
|
|
|
|
if (Animator.open && Timeline.selected_animator && Timeline.selected_animator.element == this) {
|
|
|
|
Timeline.selected_animator.selected = false;
|
|
|
|
}
|
|
|
|
return super.unselect(...args);
|
|
|
|
}
|
2021-01-04 20:45:33 +08:00
|
|
|
flip(axis, center) {
|
2022-09-23 23:47:37 +08:00
|
|
|
var offset = this.position[axis] - center
|
|
|
|
this.position[axis] = center - offset;
|
2021-01-04 20:45:33 +08:00
|
|
|
// 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();
|
2022-12-18 20:29:46 +08:00
|
|
|
this.preview_controller.updateTransform(this);
|
2021-01-04 20:45:33 +08:00
|
|
|
return this;
|
|
|
|
}
|
2021-11-30 01:15:08 +08:00
|
|
|
getWorldCenter(with_animation) {
|
2021-09-15 23:50:46 +08:00
|
|
|
var pos = Reusable.vec1.set(0, 0, 0);
|
|
|
|
var q = Reusable.quat1.set(0, 0, 0, 1);
|
2021-01-04 20:45:33 +08:00
|
|
|
if (this.parent instanceof Group) {
|
|
|
|
THREE.fastWorldPosition(this.parent.mesh, pos);
|
|
|
|
this.parent.mesh.getWorldQuaternion(q);
|
2021-09-15 23:50:46 +08:00
|
|
|
var offset2 = Reusable.vec2.fromArray(this.parent.origin).applyQuaternion(q);
|
2021-01-04 20:45:33 +08:00
|
|
|
pos.sub(offset2);
|
|
|
|
}
|
2021-12-18 21:36:12 +08:00
|
|
|
let offset;
|
|
|
|
if (with_animation && Animation.selected) {
|
|
|
|
offset = Reusable.vec3.copy(this.mesh.position);
|
|
|
|
if (this.parent instanceof Group) {
|
|
|
|
offset.x += this.parent.origin[0];
|
|
|
|
offset.y += this.parent.origin[1];
|
|
|
|
offset.z += this.parent.origin[2];
|
|
|
|
}
|
|
|
|
} else {
|
2022-09-23 23:47:37 +08:00
|
|
|
offset = Reusable.vec3.fromArray(this.position);
|
2021-12-18 21:36:12 +08:00
|
|
|
}
|
|
|
|
offset.applyQuaternion(q);
|
2021-01-04 20:45:33 +08:00
|
|
|
pos.add(offset);
|
|
|
|
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NullObject.prototype.title = tl('data.null_object');
|
|
|
|
NullObject.prototype.type = 'null_object';
|
|
|
|
NullObject.prototype.icon = 'fa far fa-circle';
|
|
|
|
//NullObject.prototype.name_regex = 'a-z0-9_'
|
|
|
|
NullObject.prototype.movable = true;
|
|
|
|
NullObject.prototype.visibility = true;
|
|
|
|
NullObject.prototype.buttons = [
|
|
|
|
//Outliner.buttons.export,
|
|
|
|
Outliner.buttons.locked,
|
2022-10-03 19:05:00 +08:00
|
|
|
Outliner.buttons.visibility,
|
2021-01-04 20:45:33 +08:00
|
|
|
];
|
2021-12-03 20:32:33 +08:00
|
|
|
NullObject.prototype.needsUniqueName = true;
|
2021-01-04 20:45:33 +08:00
|
|
|
NullObject.prototype.menu = new Menu([
|
2021-11-27 04:30:12 +08:00
|
|
|
'set_ik_target',
|
2021-12-03 20:32:33 +08:00
|
|
|
{
|
|
|
|
id: 'lock_ik_target_rotation',
|
|
|
|
name: 'menu.null_object.lock_ik_target_rotation',
|
|
|
|
icon: null_object => null_object.lock_ik_target_rotation ? 'check_box' : 'check_box_outline_blank',
|
|
|
|
click(clicked_null_object) {
|
|
|
|
let value = !clicked_null_object.lock_ik_target_rotation;
|
2021-12-08 01:31:16 +08:00
|
|
|
let affected = NullObject.selected.filter(null_object => null_object.lock_ik_target_rotation != value);
|
2021-12-03 20:32:33 +08:00
|
|
|
Undo.initEdit({elements: affected});
|
|
|
|
affected.forEach(null_object => {
|
|
|
|
null_object.lock_ik_target_rotation = value;
|
|
|
|
})
|
|
|
|
Undo.finishEdit('Change null object lock ik target rotation option');
|
2021-12-21 05:26:07 +08:00
|
|
|
if (Modes.animate) Animator.preview();
|
2021-12-03 20:32:33 +08:00
|
|
|
}
|
|
|
|
},
|
2021-11-27 04:30:12 +08:00
|
|
|
'_',
|
2022-06-22 05:30:18 +08:00
|
|
|
...Outliner.control_menu_group,
|
2021-06-15 05:42:11 +08:00
|
|
|
'_',
|
2021-01-04 20:45:33 +08:00
|
|
|
'rename',
|
|
|
|
'delete'
|
|
|
|
])
|
|
|
|
|
|
|
|
new Property(NullObject, 'string', 'name', {default: 'null_object'})
|
2022-09-23 23:47:37 +08:00
|
|
|
new Property(NullObject, 'vector', 'position')
|
2021-11-27 04:30:12 +08:00
|
|
|
new Property(NullObject, 'string', 'ik_target', {condition: () => Format.animation_mode});
|
2021-12-03 20:32:33 +08:00
|
|
|
new Property(NullObject, 'boolean', 'lock_ik_target_rotation')
|
2022-10-03 19:05:00 +08:00
|
|
|
new Property(NullObject, 'boolean', 'visibility', {default: true});
|
2021-10-09 22:29:33 +08:00
|
|
|
new Property(NullObject, 'boolean', 'locked');
|
2021-06-24 23:48:25 +08:00
|
|
|
|
2021-07-08 20:30:32 +08:00
|
|
|
OutlinerElement.registerType(NullObject, 'null_object');
|
2021-01-04 20:45:33 +08:00
|
|
|
|
2022-10-03 19:05:00 +08:00
|
|
|
(function() {
|
|
|
|
|
|
|
|
const map = new THREE.TextureLoader().load( 'assets/null_object.png' );
|
|
|
|
map.magFilter = map.minFilter = THREE.NearestFilter;
|
|
|
|
|
2021-11-21 19:07:43 +08:00
|
|
|
new NodePreviewController(NullObject, {
|
|
|
|
setup(element) {
|
2022-10-03 19:05:00 +08:00
|
|
|
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';
|
2021-11-21 19:07:43 +08:00
|
|
|
element.mesh.fix_position = new THREE.Vector3();
|
2022-10-03 19:05:00 +08:00
|
|
|
this.updateTransform(element);
|
|
|
|
|
|
|
|
this.dispatchEvent('setup', {element});
|
2022-05-26 04:41:54 +08:00
|
|
|
this.dispatchEvent('update_selection', {element});
|
2021-11-21 19:07:43 +08:00
|
|
|
},
|
|
|
|
updateTransform(element) {
|
|
|
|
NodePreviewController.prototype.updateTransform(element);
|
2022-10-03 20:59:20 +08:00
|
|
|
|
2021-11-21 19:07:43 +08:00
|
|
|
element.mesh.fix_position.copy(element.mesh.position);
|
2022-05-26 04:41:54 +08:00
|
|
|
|
2022-10-03 20:59:20 +08:00
|
|
|
this.updateWindowSize(element);
|
|
|
|
|
2022-05-26 04:41:54 +08:00
|
|
|
this.dispatchEvent('update_transform', {element});
|
2022-10-03 19:05:00 +08:00
|
|
|
},
|
|
|
|
updateSelection(element) {
|
|
|
|
let {mesh} = element;
|
|
|
|
|
|
|
|
mesh.material.color.set(element.selected ? gizmo_colors.outline : CustomTheme.data.colors.text);
|
2022-11-04 04:56:14 +08:00
|
|
|
mesh.material.depthTest = !element.selected;
|
|
|
|
mesh.renderOrder = element.selected ? 100 : 0;
|
2022-10-03 19:05:00 +08:00
|
|
|
|
|
|
|
this.dispatchEvent('update_selection', {element});
|
2022-10-03 20:59:20 +08:00
|
|
|
},
|
|
|
|
updateWindowSize(element) {
|
|
|
|
let size = 17 / Preview.selected.height;
|
|
|
|
element.mesh.scale.set(size, size, size);
|
2021-11-21 19:07:43 +08:00
|
|
|
}
|
|
|
|
})
|
2022-10-03 19:05:00 +08:00
|
|
|
|
|
|
|
})()
|
2021-09-18 04:32:53 +08:00
|
|
|
|
2021-01-04 20:45:33 +08:00
|
|
|
BARS.defineActions(function() {
|
|
|
|
new Action('add_null_object', {
|
|
|
|
icon: 'far.fa-circle',
|
|
|
|
category: 'edit',
|
2022-11-20 20:45:45 +08:00
|
|
|
condition: () => Format.animation_mode && Modes.edit,
|
2021-01-04 20:45:33 +08:00
|
|
|
click: function () {
|
|
|
|
var objs = []
|
|
|
|
Undo.initEdit({elements: objs, outliner: true});
|
|
|
|
var null_object = new NullObject().addTo(Group.selected||selected[0]).init();
|
2021-12-03 20:32:33 +08:00
|
|
|
null_object.select().createUniqueName();
|
2021-01-04 20:45:33 +08:00
|
|
|
objs.push(null_object);
|
2021-06-06 15:28:22 +08:00
|
|
|
Undo.finishEdit('Add null object');
|
2021-01-04 20:45:33 +08:00
|
|
|
Vue.nextTick(function() {
|
|
|
|
if (settings.create_rename.value) {
|
|
|
|
null_object.rename();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
2021-11-27 04:30:12 +08:00
|
|
|
|
|
|
|
new Action('set_ik_target', {
|
|
|
|
icon: 'fa-paperclip',
|
|
|
|
category: 'edit',
|
2022-11-18 05:16:53 +08:00
|
|
|
condition() {
|
|
|
|
let action = BarItems.set_ik_target;
|
|
|
|
return NullObject.selected.length && action.children(action).length
|
|
|
|
},
|
2022-09-07 23:59:48 +08:00
|
|
|
searchable: true,
|
2021-11-27 04:30:12 +08:00
|
|
|
children() {
|
2021-12-10 18:16:00 +08:00
|
|
|
let nodes = [];
|
2021-12-19 04:20:34 +08:00
|
|
|
iterate(NullObject.selected[0].getParentArray(), 0);
|
2021-11-27 04:30:12 +08:00
|
|
|
|
2021-12-19 04:20:34 +08:00
|
|
|
function iterate(arr, level) {
|
2021-11-27 04:30:12 +08:00
|
|
|
arr.forEach(node => {
|
|
|
|
if (node instanceof Group) {
|
2021-12-19 04:20:34 +08:00
|
|
|
if (level) nodes.push(node);
|
|
|
|
iterate(node.children, level+1);
|
2021-11-27 04:30:12 +08:00
|
|
|
}
|
2021-12-10 18:16:00 +08:00
|
|
|
if (node instanceof Locator) {
|
2021-12-19 04:20:34 +08:00
|
|
|
if (level) nodes.push(node);
|
2021-12-10 18:16:00 +08:00
|
|
|
}
|
2021-11-27 04:30:12 +08:00
|
|
|
})
|
|
|
|
}
|
2021-12-10 18:16:00 +08:00
|
|
|
return nodes.map(node => {
|
2021-11-27 04:30:12 +08:00
|
|
|
return {
|
2021-12-10 18:16:00 +08:00
|
|
|
name: node.name + (node.uuid == NullObject.selected[0].ik_target ? ' (✔)' : ''),
|
2022-09-07 23:59:48 +08:00
|
|
|
icon: node instanceof Locator ? 'fa-anchor' : 'folder',
|
|
|
|
color: markerColors[node.color % markerColors.length] && markerColors[node.color % markerColors.length].standard,
|
2021-11-27 04:30:12 +08:00
|
|
|
click() {
|
2021-11-30 01:15:08 +08:00
|
|
|
Undo.initEdit({elements: NullObject.selected});
|
2021-11-27 04:30:12 +08:00
|
|
|
NullObject.selected.forEach(null_object => {
|
2021-12-10 18:16:00 +08:00
|
|
|
null_object.ik_target = node.uuid;
|
2021-11-27 04:30:12 +08:00
|
|
|
})
|
2021-11-30 01:15:08 +08:00
|
|
|
Undo.finishEdit('Set IK target');
|
2021-11-27 04:30:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
|
|
|
click(event) {
|
2022-09-07 23:59:48 +08:00
|
|
|
new Menu('set_ik_target', this.children(this), {searchable: true}).show(event.target, this);
|
2021-11-27 04:30:12 +08:00
|
|
|
}
|
|
|
|
})
|
2021-01-04 20:45:33 +08:00
|
|
|
})
|