mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2024-12-23 14:32:59 -05:00
Merge remote-tracking branch 'refs/remotes/LLK/develop' into bug/escapeFromConditionalBranches
This commit is contained in:
commit
682dec493f
3 changed files with 174 additions and 4 deletions
|
@ -31,7 +31,7 @@ npm start
|
|||
```
|
||||
|
||||
## Playground
|
||||
To run the Playground, make sure the dev server's running and go to [http://localhost:8073/](http://localhost:8073/) - you will be directed to the playground, which demonstrates various tools and internal state.
|
||||
To run the Playground, make sure the dev server's running and go to [http://localhost:8073/playground/](http://localhost:8073/playground/) - you will be directed to the playground, which demonstrates various tools and internal state.
|
||||
|
||||
![VM Playground Screenshot](https://i.imgur.com/nOCNqEc.gif)
|
||||
|
||||
|
@ -50,7 +50,7 @@ npm run build
|
|||
```
|
||||
|
||||
## How to include in a Node.js App
|
||||
For an extended setup example, check out the /playground directory, which includes a fully running VM instance.
|
||||
For an extended setup example, check out the /src/playground directory, which includes a fully running VM instance.
|
||||
```js
|
||||
var VirtualMachine = require('scratch-vm');
|
||||
var vm = new VirtualMachine();
|
||||
|
|
|
@ -42,7 +42,7 @@
|
|||
"scratch-render": "latest",
|
||||
"script-loader": "0.7.0",
|
||||
"stats.js": "0.17.0",
|
||||
"tap": "10.1.1",
|
||||
"tap": "10.1.2",
|
||||
"travis-after-all": "1.4.4",
|
||||
"webpack": "2.2.1",
|
||||
"webpack-dev-server": "2.3.0"
|
||||
|
|
|
@ -1,8 +1,178 @@
|
|||
var test = require('tap').test;
|
||||
var Sequencer = require('../../src/engine/sequencer');
|
||||
var Runtime = require('../../src/engine/runtime');
|
||||
var Thread = require('../../src/engine/thread');
|
||||
var RenderedTarget = require('../../src/sprites/rendered-target');
|
||||
var Sprite = require('../../src/sprites/sprite');
|
||||
|
||||
test('spec', function (t) {
|
||||
t.type(Sequencer, 'function');
|
||||
// @todo
|
||||
|
||||
var r = new Runtime();
|
||||
var s = new Sequencer(r);
|
||||
|
||||
t.type(s, 'object');
|
||||
t.ok(s instanceof Sequencer);
|
||||
|
||||
t.type(s.stepThreads, 'function');
|
||||
t.type(s.stepThread, 'function');
|
||||
t.type(s.stepToBranch, 'function');
|
||||
t.type(s.stepToProcedure, 'function');
|
||||
t.type(s.retireThread, 'function');
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
||||
var randomString = function () {
|
||||
var top = Math.random().toString(36);
|
||||
return top.substring(7);
|
||||
};
|
||||
|
||||
var generateBlock = function (id) {
|
||||
var block = {fields: Object,
|
||||
id: id,
|
||||
inputs: {},
|
||||
STEPS: Object,
|
||||
block: 'fakeBlock',
|
||||
name: 'fakeName',
|
||||
next: null,
|
||||
opcode: 'procedures_defnoreturn',
|
||||
mutation: {proccode: 'fakeCode'},
|
||||
parent: null,
|
||||
shadow: false,
|
||||
topLevel: true,
|
||||
x: 0,
|
||||
y: 0
|
||||
};
|
||||
return block;
|
||||
};
|
||||
|
||||
var generateBlockInput = function (id, next, inp) {
|
||||
var block = {fields: Object,
|
||||
id: id,
|
||||
inputs: {SUBSTACK: {block: inp, name: 'SUBSTACK'}},
|
||||
STEPS: Object,
|
||||
block: 'fakeBlock',
|
||||
name: 'fakeName',
|
||||
next: next,
|
||||
opcode: 'procedures_defnoreturn',
|
||||
mutation: {proccode: 'fakeCode'},
|
||||
parent: null,
|
||||
shadow: false,
|
||||
topLevel: true,
|
||||
x: 0,
|
||||
y: 0
|
||||
};
|
||||
return block;
|
||||
};
|
||||
|
||||
var generateThread = function (runtime) {
|
||||
var s = new Sprite();
|
||||
var rt = new RenderedTarget(s, runtime);
|
||||
var th = new Thread(randomString());
|
||||
|
||||
var next = randomString();
|
||||
var inp = randomString();
|
||||
var name = th.topBlock;
|
||||
|
||||
rt.blocks.createBlock(generateBlockInput(name, next, inp));
|
||||
th.pushStack(name);
|
||||
rt.blocks.createBlock(generateBlock(inp));
|
||||
|
||||
for (var i = 0; i < 10; i++) {
|
||||
name = next;
|
||||
next = randomString();
|
||||
inp = randomString();
|
||||
|
||||
rt.blocks.createBlock(generateBlockInput(name, next, inp));
|
||||
th.pushStack(name);
|
||||
rt.blocks.createBlock(generateBlock(inp));
|
||||
}
|
||||
rt.blocks.createBlock(generateBlock(next));
|
||||
th.pushStack(next);
|
||||
th.target = rt;
|
||||
|
||||
runtime.threads.push(th);
|
||||
|
||||
return th;
|
||||
};
|
||||
|
||||
test('stepThread', function (t) {
|
||||
var r = new Runtime();
|
||||
var s = new Sequencer(r);
|
||||
var th = generateThread(r);
|
||||
t.notEquals(th.status, Thread.STATUS_DONE);
|
||||
s.stepThread(th);
|
||||
t.strictEquals(th.status, Thread.STATUS_DONE);
|
||||
th = generateThread(r);
|
||||
th.status = Thread.STATUS_YIELD;
|
||||
s.stepThread(th);
|
||||
t.notEquals(th.status, Thread.STATUS_DONE);
|
||||
th.status = Thread.STATUS_PROMISE_WAIT;
|
||||
s.stepThread(th);
|
||||
t.notEquals(th.status, Thread.STATUS_DONE);
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
||||
test('stepToBranch', function (t) {
|
||||
var r = new Runtime();
|
||||
var s = new Sequencer(r);
|
||||
var th = generateThread(r);
|
||||
s.stepToBranch(th, 2, false);
|
||||
t.strictEquals(th.peekStack(), null);
|
||||
th.popStack();
|
||||
s.stepToBranch(th, 1, false);
|
||||
t.strictEquals(th.peekStack(), null);
|
||||
th.popStack();
|
||||
th.popStack();
|
||||
s.stepToBranch(th, 1, false);
|
||||
t.notEquals(th.peekStack(), null);
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
||||
test('retireThread', function (t) {
|
||||
var r = new Runtime();
|
||||
var s = new Sequencer(r);
|
||||
var th = generateThread(r);
|
||||
t.strictEquals(th.stack.length, 12);
|
||||
s.retireThread(th);
|
||||
t.strictEquals(th.stack.length, 0);
|
||||
t.strictEquals(th.status, Thread.STATUS_DONE);
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
||||
test('stepToProcedure', function (t) {
|
||||
var r = new Runtime();
|
||||
var s = new Sequencer(r);
|
||||
var th = generateThread(r);
|
||||
var expectedBlock = th.peekStack();
|
||||
s.stepToProcedure(th, '');
|
||||
t.strictEquals(th.peekStack(), expectedBlock);
|
||||
s.stepToProcedure(th, 'faceCode');
|
||||
t.strictEquals(th.peekStack(), expectedBlock);
|
||||
s.stepToProcedure(th, 'faceCode');
|
||||
th.target.blocks.getBlock(th.stack[th.stack.length - 4]).mutation.proccode = 'othercode';
|
||||
expectedBlock = th.stack[th.stack.length - 4];
|
||||
s.stepToProcedure(th, 'othercode');
|
||||
t.strictEquals(th.peekStack(), expectedBlock);
|
||||
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
||||
test('stepThreads', function (t) {
|
||||
var r = new Runtime();
|
||||
r.currentStepTime = Infinity;
|
||||
var s = new Sequencer(r);
|
||||
t.strictEquals(s.stepThreads().length, 0);
|
||||
generateThread(r);
|
||||
t.strictEquals(r.threads.length, 1);
|
||||
t.strictEquals(s.stepThreads().length, 0);
|
||||
r.threads[0].status = Thread.STATUS_RUNNING;
|
||||
t.strictEquals(s.stepThreads().length, 1);
|
||||
|
||||
t.end();
|
||||
});
|
||||
|
|
Loading…
Reference in a new issue