pointvec/scripts/manager.js
2021-09-24 18:53:18 +01:00

190 lines
6.5 KiB
JavaScript

'use strict'
/* global XMLSerializer */
/* global btoa */
/* global Image */
/* global Blob */
function Manager (client) {
// Create SVG parts
this.el = document.createElementNS('http://www.w3.org/2000/svg', 'svg')
this.el.setAttribute('xmlns', 'http://www.w3.org/2000/svg')
this.el.setAttribute('baseProfile', 'full')
this.el.setAttribute('version', '1.1')
this.el.style.fill = 'none'
this.masks = []
this.layers = []
this.install = function () {
this.masks[0] = [this.maskBg()]
this.masks[1] = [this.maskBg(), document.createElementNS('http://www.w3.org/2000/svg', 'path')]
this.masks[2] = [
this.maskBg(),
document.createElementNS('http://www.w3.org/2000/svg', 'path'),
document.createElementNS('http://www.w3.org/2000/svg', 'path')
]
const defs = document.createElementNS('http://www.w3.org/2000/svg', 'defs')
this.el.appendChild(defs)
for (const id of [2, 1, 0]) {
const maskPath = document.createElementNS('http://www.w3.org/2000/svg', 'mask')
maskPath.setAttribute('id', 'mask' + id)
for (const mask of this.masks[id]) {
maskPath.appendChild(mask)
}
defs.appendChild(maskPath)
}
this.el.appendChild(this.layers[2] = document.createElementNS('http://www.w3.org/2000/svg', 'path'))
this.el.appendChild(this.layers[1] = document.createElementNS('http://www.w3.org/2000/svg', 'path'))
this.el.appendChild(this.layers[0] = document.createElementNS('http://www.w3.org/2000/svg', 'path'))
}
this.maskBg = function () {
const bg = document.createElementNS('http://www.w3.org/2000/svg', 'rect')
bg.setAttribute('fill', 'white')
bg.setAttribute('width', '100%')
bg.setAttribute('height', '100%')
return bg
}
this.update = function () {
this.el.setAttribute('width', (client.tool.settings.size.width) + 'px')
this.el.setAttribute('height', (client.tool.settings.size.height) + 'px')
this.el.style.width = client.tool.settings.size.width
this.el.style.height = client.tool.settings.size.height
const styles = client.tool.styles
const paths = client.tool.paths()
const masked = styles.map((style) => style.mask)
const viewBox = this.minimalViewBox();
if (viewBox !== null) {
for (const id in this.masks) {
const [x, y, width, height] = viewBox
this.masks[id][0].setAttribute('x', x)
this.masks[id][0].setAttribute('y', y)
this.masks[id][0].setAttribute('width', width)
this.masks[id][0].setAttribute('height', height)
}
}
for (const id in this.layers) {
const style = styles[id]
const path = paths[id]
const layer = this.layers[id]
const nextLayer = parseInt(id) + 1
for (var i = nextLayer; i < 3; i++) {
const mask = this.masks[i][nextLayer]
if (style.mask) {
mask.setAttribute('d', path)
mask.style.strokeWidth = style.thickness
mask.style.strokeLinecap = style.strokeLinecap
mask.style.strokeLinejoin = style.strokeLinejoin
mask.style.stroke = style.color
mask.style.fill = style.fill
} else {
mask.setAttribute('d', '')
}
}
layer.style.strokeWidth = style.thickness
layer.style.strokeLinecap = style.strokeLinecap
layer.style.strokeLinejoin = style.strokeLinejoin
layer.style.stroke = style.color
layer.style.fill = style.fill
layer.setAttribute('d', style.mask ? '' : path)
const maskedAbove = masked.slice(0, id).some((mask) => mask === true)
const maskPath = !(style.mask) && maskedAbove ? 'url(#mask' + id + ')' : null
if (maskPath !== null) {
layer.setAttribute('mask', maskPath)
} else if (layer.hasAttribute('mask')) {
layer.removeAttribute('mask')
}
}
}
this.minimalViewBox = function () {
const styles = client.tool.styles
const paths = client.tool.paths()
var xMin = null
var yMin = null
var xMax = null
var yMax = null
for (const id in this.layers) {
const style = styles[id]
const canOvershoot = style.strokeLinejoin === 'miter' || style.strokeLinecap === 'square'
const offset = canOvershoot ? style.thickness : style.thickness / 2
const path = paths[id]
const matches = path.match(/\d+,\d+/g)
if (matches === null) { continue }
const coordinates = matches.map(p => p.split(/,/))
const xs = coordinates.map(p => p[0]);
const ys = coordinates.map(p => p[1]);
const xMinOfLayer = Math.min(...xs) - offset
const yMinOfLayer = Math.min(...ys) - offset
const xMaxOfLayer = Math.max(...xs) + offset
const yMaxOfLayer = Math.max(...ys) + offset
if (xMin === null || xMinOfLayer < xMin) { xMin = xMinOfLayer }
if (yMin === null || yMinOfLayer < yMin) { yMin = yMinOfLayer }
if (xMax === null || xMaxOfLayer > xMax) { xMax = xMaxOfLayer }
if (yMax === null || yMaxOfLayer > yMax) { yMax = yMaxOfLayer }
}
if (xMin === null || yMin === null || xMax === null || yMax === null) { return null }
return [xMin, yMin, xMax - xMin, yMax - yMin]
}
this.svg64 = function () {
const xml = new XMLSerializer().serializeToString(this.el)
const svg64 = btoa(xml)
const b64Start = 'data:image/svg+xml;base64,'
return b64Start + svg64
}
// Exporters
this.toPNG = function (size = client.tool.settings.size, callback) {
this.update()
const image64 = this.svg64()
const img = new Image()
const canvas = document.createElement('canvas')
canvas.width = (size.width) * 2
canvas.height = (size.height) * 2
img.onload = function () {
canvas.getContext('2d').drawImage(img, 0, 0, (size.width) * 2, (size.height) * 2)
callback(canvas.toDataURL('image/png'))
}
img.src = image64
}
this.toString = () => {
const viewBox = this.minimalViewBox();
if (viewBox !== null) {
this.el.setAttribute('width', viewBox[2] + 'px')
this.el.removeAttribute('height')
this.el.style.width = null
this.el.style.height = null
this.el.setAttribute('viewBox', viewBox.join(' '))
}
const serialized = new XMLSerializer().serializeToString(this.el)
if (viewBox !== null) {
this.el.setAttribute('width', (client.tool.settings.size.width) + 'px')
this.el.setAttribute('height', (client.tool.settings.size.height) + 'px')
this.el.style.width = client.tool.settings.size.width
this.el.style.height = client.tool.settings.size.height
this.el.removeAttribute('viewBox')
}
return serialized
}
}