Merge branch 'develop' into sensor-error

This commit is contained in:
Eric Rosenbaum 2019-01-15 10:38:51 -05:00 committed by GitHub
commit 2856b32dd5
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 74 additions and 40 deletions

View file

@ -165,7 +165,10 @@ class Scratch3ControlBlocks {
// Create clone // Create clone
const newClone = cloneTarget.makeClone(); const newClone = cloneTarget.makeClone();
if (newClone) { if (newClone) {
this.runtime.targets.push(newClone); this.runtime.addTarget(newClone);
// Place behind the original target.
newClone.goBehindOther(cloneTarget);
} }
} }

View file

@ -1560,11 +1560,14 @@ class Runtime extends EventEmitter {
} }
/** /**
* Add a target to the execution order. * Add a target to the runtime. This tracks the sprite pane
* @param {Target} executableTarget target to add * ordering of the target. The target still needs to be put
* into the correct execution order after calling this function.
* @param {Target} target target to add
*/ */
addExecutable (executableTarget) { addTarget (target) {
this.executableTargets.push(executableTarget); this.targets.push(target);
this.executableTargets.push(target);
} }
/** /**

View file

@ -69,10 +69,6 @@ class Target extends EventEmitter {
* @type {Object.<string, *>} * @type {Object.<string, *>}
*/ */
this._edgeActivatedHatValues = {}; this._edgeActivatedHatValues = {};
if (this.runtime) {
this.runtime.addExecutable(this);
}
} }
/** /**

View file

@ -13,7 +13,7 @@ const ScratchLinkDeviceAdapter = require('./scratch-link-device-adapter');
* @type {string} * @type {string}
*/ */
// eslint-disable-next-line max-len // eslint-disable-next-line max-len
const blockIconURI = ''; const blockIconURI = '';
/** /**
* Enum for Vernier godirect protocol. * Enum for Vernier godirect protocol.
@ -76,8 +76,8 @@ class GdxFor {
* Called by the runtime when user wants to scan for a peripheral. * Called by the runtime when user wants to scan for a peripheral.
*/ */
scan () { scan () {
if (this._device) { if (this._scratchLinkSocket) {
this._device.close(); this._scratchLinkSocket.disconnect();
} }
this._scratchLinkSocket = new BLE(this._runtime, this._extensionId, { this._scratchLinkSocket = new BLE(this._runtime, this._extensionId, {
@ -105,10 +105,9 @@ class GdxFor {
* Disconnect from the GDX FOR. * Disconnect from the GDX FOR.
*/ */
disconnect () { disconnect () {
if (this._device) { this._sensorsEnabled = false;
this._device.close(); if (this._scratchLinkSocket) {
this._device = null; this._scratchLinkSocket.disconnect();
this._sensorsEnabled = false;
} }
} }

View file

@ -39,10 +39,6 @@ class ScratchLinkDeviceAdapter {
const response = new DataView(array.buffer); const response = new DataView(array.buffer);
return this._deviceOnResponse(response); return this._deviceOnResponse(response);
} }
close () {
return this.scratchLinkSocket.disconnect();
}
} }
module.exports = ScratchLinkDeviceAdapter; module.exports = ScratchLinkDeviceAdapter;

View file

@ -781,14 +781,17 @@ const reorderParsedTargets = function (targets) {
// Reorder parsed targets based on the temporary targetPaneOrder property // Reorder parsed targets based on the temporary targetPaneOrder property
// and then delete it. // and then delete it.
targets.sort((a, b) => a.targetPaneOrder - b.targetPaneOrder); const reordered = targets.map((t, index) => {
t.layerOrder = index;
return t;
}).sort((a, b) => a.targetPaneOrder - b.targetPaneOrder);
// Delete the temporary target pane ordering since we shouldn't need it anymore. // Delete the temporary target pane ordering since we shouldn't need it anymore.
targets.forEach(t => { reordered.forEach(t => {
delete t.targetPaneOrder; delete t.targetPaneOrder;
}); });
return targets; return reordered;
}; };

View file

@ -1157,12 +1157,18 @@ const deserialize = function (json, runtime, zip, isSingleSprite) {
parseScratchObject(target, runtime, extensions, zip)) parseScratchObject(target, runtime, extensions, zip))
) )
.then(targets => targets // Re-sort targets back into original sprite-pane ordering .then(targets => targets // Re-sort targets back into original sprite-pane ordering
.map((t, i) => {
// Add layer order property to deserialized targets.
// This property is used to initialize executable targets in
// the correct order and is deleted in VM's installTargets function
t.layerOrder = i;
return t;
})
.sort((a, b) => a.targetPaneOrder - b.targetPaneOrder) .sort((a, b) => a.targetPaneOrder - b.targetPaneOrder)
.map(t => { .map(t => {
// Delete the temporary properties used for // Delete the temporary properties used for
// sprite pane ordering and stage layer ordering // sprite pane ordering and stage layer ordering
delete t.targetPaneOrder; delete t.targetPaneOrder;
delete t.layerOrder;
return t; return t;
})) }))
.then(targets => { .then(targets => {

View file

@ -1021,8 +1021,6 @@ class RenderedTarget extends Target {
newClone._edgeActivatedHatValues = Clone.simple(this._edgeActivatedHatValues); newClone._edgeActivatedHatValues = Clone.simple(this._edgeActivatedHatValues);
newClone.initDrawable(StageLayering.SPRITE_LAYER); newClone.initDrawable(StageLayering.SPRITE_LAYER);
newClone.updateAllDrawableProperties(); newClone.updateAllDrawableProperties();
// Place behind the current target.
newClone.goBehindOther(this);
return newClone; return newClone;
} }
@ -1046,7 +1044,6 @@ class RenderedTarget extends Target {
newTarget.effects = JSON.parse(JSON.stringify(this.effects)); newTarget.effects = JSON.parse(JSON.stringify(this.effects));
newTarget.variables = this.duplicateVariables(newTarget.blocks); newTarget.variables = this.duplicateVariables(newTarget.blocks);
newTarget.updateAllDrawableProperties(); newTarget.updateAllDrawableProperties();
newTarget.goBehindOther(this);
return newTarget; return newTarget;
}); });
} }

