mirror of
https://github.com/lubosz/overte.git
synced 2025-04-25 15:13:28 +02:00
281 lines
9.1 KiB
JavaScript
281 lines
9.1 KiB
JavaScript
//
|
|
// groups.js
|
|
//
|
|
// Created by David Rowe on 1 Aug 2017.
|
|
// Copyright 2017 High Fidelity, Inc.
|
|
//
|
|
// Distributed under the Apache License, Version 2.0.
|
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
|
//
|
|
|
|
/* global Groups:true, App, History */
|
|
|
|
Groups = function () {
|
|
// Groups and ungroups trees of entities.
|
|
|
|
"use strict";
|
|
|
|
var rootEntityIDs = [],
|
|
selections = [],
|
|
entitiesSelectedCount = 0;
|
|
|
|
if (!(this instanceof Groups)) {
|
|
return new Groups();
|
|
}
|
|
|
|
function add(selection) {
|
|
rootEntityIDs.push(selection[0].id);
|
|
selections.push(Object.clone(selection));
|
|
entitiesSelectedCount += selection.length;
|
|
}
|
|
|
|
function remove(selection) {
|
|
var index = rootEntityIDs.indexOf(selection[0].id);
|
|
|
|
entitiesSelectedCount -= selections[index].length;
|
|
rootEntityIDs.splice(index, 1);
|
|
selections.splice(index, 1);
|
|
}
|
|
|
|
function toggle(selection) {
|
|
if (selection.length === 0) {
|
|
return;
|
|
}
|
|
|
|
if (rootEntityIDs.indexOf(selection[0].id) === -1) {
|
|
add(selection);
|
|
} else {
|
|
remove(selection);
|
|
}
|
|
}
|
|
|
|
function selection(excludes) {
|
|
var result = [],
|
|
i,
|
|
lengthI,
|
|
j,
|
|
lengthJ;
|
|
|
|
for (i = 0, lengthI = rootEntityIDs.length; i < lengthI; i++) {
|
|
if (excludes.indexOf(rootEntityIDs[i]) === -1) {
|
|
for (j = 0, lengthJ = selections[i].length; j < lengthJ; j++) {
|
|
result.push(selections[i][j]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
function includes(rootEntityID) {
|
|
return rootEntityIDs.indexOf(rootEntityID) !== -1;
|
|
}
|
|
|
|
function getRootEntityIDs() {
|
|
return rootEntityIDs;
|
|
}
|
|
|
|
function groupsCount() {
|
|
return selections.length;
|
|
}
|
|
|
|
function entitiesCount() {
|
|
return entitiesSelectedCount;
|
|
}
|
|
|
|
function group() {
|
|
// Groups all selections into one.
|
|
var DYNAMIC_AND_COLLISIONLESS = { dynamic: true, collisionless: true },
|
|
rootID,
|
|
undoData = [],
|
|
redoData = [],
|
|
i,
|
|
lengthI,
|
|
j,
|
|
lengthJ;
|
|
|
|
// If the first group has physics (i.e., root entity is dynamic) make all entities in child groups dynamic and
|
|
// collisionless. (Don't need to worry about other groups physics properties because only those of the first entity in
|
|
// the linkset are used by High Fidelity.) See Selection.applyPhysics().
|
|
if (selections[0][0].dynamic) {
|
|
for (i = 1, lengthI = selections.length; i < lengthI; i++) {
|
|
for (j = 0, lengthJ = selections[i].length; j < lengthJ; j++) {
|
|
undoData.push({
|
|
entityID: selections[i][j].id,
|
|
properties: {
|
|
dynamic: selections[i][j].dynamic,
|
|
collisionless: selections[i][j].collisionless
|
|
}
|
|
});
|
|
Entities.editEntity(selections[i][j].id, DYNAMIC_AND_COLLISIONLESS);
|
|
selections[i][j].dynamic = true;
|
|
selections[i][j].collisionless = true;
|
|
redoData.push({
|
|
entityID: selections[i][j].id,
|
|
properties: DYNAMIC_AND_COLLISIONLESS
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
// Make the first entity in the first group the root and link the first entities of all other groups to it.
|
|
rootID = rootEntityIDs[0];
|
|
for (i = 1, lengthI = rootEntityIDs.length; i < lengthI; i++) {
|
|
undoData.push({
|
|
entityID: rootEntityIDs[i],
|
|
properties: { parentID: Uuid.NULL }
|
|
});
|
|
Entities.editEntity(rootEntityIDs[i], {
|
|
parentID: rootID
|
|
});
|
|
redoData.push({
|
|
entityID: rootEntityIDs[i],
|
|
properties: { parentID: rootID }
|
|
});
|
|
}
|
|
|
|
// Update selection.
|
|
rootEntityIDs.splice(1, rootEntityIDs.length - 1);
|
|
for (i = 1, lengthI = selections.length; i < lengthI; i++) {
|
|
selections[i][0].parentID = rootID;
|
|
selections[0] = selections[0].concat(selections[i]);
|
|
}
|
|
selections.splice(1, selections.length - 1);
|
|
|
|
// Add history entry.
|
|
History.push(
|
|
{ setProperties: undoData },
|
|
{ setProperties: redoData }
|
|
);
|
|
}
|
|
|
|
function ungroup() {
|
|
// Ungroups the first and assumed to be only selection.
|
|
// If the first entity in the selection has a mix of solo and group children then just the group children are unlinked,
|
|
// otherwise all are unlinked.
|
|
var rootID,
|
|
childrenIDs = [],
|
|
childrenIndexes = [],
|
|
childrenIndexIsGroup = [],
|
|
previousWasGroup,
|
|
hasSoloChildren = false,
|
|
hasGroupChildren = false,
|
|
isUngroupAll,
|
|
NONDYNAMIC_AND_NONCOLLISIONLESS = { dynamic: false, collisionless: false },
|
|
undoData = [],
|
|
redoData = [],
|
|
i,
|
|
lengthI,
|
|
j,
|
|
lengthJ;
|
|
|
|
function updateGroupInformation() {
|
|
var childrenIndexesLength = childrenIndexes.length;
|
|
if (childrenIndexesLength > 1) {
|
|
previousWasGroup = childrenIndexes[childrenIndexesLength - 2] < i - 1;
|
|
childrenIndexIsGroup.push(previousWasGroup);
|
|
if (previousWasGroup) {
|
|
hasGroupChildren = true;
|
|
} else {
|
|
hasSoloChildren = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (entitiesSelectedCount === 0) {
|
|
App.log("ERROR: Groups: Nothing to ungroup!");
|
|
return;
|
|
}
|
|
if (entitiesSelectedCount === 1) {
|
|
App.log("ERROR: Groups: Cannot ungroup sole entity!");
|
|
return;
|
|
}
|
|
|
|
// Compile information on immediate children.
|
|
rootID = rootEntityIDs[0];
|
|
for (i = 1, lengthI = selections[0].length; i < lengthI; i++) {
|
|
if (selections[0][i].parentID === rootID) {
|
|
childrenIDs.push(selections[0][i].id);
|
|
childrenIndexes.push(i);
|
|
updateGroupInformation();
|
|
}
|
|
}
|
|
childrenIndexes.push(selections[0].length); // Special extra item at end to aid updating selection.
|
|
updateGroupInformation();
|
|
|
|
// Unlink children.
|
|
isUngroupAll = hasSoloChildren !== hasGroupChildren;
|
|
for (i = childrenIDs.length - 1; i >= 0; i--) {
|
|
if (isUngroupAll || childrenIndexIsGroup[i]) {
|
|
undoData.push({
|
|
entityID: childrenIDs[i],
|
|
properties: { parentID: selections[0][childrenIndexes[i]].parentID }
|
|
});
|
|
Entities.editEntity(childrenIDs[i], {
|
|
parentID: Uuid.NULL
|
|
});
|
|
redoData.push({
|
|
entityID: childrenIDs[i],
|
|
properties: { parentID: Uuid.NULL }
|
|
});
|
|
rootEntityIDs.push(childrenIDs[i]);
|
|
selections[0][childrenIndexes[i]].parentID = Uuid.NULL;
|
|
selections.push(selections[0].splice(childrenIndexes[i], childrenIndexes[i + 1] - childrenIndexes[i]));
|
|
}
|
|
}
|
|
|
|
// If root group has physics, reset child groups to defaults for dynamic and collisionless. See
|
|
// Selection.applyPhysics().
|
|
if (selections[0][0].dynamic) {
|
|
for (i = 1, lengthI = selections.length; i < lengthI; i++) {
|
|
for (j = 0, lengthJ = selections[i].length; j < lengthJ; j++) {
|
|
undoData.push({
|
|
entityID: selections[i][j].id,
|
|
properties: {
|
|
dynamic: selections[i][j].dynamic,
|
|
collisionless: selections[i][j].collisionless
|
|
}
|
|
});
|
|
Entities.editEntity(selections[i][j].id, NONDYNAMIC_AND_NONCOLLISIONLESS);
|
|
selections[i][j].dynamic = false;
|
|
selections[i][j].collisionless = false;
|
|
redoData.push({
|
|
entityID: selections[i][j].id,
|
|
properties: NONDYNAMIC_AND_NONCOLLISIONLESS
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add history entry.
|
|
History.push(
|
|
{ setProperties: undoData },
|
|
{ setProperties: redoData }
|
|
);
|
|
}
|
|
|
|
function clear() {
|
|
rootEntityIDs = [];
|
|
selections = [];
|
|
entitiesSelectedCount = 0;
|
|
}
|
|
|
|
function destroy() {
|
|
clear();
|
|
}
|
|
|
|
return {
|
|
toggle: toggle,
|
|
selection: selection,
|
|
includes: includes,
|
|
rootEntityIDs: getRootEntityIDs,
|
|
groupsCount: groupsCount,
|
|
entitiesCount: entitiesCount,
|
|
group: group,
|
|
ungroup: ungroup,
|
|
clear: clear,
|
|
destroy: destroy
|
|
};
|
|
};
|
|
|
|
Groups.prototype = {};
|