Skip to content

Commit

Permalink
fix: dragging variables from flyout (#5434)
Browse files Browse the repository at this point in the history
* fix: dragging variables from flyout

* fix: rename positionBlock_ to positionNewBlock_

* fix: type

* fix: try alternative method for handling variables in flyout
  • Loading branch information
BeksOmega authored Sep 10, 2021
1 parent d38c08d commit 3156270
Show file tree
Hide file tree
Showing 7 changed files with 175 additions and 35 deletions.
2 changes: 1 addition & 1 deletion core/block.js
Original file line number Diff line number Diff line change
Expand Up @@ -1068,7 +1068,7 @@ Blockly.Block.prototype.getField = function(name) {

/**
* Return all variables referenced by this block.
* @return {!Array<string>} List of variable names.
* @return {!Array<string>} List of variable ids.
*/
Blockly.Block.prototype.getVars = function() {
var vars = [];
Expand Down
5 changes: 4 additions & 1 deletion core/field.js
Original file line number Diff line number Diff line change
Expand Up @@ -418,10 +418,13 @@ Blockly.Field.prototype.toXml = function(fieldElement) {
/**
* Saves this fields value as something which can be serialized to JSON. Should
* only be called by the serialization system.
* @param {boolean=} _doFullSerialization If true, this signals to the field that
* if it normally just saves a reference to some state (eg variable fields)
* it should instead serialize the full state of the thing being referenced.
* @return {*} JSON serializable state.
* @package
*/
Blockly.Field.prototype.saveState = function() {
Blockly.Field.prototype.saveState = function(_doFullSerialization) {
var legacyState = this.saveLegacyState(Blockly.Field);
if (legacyState !== null) {
return legacyState;
Expand Down
14 changes: 11 additions & 3 deletions core/field_variable.js
Original file line number Diff line number Diff line change
Expand Up @@ -196,20 +196,28 @@ Blockly.FieldVariable.prototype.toXml = function(fieldElement) {

/**
* Saves this field's value.
* @return {*} The ID of the variable referenced by this field.
* @param {boolean=} doFullSerialization If true, the variable field will
* serialize the full state of the field being referenced (ie ID, name,
* and type) rather than just a reference to it (ie ID).
* @return {*} The state of the variable field.
* @override
* @package
*/
Blockly.FieldVariable.prototype.saveState = function() {
Blockly.FieldVariable.prototype.saveState = function(doFullSerialization) {
var legacyState = this.saveLegacyState(Blockly.Field);
if (legacyState !== null) {
return legacyState;
}
// Make sure the variable is initialized.
this.initModel();
return {
var state = {
'id': this.variable_.getId()
};
if (doFullSerialization) {
state['name'] = this.variable_.name;
state['type'] = this.variable_.type;
}
return state;
};

/**
Expand Down
18 changes: 16 additions & 2 deletions core/flyout_base.js
Original file line number Diff line number Diff line change
Expand Up @@ -1055,7 +1055,8 @@ Blockly.Flyout.prototype.isScrollable = function() {
* @private
*/
Blockly.Flyout.prototype.placeNewBlock_ = function(oldBlock) {
var targetWorkspace = this.targetWorkspace;
var targetWorkspace =
/** @type {!Blockly.WorkspaceSvg} */ (this.targetWorkspace);
var svgRootOld = oldBlock.getSvgRoot();
if (!svgRootOld) {
throw Error('oldBlock is not rendered.');
Expand All @@ -1069,6 +1070,20 @@ Blockly.Flyout.prototype.placeNewBlock_ = function(oldBlock) {
var block = /** @type {!Blockly.BlockSvg} */
(Blockly.serialization.blocks.load(json, targetWorkspace));

this.positionNewBlock_(oldBlock, block);

return block;
};

/**
* Positions a block on the target workspace.
* @param {!Blockly.BlockSvg} oldBlock The flyout block being copied.
* @param {!Blockly.BlockSvg} block The block to posiiton.
* @private
*/
Blockly.Flyout.prototype.positionNewBlock_ = function(oldBlock, block) {
var targetWorkspace = this.targetWorkspace;

// The offset in pixels between the main workspace's origin and the upper left
// corner of the injection div.
var mainOffsetPixels = targetWorkspace.getOriginOffsetInPixels();
Expand Down Expand Up @@ -1096,7 +1111,6 @@ Blockly.Flyout.prototype.placeNewBlock_ = function(oldBlock) {
finalOffset.scale(1 / targetWorkspace.scale);

block.moveBy(finalOffset.x, finalOffset.y);
return block;
};

/**
Expand Down
44 changes: 29 additions & 15 deletions core/serialization/blocks.js
Original file line number Diff line number Diff line change
Expand Up @@ -71,14 +71,19 @@ exports.State = State;
* Returns the state of the given block as a plain JavaScript object.
* @param {!Block} block The block to serialize.
* @param {{addCoordinates: (boolean|undefined), addInputBlocks:
* (boolean|undefined), addNextBlocks: (boolean|undefined)}=} param1
* (boolean|undefined), addNextBlocks: (boolean|undefined),
* doFullSerialization: (boolean|undefined)}=} param1
* addCoordinates: If true, the coordinates of the block are added to the
* serialized state. False by default.
* addinputBlocks: If true, children of the block which are connected to
* inputs will be serialized. True by default.
* addNextBlocks: If true, children of the block which are connected to the
* block's next connection (if it exists) will be serialized.
* True by default.
* doFullSerialization: If true, fields that normally just save a reference
* to some external state (eg variables) will instead serialize all of the
* info about that state. This supports deserializing the block into a
* workspace where that state doesn't yet exist. True by default.
* @return {?State} The serialized state of the block, or null if the block
* could not be serialied (eg it was an insertion marker).
*/
Expand All @@ -87,7 +92,8 @@ const save = function(
{
addCoordinates = false,
addInputBlocks = true,
addNextBlocks = true
addNextBlocks = true,
doFullSerialization = true,
} = {}
) {
if (block.isInsertionMarker()) {
Expand All @@ -105,12 +111,12 @@ const save = function(
saveAttributes(block, state);
saveExtraState(block, state);
saveIcons(block, state);
saveFields(block, state);
saveFields(block, state, doFullSerialization);
if (addInputBlocks) {
saveInputBlocks(block, state);
saveInputBlocks(block, state, doFullSerialization);
}
if (addNextBlocks) {
saveNextBlocks(block, state);
saveNextBlocks(block, state, doFullSerialization);
}

return state;
Expand Down Expand Up @@ -195,8 +201,11 @@ const saveIcons = function(block, state) {
* Adds the state of all of the fields on the block to the given state object.
* @param {!Block} block The block to serialize the field state of.
* @param {!State} state The state object to append to.
* @param {boolean} doFullSerialization Whether or not to serialize the full
* state of the field (rather than possibly saving a reference to some
* state).
*/
const saveFields = function(block, state) {
const saveFields = function(block, state, doFullSerialization) {
let hasFieldState = false;
const fields = Object.create(null);
for (let i = 0; i < block.inputList.length; i++) {
Expand All @@ -205,7 +214,7 @@ const saveFields = function(block, state) {
const field = input.fieldRow[j];
if (field.isSerializable()) {
hasFieldState = true;
fields[field.name] = field.saveState();
fields[field.name] = field.saveState(doFullSerialization);
}
}
}
Expand All @@ -219,16 +228,17 @@ const saveFields = function(block, state) {
* connected to inputs) to the given state object.
* @param {!Block} block The block to serialize the input blocks of.
* @param {!State} state The state object to append to.
* @param {boolean} doFullSerialization Whether or not to do full serialization.
*/
const saveInputBlocks = function(block, state) {
const saveInputBlocks = function(block, state, doFullSerialization) {
const inputs = Object.create(null);
for (let i = 0; i < block.inputList.length; i++) {
const input = block.inputList[i];
if (input.type === inputTypes.DUMMY) {
continue;
}
const connectionState =
saveConnection(/** @type {!Connection} */ (input.connection));
const connectionState = saveConnection(
/** @type {!Connection} */ (input.connection), doFullSerialization);
if (connectionState) {
inputs[input.name] = connectionState;
}
Expand All @@ -244,12 +254,14 @@ const saveInputBlocks = function(block, state) {
* state object.
* @param {!Block} block The block to serialize the next blocks of.
* @param {!State} state The state object to append to.
* @param {boolean} doFullSerialization Whether or not to do full serialization.
*/
const saveNextBlocks = function(block, state) {
const saveNextBlocks = function(block, state, doFullSerialization) {
if (!block.nextConnection) {
return;
}
const connectionState = saveConnection(block.nextConnection);
const connectionState = saveConnection(
block.nextConnection, doFullSerialization);
if (connectionState) {
state['next'] = connectionState;
}
Expand All @@ -262,8 +274,9 @@ const saveNextBlocks = function(block, state) {
* blocks of.
* @return {?ConnectionState} An object containing the state of any connected
* shadow block, or any connected real block.
* @param {boolean} doFullSerialization Whether or not to do full serialization.
*/
const saveConnection = function(connection) {
const saveConnection = function(connection, doFullSerialization) {
const shadow = connection.getShadowState(true);
const child = connection.targetBlock();
if (!shadow && !child) {
Expand All @@ -274,7 +287,7 @@ const saveConnection = function(connection) {
state['shadow'] = shadow;
}
if (child && !child.isShadow()) {
state['block'] = save(child);
state['block'] = save(child, {doFullSerialization});
}
return state;
};
Expand Down Expand Up @@ -640,7 +653,8 @@ class BlockSerializer {
save(workspace) {
const blockState = [];
for (const block of workspace.getTopBlocks(false)) {
const state = saveBlock(block, {addCoordinates: true});
const state = saveBlock(
block, {addCoordinates: true, doFullSerialization: false});
if (state) {
blockState.push(state);
}
Expand Down
55 changes: 42 additions & 13 deletions tests/mocha/field_variable_test.js
Original file line number Diff line number Diff line change
Expand Up @@ -389,21 +389,50 @@ suite('Variable Fields', function() {
workspaceTeardown.call(this, this.workspace);
});

test('Untyped', function() {
const block = this.workspace.newBlock('row_block');
const field = new Blockly.FieldVariable('x');
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(jso['fields'], {'VAR': {'id': 'id2'}});
suite('Full', function() {
test('Untyped', function() {
const block = this.workspace.newBlock('row_block');
const field = new Blockly.FieldVariable('x');
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(
jso['fields'], {'VAR': {'id': 'id2', 'name': 'x', 'type': ''}});
});

test('Typed', function() {
const block = this.workspace.newBlock('row_block');
const field =
new Blockly.FieldVariable('x', undefined, undefined, 'String');
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(
jso['fields'], {'VAR': {'id': 'id2', 'name': 'x', 'type': 'String'}});
});
});

test('Typed', function() {
const block = this.workspace.newBlock('row_block');
const field =
new Blockly.FieldVariable('x', undefined, undefined, ['String']);
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(jso['fields'], {'VAR': {'id': 'id2'}});
suite('Not full', function() {
test('Untyped', function() {
const block = this.workspace.newBlock('row_block');
const field = new Blockly.FieldVariable('x');
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(
block, {doFullSerialization: false});
chai.assert.deepEqual(jso['fields'], {'VAR': {'id': 'id2'}});
chai.assert.isUndefined(jso['fields']['VAR']['name']);
chai.assert.isUndefined(jso['fields']['VAR']['type']);
});

test('Typed', function() {
const block = this.workspace.newBlock('row_block');
const field =
new Blockly.FieldVariable('x', undefined, undefined, 'String');
block.getInput('INPUT').appendField(field, 'VAR');
const jso = Blockly.serialization.blocks.save(
block, {doFullSerialization: false});
chai.assert.deepEqual(jso['fields'], {'VAR': {'id': 'id2'}});
chai.assert.isUndefined(jso['fields']['VAR']['name']);
chai.assert.isUndefined(jso['fields']['VAR']['type']);
});
});
});

Expand Down
72 changes: 72 additions & 0 deletions tests/mocha/jso_serialization_test.js
Original file line number Diff line number Diff line change
Expand Up @@ -694,6 +694,78 @@ suite('JSO Serialization', function() {
});
});
});

suite('Do full serialization', function() {
suite('True', function() {
test('Single block', function() {
var block = this.workspace.newBlock('variables_get');
var jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(
jso['fields']['VAR'], {'id': 'id2', 'name': 'item', 'type': ''});
});

test('Input block', function() {
var block = this.workspace.newBlock('row_block');
var childBlock = this.workspace.newBlock('variables_get');
block.getInput('INPUT').connection.connect(
childBlock.outputConnection);
var jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(
jso['inputs']['INPUT']['block']['fields']['VAR'],
{'id': 'id4', 'name': 'item', 'type': ''});
});

test('Next block', function() {
var block = this.workspace.newBlock('stack_block');
var childBlock = this.workspace.newBlock('variables_set');
block.nextConnection.connect(childBlock.previousConnection);
var jso = Blockly.serialization.blocks.save(block);
chai.assert.deepEqual(
jso['next']['block']['fields']['VAR'],
{'id': 'id4', 'name': 'item', 'type': ''});
});
});

suite('False', function() {
test('Single block', function() {
var block = this.workspace.newBlock('variables_get');
var jso = Blockly.serialization.blocks.save(
block, {doFullSerialization: false});
chai.assert.deepEqual(jso['fields']['VAR'], {'id': 'id2'});
chai.assert.isUndefined(jso['fields']['VAR']['name']);
chai.assert.isUndefined(jso['fields']['VAR']['type']);
});

test('Input block', function() {
var block = this.workspace.newBlock('row_block');
var childBlock = this.workspace.newBlock('variables_get');
block.getInput('INPUT').connection.connect(
childBlock.outputConnection);
var jso = Blockly.serialization.blocks.save(
block, {doFullSerialization: false});
chai.assert.deepEqual(
jso['inputs']['INPUT']['block']['fields']['VAR'], {'id': 'id4'});
chai.assert.isUndefined(
jso['inputs']['INPUT']['block']['fields']['VAR']['name']);
chai.assert.isUndefined(
jso['inputs']['INPUT']['block']['fields']['VAR']['type']);
});

test('Next block', function() {
var block = this.workspace.newBlock('stack_block');
var childBlock = this.workspace.newBlock('variables_set');
block.nextConnection.connect(childBlock.previousConnection);
var jso = Blockly.serialization.blocks.save(
block, {doFullSerialization: false});
chai.assert.deepEqual(
jso['next']['block']['fields']['VAR'], {'id': 'id4'});
chai.assert.isUndefined(
jso['next']['block']['fields']['VAR']['name']);
chai.assert.isUndefined(
jso['next']['block']['fields']['VAR']['type']);
});
});
});
});

suite('Variables', function() {
Expand Down

0 comments on commit 3156270

Please sign in to comment.