mirror of
https://github.com/google/blockly.git
synced 2026-01-08 09:30:06 +01:00
557 lines
19 KiB
JavaScript
557 lines
19 KiB
JavaScript
/**
|
|
* @license
|
|
* Visual Blocks Editor
|
|
*
|
|
* Copyright 2016 Massachusetts Institute of Technology
|
|
* All rights reserved.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
/**
|
|
* @fileoverview A div that floats on top of the workspace, for drop-down menus.
|
|
* The drop-down can be kept inside the workspace, animate in/out, etc.
|
|
* @author tmickel@mit.edu (Tim Mickel)
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
goog.provide('Blockly.DropDownDiv');
|
|
|
|
goog.require('Blockly.utils.math');
|
|
|
|
goog.require('goog.style');
|
|
|
|
|
|
/**
|
|
* Class for drop-down div.
|
|
* @constructor
|
|
*/
|
|
Blockly.DropDownDiv = function() {
|
|
};
|
|
|
|
/**
|
|
* The div element. Set once by Blockly.DropDownDiv.createDom.
|
|
* @type {Element}
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.DIV_ = null;
|
|
|
|
/**
|
|
* Drop-downs will appear within the bounds of this element if possible.
|
|
* Set in Blockly.DropDownDiv.setBoundsElement.
|
|
* @type {Element}
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.boundsElement_ = null;
|
|
|
|
/**
|
|
* The object currently using the drop-down.
|
|
* @type {Object}
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.owner_ = null;
|
|
|
|
/**
|
|
* Whether the dropdown was positioned to a field or the source block.
|
|
* @type {boolean}
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.positionToField_ = null;
|
|
|
|
/**
|
|
* Arrow size in px. Should match the value in CSS (need to position pre-render).
|
|
* @type {number}
|
|
* @const
|
|
*/
|
|
Blockly.DropDownDiv.ARROW_SIZE = 16;
|
|
|
|
/**
|
|
* Drop-down border size in px. Should match the value in CSS (need to position the arrow).
|
|
* @type {number}
|
|
* @const
|
|
*/
|
|
Blockly.DropDownDiv.BORDER_SIZE = 1;
|
|
|
|
/**
|
|
* Amount the arrow must be kept away from the edges of the main drop-down div, in px.
|
|
* @type {number}
|
|
* @const
|
|
*/
|
|
Blockly.DropDownDiv.ARROW_HORIZONTAL_PADDING = 12;
|
|
|
|
/**
|
|
* Amount drop-downs should be padded away from the source, in px.
|
|
* @type {number}
|
|
* @const
|
|
*/
|
|
Blockly.DropDownDiv.PADDING_Y = 16;
|
|
|
|
/**
|
|
* Length of animations in seconds.
|
|
* @type {number}
|
|
* @const
|
|
*/
|
|
Blockly.DropDownDiv.ANIMATION_TIME = 0.25;
|
|
|
|
/**
|
|
* Timer for animation out, to be cleared if we need to immediately hide
|
|
* without disrupting new shows.
|
|
* @type {number}
|
|
*/
|
|
Blockly.DropDownDiv.animateOutTimer_ = null;
|
|
|
|
/**
|
|
* Callback for when the drop-down is hidden.
|
|
* @type {Function}
|
|
*/
|
|
Blockly.DropDownDiv.onHide_ = 0;
|
|
|
|
/**
|
|
* Create and insert the DOM element for this div.
|
|
*/
|
|
Blockly.DropDownDiv.createDom = function() {
|
|
if (Blockly.DropDownDiv.DIV_) {
|
|
return; // Already created.
|
|
}
|
|
var div = document.createElement('div');
|
|
div.className = 'blocklyDropDownDiv';
|
|
document.body.appendChild(div);
|
|
Blockly.DropDownDiv.DIV_ = div;
|
|
|
|
var content = document.createElement('div');
|
|
content.className = 'blocklyDropDownContent';
|
|
div.appendChild(content);
|
|
Blockly.DropDownDiv.content_ = content;
|
|
|
|
var arrow = document.createElement('div');
|
|
arrow.className = 'blocklyDropDownArrow';
|
|
div.appendChild(arrow);
|
|
Blockly.DropDownDiv.arrow_ = arrow;
|
|
|
|
Blockly.DropDownDiv.DIV_.style.opacity = 0;
|
|
|
|
// Transition animation for transform: translate() and opacity.
|
|
Blockly.DropDownDiv.DIV_.style.transition = 'transform ' +
|
|
Blockly.DropDownDiv.ANIMATION_TIME + 's, ' +
|
|
'opacity ' + Blockly.DropDownDiv.ANIMATION_TIME + 's';
|
|
};
|
|
|
|
/**
|
|
* Set an element to maintain bounds within. Drop-downs will appear
|
|
* within the box of this element if possible.
|
|
* @param {Element} boundsElement Element to bind drop-down to.
|
|
*/
|
|
Blockly.DropDownDiv.setBoundsElement = function(boundsElement) {
|
|
Blockly.DropDownDiv.boundsElement_ = boundsElement;
|
|
};
|
|
|
|
/**
|
|
* Provide the div for inserting content into the drop-down.
|
|
* @return {Element} Div to populate with content
|
|
*/
|
|
Blockly.DropDownDiv.getContentDiv = function() {
|
|
return Blockly.DropDownDiv.content_;
|
|
};
|
|
|
|
/**
|
|
* Clear the content of the drop-down.
|
|
*/
|
|
Blockly.DropDownDiv.clearContent = function() {
|
|
Blockly.DropDownDiv.content_.innerHTML = '';
|
|
Blockly.DropDownDiv.content_.style.width = '';
|
|
};
|
|
|
|
/**
|
|
* Set the colour for the drop-down.
|
|
* @param {string} backgroundColour Any CSS colour for the background.
|
|
* @param {string} borderColour Any CSS colour for the border.
|
|
*/
|
|
Blockly.DropDownDiv.setColour = function(backgroundColour, borderColour) {
|
|
Blockly.DropDownDiv.DIV_.style.backgroundColor = backgroundColour;
|
|
Blockly.DropDownDiv.DIV_.style.borderColor = borderColour;
|
|
};
|
|
|
|
/**
|
|
* Set the category for the drop-down.
|
|
* @param {string} category The new category for the drop-down.
|
|
*/
|
|
Blockly.DropDownDiv.setCategory = function(category) {
|
|
Blockly.DropDownDiv.DIV_.setAttribute('data-category', category);
|
|
};
|
|
|
|
/**
|
|
* Shortcut to show and place the drop-down with positioning determined
|
|
* by a particular block. The primary position will be below the block,
|
|
* and the secondary position above the block. Drop-down will be
|
|
* constrained to the block's workspace.
|
|
* @param {!Blockly.Field} field The field showing the drop-down.
|
|
* @param {!Blockly.Block} block Block to position the drop-down around.
|
|
* @param {Function=} opt_onHide Optional callback for when the drop-down is
|
|
* hidden.
|
|
* @param {number} opt_secondaryYOffset Optional Y offset for above-block
|
|
* positioning.
|
|
* @return {boolean} True if the menu rendered below block; false if above.
|
|
*/
|
|
Blockly.DropDownDiv.showPositionedByBlock = function(field, block,
|
|
opt_onHide, opt_secondaryYOffset) {
|
|
var scale = block.workspace.scale;
|
|
var bBox = {width: block.width, height: block.height};
|
|
bBox.width *= scale;
|
|
bBox.height *= scale;
|
|
var position = block.getSvgRoot().getBoundingClientRect();
|
|
// If we can fit it, render below the block.
|
|
var primaryX = position.left + bBox.width / 2;
|
|
var primaryY = position.top + bBox.height;
|
|
// If we can't fit it, render above the entire parent block.
|
|
var secondaryX = primaryX;
|
|
var secondaryY = position.top;
|
|
if (opt_secondaryYOffset) {
|
|
secondaryY += opt_secondaryYOffset;
|
|
}
|
|
// Set bounds to workspace; show the drop-down.
|
|
Blockly.DropDownDiv.setBoundsElement(block.workspace.getParentSvg().parentNode);
|
|
return Blockly.DropDownDiv.show(field, primaryX, primaryY, secondaryX, secondaryY, opt_onHide);
|
|
};
|
|
|
|
|
|
/**
|
|
* Shortcut to show and place the drop-down with positioning determined
|
|
* by a particular field. The primary position will be below the field,
|
|
* and the secondary position above the field. Drop-down will be
|
|
* constrained to the block's workspace.
|
|
* @param {!Object} owner The object showing the drop-down.
|
|
* @param {Function=} opt_onHide Optional callback for when the drop-down is
|
|
* hidden.
|
|
* @param {number=} opt_secondaryYOffset Optional Y offset for above-block
|
|
* positioning.
|
|
* @return {boolean} True if the menu rendered below block; false if above.
|
|
*/
|
|
Blockly.DropDownDiv.showPositionedByField = function(owner,
|
|
opt_onHide, opt_secondaryYOffset) {
|
|
var position = owner.fieldGroup_.getBoundingClientRect();
|
|
// If we can fit it, render below the block.
|
|
var primaryX = position.left + position.width / 2;
|
|
var primaryY = position.bottom;
|
|
// If we can't fit it, render above the entire parent block.
|
|
var secondaryX = primaryX;
|
|
var secondaryY = position.top;
|
|
if (opt_secondaryYOffset) {
|
|
secondaryY += opt_secondaryYOffset;
|
|
}
|
|
// Set bounds to workspace; show the drop-down.
|
|
Blockly.DropDownDiv.positionToField_ = true;
|
|
Blockly.DropDownDiv.setBoundsElement(
|
|
owner.getSourceBlock().workspace.getParentSvg().parentNode);
|
|
return Blockly.DropDownDiv.show(
|
|
owner, primaryX, primaryY, secondaryX, secondaryY, opt_onHide);
|
|
};
|
|
|
|
|
|
/**
|
|
* Show and place the drop-down.
|
|
* The drop-down is placed with an absolute "origin point" (x, y) - i.e.,
|
|
* the arrow will point at this origin and box will positioned below or above it.
|
|
* If we can maintain the container bounds at the primary point, the arrow will
|
|
* point there, and the container will be positioned below it.
|
|
* If we can't maintain the container bounds at the primary point, fall-back to the
|
|
* secondary point and position above.
|
|
* @param {Object} owner The object showing the drop-down
|
|
* @param {number} primaryX Desired origin point x, in absolute px
|
|
* @param {number} primaryY Desired origin point y, in absolute px
|
|
* @param {number} secondaryX Secondary/alternative origin point x, in absolute px
|
|
* @param {number} secondaryY Secondary/alternative origin point y, in absolute px
|
|
* @param {Function=} opt_onHide Optional callback for when the drop-down is hidden
|
|
* @return {boolean} True if the menu rendered at the primary origin point.
|
|
*/
|
|
Blockly.DropDownDiv.show = function(owner, primaryX, primaryY,
|
|
secondaryX, secondaryY, opt_onHide) {
|
|
Blockly.DropDownDiv.owner_ = owner;
|
|
Blockly.DropDownDiv.onHide_ = opt_onHide;
|
|
var metrics = Blockly.DropDownDiv.getPositionMetrics(primaryX, primaryY,
|
|
secondaryX, secondaryY);
|
|
// Update arrow CSS.
|
|
Blockly.DropDownDiv.arrow_.style.transform = 'translate(' +
|
|
metrics.arrowX + 'px,' + metrics.arrowY + 'px) rotate(45deg)';
|
|
Blockly.DropDownDiv.arrow_.setAttribute('class', metrics.arrowAtTop ?
|
|
'blocklyDropDownArrow arrowTop' : 'blocklyDropDownArrow arrowBottom');
|
|
Blockly.DropDownDiv.arrow_.style.display = metrics.arrowVisible ? '' : 'none';
|
|
|
|
// When we change `translate` multiple times in close succession,
|
|
// Chrome may choose to wait and apply them all at once.
|
|
// Since we want the translation to initial X, Y to be immediate,
|
|
// and the translation to final X, Y to be animated,
|
|
// we saw problems where both would be applied after animation was turned on,
|
|
// making the dropdown appear to fly in from (0, 0).
|
|
// Using both `left`, `top` for the initial translation and then `translate`
|
|
// for the animated transition to final X, Y is a workaround.
|
|
|
|
Blockly.DropDownDiv.positionInternal_(
|
|
metrics.initialX, metrics.initialY,
|
|
metrics.finalX, metrics.finalY);
|
|
return metrics.arrowAtTop;
|
|
};
|
|
|
|
/**
|
|
* Get sizing info about the bounding element.
|
|
* @return {!Object} An object containing size information about the bounding
|
|
* element (bounding box and width/height).
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.getBoundsInfo_ = function() {
|
|
var boundPosition = Blockly.DropDownDiv.boundsElement_.getBoundingClientRect();
|
|
var boundSize = goog.style.getSize(Blockly.DropDownDiv.boundsElement_);
|
|
|
|
return {
|
|
left: boundPosition.left,
|
|
right: boundPosition.left + boundSize.width,
|
|
top: boundPosition.top,
|
|
bottom: boundPosition.top + boundSize.height,
|
|
width: boundSize.width,
|
|
height: boundSize.height
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Helper to position the drop-down and the arrow, maintaining bounds.
|
|
* See explanation of origin points in Blockly.DropDownDiv.show.
|
|
* @param {number} primaryX Desired origin point x, in absolute px
|
|
* @param {number} primaryY Desired origin point y, in absolute px
|
|
* @param {number} secondaryX Secondary/alternative origin point x, in absolute px
|
|
* @param {number} secondaryY Secondary/alternative origin point y, in absolute px
|
|
* @return {Object} Various final metrics, including rendered positions for drop-down and arrow.
|
|
*/
|
|
Blockly.DropDownDiv.getPositionMetrics = function(primaryX, primaryY,
|
|
secondaryX, secondaryY) {
|
|
var boundsInfo = Blockly.DropDownDiv.getBoundsInfo_();
|
|
var div = Blockly.DropDownDiv.DIV_;
|
|
var divSize = goog.style.getSize(div);
|
|
|
|
// First decide if we will render at primary or secondary position
|
|
// i.e., above or below
|
|
// renderX, renderY will eventually be the final rendered position of the box.
|
|
var renderX, renderY, renderedSecondary, renderedTertiary;
|
|
// Can the div fit inside the bounds if we render below the primary point?
|
|
if (primaryY + divSize.height > boundsInfo.bottom) {
|
|
// We can't fit below in terms of y. Can we fit above?
|
|
if (secondaryY - divSize.height < boundsInfo.top) {
|
|
// We also can't fit above, so just render at the top of the screen.
|
|
renderX = primaryX;
|
|
renderY = 0;
|
|
renderedSecondary = false;
|
|
renderedTertiary = true;
|
|
} else {
|
|
// We can fit above, render secondary
|
|
renderX = secondaryX;
|
|
renderY = secondaryY - divSize.height - Blockly.DropDownDiv.PADDING_Y;
|
|
renderedSecondary = true;
|
|
}
|
|
} else {
|
|
// We can fit below, render primary
|
|
renderX = primaryX;
|
|
renderY = primaryY + Blockly.DropDownDiv.PADDING_Y;
|
|
renderedSecondary = false;
|
|
}
|
|
|
|
var centerX = renderX;
|
|
// The dropdown's X position is at the top-left of the dropdown rect, but the
|
|
// dropdown should appear centered relative to the desired origin point.
|
|
renderX -= divSize.width / 2;
|
|
// Fit horizontally in the bounds.
|
|
renderX = Blockly.utils.math.clamp(
|
|
boundsInfo.left, renderX, boundsInfo.right - divSize.width);
|
|
|
|
// Calculate the absolute arrow X. The arrow wants to be as close to the
|
|
// origin point as possible. The arrow may not be centered in the dropdown div.
|
|
var absoluteArrowX = centerX - Blockly.DropDownDiv.ARROW_SIZE / 2;
|
|
// Keep in overall bounds
|
|
absoluteArrowX = Blockly.utils.math.clamp(
|
|
boundsInfo.left, absoluteArrowX, boundsInfo.right);
|
|
|
|
// Convert the arrow position to be relative to the top left corner of the div.
|
|
var relativeArrowX = absoluteArrowX - renderX;
|
|
|
|
// Pad the arrow by some pixels, primarily so that it doesn't render on top
|
|
// of a rounded border.
|
|
relativeArrowX = Blockly.utils.math.clamp(
|
|
Blockly.DropDownDiv.ARROW_HORIZONTAL_PADDING,
|
|
relativeArrowX,
|
|
divSize.width - Blockly.DropDownDiv.ARROW_HORIZONTAL_PADDING -
|
|
Blockly.DropDownDiv.ARROW_SIZE);
|
|
|
|
var arrowY = (renderedSecondary) ?
|
|
divSize.height - Blockly.DropDownDiv.BORDER_SIZE : 0;
|
|
arrowY -= (Blockly.DropDownDiv.ARROW_SIZE / 2) +
|
|
Blockly.DropDownDiv.BORDER_SIZE;
|
|
|
|
var initialY;
|
|
if (renderedSecondary) {
|
|
initialY = secondaryY - divSize.height; // No padding on Y
|
|
} else {
|
|
initialY = primaryY; // No padding on Y
|
|
}
|
|
|
|
return {
|
|
initialX: renderX, // X position remains constant during animation.
|
|
initialY : initialY,
|
|
finalX: renderX,
|
|
finalY: renderY,
|
|
arrowX: relativeArrowX,
|
|
arrowY: arrowY,
|
|
arrowAtTop: !renderedSecondary,
|
|
arrowVisible: !renderedTertiary
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Is the container visible?
|
|
* @return {boolean} True if visible.
|
|
*/
|
|
Blockly.DropDownDiv.isVisible = function() {
|
|
return !!Blockly.DropDownDiv.owner_;
|
|
};
|
|
|
|
/**
|
|
* Hide the menu only if it is owned by the provided object.
|
|
* @param {Object} owner Object which must be owning the drop-down to hide.
|
|
* @return {boolean} True if hidden.
|
|
*/
|
|
Blockly.DropDownDiv.hideIfOwner = function(owner) {
|
|
if (Blockly.DropDownDiv.owner_ === owner) {
|
|
Blockly.DropDownDiv.hide();
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Hide the menu, triggering animation.
|
|
*/
|
|
Blockly.DropDownDiv.hide = function() {
|
|
// Start the animation by setting the translation and fading out.
|
|
var div = Blockly.DropDownDiv.DIV_;
|
|
// Reset to (initialX, initialY) - i.e., no translation.
|
|
div.style.transform = 'translate(0px, 0px)';
|
|
div.style.opacity = 0;
|
|
Blockly.DropDownDiv.animateOutTimer_ = setTimeout(function() {
|
|
// Finish animation - reset all values to default.
|
|
Blockly.DropDownDiv.hideWithoutAnimation();
|
|
}, Blockly.DropDownDiv.ANIMATION_TIME * 1000);
|
|
if (Blockly.DropDownDiv.onHide_) {
|
|
Blockly.DropDownDiv.onHide_();
|
|
Blockly.DropDownDiv.onHide_ = null;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Hide the menu, without animation.
|
|
*/
|
|
Blockly.DropDownDiv.hideWithoutAnimation = function() {
|
|
if (!Blockly.DropDownDiv.isVisible()) {
|
|
return;
|
|
}
|
|
if (Blockly.DropDownDiv.animateOutTimer_) {
|
|
clearTimeout(Blockly.DropDownDiv.animateOutTimer_);
|
|
}
|
|
|
|
// Reset style properties in case this gets called directly
|
|
// instead of hide() - see discussion on #2551.
|
|
var div = Blockly.DropDownDiv.DIV_;
|
|
div.style.transform = '';
|
|
div.style.left = '';
|
|
div.style.top = '';
|
|
div.style.opacity = 0;
|
|
div.style.display = 'none';
|
|
|
|
Blockly.DropDownDiv.clearContent();
|
|
Blockly.DropDownDiv.owner_ = null;
|
|
if (Blockly.DropDownDiv.onHide_) {
|
|
Blockly.DropDownDiv.onHide_();
|
|
Blockly.DropDownDiv.onHide_ = null;
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Set the dropdown div's position.
|
|
* @param {number} initialX Initial Horizontal location
|
|
* (window coordinates, not body).
|
|
* @param {number} initialY Initial Vertical location
|
|
* (window coordinates, not body).
|
|
* @param {number} finalX Final Horizontal location
|
|
* (window coordinates, not body).
|
|
* @param {number} finalY Final Vertical location
|
|
* (window coordinates, not body).
|
|
* @private
|
|
*/
|
|
Blockly.DropDownDiv.positionInternal_ = function(initialX, initialY, finalX, finalY) {
|
|
initialX = Math.floor(initialX);
|
|
initialY = Math.floor(initialY);
|
|
finalX = Math.floor(finalX);
|
|
finalY = Math.floor(finalY);
|
|
|
|
var div = Blockly.DropDownDiv.DIV_;
|
|
// First apply initial translation.
|
|
div.style.left = initialX + 'px';
|
|
div.style.top = initialY + 'px';
|
|
|
|
// Show the div.
|
|
div.style.display = 'block';
|
|
div.style.opacity = 1;
|
|
// Add final translate, animated through `transition`.
|
|
// Coordinates are relative to (initialX, initialY),
|
|
// where the drop-down is absolutely positioned.
|
|
var dx = finalX - initialX;
|
|
var dy = finalY - initialY;
|
|
div.style.transform = 'translate(' + dx + 'px,' + dy + 'px)';
|
|
};
|
|
|
|
/**
|
|
* Repositions the dropdownDiv on window resize. If it doesn't know how to
|
|
* calculate the new position, it wll just hide it instead.
|
|
*/
|
|
Blockly.DropDownDiv.repositionForWindowResize = function() {
|
|
// This condition mainly catches the dropdown div when it is being used as a
|
|
// dropdown. It is important not to close it in this case because on Android,
|
|
// when a field is focused, the soft keyboard opens triggering a window resize
|
|
// event and we want the dropdown div to stick around so users can type into it.
|
|
if (Blockly.DropDownDiv.owner_) {
|
|
var block = Blockly.DropDownDiv.owner_.getSourceBlock();
|
|
var scale = block.workspace.scale;
|
|
var bBox = {
|
|
width: Blockly.DropDownDiv.positionToField_ ?
|
|
Blockly.DropDownDiv.owner_.size_.width : block.width,
|
|
height: Blockly.DropDownDiv.positionToField_ ?
|
|
Blockly.DropDownDiv.owner_.size_.height : block.height
|
|
};
|
|
bBox.width *= scale;
|
|
bBox.height *= scale;
|
|
var position = Blockly.DropDownDiv.positionToField_ ?
|
|
Blockly.DropDownDiv.owner_.fieldGroup_.getBoundingClientRect() :
|
|
block.getSvgRoot().getBoundingClientRect();
|
|
// If we can fit it, render below the block.
|
|
var primaryX = position.left + bBox.width / 2;
|
|
var primaryY = position.top + bBox.height;
|
|
// If we can't fit it, render above the entire parent block.
|
|
var secondaryX = primaryX;
|
|
var secondaryY = position.top;
|
|
var metrics = Blockly.DropDownDiv.getPositionMetrics(
|
|
primaryX, primaryY, secondaryX, secondaryY);
|
|
Blockly.DropDownDiv.positionInternal_(
|
|
metrics.initialX, metrics.initialY,
|
|
metrics.finalX, metrics.finalY);
|
|
} else {
|
|
Blockly.DropDownDiv.hide();
|
|
}
|
|
};
|