UNPKG

vue-data-ui

Version:

A user-empowering data visualization Vue 3 components library for eloquent data storytelling

374 lines (373 loc) 17.3 kB
import { ref as c, computed as ge, watch as X, nextTick as le, onMounted as pe, onBeforeUnmount as me, createElementBlock as Y, createCommentVNode as q, openBlock as z, normalizeStyle as x, createElementVNode as k, withDirectives as ae, createVNode as R, normalizeClass as G, toDisplayString as Ae, unref as he, vModelText as oe } from "vue"; import T from "./BaseIcon-Ba5t14Aj.js"; import { C as we } from "./ColorPicker-8-sXft1r.js"; import { l as be, f as xe } from "./index-BLtEpj8j.js"; const ye = { class: "vue-ui-pen-and-paper-action", style: { padding: "0 !important" } }, Ce = ["disabled"], Be = { __name: "PenAndPaper", props: { svgRef: { type: [Object, null, void 0], required: !0 }, color: { type: String, default: "#2D353C" }, backgroundColor: { type: String, default: "#FFFFFF" }, active: { type: Boolean, default: !1 }, scale: { type: Number, default: 1 } }, emits: ["close"], setup(d, { emit: se }) { const s = d, ue = se, b = c([]), E = c([]), S = c(s.color), $ = c(2), M = c(!1), P = c(""), o = c(null), A = c(null), F = c(null), h = c("draw"), Q = c(!1), f = c(null), H = c({ x: 0, y: 0 }), N = c([""]), D = c({ row: 0, col: 0 }), L = c(16), W = c("url('') 5 5, auto"); function O(l) { if (!o.value || h.value !== "text" || Q.value) return; const { x: e, y: n } = V(l); H.value = { x: e, y: n }, N.value = [""], D.value = { row: 0, col: 0 }; const t = document.createElementNS("http://www.w3.org/2000/svg", "text"); t.setAttribute("x", e), t.setAttribute("y", n), t.setAttribute("fill", S.value), t.setAttribute("font-size", L.value * s.scale), t.setAttribute("font-family", "sans-serif"), t.setAttribute("class", "vue-data-ui-doodle"), t.setAttribute("dominant-baseline", "hanging"), t.setAttribute("pointer-events", "all"); const a = document.createElementNS("http://www.w3.org/2000/svg", "tspan"); a.setAttribute("x", e), a.setAttribute("dy", "0"), a.textContent = "", t.appendChild(a), t.style.pointerEvents = "none", t.style.userSelect = "none", o.value.appendChild(t), f.value = t, Q.value = !0, window.addEventListener("keydown", j), window.addEventListener("mousedown", _, !0), J(), Z(); } function j(l) { if (!Q.value) return; let { row: e, col: n } = D.value, t = N.value.slice(), a = !1; if (l.key === "Enter") { const u = t[e], r = u.slice(0, n), v = u.slice(n); t.splice(e, 1, r, v), e += 1, n = 0, a = !0, l.preventDefault(); } else if (l.key === "Backspace") { if (n > 0) t[e] = t[e].slice(0, n - 1) + t[e].slice(n), n -= 1, a = !0; else if (e > 0) { const u = t[e - 1].length; t[e - 1] += t[e], t.splice(e, 1), e -= 1, n = u, a = !0; } l.preventDefault(); } else if (l.key === "Delete") n < t[e].length ? (t[e] = t[e].slice(0, n) + t[e].slice(n + 1), a = !0) : e < t.length - 1 && (t[e] += t[e + 1], t.splice(e + 1, 1), a = !0), l.preventDefault(); else if (l.key === "ArrowLeft") n > 0 ? n -= 1 : e > 0 && (e -= 1, n = t[e].length), a = !0, l.preventDefault(); else if (l.key === "ArrowRight") n < t[e].length ? n += 1 : e < t.length - 1 && (e += 1, n = 0), a = !0, l.preventDefault(); else if (l.key === "ArrowUp") e > 0 && (e -= 1, n = Math.min(n, t[e].length), a = !0), l.preventDefault(); else if (l.key === "ArrowDown") e < t.length - 1 && (e += 1, n = Math.min(n, t[e].length), a = !0), l.preventDefault(); else if (l.key.length === 1 && !l.ctrlKey && !l.metaKey && !l.altKey) t[e] = t[e].slice(0, n) + l.key + t[e].slice(n), n += 1, a = !0, l.preventDefault(); else if (l.key === "Escape") { ee(!0); return; } else l.key === "Tab" && l.preventDefault(); a && (N.value = t, D.value = { row: e, col: n }, J(), Z()); } function J() { const l = f.value, { x: e } = H.value; for (; l.firstChild; ) l.removeChild(l.firstChild); N.value.forEach((n, t) => { const a = document.createElementNS("http://www.w3.org/2000/svg", "tspan"); a.setAttribute("x", e), a.setAttribute("dy", t === 0 ? "0" : `${L.value * 1.2 * s.scale}`), a.textContent = n.length ? n : "​", l.appendChild(a); }); } function Z() { const l = o.value.querySelector(".vue-data-ui-svg-caret"); l && o.value.removeChild(l); const e = f.value; if (!e) return; const { x: n, y: t } = H.value, { row: a, col: u } = D.value, r = L.value * s.scale, v = e.childNodes[a]; if (!v) return; let g = v.textContent.slice(0, u); g.endsWith(" ") && (g += " "); const i = document.createElementNS("http://www.w3.org/2000/svg", "text"); i.setAttribute("x", n), i.setAttribute("y", t), i.setAttribute("font-size", r), i.setAttribute("font-family", "sans-serif"), i.textContent = g || "", o.value.appendChild(i); const p = i.getBBox(); o.value.removeChild(i); let m = t + a * r * 1.2, C = n + p.width; const w = document.createElementNS("http://www.w3.org/2000/svg", "rect"); w.setAttribute("x", C), w.setAttribute("y", m), w.setAttribute("width", 2), w.setAttribute("height", r), w.setAttribute("fill", S.value), w.setAttribute("class", "vue-data-ui-svg-caret"), o.value.appendChild(w); } function _(l) { if (f.value && !f.value.contains(l.target)) { const e = f.value.children; e.length === 1 && (e[0].textContent === "" || e[0].textContent === "​") && f.value.remove(), ee(!1); } } function ee(l = !1) { window.removeEventListener("keydown", j), window.removeEventListener("mousedown", _, !0); const e = o.value.querySelector(".vue-data-ui-svg-caret"); e && o.value.removeChild(e); const n = f.value?.children; let t = !1; if (n && n.length === 1) { const a = n[0].textContent; t = !a || a === "​"; } l || t ? f.value && o.value.contains(f.value) && o.value.removeChild(f.value) : f.value && o.value.contains(f.value) && b.value.push(f.value), Q.value = !1, f.value = null, N.value = [""], D.value = { row: 0, col: 0 }; } const y = ge(() => be(s.color, 0.6)); function te() { if (!o.value) return; const l = o.value.querySelector(".vue-data-ui-mask"); if (l && o.value.removeChild(l), s.active) { const e = document.createElementNS("http://www.w3.org/2000/svg", "rect"); e.setAttribute("class", "vue-data-ui-mask"), e.setAttribute("width", "100%"), e.setAttribute("height", "100%"), e.setAttribute("fill", "transparent"), e.setAttribute("pointer-events", "all"), o.value.insertBefore(e, o.value.firstChild); } } function V(l) { const e = s.svgRef; if (!e) return { x: 0, y: 0 }; const n = e.createSVGPoint(); n.x = l.clientX, n.y = l.clientY; const t = e.getScreenCTM()?.inverse(); return t ? n.matrixTransform(t) : { x: 0, y: 0 }; } function re(l) { const e = l.trim().split(/\s+/); if (e.length < 4) return l; const n = e.slice(1).map(Number); if (n.length % 2 !== 0) return l; const t = ie(n), a = [`M ${t[0]} ${t[1]}`]; for (let v = 2; v < t.length - 2; v += 2) { const g = t[v - 2], i = t[v - 1], p = t[v], m = t[v + 1], C = (g + p) / 2, w = (i + m) / 2; a.push(`Q ${g} ${i} ${C} ${w}`); } const u = t[t.length - 2], r = t[t.length - 1]; return a.push(`L ${u} ${r}`), a.join(" "); } function ie(l, e = 1) { const n = [...l]; for (let t = 2; t < l.length - 2; t += 2) { const a = l[t], u = l[t + 1], r = l[t - 2], v = l[t - 1], g = l[t + 2], i = l[t + 3]; n[t] = a + e * ((r + g) / 2 - a), n[t + 1] = u + e * ((v + i) / 2 - u); } return n; } function ve(l) { const e = l.trim().split(/\s+/); let n = "", t = "", a = null, u = null; for (let r = 0; r < e.length; r += 1) { const v = e[r]; if (isNaN(v)) { if (t = v, t === "M" || t === "L") a = parseFloat(e[++r]), u = parseFloat(e[++r]), n += `${t}${a} ${u}`; else if (t === "Q") { const g = parseFloat(e[++r]), i = parseFloat(e[++r]), p = parseFloat(e[++r]), m = parseFloat(e[++r]); g === a && i === u ? n += `t${p - a} ${m - u}` : n += `q${g - a} ${i - u} ${p - a} ${m - u}`, a = p, u = m; } } else { const g = parseFloat(v), i = parseFloat(e[++r]); if (t === "L") { const p = g - a, m = i - u; p === 0 ? n += `v${m}` : m === 0 ? n += `h${p}` : n += `l${p} ${m}`, a = g, u = i; } else if (t === "Q") { const p = g, m = i, C = parseFloat(e[++r]), w = parseFloat(e[++r]); p === a && m === u ? n += `t${C - a} ${w - u}` : n += `q${p - a} ${m - u} ${C - a} ${w - u}`, a = C, u = w; } } } return n; } function I(l) { if (h.value !== "draw" || !s.active || !o.value) return; M.value = !0; const { x: e, y: n } = V(l); F.value = { x: e, y: n }, P.value = `M ${e} ${n}`, A.value = document.createElementNS("http://www.w3.org/2000/svg", "path"), A.value.setAttribute("stroke", S.value), A.value.setAttribute("stroke-width", $.value * s.scale), A.value.setAttribute("fill", "none"), A.value.setAttribute("stroke-linecap", "round"), A.value.setAttribute("stroke-linejoin", "round"), A.value.setAttribute("class", "vue-data-ui-doodle"), o.value.appendChild(A.value); } function K(l) { if (!M.value || !o.value || !A.value) return; const { x: e, y: n } = V(l); P.value += ` ${e} ${n}`, A.value.setAttribute("d", P.value); } function B(l) { if (M.value && o.value && A.value) { const { x: e, y: n } = V(l); if (F.value && F.value.x === e && F.value.y === n) { const t = document.createElementNS("http://www.w3.org/2000/svg", "circle"); t.setAttribute("cx", e), t.setAttribute("cy", n), t.setAttribute("r", $.value * s.scale / 2), t.setAttribute("fill", S.value), t.setAttribute("class", "vue-data-ui-doodle"), o.value.appendChild(t), b.value.push(t); } else { const t = A.value; t.setAttribute("d", ve(re(P.value))), b.value.push(t); } E.value = [], A.value = ""; } M.value = !1; } function ce() { if (b.value.length > 0) { const l = b.value.pop(); E.value.push(l), o.value && o.value.removeChild(l); } } function de() { if (E.value.length > 0) { const l = E.value.pop(); b.value.push(l), o.value && o.value.appendChild(l); } } function fe() { o.value && (o.value.innerHTML = ""), b.value = [], E.value = [], te(); } X(h, () => { s.active && (U(), ne(), h.value === "text" ? o.value.style.cursor = "text" : o.value.style.cursor = W.value); }); function ne() { !s.svgRef || !s.active || (h.value === "draw" ? (s.svgRef.addEventListener("mousedown", I), s.svgRef.addEventListener("mousemove", K), s.svgRef.addEventListener("mouseup", B), s.svgRef.addEventListener("mouseleave", B), s.svgRef.addEventListener("touchstart", I, { passive: !1 }), s.svgRef.addEventListener("touchmove", K, { passive: !1 }), s.svgRef.addEventListener("touchend", B)) : h.value === "text" && s.svgRef.addEventListener("mousedown", O), o.value && (o.value.style.pointerEvents = "auto")); } function U() { s.svgRef && (s.svgRef.removeEventListener("mousedown", I), s.svgRef.removeEventListener("mousemove", K), s.svgRef.removeEventListener("mouseup", B), s.svgRef.removeEventListener("mouseleave", B), s.svgRef.removeEventListener("touchstart", I), s.svgRef.removeEventListener("touchmove", K), s.svgRef.removeEventListener("touchend", B), s.svgRef.removeEventListener("mousedown", O), o.value && (o.value.style.pointerEvents = "none")); } return X(() => s.active, (l) => { l ? ne() : U(); }), X(() => s.active, () => { le(() => { te(); }); }), pe(() => { le(() => { s.svgRef && (o.value = document.createElementNS("http://www.w3.org/2000/svg", "g"), o.value.setAttribute("class", "vue-data-ui-doodles"), o.value.style.cursor = W.value, s.svgRef.appendChild(o.value), U()); }); }), me(() => { o.value && s.svgRef && (s.svgRef.removeChild(o.value), U()); }), (l, e) => d.active ? (z(), Y("div", { key: 0, "data-dom-to-png-ignore": "", class: "vue-ui-pen-and-paper-actions", style: x({ backgroundColor: d.backgroundColor }) }, [ k("button", { class: "vue-ui-pen-and-paper-action", onClick: e[0] || (e[0] = (n) => ue("close")), style: x({ backgroundColor: d.backgroundColor, border: `1px solid ${y.value}` }) }, [ R(T, { name: "close", stroke: d.color }, null, 8, ["stroke"]) ], 4), k("button", ye, [ R(we, { value: S.value, "onUpdate:value": e[1] || (e[1] = (n) => S.value = n), backgroundColor: d.backgroundColor, buttonBorderColor: y.value }, null, 8, ["value", "backgroundColor", "buttonBorderColor"]) ]), k("button", { class: G(["vue-ui-pen-and-paper-action", { "vue-ui-pen-and-paper-action-active": h.value === "text" }]), onClick: e[2] || (e[2] = (n) => h.value = h.value === "text" ? "draw" : "text"), style: x({ backgroundColor: d.backgroundColor, border: `1px solid ${y.value}` }) }, [ R(T, { name: h.value === "draw" ? "annotator" : "text", stroke: d.color }, null, 8, ["name", "stroke"]), k("div", { style: x({ position: "absolute", bottom: "-20px", color: y.value, width: "100%", textAlign: "center", fontSize: "12px", fontVariantNumeric: "tabular-nums" }) }, Ae(he(xe)({ v: h.value === "draw" ? $.value : L.value, s: "px", r: 1 })), 5) ], 6), k("button", { class: G(["vue-ui-pen-and-paper-action", { "vue-ui-pen-and-paper-action-disabled": !b.value.length }]), disabled: !b.value.length, onClick: ce, style: x({ backgroundColor: d.backgroundColor, border: `1px solid ${y.value}`, marginTop: "20px" }) }, [ R(T, { name: "restart", stroke: d.color }, null, 8, ["stroke"]) ], 14, Ce), k("button", { class: G(["vue-ui-pen-and-paper-action", { "vue-ui-pen-and-paper-action-disabled": !E.value.length }]), onClick: de, style: x({ backgroundColor: d.backgroundColor, border: `1px solid ${y.value}` }) }, [ R(T, { name: "restart", stroke: d.color, style: { transform: "scaleX(-1)" } }, null, 8, ["stroke"]) ], 6), k("button", { class: G(["vue-ui-pen-and-paper-action", { "vue-ui-pen-and-paper-action-disabled": !b.value.length }]), onClick: fe, style: x({ backgroundColor: d.backgroundColor, border: `1px solid ${y.value}` }) }, [ R(T, { name: "trash", stroke: d.color }, null, 8, ["stroke"]) ], 6), h.value === "draw" ? ae((z(), Y("input", { key: 0, ref: "range", type: "range", class: "vertical-range", min: 0.5, max: 12, step: 0.1, "onUpdate:modelValue": e[3] || (e[3] = (n) => $.value = n), style: x({ accentColor: d.color }) }, null, 4)), [ [oe, $.value] ]) : q("", !0), h.value === "text" ? ae((z(), Y("input", { key: 1, ref: "range", type: "range", class: "vertical-range", min: 3, max: 48, step: 0.1, "onUpdate:modelValue": e[4] || (e[4] = (n) => L.value = n), style: x({ accentColor: d.color }) }, null, 4)), [ [oe, L.value] ]) : q("", !0) ], 4)) : q("", !0); } }; export { Be as default };