import { isImage } from '../modules/core/regex.js' import { nodeOrNew, register, wrapWithAttrCheck } from '../utils/adopter.js' import { off, on } from '../modules/core/event.js' import { registerAttrHook } from '../modules/core/attr.js' import { registerMethods } from '../utils/methods.js' import { xlink } from '../modules/core/namespaces.js' import Pattern from './Pattern.js' import Shape from './Shape.js' import globals from '../utils/window.js' const { window } = globals export default class Image extends Shape { constructor (node) { super(nodeOrNew('image', node), node) } // (re)load image load (url, callback) { if (!url) return this var img = new window.Image() on(img, 'load', function (e) { var p = this.parent(Pattern) // ensure image size if (this.width() === 0 && this.height() === 0) { this.size(img.width, img.height) } if (p instanceof Pattern) { // ensure pattern size if not set if (p.width() === 0 && p.height() === 0) { p.size(this.width(), this.height()) } } if (typeof callback === 'function') { callback.call(this, { width: img.width, height: img.height, ratio: img.width / img.height, url: url }) } }, this) on(img, 'load error', function () { // dont forget to unbind memory leaking events off(img) }) return this.attr('href', (img.src = url), xlink) } } registerAttrHook(function (attr, val, _this) { // convert image fill and stroke to patterns if (attr === 'fill' || attr === 'stroke') { if (isImage.test(val)) { val = _this.doc().defs().image(val) } } if (val instanceof Image) { val = _this.doc().defs().pattern(0, 0, (pattern) => { pattern.add(val) }) } return val }) registerMethods({ Container: { // create image element, load image and set its size image: wrapWithAttrCheck(function (source, callback) { return this.put(new Image()).size(0, 0).load(source, callback) }) } }) register(Image)