function motif(options = {}) {
let {
type,
scale,
size,
angle,
fill,
stroke,
strokeWidth,
dash,
background,
border,
patchSize,
custom
} = options;
type = type ?? "line";
scale = scale ?? 1;
size = size ?? 20;
angle = angle ?? 0;
fill = fill ?? "black";
stroke = stroke ?? "transparent";
strokeWidth = strokeWidth ?? 0;
dash = dash ?? [];
background = background ?? "white";
border = border ?? 0;
patchSize = patchSize ?? false;
const tile = scale * 10;
const getShapeArea = () => (size / 100) * (tile * tile);
let externalContext;
const dpi = devicePixelRatio || 2;
const shapePath = new Map([
["line", line],
["plaid", plaid],
["circle", circle],
["plus", plus],
["cross", plus],
["triangle", triangle],
["square", square],
["diamond", square],
["custom", custom?.shape]
]);
const tippingPoint = new Map([
["line", 50],
["plaid", 50],
["circle", 78],
["triangle", 43],
["plus", 43],
["cross", 55],
["square", 50],
["diamond", 50],
["custom", custom?.patch ?? 50]
]);
if (patchSize && size > tippingPoint.get(type)) {
const back = background;
background = fill;
fill = back;
size = 100 - size;
}
const pathFunction = shapePath.get(type);
const path =
custom?.shape && custom?.shape.length === 3
? pathFunction(tile, getShapeArea(), d3.path())
: pathFunction(tile, getShapeArea());
const id = `${type}-${
Math.round(performance.now()) + Math.random().toFixed(3)
}`;
const rotateShape = ["cross", "diamond"].includes(type) ? 45 : 0;
const pattern = svg`<pattern id="${id}"
width="${tile}" height="${tile}"
viewBox="${-tile / 2} ${-tile / 2} ${tile} ${tile}"
patternUnits="userSpaceOnUse"
patternTransform="rotate(${angle})">
<rect fill="${background}"
x=${-tile / 2} y=${-tile / 2}
width="${tile}" height="${tile}"/>
<path d="${path}"
transform="rotate(${rotateShape})"
fill="${fill}" stroke="${stroke}" strokeWidth="${strokeWidth}"/>
</pattern>`;
const defs = svg`<defs>${pattern}</defs>`;
const url = `url(#${id})`;
function get_canvas_with_base_tile() {
let canvas;
const tile_dpi = tile * dpi;
if (typeof window.OffscreenCanvas !== "undefined") {
canvas = new window.OffscreenCanvas(tile_dpi, tile_dpi);
} else {
canvas = document.createElement("canvas");
canvas.width = canvas.height = tile_dpi;
}
const ctx = canvas.getContext("2d");
ctx.fillStyle = background;
ctx.strokeStyle = border === 0 ? "transparent" : "black";
ctx.lineWidth = border;
ctx.rect(0, 0, tile_dpi, tile_dpi);
ctx.fill();
ctx.stroke();
ctx.beginPath();
ctx.translate(tile_dpi / 2, tile_dpi / 2);
ctx.scale(dpi, dpi);
ctx.rotate(degreeToRadian(rotateShape));
ctx.strokeStyle = strokeWidth === 0 ? "transparent" : stroke;
ctx.setLineDash(dash);
ctx.lineWidth = strokeWidth;
ctx.lineCap = "round";
ctx.lineJoin = "round";
ctx.fillStyle = fill;
const path2D = new Path2D(path);
ctx.stroke(path2D);
ctx.fill(path2D);
ctx.resetTransform();
return canvas;
}
function context(ctx) {
const canvasBaseTile = get_canvas_with_base_tile();
const pattern = ctx.createPattern(canvasBaseTile, "repeat");
externalContext = ctx;
pattern.apply = apply;
pattern.defs = defs;
pattern.pattern = pattern;
pattern.url = url;
return pattern;
}
function apply() {
const angleRadian = degreeToRadian(angle);
externalContext.scale(1 / dpi, 1 / dpi);
externalContext.rotate(angleRadian);
externalContext.fill();
externalContext.resetTransform();
externalContext.scale(dpi, dpi);
}
return { defs, pattern, url, context, apply };
}