View file

@ -495,11 +495,18 @@ class VirtualMachine extends EventEmitter {
return Promise.all(extensionPromises).then(() => { return Promise.all(extensionPromises).then(() => {
targets.forEach(target => { targets.forEach(target => {
this.runtime.targets.push(target); this.runtime.addTarget(target);
(/** @type RenderedTarget */ target).updateAllDrawableProperties(); (/** @type RenderedTarget */ target).updateAllDrawableProperties();
// Ensure unique sprite name // Ensure unique sprite name
if (target.isSprite()) this.renameSprite(target.id, target.getName()); if (target.isSprite()) this.renameSprite(target.id, target.getName());
}); });
// Sort the executable targets by layerOrder.
// Remove layerOrder property after use.
this.runtime.executableTargets.sort((a, b) => a.layerOrder - b.layerOrder);
targets.forEach(target => {
delete target.layerOrder;
});
// Select the first target for editing, e.g., the first sprite. // Select the first target for editing, e.g., the first sprite.
if (wholeProject && (targets.length > 1)) { if (wholeProject && (targets.length > 1)) {
this.editingTarget = targets[1]; this.editingTarget = targets[1];
@ -1016,7 +1023,8 @@ class VirtualMachine extends EventEmitter {
throw new Error('No sprite associated with this target.'); throw new Error('No sprite associated with this target.');
} }
return target.duplicate().then(newTarget => { return target.duplicate().then(newTarget => {
this.runtime.targets.push(newTarget); this.runtime.addTarget(newTarget);
newTarget.goBehindOther(target);
this.setEditingTarget(newTarget.id); this.setEditingTarget(newTarget.id);
}); });
} }

View file

@ -192,7 +192,8 @@ test('edge activated hat should trigger for both sprites when sprite is cloned',
val + Object.keys(target._edgeActivatedHatValues).length, 0); val + Object.keys(target._edgeActivatedHatValues).length, 0);
t.equal(numTargetEdgeHats, 1); t.equal(numTargetEdgeHats, 1);
vm.runtime.targets.push(vm.runtime.targets[1].makeClone()); const cloneTarget = vm.runtime.targets[1].makeClone();
vm.runtime.addTarget(cloneTarget);
vm.runtime._step(); vm.runtime._step();
// Check that the runtime's _edgeActivatedHatValues object has two separate keys // Check that the runtime's _edgeActivatedHatValues object has two separate keys

View file

@ -54,7 +54,8 @@ test('#760 - broadcastAndWait', t => {
const tgt = new Target(rt, b); const tgt = new Target(rt, b);
tgt.isStage = true; tgt.isStage = true;
tgt.createVariable('testBroadcastID', 'message', Variable.BROADCAST_MESSAGE_TYPE); tgt.createVariable('testBroadcastID', 'message', Variable.BROADCAST_MESSAGE_TYPE);
rt.targets.push(tgt);
rt.addTarget(tgt);
let th = rt._pushThread('broadcastAndWaitBlock', t); let th = rt._pushThread('broadcastAndWaitBlock', t);
const util = new BlockUtility(); const util = new BlockUtility();

View file

@ -48,7 +48,7 @@ const testCostume = (costumes, arg, currentCostume = 1, isStage = false) => {
if (isStage) { if (isStage) {
target.isStage = true; target.isStage = true;
rt.targets.push(target); rt.addTarget(target);
looks.switchBackdrop({BACKDROP: arg}, {target}); looks.switchBackdrop({BACKDROP: arg}, {target});
} else { } else {
looks.switchCostume({COSTUME: arg}, {target}); looks.switchCostume({COSTUME: arg}, {target});

View file

@ -188,7 +188,7 @@ test('renameVariable calls cloud io device\'s requestRenameVariable function', t
target.isStage = true; target.isStage = true;
const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true); const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true);
target.variables[mockCloudVar.id] = mockCloudVar; target.variables[mockCloudVar.id] = mockCloudVar;
runtime.targets.push(target); runtime.addTarget(target);
target.renameVariable('foo', 'bar2'); target.renameVariable('foo', 'bar2');
@ -215,7 +215,7 @@ test('renameVariable does not call cloud io device\'s requestRenameVariable func
const target = new Target(runtime); const target = new Target(runtime);
const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true); const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true);
target.variables[mockCloudVar.id] = mockCloudVar; target.variables[mockCloudVar.id] = mockCloudVar;
runtime.targets.push(target); runtime.addTarget(target);
target.renameVariable('foo', 'bar2'); target.renameVariable('foo', 'bar2');
@ -266,7 +266,7 @@ test('deleteVariable calls cloud io device\'s requestRenameVariable function', t
target.isStage = true; target.isStage = true;
const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true); const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true);
target.variables[mockCloudVar.id] = mockCloudVar; target.variables[mockCloudVar.id] = mockCloudVar;
runtime.targets.push(target); runtime.addTarget(target);
target.deleteVariable('foo'); target.deleteVariable('foo');
@ -288,7 +288,7 @@ test('deleteVariable calls cloud io device\'s requestRenameVariable function', t
const target = new Target(runtime); const target = new Target(runtime);
const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true); const mockCloudVar = new Variable('foo', 'bar', Variable.SCALAR_TYPE, true);
target.variables[mockCloudVar.id] = mockCloudVar; target.variables[mockCloudVar.id] = mockCloudVar;
runtime.targets.push(target); runtime.addTarget(target);
target.deleteVariable('foo'); target.deleteVariable('foo');

View file

@ -1,6 +1,7 @@
const test = require('tap').test; const test = require('tap').test;
const path = require('path'); const path = require('path');
const VirtualMachine = require('../../src/index'); const VirtualMachine = require('../../src/index');
const Runtime = require('../../src/engine/runtime');
const sb3 = require('../../src/serialization/sb3'); const sb3 = require('../../src/serialization/sb3');
const readFileToBuffer = require('../fixtures/readProjectFile').readFileToBuffer; const readFileToBuffer = require('../fixtures/readProjectFile').readFileToBuffer;
const exampleProjectPath = path.resolve(__dirname, '../fixtures/clone-cleanup.sb2'); const exampleProjectPath = path.resolve(__dirname, '../fixtures/clone-cleanup.sb2');
@ -147,10 +148,10 @@ test('deserialize sb3 project with comments - no duplicate id serialization', t
}); });
}); });
test('serialize sb3 preserves sprite layer order', t => { test('serializing and deserializing sb3 preserves sprite layer order', t => {
const vm = new VirtualMachine(); const vm = new VirtualMachine();
vm.attachRenderer(new FakeRenderer()); vm.attachRenderer(new FakeRenderer());
vm.loadProject(readFileToBuffer(path.resolve(__dirname, '../fixtures/ordering.sb2'))) return vm.loadProject(readFileToBuffer(path.resolve(__dirname, '../fixtures/ordering.sb2')))
.then(() => { .then(() => {
// Target get layer order needs a renderer, // Target get layer order needs a renderer,
// fake the numbers we would get back from the // fake the numbers we would get back from the
@ -182,8 +183,28 @@ test('serialize sb3 preserves sprite layer order', t => {
t.equal(result.targets[2].layerOrder, 1); t.equal(result.targets[2].layerOrder, 1);
t.equal(result.targets[3].layerOrder, 3); t.equal(result.targets[3].layerOrder, 3);
t.end(); return result;
}); })
.then(serializedObject =>
sb3.deserialize(
JSON.parse(JSON.stringify(serializedObject)), new Runtime(), null, false)
.then(({targets}) => {
// First check that the sprites are ordered correctly (as they would
// appear in the target pane)
t.equal(targets[0].sprite.name, 'Stage');
t.equal(targets[1].sprite.name, 'First');
t.equal(targets[2].sprite.name, 'Second');
t.equal(targets[3].sprite.name, 'Third');
// Check that they are in the correct layer order (as they would render
// back to front on the stage)
t.equal(targets[0].layerOrder, 0);
t.equal(targets[1].layerOrder, 2);
t.equal(targets[2].layerOrder, 1);
t.equal(targets[3].layerOrder, 3);
t.end();
}));
}); });
test('serializeBlocks', t => { test('serializeBlocks', t => {