cahir
Version:
flexible interface for method chaining using Proxy and tagged template literals
455 lines (451 loc) • 26.5 kB
JavaScript
import ch from "../../collections/DOM/ch.0.0.9.es.js";
const manaOrb = ((symbols) => function manaOrb({name, attrs, styles, props, data, el, proto}) {
if (!el[symbols.initialized]){
proto[symbols.initialized] = true;
proto[symbols.nuf] = [null, void(0), false];
proto[symbols.nue] = [null, void(0), ""];
proto.remove = function(){
return ch(this).animate([{
transform: "translate(0%, -100%)",
opacity: 0
}], {duration:1000, easing: "ease-in-out", fill: "both"}).lastOp.then(() => (this.parentNode.removeChild(this), this));
}
proto.filterOpacity = function(v) {
/*const shadow = this.shadowRoot,
[filter, pattern, rect] = shadow.querySelectorAll("#displacement-filter, #imagePattern, #filter-rect");*/
this.setAttribute("data-filter-opacity", v);
return this;
}
proto.color = function(f, v) {
if (f instanceof Array){
f.forEach(([f,v]) => this.color(f, v))
return this;
}
const colors = Object.fromEntries(
(ch(this).gatr("data-colors") ?? "")
.split(",")
.filter(Boolean)
.reduce((ac,d,i,a) => {
if(!(i % 2)){
ac.push([d, a[i+1]])
}
return ac;
},[]));
colors[f] = v;
return ch.satr("data-colors", `${Object.entries(colors).flat()}`).selected;
}
proto.css = function (str, pos) {
const shadow = this.shadowRoot,
style = shadow.querySelector("style[data-for=mana-orb]"),
sheet = style.sheet;
pos = pos ?? sheet.cssRules.length;
sheet.insertRule(str, pos);
return this;
}
proto.play = function() {
this.removeAttribute("data-paused"); return this;
}
proto.pause = function() {
this.setAttribute("data-paused", ""); return this;
}
proto[symbols.upAttrs] = async function({values:v, el}){
ch(el);
v.colors = el.colors = Object.fromEntries(
(ch.gatr("data-colors") ?? "")
.split(",")
.filter(Boolean)
.reduce((ac,d,i,a) => {
if(!(i % 2)){
ac.push([d, a[i+1]])
}
return ac;
},[]));
v.borderRadius = ch.gatr("data-border-radius");
v.backgroundOpacity = ch.gatr("data-background-opacity");
v.widthRatio = ch.gatr("data-width-ratio");
v.aspectRatio = ch.gatr("data-aspect-ratio");
v.padding = ch.gatr("data-padding");
v.fadein = el.fadein = (ch.gatr("data-fadein") ?? null) !== null ? 1 : 0;
v.hrefTexture = ch.gatr("data-href-texture");
v.hrefOverlay = ch.gatr("data-href-overlay");
el[symbols.nue].includes(v.overlayOpacity = ch.gatr("data-overlay-opacity"))
? v.overlayOpacity = 1
: void(0);
//animateTransform
v.animDur = ch.gatr("data-duration") || "3s";
v.animBegin = ch.gatr("data-begin") || "0s";
v.animFrom = ch.gatr("data-from") || "0,0";
v.animTo = ch.gatr("data-to") || "1,0";
v.animFill = ch.gatr("data-fill") || "freeze";
v.animRepCount = ch.gatr("data-repeat-count") || "indefinite";
v.paused = el.hasAttribute("data-paused"); //TODO: add to ch 'hatr'
//progress
v.progress = ch.gatr("data-progress") || "1";
v.progressRight = ch.gatr("data-progress-right") || "1";
//stroke
v.strokeWidth = (+ch.gatr("data-stroke-width") || 0) * 2;
//filter
v.filterOpacity = ch.gatr("data-filter-opacity") ?? 1;
el[symbols.nue].includes(v.filterScale = ch.gatr("data-filter-scale"))
? v.filterScale = 0.65
: void(0);
return v;
};
proto[symbols.upClipPath] = async function({values:v, el}){
//console.log("upClipPath", el);
el[symbols.clipG] = el[symbols.clipG] || el?.shadowRoot?.querySelector("#clip-g");
el[symbols.strokeG] = el[symbols.strokeG] || el?.shadowRoot?.querySelector("#stroke-g");
el[symbols.strokeGParent] = el[symbols.strokeGParent] || el?.shadowRoot?.querySelector("#stroke-g-parent");
if(el.firstElementChild) {
el[symbols.cusClipPath] = el[symbols.cusClipPath] || el?.shadowRoot?.querySelector("#cusClip");
if (!el[symbols.cusClipPath]){return}
ch(el[symbols.cusClipPath])`
>> innerHTML outerHTML: ${el.firstElementChild.outerHTML}
-> ${el[symbols.strokeG]}
>> innerHTML ${({values:vv}) => vv.outerHTML}
-> ${el[symbols.strokeGParent]}
satr clip-path ${"url(#cusClip)"}
-> ${el[symbols.clipG]}
satr clip-path ${"url(#cusClip)"}`
} else {
el[symbols.defClipPath] = el[symbols.defClipPath] || el?.shadowRoot?.querySelector("#defClip");
ch(el[symbols.clipG]).satr("clip-path", "url(#defClip)")
(el[symbols.strokeGParent]).satr("clip-path", "url(#defClip)")
(el[symbols.strokeG]).set("innerHTML", el[symbols.defClipPath].firstElementChild.outerHTML);
}
};
proto[symbols.upStyle] = function({values:v, el, selector, prop, val}){
//console.log("upStyle", el);
selector = selector.trim();
el[symbols.stylesheet] = el[symbols.stylesheet] || el?.shadowRoot?.styleSheets?.[0];
if (!el[symbols.stylesheet]){return proto[symbols.upStyle]}
for (const {style, selectorText} of el[symbols.stylesheet].cssRules){
if (selectorText?.trim() !== selector){continue}
if (el[symbols.nuf].includes(val)) {
style.removeProperty(prop);
} else {
style.setProperty(prop, val);
}
break;
}
return proto[symbols.upStyle];
}
proto[symbols.upStyles] = function({values:v, el, state}){
//console.log("upStyles", el, state);
el[symbols.upStyle]
({values:v, el, selector: ":host", prop: "height", val:`calc(var(--currWidth) * ${v.aspectRatio || 1} * 1px)`})
({values:v, el, selector: ":host", prop: "padding", val:`${v.padding || "var(--padding, 4px)"}`})
({values:v, el, selector: ":host", prop: "color", val:`${v.colors.font || "var(--font-color, DarkSlateGray)"}`})
({values:v, el, selector: ":host", prop: "background-color", val:`${v.colors.background || "var(--bg-color, #f9f9f9)"}`})
({values:v, el, selector: ":host", prop: "border-radius", val:`${v.borderRadius || "var(--border-radius, 4px)"}`})
({values:v, el, selector: ":host", prop: "animation", val:`${
v.fadein
? "animation: fadein-translate-y 0.4s ease-in-out 0s 1 normal forwards running;"
: false
}`})
({values:v, el, selector: ":host", prop: "--state", val:`${state}`})
({values:v, el, selector: ":host", prop: "transform", val:`translate(0px, ${(1 - state) * -100}px)`})
({values:v, el, selector: ":host", prop: "opacity", val:`${state}`})
({values:v, el, selector: "path", prop: "stroke", val:`${v.colors.stroke || "var(--stroke-color, #9098a9)"}`})
({values:v, el, selector: "path", prop: "fill", val:`${v.colors.fill || "none"}`})
};
proto[symbols.upSvg] = async function({values:v, el}){
//console.log("upSvg", el);
if(!v.svg){return}
el[symbols.pattern] = el[symbols.pattern] || v.svg.querySelector("#imagePattern");
el[symbols.animateTransform] = el[symbols.animateTransform] || v.svg.querySelector("animateTransform");
el[symbols.svgImage] = el[symbols.svgImage] || v.svg.querySelector("#source-image");
el[symbols.svgOverlay] = el[symbols.svgOverlay] || v.svg.querySelector("#source-overlay");
el[symbols.maskG] = el[symbols.maskG] || v.svg.querySelector("#mask-g");
el[symbols.maskRect] = el[symbols.maskRect] || v.svg.querySelector("#progress-rect");
el[symbols.strokeG] = el[symbols.strokeG] || el?.shadowRoot?.querySelector("#stroke-g");
el[symbols.filterRect] = el[symbols.filterRect] || el?.shadowRoot?.querySelector("#filter-rect");
el[symbols.displacementFe] = el[symbols.displacementFe] || el?.shadowRoot?.querySelector("#displacement-fe");
ch(el[symbols.animateTransform])
.satr([["dur", v.animDur], ["begin", v.animBegin], ["from", v.animFrom], ["to", v.animTo], ["fill", v.animFill], ["repeatCount", v.animRepCount]])
.exec(() => {
const
a = !!el[symbols.isPaused],
b = !!v.paused;
switch (true) {
case a && b:
case !a && !b:
break;
case !a && b:
el[symbols.isPaused] = 1;
//ch.selected.endElement(); iOS 18 bug
el[symbols.animateTransform].remove();
break;
case a && !b:
el[symbols.isPaused] = 0;
//ch.selected.beginElement(); iOS 18 bug
el[symbols.pattern].appendChild(el[symbols.animateTransform]);
}
})
(el[symbols.svgImage])
.satr("href", v.hrefTexture)
(el[symbols.svgOverlay])
.satr([["href", v.hrefOverlay || ""], ["opacity", v.overlayOpacity]])
(el[symbols.maskRect])
.satr("transform", `translate(${v.progressRight - 1}, ${1 - v.progress})`)
(el[symbols.strokeG])
.style([["stroke", `${v.colors.stroke || "var(--stroke-color, #9098a9)"}`], ["stroke-width", v.strokeWidth / 100/*v.svg.getScreenCTM().a*/]])
(el[symbols.filterRect])
.satr("opacity", v.filterOpacity)
(el[symbols.displacementFe])
.satr("scale", v.filterScale);
v.calcResize?.();
};
proto.update = async function({values, el, state}, mList, obs){
//console.log("update", el);
//prevent recursion
if (mList.every(d => d.attributeName === "style")) {
return;
}
LOOP:
for (const mutRec of mList) {
//console.log(mList);
switch (`${mutRec.type}-${mutRec?.target?.assignedSlot ? "slot" : "host"}` ) {
case "childList-host":
//console.log("childList-host");
el[symbols.upClipPath]({values, el});
break;
case "attributes-host":
//console.log("attributes-host");
await el[symbols.upAttrs]({values, el});
await el[symbols.upStyles]({values, el, state});
await el[symbols.upSvg]({values, el})
break;
case "characterData-host":
//console.log("characterData-host");
el[symbols.upClipPath]({values, el});
case "childList-slot":
//console.log("childList-slot");
el[symbols.upClipPath]({values, el});
break;
case "attributes-slot":
//console.log("attributes-slot");
el[symbols.upClipPath]({values, el});
break;
case "characterData-slot":
//console.log("characterData-slot");
el[symbols.upClipPath]({values, el});
break;
default:
continue LOOP;
}
}
}
}
const shadow = el[symbols.shadow] = el.attachShadow({ mode: "open" });
let state = 0,
toggle,
calcResize;
ch(el)`
=> ${({values:v}) => () => {
const updateDelay = +ch.gatr("data-update-delay") || 50,
resizeDelay = +ch.gatr("data-resize-delay") || 500,
toggleDelay = +ch.gatr("data-toggle-delay") || 50;
v.resizeDelay = resizeDelay;
v.toggleDelay = toggleDelay;
el[symbols.upAttrs] = ch.throttle(el[symbols.upAttrs], {delay: updateDelay});
el[symbols.upClipPath] = ch.throttle(el[symbols.upClipPath], {delay: updateDelay});
el[symbols.upStyles] = ch.throttle(el[symbols.upStyles], {delay: updateDelay});
el[symbols.upSvg] = ch.throttle(el[symbols.upSvg], {delay: updateDelay});
}}
=> ${({values}) => async () => {
return el[symbols.upAttrs]({values, el})
}}
|> await ${({values:v}) => async () => {
v.mutObs = el.mutObs ?? new MutationObserver(el.update.bind(el, {values:v, el, get state(){return state}}))
}}
|> await ${({values:v}) => async () => {
calcResize = v.calcResize = ch.throttle(function(){
const parent = el?.parentElement;
if(!parent){return}
const
scrollBarWidth = parent.offsetWidth - parent.clientWidth,
scrollBarHeight = parent.offsetHeight - parent.clientHeight,
{width: w, height: h} = el?.parentElement?.getBoundingClientRect();
ch(el).style("--currWidth", (w - scrollBarWidth) * (+v.widthRatio || 0.95))
.style("--currHeight", (h - scrollBarHeight));
}, {delay: v.resizeDelay ?? 500});
(v.robserver = new ResizeObserver(calcResize)).observe(el?.parentElement, {box: "border-box"});
}}
|> await ${({values:v}) => async () => calcResize()}
|> await ${({values:v}) => async () => {
ch(shadow)`
+> ${ch.dom`
<style data-for="mana-orb">
*,
*:after,
*:before {
box-sizing: border-box;
}
:host {
box-sizing: border-box;
width: calc(var(--currWidth) * 1px);
height: calc(var(--currWidth) * ${v.aspectRatio || 1} * 1px);
margin: auto;
padding: ${v.padding || "var(--padding, 4px)"};
overflow-y: auto;
color: ${v.colors.font || "var(--font-color, DarkSlateGray)"};
background-color: ${v.colors.background || "var(--bg-color, #f9f9f9)"};
border-radius: ${v.borderRadius || "var(--border-radius, 4px)"};
display: grid;
position: relative;
font-size: 1rem;
container: component-container;
container-type: inline-size;
grid-template-columns: repeat(2, 1fr);
grid-template-rows: repeat(2, 1fr);
grid-template-areas:
"q q"
"q q";
${
v.fadein
? "animation: fadein-translate-y 0.4s ease-in-out 0s 1 normal forwards running;"
: ""
}
--state: ${state};
transform: translate(0px, ${(1 - state) * -100}px);
opacity: ${state};
}
svg, path, rect, circle {
transition: all 0.4s ease;
}
svg {
display: block;
position: absolute;
inset: 0;
pointer-events: none;
width: 100%;
height: 100%;
transform: translate(0px, -0px);
}
path {
fill: ${v.colors.fill || "none"};
stroke: ${v.colors.stroke || "var(--stroke-color, #9098a9)"};
stroke-linejoin: round;
stroke-linecap: round;
}
@keyframes fadein-translate-y {
0% {
visibility: visible;
transform: translate(0%, 50%);
opacity: 0;
}
100% {
transform: translate(0%,0%);
opacity: 1;
visibility: visible;
}
}
/*@container component-container (max-width: 100px) {
svg {
}
}*/
</style>
`}`
}}
|> await ${({values:v}) => async () => ch(shadow).append(v.svg = ch.dom`
<svg id="main-svg" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1 1" preserveAspectRatio="none">
<defs>
<foreignObject>
<slot></slot>
</foreignObject>
<filter id="displacement-filter" filterUnits="objectBoundingBox" primitiveUnits="objectBoundingBox" color-interpolation-filters="sRGB">
<feImage id="displacement-map" href="" result="displacementMap" preserveAspectRatio="none" />
<feDisplacementMap id="displacement-fe" in="SourceGraphic" in2="displacementMap" scale="0.65" xChannelSelector="R" yChannelSelector="B" />
</filter>
<pattern id="imagePattern" patternTransform="translate(0,0)" patternContentUnits="objectBoundingBox" patternUnits="objectBoundingBox" width="1" height="1">
<image id="source-image" href="${v.hrefTexture}" width="1" height="1" preserveAspectRatio="none"/>
<animateTransform id="anim-pattern" begin="0s" attributeName="patternTransform" type="translate" from="0,0" to="1,0" dur="3s" repeatCount="indefinite" fill="freeze"/>
</pattern>
<clipPath id="defClip" clipPathUnits="userSpaceOnUse">
<circle cx="0.5" cy="0.5" r="0.5"></circle>
</clipPath>
<clipPath id="cusClip" clipPathUnits="userSpaceOnUse"></clipPath>
<mask id="progress-mask" maskUnits="objectBoundingBox" preserveAspectRatio="none">
<rect
id="progress-rect"
width="1"
height="1"
x="0"
y="0"
fill="white"
transform="translate(0, 0)"
/>
</mask>
</defs>
<g id="mask-g" mask="url(#progress-mask)">
<g id="clip-g" clip-path="url(#defClip)">
<rect id="filter-rect" width="1" height="1" fill="url(#imagePattern)" filter="url(#displacement-filter)" />
</g>
</g>
<g clip-path="url(#defClip)" id="stroke-g-parent">
<image id="source-overlay" href="${v.hrefOverlay}" x="0" y="0" width="1" height="1" preserveAspectRatio="xMidYMid slice" />
<g id="stroke-g"style="transition: all 0.4s ease; fill: transparent; stroke: ${v.colors.stroke || "var(--stroke-color, #9098a9)"}; stroke-width: 0;">
</g>
</g>
</svg>
`)}
|> await ${({values:v}) => async () => {
v.svg = shadow.querySelector("svg");
toggle = v.toggle = el.toggle = ch.throttle(function(){
//console.log("toggle", el);
state = state ? 0 : 1;
const display = ["none", "grid"][state];
ch(el).style("--state", state)
`=> ${() => () => (!state || el.getAnimations().filter(a => a.playState !== "finished").length) && ch.cancelAnimate({commit: ["transform", "opacity"]})}`
`=> ${() => () => state && el.style.setProperty("display", display, "important")}`
.animate([{
transform: `translate(0px, ${(1 - state) * -100}px)`,
opacity: state,
display
}], {duration: 400, fill:"both", easing: "ease-in-out"})
.lastOp
.then(() => el.style.setProperty("display", display, "important"))
.catch(() => {})
}, {delay: v.toggleDelay ?? 50})
}}
|> await ${({values}) => async() => {
return Promise.all([
el[symbols.upClipPath]({values, el}),
el[symbols.upStyles]({values, el, state}),
el[symbols.upSvg]({values, el})
]);
}}
|> await ${({values:v}) => async () => v.toggle()}
|> await ${({values:v}) => async () => {shadow.querySelector("style").sheet.cssRules[1].style.setProperty("transition", "all 0.4s ease")}}
|> await ${({values:v}) => async () => v.mutObs.observe(el, {childList: true, subtree: true, attributes: true, characterData: true})}
`
})(
{
initialized: Symbol("initialized"),
cusClipPath: Symbol("cusClipPath"),
defClipPath: Symbol("defClipPath"),
clipG: Symbol("clipG"),
maskG: Symbol("maskG"),
strokeG: Symbol("strokeG"),
strokeGParent: Symbol("strokeGParent"),
maskRect: Symbol("maskRect"),
filterRect: Symbol("filterRect"),
displacementFe: Symbol("displacementFe"),
shadow: Symbol("shadow"),
upAttrs: Symbol("upAttrs"),
upSvg: Symbol("upSvg"),
upStyle: Symbol("upStyle"),
upStyles: Symbol("upStyles"),
upClipPath: Symbol("upClipPath"),
stylesheet: Symbol("stylesheet"),
svgImage: Symbol("svgImage"),
svgOverlay: Symbol("svgOverlay"),
pattern: Symbol("pattern"),
animateTransform: Symbol("animateTransform"),
isPaused: Symbol("isPaused"),
nuf: Symbol("nuf"),
nue: Symbol("nue")
}
)
ch.adopt("mana-orb", manaOrb)`<mana-orb ${{}}/>`