143 lines
4.9 KiB
JavaScript
143 lines
4.9 KiB
JavaScript
const twgl = require('twgl.js');
|
|
|
|
const Silhouette = require('./Silhouette');
|
|
const Skin = require('./Skin');
|
|
const SvgRenderer = require('scratch-svg-renderer');
|
|
|
|
const MAX_TEXTURE_DIMENSION = 2048;
|
|
|
|
class SVGSkin extends Skin {
|
|
/**
|
|
* Create a new SVG skin.
|
|
* @param {!int} id - The ID for this Skin.
|
|
* @param {!RenderWebGL} renderer - The renderer which will use this skin.
|
|
* @constructor
|
|
* @extends Skin
|
|
*/
|
|
constructor (id, renderer) {
|
|
super(id);
|
|
|
|
/** @type {RenderWebGL} */
|
|
this._renderer = renderer;
|
|
|
|
/** @type {SvgRenderer} */
|
|
this._svgRenderer = new SvgRenderer();
|
|
|
|
/** @type {WebGLTexture} */
|
|
this._texture = null;
|
|
|
|
/** @type {number} */
|
|
this._textureScale = 1;
|
|
|
|
/** @type {Number} */
|
|
this._maxTextureScale = 0;
|
|
|
|
this._silhouette = new Silhouette();
|
|
}
|
|
|
|
/**
|
|
* Dispose of this object. Do not use it after calling this method.
|
|
*/
|
|
dispose () {
|
|
if (this._texture) {
|
|
this._renderer.gl.deleteTexture(this._texture);
|
|
this._texture = null;
|
|
}
|
|
super.dispose();
|
|
}
|
|
|
|
/**
|
|
* @return {Array<number>} the natural size, in Scratch units, of this skin.
|
|
*/
|
|
get size () {
|
|
return this._svgRenderer.size;
|
|
}
|
|
|
|
/**
|
|
* Set the origin, in object space, about which this Skin should rotate.
|
|
* @param {number} x - The x coordinate of the new rotation center.
|
|
* @param {number} y - The y coordinate of the new rotation center.
|
|
*/
|
|
setRotationCenter (x, y) {
|
|
const viewOffset = this._svgRenderer.viewOffset;
|
|
super.setRotationCenter(x - viewOffset[0], y - viewOffset[1]);
|
|
}
|
|
|
|
/**
|
|
* @param {Array<number>} scale - The scaling factors to be used, each in the [0,100] range.
|
|
* @return {WebGLTexture} The GL texture representation of this skin when drawing at the given scale.
|
|
*/
|
|
// eslint-disable-next-line no-unused-vars
|
|
getTexture (scale) {
|
|
// The texture only ever gets uniform scale. Take the larger of the two axes.
|
|
const scaleMax = scale ? Math.max(Math.abs(scale[0]), Math.abs(scale[1])) : 100;
|
|
const requestedScale = Math.min(scaleMax / 100, this._maxTextureScale);
|
|
let newScale = this._textureScale;
|
|
while ((newScale < this._maxTextureScale) && (requestedScale >= 1.5 * newScale)) {
|
|
newScale *= 2;
|
|
}
|
|
if (this._textureScale !== newScale) {
|
|
this._textureScale = newScale;
|
|
this._svgRenderer._draw(this._textureScale, () => {
|
|
if (this._textureScale === newScale) {
|
|
const gl = this._renderer.gl;
|
|
gl.bindTexture(gl.TEXTURE_2D, this._texture);
|
|
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this._svgRenderer.canvas);
|
|
}
|
|
});
|
|
}
|
|
|
|
return this._texture;
|
|
}
|
|
|
|
/**
|
|
* Set the contents of this skin to a snapshot of the provided SVG data.
|
|
* @param {string} svgData - new SVG to use.
|
|
* @param {Array<number>} [rotationCenter] - Optional rotation center for the SVG. If not supplied, it will be
|
|
* calculated from the bounding box
|
|
* @fires Skin.event:WasAltered
|
|
*/
|
|
setSVG (svgData, rotationCenter) {
|
|
this._svgRenderer.fromString(svgData, 1, () => {
|
|
const gl = this._renderer.gl;
|
|
this._textureScale = this._maxTextureScale = 1;
|
|
if (this._texture) {
|
|
gl.bindTexture(gl.TEXTURE_2D, this._texture);
|
|
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, this._svgRenderer.canvas);
|
|
this._silhouette.update(this._svgRenderer.canvas);
|
|
} else {
|
|
// TODO: mipmaps?
|
|
const textureOptions = {
|
|
auto: true,
|
|
wrap: gl.CLAMP_TO_EDGE,
|
|
src: this._svgRenderer.canvas
|
|
};
|
|
|
|
this._texture = twgl.createTexture(gl, textureOptions);
|
|
this._silhouette.update(this._svgRenderer.canvas);
|
|
}
|
|
|
|
const maxDimension = Math.max(this._svgRenderer.canvas.width, this._svgRenderer.canvas.height);
|
|
let testScale = 2;
|
|
for (testScale; maxDimension * testScale <= MAX_TEXTURE_DIMENSION; testScale *= 2) {
|
|
this._maxTextureScale = testScale;
|
|
}
|
|
|
|
if (typeof rotationCenter === 'undefined') rotationCenter = this.calculateRotationCenter();
|
|
this.setRotationCenter.apply(this, rotationCenter);
|
|
this.emit(Skin.Events.WasAltered);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Does this point touch an opaque or translucent point on this skin?
|
|
* @param {twgl.v3} vec A texture coordinate.
|
|
* @return {boolean} Did it touch?
|
|
*/
|
|
isTouching (vec) {
|
|
return this._silhouette.isTouching(vec);
|
|
}
|
|
}
|
|
|
|
module.exports = SVGSkin;
|