0
0
Fork 0
mirror of https://github.com/liabru/matter-js.git synced 2025-01-21 17:14:38 -05:00

further work on Matter.Inspector

This commit is contained in:
liabru 2014-04-28 17:26:03 +01:00
parent f80866c6de
commit 00559ba8d3
5 changed files with 233 additions and 90 deletions

View file

@ -43,11 +43,14 @@ body .dg li.title {
background: #444 url() 10px 10px no-repeat;
}
body .dg .cr.function:hover,
body .dg .cr.boolean:hover {
background: #3d3d3d;
}
body .dg .cr.function:hover {
background: #555;
}
body .dg .c select {
margin-top: 2px;
margin-left: -5px;

View file

@ -2,6 +2,14 @@
cursor: move !important;
}
.ins-cursor-rotate canvas {
cursor: ew-resize !important;
}
.ins-cursor-scale canvas {
cursor: nwse-resize !important;
}
.ins-container {
position: fixed;
overflow: auto;
@ -9,7 +17,6 @@
height: 98%;
left: 0;
background: #3d3d3d;
border-right: 2px solid #444;
padding: 1% 20px;
font-family: Arial;
font-size: 12px;
@ -18,14 +25,11 @@
.ins-world-tree {
overflow: auto;
/* height: 90%; */
/* position: relative; */
position: absolute;
left: -17px;
right: 0px;
top: 92px;
bottom: 8px;
/* width: 260px; */
}
.ins-control-group {
@ -57,26 +61,40 @@
.jstree-default .jstree-wholerow-clicked,
.jstree-default .jstree-clicked {
/*background: #4a4a4a;*/
/*background: #373737;*/
background: transparent;
border-radius: 0;
box-shadow: none;
transition: none;
}
.jstree-default .jstree-clicked:before {
.jstree-default .jstree-open > .jstree-children > .jstree-node > .jstree-anchor:before {
content: '';
display: block;
/* width: 100%; */
position: absolute;
z-index: -1;
left: 0;
right: 0;
right: 2px;
height: 26px;
background: #373737;
background: transparent;
border-radius: 0;
box-shadow: none;
border-right: none;
}
.jstree-anchor.jstree-clicked:before {
background: #373737 !important;
}
.jstree-node-type-body > .jstree-anchor:before {
border-right: 3px solid #e1115f !important;
}
.jstree-node-type-constraint > .jstree-anchor:before {
border-right: 3px solid #1ed36f !important;
}
.jstree-node-type-composite > .jstree-anchor:before {
border-right: 3px solid #2fa1d6 !important;
}
.jstree-default .jstree-node,
@ -114,16 +132,20 @@
.ins-container *::-webkit-scrollbar-thumb:vertical {
border-left: 5px solid rgba(0,0,0,0.2);
width: 6px;
/*background: rgba(0,0,0,0.2);*/
}
.ins-container *::-webkit-scrollbar-thumb:horizontal {
border-top: 5px solid rgba(0,0,0,0.2);
height: 6px;
/*background: rgba(0,0,0,0.2);*/
}
.ins-container *::-webkit-scrollbar-track,
.ins-container *::-webkit-scrollbar-corner {
background: transparent;
}
#vakata-dnd {
font-family: Arial;
font-size: 12px;
color: #aaa;
}

View file

@ -21,6 +21,10 @@ h1 {
font-size: 30px;
}
.is-mobile h1 {
font-size: 18px;
}
.nav-sep {
padding: 0 5px;
}
@ -35,7 +39,7 @@ h1 {
.container {
max-width: 800px;
margin: 0 auto;
padding: 6% 280px 0 280px;
padding: 64px 280px 0 280px;
}
.is-mobile .container {

View file

@ -6,16 +6,21 @@
<meta name="viewport" content="width=device-width,minimal-ui">
<meta name="robots" content="noindex">
<!-- optional libs, only required for Matter.Gui -->
<link rel="stylesheet" href="./js/lib/jstree/themes/default/style.css" type="text/css">
<link rel="stylesheet" href="./css/matter-inspector.css" type="text/css">
<script type="text/javascript" src="./js/lib/pixi.dev.js"></script>
<!-- only required if using Matter.Gui -->
<link rel="stylesheet" href="./css/matter-gui.css" type="text/css">
<script type="text/javascript" src="./js/lib/dat.gui.min.js"></script>
<script type="text/javascript" src="./js/lib/resurrect.js"></script>
<!-- only required if using Matter.Inspector -->
<link rel="stylesheet" href="./js/lib/jstree/themes/default/style.css" type="text/css">
<link rel="stylesheet" href="./css/matter-inspector.css" type="text/css">
<script type="text/javascript" src="./js/lib/jquery-1.11.0.min.js"></script>
<script type="text/javascript" src="./js/lib/jstree/jstree.min.js"></script>
<script type="text/javascript" src="./js/lib/keymaster.js"></script>
<!-- only required if using Matter.RenderPixi -->
<script type="text/javascript" src="./js/lib/pixi.dev.js"></script>
<!-- matter.js demo code -->
<script type="text/javascript" src="./js/lib/matter-dev.js"></script>
<script type="text/javascript" src="./js/Demo.js"></script>

View file

@ -21,7 +21,7 @@ var Inspector = {};
*/
Inspector.create = function(engine, options) {
if (!jQuery || !$.fn.jstree) {
console.log("Could not create inspector. Check jQuery and jsTree libraries are loaded first.");
console.log('Could not create inspector. Check jQuery and jsTree libraries are loaded first.');
return;
}
@ -60,38 +60,38 @@ var Inspector = {};
var engine = inspector.engine;
var worldTreeOptions = {
"core": {
"check_callback": true
'core': {
'check_callback': true
},
"dnd": {
"copy": false
'dnd': {
'copy': false
},
"types": {
"body": {
"valid_children": []
'types': {
'body': {
'valid_children': []
},
"constraint": {
"valid_children": []
'constraint': {
'valid_children': []
},
"composite": {
"valid_children": []
'composite': {
'valid_children': []
},
"bodies": {
"valid_children": ['body']
'bodies': {
'valid_children': ['body']
},
"constraints": {
"valid_children": ['constraint']
'constraints': {
'valid_children': ['constraint']
},
"composites": {
"valid_children": ['composite']
'composites': {
'valid_children': ['composite']
}
},
"plugins" : ["dnd", "types", "unique"]
'plugins' : ['dnd', 'types', 'unique']
};
var $inspectorContainer = $('<div class="ins-container">'),
$worldTree = $('<div class="ins-world-tree">').jstree(worldTreeOptions),
$buttonGroup = $('<div class="ins-control-group">')
$buttonGroup = $('<div class="ins-control-group">'),
$importButton = $('<button class="ins-import-button ins-button">Import</button>'),
$exportButton = $('<button class="ins-export-button ins-button">Export</button>'),
$pauseButton = $('<button class="ins-pause-button ins-button">Pause</button>');
@ -115,7 +115,7 @@ var Inspector = {};
controls.worldTree.on('changed.jstree', function(event, data) {
var selected = [],
worldTree = controls.worldTree.data("jstree");
worldTree = controls.worldTree.data('jstree');
if (data.action !== 'select_node')
return;
@ -129,20 +129,24 @@ var Inspector = {};
var nodeId = data.selected[i],
objectType = nodeId.split('_')[0],
objectId = nodeId.split('_')[1],
worldObject = Composite.get(engine.world, objectType, objectId);
worldObject = Composite.get(engine.world, objectId, objectType);
switch (objectType) {
case 'body':
case 'constraint':
case 'body':
case 'constraint':
selected.push(worldObject);
break;
case 'composite':
case 'composites':
case 'bodies':
case 'constraints':
case 'composite':
case 'composites':
case 'bodies':
case 'constraints':
if (objectType === 'composite')
selected.push(worldObject);
var node = worldTree.get_node(nodeId),
children = worldTree.get_node(nodeId).children;
@ -151,6 +155,7 @@ var Inspector = {};
worldTree.select_node(children[j], false);
break;
}
}
@ -159,12 +164,28 @@ var Inspector = {};
}, 1);
});
$(document).on('dnd_move.vakata', function(event, data) {
var worldTree = controls.worldTree.data("jstree"),
parentNodeId = worldTree.get_parent(data.data.nodes[0]),
objectType = parentNodeId.split('_')[0];
$(document).on('dnd_stop.vakata', function(event, data) {
var worldTree = controls.worldTree.data('jstree');
// TODO: composite.move
for (var i = 0; i < data.data.nodes.length; i++) {
var node = worldTree.get_node(data.data.nodes[i]),
parentNode = worldTree.get_node(worldTree.get_parent(data.data.nodes[i])),
prevCompositeId = node.data.compositeId,
newCompositeId = parentNode.data.compositeId;
if (prevCompositeId === newCompositeId)
continue;
var nodeId = data.data.nodes[i],
objectType = nodeId.split('_')[0],
objectId = nodeId.split('_')[1],
worldObject = Composite.get(engine.world, objectId, objectType),
prevComposite = Composite.get(engine.world, prevCompositeId, 'composite'),
newComposite = Composite.get(engine.world, newCompositeId, 'composite');
Composite.move(prevComposite, worldObject, newComposite);
node.data.compositeId = newCompositeId;
}
});
controls.pauseButton.click(function() {
@ -194,7 +215,8 @@ var Inspector = {};
Events.on(engine, 'tick', function() {
if (engine.world.isModified) {
var data = _generateCompositeTreeNode(engine.world);
_updateTree(controls.worldTree.data("jstree"), data);
_updateTree(controls.worldTree.data('jstree'), data);
_setSelectedObjects(inspector, []);
}
if (key.isPressed('del') || key.isPressed('backspace')) {
@ -230,18 +252,18 @@ var Inspector = {};
bodies = Composite.allBodies(engine.world),
constraints = Composite.allConstraints(engine.world),
isUnionSelect = _key.shift || _key.control,
worldTree = inspector.controls.worldTree.data("jstree");
worldTree = inspector.controls.worldTree.data('jstree'),
i;
$body.removeClass('ins-cursor-move');
$body.removeClass('ins-cursor-move ins-cursor-rotate ins-cursor-scale');
if (mouse.button === 0) {
var hasSelected = false;
for (var i = 0; i < bodies.length; i++) {
for (i = 0; i < bodies.length; i++) {
var body = bodies[i];
if (Bounds.contains(body.bounds, mouse.position)
&& Vertices.contains(body.vertices, mouse.position)) {
if (Bounds.contains(body.bounds, mouse.position) && Vertices.contains(body.vertices, mouse.position)) {
if (isUnionSelect) {
_addSelectedObject(inspector, body);
@ -288,6 +310,8 @@ var Inspector = {};
}
if (!hasSelected) {
var worldTree = inspector.controls.worldTree.data('jstree');
worldTree.deselect_all(true);
_setSelectedObjects(inspector, []);
inspector.selectStart = Common.clone(mouse.position);
@ -305,7 +329,7 @@ var Inspector = {};
if (mouse.button === 0 && inspector.selected.length > 0) {
$body.addClass('ins-cursor-move');
for (var i = 0; i < inspector.selected.length; i++) {
for (i = 0; i < inspector.selected.length; i++) {
var item = inspector.selected[i],
data = item.data;
@ -330,11 +354,70 @@ var Inspector = {};
}
});
var mousePrevPosition = { x: 0, y: 0 };
Events.on(engine, 'mousemove', function(event) {
var mouse = event.mouse,
selected = inspector.selected;
selected = inspector.selected,
item,
data,
i;
if (inspector.isPaused) {
$body.removeClass('ins-cursor-move ins-cursor-rotate ins-cursor-scale');
if (_key.shift && _key.isPressed('r')) {
$body.addClass('ins-cursor-rotate');
// roate mode
for (i = 0; i < selected.length; i++) {
item = selected[i];
data = item.data;
switch (data.type) {
case 'body':
var angle = Common.sign(mouse.position.x - mousePrevPosition.x) * 0.05;
Body.rotate(data, angle);
break;
}
}
mousePrevPosition = Common.clone(mouse.position);
return;
}
if (_key.shift && _key.isPressed('s')) {
$body.addClass('ins-cursor-scale');
// scale mode
for (i = 0; i < selected.length; i++) {
item = selected[i];
data = item.data;
switch (data.type) {
case 'body':
var scale = 1 + Common.sign(mouse.position.x - mousePrevPosition.x) * 0.02;
Body.scale(data, scale, scale);
if (data.circleRadius)
data.circleRadius *= scale;
break;
}
}
mousePrevPosition = Common.clone(mouse.position);
return;
}
if (mouse.button !== 0 || mouse.sourceEvents.mousedown || mouse.sourceEvents.mouseup)
return;
@ -346,12 +429,16 @@ var Inspector = {};
return;
}
for (var i = 0; i < selected.length; i++) {
var item = selected[i],
data = item.data;
$body.addClass('ins-cursor-move');
// translate mode
for (i = 0; i < selected.length; i++) {
item = selected[i];
data = item.data;
switch (data.type) {
case 'body':
case 'body':
var delta = {
x: mouse.position.x - data.position.x - item.mousedownOffset.x,
@ -362,7 +449,7 @@ var Inspector = {};
break;
case 'constraint':
case 'constraint':
var point = data.pointA;
if (data.bodyA)
@ -377,6 +464,7 @@ var Inspector = {};
data.length = Vector.magnitude(Vector.sub(initialPointA, initialPointB));
break;
}
}
}
@ -395,8 +483,7 @@ var Inspector = {};
var doPrevent = false;
if (event.keyCode === 8) {
var d = event.srcElement || event.target;
if ((d.tagName.toUpperCase() === 'INPUT' && (d.type.toUpperCase() === 'TEXT' || d.type.toUpperCase() === 'PASSWORD' || d.type.toUpperCase() === 'FILE' || d.type.toUpperCase() === 'EMAIL' ))
|| d.tagName.toUpperCase() === 'TEXTAREA') {
if ((d.tagName.toUpperCase() === 'INPUT' && (d.type.toUpperCase() === 'TEXT' || d.type.toUpperCase() === 'PASSWORD' || d.type.toUpperCase() === 'FILE' || d.type.toUpperCase() === 'EMAIL' )) || d.tagName.toUpperCase() === 'TEXTAREA') {
doPrevent = d.readOnly || d.disabled;
}
else {
@ -423,19 +510,20 @@ var Inspector = {};
};
var _setSelectedObjects = function(inspector, objects) {
var worldTree = inspector.controls.worldTree.data("jstree"),
selectedItems = [];
var worldTree = inspector.controls.worldTree.data('jstree'),
selectedItems = [],
data;
for (var i = 0; i < inspector.selected.length; i++) {
var data = inspector.selected[i].data;
worldTree.deselect_node(data.type + "_" + data.id, true);
data = inspector.selected[i].data;
worldTree.deselect_node(data.type + '_' + data.id, true);
}
inspector.selected = [];
console.clear();
for (i = 0; i < objects.length; i++) {
var data = objects[i];
data = objects[i];
// add the object to the selection
_addSelectedObject(inspector, data);
@ -450,20 +538,21 @@ var Inspector = {};
};
var _addSelectedObject = function(inspector, object) {
var worldTree = inspector.controls.worldTree.data("jstree");
var worldTree = inspector.controls.worldTree.data('jstree');
inspector.selected.push({
data: object
});
worldTree.select_node(object.type + "_" + object.id, true);
worldTree.select_node(object.type + '_' + object.id, true);
};
var _render = function(inspector) {
var engine = inspector.engine,
mouse = engine.input.mouse,
context = engine.render.context,
selected = inspector.selected;
selected = inspector.selected,
bounds;
for (var i = 0; i < selected.length; i++) {
var item = selected[i].data;
@ -473,10 +562,11 @@ var Inspector = {};
context.strokeStyle = 'rgba(255,165,0,0.8)';
switch (item.type) {
case 'body':
case 'body':
// render body selections
var bounds = item.bounds;
bounds = item.bounds;
context.beginPath();
context.rect(Math.floor(bounds.min.x - 3), Math.floor(bounds.min.y - 3),
Math.floor(bounds.max.x - bounds.min.x + 6), Math.floor(bounds.max.y - bounds.min.y + 6));
@ -485,7 +575,7 @@ var Inspector = {};
break;
case 'constraint':
case 'constraint':
// render constraint selections
var point = item.pointA;
@ -497,6 +587,7 @@ var Inspector = {};
context.stroke();
break;
}
context.translate(-0.5, -0.5);
@ -508,7 +599,7 @@ var Inspector = {};
context.lineWidth = 1;
context.setLineDash([1,2]);
context.strokeStyle = 'rgba(255,165,0,0.5)';
var bounds = inspector.selectBounds;
bounds = inspector.selectBounds;
context.beginPath();
context.rect(Math.floor(bounds.min.x), Math.floor(bounds.min.y),
Math.floor(bounds.max.x - bounds.min.x), Math.floor(bounds.max.y - bounds.min.y));
@ -525,9 +616,12 @@ var Inspector = {};
tree.refresh(-1);
};
var _generateCompositeTreeNode = function(composite) {
var _generateCompositeTreeNode = function(composite, compositeId) {
var node = {
id: 'composite_' + composite.id,
data: {
compositeId: compositeId,
},
type: 'composite',
text: (composite.label ? composite.label : 'Composite') + ' ' + composite.id,
children: [],
@ -536,25 +630,28 @@ var Inspector = {};
}
};
var childNode = _generateBodiesTreeNode(composite.bodies);
var childNode = _generateBodiesTreeNode(composite.bodies, composite.id);
childNode.id = 'bodies_' + composite.id;
node.children.push(childNode);
childNode = _generateConstraintsTreeNode(composite.constraints);
childNode = _generateConstraintsTreeNode(composite.constraints, composite.id);
childNode.id = 'constraints_' + composite.id;
node.children.push(childNode);
childNode = _generateCompositesTreeNode(composite.composites);
childNode = _generateCompositesTreeNode(composite.composites, composite.id);
childNode.id = 'composites_' + composite.id;
node.children.push(childNode);
return node;
};
var _generateCompositesTreeNode = function(composites) {
var _generateCompositesTreeNode = function(composites, compositeId) {
var node = {
type: 'composites',
text: 'Composites',
data: {
compositeId: compositeId,
},
children: [],
'li_attr': {
'class': 'jstree-node-type-composites'
@ -563,16 +660,19 @@ var Inspector = {};
for (var i = 0; i < composites.length; i++) {
var composite = composites[i];
node.children.push(_generateCompositeTreeNode(composite));
node.children.push(_generateCompositeTreeNode(composite, compositeId));
}
return node;
};
var _generateBodiesTreeNode = function(bodies) {
var _generateBodiesTreeNode = function(bodies, compositeId) {
var node = {
type: 'bodies',
text: 'Bodies',
data: {
compositeId: compositeId,
},
children: [],
'li_attr': {
'class': 'jstree-node-type-bodies'
@ -584,6 +684,9 @@ var Inspector = {};
node.children.push({
type: 'body',
id: 'body_' + body.id,
data: {
compositeId: compositeId,
},
text: (body.label ? body.label : 'Body') + ' ' + body.id,
'li_attr': {
'class': 'jstree-node-type-body'
@ -594,10 +697,13 @@ var Inspector = {};
return node;
};
var _generateConstraintsTreeNode = function(constraints) {
var _generateConstraintsTreeNode = function(constraints, compositeId) {
var node = {
type: 'constraints',
text: 'Constraints',
data: {
compositeId: compositeId,
},
children: [],
'li_attr': {
'class': 'jstree-node-type-constraints'
@ -609,6 +715,9 @@ var Inspector = {};
node.children.push({
type: 'constraint',
id: 'constraint_' + constraint.id,
data: {
compositeId: compositeId,
},
text: (constraint.label ? constraint.label : 'Constraint') + ' ' + constraint.id,
'li_attr': {
'class': 'jstree-node-type-constraint'
@ -642,7 +751,7 @@ var Inspector = {};
var blob = new Blob([json], { type: 'application/json' }),
anchor = document.createElement('a');
anchor.download = "world.json";
anchor.download = 'world.json';
anchor.href = (window.webkitURL || window.URL).createObjectURL(blob);
anchor.dataset.downloadurl = ['application/json', anchor.download, anchor.href].join(':');
anchor.click();
@ -675,11 +784,11 @@ var Inspector = {};
}
Events.trigger(inspector, 'import');
}
};
reader.readAsText(file);
} else {
alert("File not supported, JSON text files only");
alert('File not supported, JSON text files only');
}
});