2017-10-12 18:35:30 -04:00
|
|
|
import paper from '@scratch/paper';
|
2017-11-02 17:19:42 -04:00
|
|
|
import log from '../log/log';
|
2018-04-16 18:08:17 -04:00
|
|
|
import {ART_BOARD_WIDTH, ART_BOARD_HEIGHT} from './view';
|
2020-02-06 18:11:10 -05:00
|
|
|
import {isGroupItem} from './item';
|
|
|
|
import costumeAnchorIcon from './icons/costume-anchor.svg';
|
|
|
|
|
|
|
|
const CROSSHAIR_SIZE = 28;
|
2017-09-11 14:23:30 -04:00
|
|
|
|
2017-11-02 17:19:42 -04:00
|
|
|
const _getLayer = function (layerString) {
|
|
|
|
for (const layer of paper.project.layers) {
|
|
|
|
if (layer.data && layer.data[layerString]) {
|
2017-09-11 14:23:30 -04:00
|
|
|
return layer;
|
|
|
|
}
|
|
|
|
}
|
2017-11-02 17:19:42 -04:00
|
|
|
};
|
2017-09-11 14:23:30 -04:00
|
|
|
|
2017-11-02 17:19:42 -04:00
|
|
|
const _getPaintingLayer = function () {
|
|
|
|
return _getLayer('isPaintingLayer');
|
2017-09-11 14:23:30 -04:00
|
|
|
};
|
|
|
|
|
2018-06-21 10:22:24 -04:00
|
|
|
/**
|
|
|
|
* Creates a canvas with width and height matching the art board size.
|
|
|
|
* @param {?number} width Width of the canvas. Defaults to ART_BOARD_WIDTH.
|
|
|
|
* @param {?number} height Height of the canvas. Defaults to ART_BOARD_HEIGHT.
|
|
|
|
* @return {HTMLCanvasElement} the canvas
|
|
|
|
*/
|
|
|
|
const createCanvas = function (width, height) {
|
|
|
|
const canvas = document.createElement('canvas');
|
|
|
|
canvas.width = width ? width : ART_BOARD_WIDTH;
|
|
|
|
canvas.height = height ? height : ART_BOARD_HEIGHT;
|
|
|
|
canvas.getContext('2d').imageSmoothingEnabled = false;
|
|
|
|
return canvas;
|
|
|
|
};
|
|
|
|
|
2018-04-06 14:41:30 -04:00
|
|
|
const clearRaster = function () {
|
2018-04-05 17:37:40 -04:00
|
|
|
const layer = _getLayer('isRasterLayer');
|
2018-04-06 14:41:30 -04:00
|
|
|
layer.removeChildren();
|
|
|
|
|
2018-04-05 17:37:40 -04:00
|
|
|
// Generate blank raster
|
2018-06-21 10:22:24 -04:00
|
|
|
const raster = new paper.Raster(createCanvas());
|
|
|
|
raster.canvas.getContext('2d').imageSmoothingEnabled = false;
|
2018-04-06 14:41:30 -04:00
|
|
|
raster.parent = layer;
|
|
|
|
raster.guide = true;
|
|
|
|
raster.locked = true;
|
2018-05-01 16:18:24 -04:00
|
|
|
raster.position = new paper.Point(ART_BOARD_WIDTH / 2, ART_BOARD_HEIGHT / 2);
|
2018-04-06 14:41:30 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
const getRaster = function () {
|
2018-04-04 23:54:41 -04:00
|
|
|
const layer = _getLayer('isRasterLayer');
|
|
|
|
// Generate blank raster
|
|
|
|
if (layer.children.length === 0) {
|
2018-05-01 16:18:24 -04:00
|
|
|
clearRaster();
|
2018-04-04 23:54:41 -04:00
|
|
|
}
|
2018-04-05 17:37:40 -04:00
|
|
|
return _getLayer('isRasterLayer').children[0];
|
|
|
|
};
|
|
|
|
|
2020-02-06 18:11:10 -05:00
|
|
|
const getDragCrosshairLayer = function () {
|
|
|
|
return _getLayer('isDragCrosshairLayer');
|
|
|
|
};
|
|
|
|
|
2018-08-08 18:27:01 -04:00
|
|
|
const getBackgroundGuideLayer = function () {
|
2017-11-02 17:19:42 -04:00
|
|
|
return _getLayer('isBackgroundGuideLayer');
|
|
|
|
};
|
|
|
|
|
2018-04-09 17:47:11 -04:00
|
|
|
const _makeGuideLayer = function () {
|
|
|
|
const guideLayer = new paper.Layer();
|
|
|
|
guideLayer.data.isGuideLayer = true;
|
|
|
|
return guideLayer;
|
|
|
|
};
|
|
|
|
|
2017-11-02 17:19:42 -04:00
|
|
|
const getGuideLayer = function () {
|
2018-04-09 17:47:11 -04:00
|
|
|
let layer = _getLayer('isGuideLayer');
|
|
|
|
if (!layer) {
|
|
|
|
layer = _makeGuideLayer();
|
|
|
|
_getPaintingLayer().activate();
|
|
|
|
}
|
|
|
|
return layer;
|
2017-11-02 17:19:42 -04:00
|
|
|
};
|
|
|
|
|
2020-02-06 18:11:10 -05:00
|
|
|
const setGuideItem = function (item) {
|
|
|
|
item.locked = true;
|
|
|
|
item.guide = true;
|
|
|
|
if (isGroupItem(item)) {
|
|
|
|
for (let i = 0; i < item.children.length; i++) {
|
|
|
|
setGuideItem(item.children[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-11-02 17:19:42 -04:00
|
|
|
/**
|
|
|
|
* Removes the guide layers, e.g. for purposes of exporting the image. Must call showGuideLayers to re-add them.
|
2018-04-09 17:47:11 -04:00
|
|
|
* @param {boolean} includeRaster true if the raster layer should also be hidden
|
2017-11-02 17:19:42 -04:00
|
|
|
* @return {object} an object of the removed layers, which should be passed to showGuideLayers to re-add them.
|
|
|
|
*/
|
2018-04-09 17:47:11 -04:00
|
|
|
const hideGuideLayers = function (includeRaster) {
|
2018-08-08 18:27:01 -04:00
|
|
|
const backgroundGuideLayer = getBackgroundGuideLayer();
|
2020-02-06 18:11:10 -05:00
|
|
|
const dragCrosshairLayer = getDragCrosshairLayer();
|
2017-11-02 17:19:42 -04:00
|
|
|
const guideLayer = getGuideLayer();
|
2020-02-06 18:11:10 -05:00
|
|
|
dragCrosshairLayer.remove();
|
2017-11-02 17:19:42 -04:00
|
|
|
guideLayer.remove();
|
|
|
|
backgroundGuideLayer.remove();
|
2018-04-09 17:47:11 -04:00
|
|
|
let rasterLayer;
|
|
|
|
if (includeRaster) {
|
|
|
|
rasterLayer = _getLayer('isRasterLayer');
|
|
|
|
rasterLayer.remove();
|
|
|
|
}
|
2017-11-02 17:19:42 -04:00
|
|
|
return {
|
2020-02-06 18:11:10 -05:00
|
|
|
dragCrosshairLayer: dragCrosshairLayer,
|
2017-11-02 17:19:42 -04:00
|
|
|
guideLayer: guideLayer,
|
2018-04-09 17:47:11 -04:00
|
|
|
backgroundGuideLayer: backgroundGuideLayer,
|
|
|
|
rasterLayer: rasterLayer
|
2017-11-02 17:19:42 -04:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add back the guide layers removed by calling hideGuideLayers. This must be done before any editing operations are
|
|
|
|
* taken in the paint editor.
|
|
|
|
* @param {!object} guideLayers object of the removed layers, which was returned by hideGuideLayers
|
|
|
|
*/
|
|
|
|
const showGuideLayers = function (guideLayers) {
|
|
|
|
const backgroundGuideLayer = guideLayers.backgroundGuideLayer;
|
2020-02-06 18:11:10 -05:00
|
|
|
const dragCrosshairLayer = guideLayers.dragCrosshairLayer;
|
2017-11-02 17:19:42 -04:00
|
|
|
const guideLayer = guideLayers.guideLayer;
|
2018-04-09 17:47:11 -04:00
|
|
|
const rasterLayer = guideLayers.rasterLayer;
|
|
|
|
if (rasterLayer && !rasterLayer.index) {
|
|
|
|
paper.project.addLayer(rasterLayer);
|
|
|
|
rasterLayer.sendToBack();
|
|
|
|
}
|
2017-11-02 17:19:42 -04:00
|
|
|
if (!backgroundGuideLayer.index) {
|
|
|
|
paper.project.addLayer(backgroundGuideLayer);
|
|
|
|
backgroundGuideLayer.sendToBack();
|
|
|
|
}
|
2020-02-06 18:11:10 -05:00
|
|
|
if (!dragCrosshairLayer.index) {
|
|
|
|
paper.project.addLayer(dragCrosshairLayer);
|
|
|
|
dragCrosshairLayer.bringToFront();
|
|
|
|
}
|
2017-11-02 17:19:42 -04:00
|
|
|
if (!guideLayer.index) {
|
|
|
|
paper.project.addLayer(guideLayer);
|
|
|
|
guideLayer.bringToFront();
|
|
|
|
}
|
|
|
|
if (paper.project.activeLayer !== _getPaintingLayer()) {
|
|
|
|
log.error(`Wrong active layer`);
|
|
|
|
log.error(paper.project.activeLayer.data);
|
2017-10-24 13:21:57 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const _makePaintingLayer = function () {
|
|
|
|
const paintingLayer = new paper.Layer();
|
|
|
|
paintingLayer.data.isPaintingLayer = true;
|
|
|
|
return paintingLayer;
|
|
|
|
};
|
|
|
|
|
2018-04-05 17:37:40 -04:00
|
|
|
const _makeRasterLayer = function () {
|
|
|
|
const rasterLayer = new paper.Layer();
|
|
|
|
rasterLayer.data.isRasterLayer = true;
|
2018-04-06 14:41:30 -04:00
|
|
|
clearRaster();
|
2018-04-05 17:37:40 -04:00
|
|
|
return rasterLayer;
|
|
|
|
};
|
|
|
|
|
2018-04-07 01:59:52 -04:00
|
|
|
const _makeBackgroundPaper = function (width, height, color) {
|
2018-04-07 02:42:32 -04:00
|
|
|
// creates a checkerboard path of width * height squares in color on white
|
2018-04-07 02:20:59 -04:00
|
|
|
let x = 0;
|
|
|
|
let y = 0;
|
2018-04-07 02:25:33 -04:00
|
|
|
const pathPoints = [];
|
2018-04-07 01:59:52 -04:00
|
|
|
while (x < width) {
|
2018-04-07 02:20:59 -04:00
|
|
|
pathPoints.push(new paper.Point(x, y));
|
2018-04-07 01:59:52 -04:00
|
|
|
x++;
|
2018-04-07 02:20:59 -04:00
|
|
|
pathPoints.push(new paper.Point(x, y));
|
|
|
|
y = y === 0 ? height : 0;
|
2018-04-07 01:59:52 -04:00
|
|
|
}
|
|
|
|
y = height - 1;
|
|
|
|
x = width;
|
|
|
|
while (y > 0) {
|
2018-04-07 02:20:59 -04:00
|
|
|
pathPoints.push(new paper.Point(x, y));
|
|
|
|
x = (x === 0 ? width : 0);
|
|
|
|
pathPoints.push(new paper.Point(x, y));
|
2018-04-07 01:59:52 -04:00
|
|
|
y--;
|
|
|
|
}
|
2018-04-07 02:20:59 -04:00
|
|
|
const vRect = new paper.Shape.Rectangle(new paper.Point(0, 0), new paper.Point(120, 90));
|
2018-04-07 01:59:52 -04:00
|
|
|
vRect.fillColor = '#fff';
|
|
|
|
vRect.guide = true;
|
|
|
|
vRect.locked = true;
|
2018-04-07 02:20:59 -04:00
|
|
|
const vPath = new paper.Path(pathPoints);
|
2018-04-07 01:59:52 -04:00
|
|
|
vPath.fillRule = 'evenodd';
|
|
|
|
vPath.fillColor = color;
|
|
|
|
vPath.guide = true;
|
|
|
|
vPath.locked = true;
|
|
|
|
const vGroup = new paper.Group([vRect, vPath]);
|
|
|
|
return vGroup;
|
2018-04-07 02:20:59 -04:00
|
|
|
};
|
2018-04-07 01:59:52 -04:00
|
|
|
|
2020-02-06 18:11:10 -05:00
|
|
|
// Helper function for drawing a crosshair
|
|
|
|
const _makeCrosshair = function (opacity, parent) {
|
|
|
|
paper.project.importSVG(costumeAnchorIcon, {
|
|
|
|
applyMatrix: false,
|
|
|
|
onLoad: function (item) {
|
|
|
|
item.position = new paper.Point(ART_BOARD_WIDTH / 2, ART_BOARD_HEIGHT / 2);
|
|
|
|
item.opacity = opacity;
|
|
|
|
item.parent = parent;
|
|
|
|
parent.dragCrosshair = item;
|
|
|
|
item.scale(CROSSHAIR_SIZE / item.bounds.width / paper.view.zoom);
|
|
|
|
setGuideItem(item);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const _makeDragCrosshairLayer = function () {
|
|
|
|
const dragCrosshairLayer = new paper.Layer();
|
|
|
|
_makeCrosshair(1, dragCrosshairLayer);
|
|
|
|
dragCrosshairLayer.data.isDragCrosshairLayer = true;
|
|
|
|
dragCrosshairLayer.visible = false;
|
|
|
|
return dragCrosshairLayer;
|
|
|
|
};
|
|
|
|
|
2019-05-29 15:24:07 -04:00
|
|
|
const _makeBackgroundGuideLayer = function () {
|
|
|
|
const guideLayer = new paper.Layer();
|
|
|
|
guideLayer.locked = true;
|
|
|
|
|
|
|
|
const vBackground = _makeBackgroundPaper(120, 90, '#E5E5E5');
|
|
|
|
vBackground.position = new paper.Point(ART_BOARD_WIDTH / 2, ART_BOARD_HEIGHT / 2);
|
|
|
|
vBackground.scaling = new paper.Point(8, 8);
|
|
|
|
vBackground.guide = true;
|
|
|
|
vBackground.locked = true;
|
|
|
|
|
2020-02-06 18:11:10 -05:00
|
|
|
_makeCrosshair(0.25, guideLayer);
|
2017-10-24 13:21:57 -04:00
|
|
|
|
|
|
|
guideLayer.data.isBackgroundGuideLayer = true;
|
|
|
|
return guideLayer;
|
|
|
|
};
|
|
|
|
|
|
|
|
const setupLayers = function () {
|
2017-11-02 17:19:42 -04:00
|
|
|
const backgroundGuideLayer = _makeBackgroundGuideLayer();
|
2018-04-05 17:37:40 -04:00
|
|
|
_makeRasterLayer();
|
2017-11-02 17:19:42 -04:00
|
|
|
const paintLayer = _makePaintingLayer();
|
2020-02-06 18:11:10 -05:00
|
|
|
const dragCrosshairLayer = _makeDragCrosshairLayer();
|
2017-11-02 17:19:42 -04:00
|
|
|
const guideLayer = _makeGuideLayer();
|
|
|
|
backgroundGuideLayer.sendToBack();
|
2020-02-06 18:11:10 -05:00
|
|
|
dragCrosshairLayer.bringToFront();
|
2017-11-02 17:19:42 -04:00
|
|
|
guideLayer.bringToFront();
|
|
|
|
paintLayer.activate();
|
2017-10-24 13:21:57 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
export {
|
2020-02-06 18:11:10 -05:00
|
|
|
CROSSHAIR_SIZE,
|
2018-06-21 10:22:24 -04:00
|
|
|
createCanvas,
|
2017-11-02 17:19:42 -04:00
|
|
|
hideGuideLayers,
|
|
|
|
showGuideLayers,
|
2020-02-06 18:11:10 -05:00
|
|
|
getDragCrosshairLayer,
|
2017-10-24 13:21:57 -04:00
|
|
|
getGuideLayer,
|
2018-08-08 18:27:01 -04:00
|
|
|
getBackgroundGuideLayer,
|
2018-04-06 14:41:30 -04:00
|
|
|
clearRaster,
|
2018-04-05 17:37:40 -04:00
|
|
|
getRaster,
|
2020-02-06 18:11:10 -05:00
|
|
|
setGuideItem,
|
2017-10-24 13:21:57 -04:00
|
|
|
setupLayers
|
|
|
|
};
|