1
0
mirror of https://github.com/jaandrle/deka-dom-el synced 2025-04-03 20:35:53 +02:00
This commit is contained in:
Jan Andrle 2025-02-28 19:53:07 +01:00
parent b53f3926b3
commit 3168f452ae
Signed by: jaandrle
GPG Key ID: B3A25AED155AFFAB
6 changed files with 29 additions and 31 deletions

View File

@ -20,7 +20,7 @@ export const enviroment= {
M: globalThis.MutationObserver, M: globalThis.MutationObserver,
q: p=> p || Promise.resolve(), q: p=> p || Promise.resolve(),
}; };
import { isUndef } from './helpers.js'; import { isInstance, isUndef } from './helpers.js';
/** /**
* Handles attribute setting with special undefined handling * Handles attribute setting with special undefined handling
@ -43,7 +43,7 @@ function setDeleteAttr(obj, prop, val){
Reflect.set(obj, prop, val); Reflect.set(obj, prop, val);
if(!isUndef(val)) return; if(!isUndef(val)) return;
Reflect.deleteProperty(obj, prop); Reflect.deleteProperty(obj, prop);
if(obj instanceof enviroment.H && obj.getAttribute(prop)==="undefined") if(isInstance(obj, enviroment.H) && obj.getAttribute(prop)==="undefined")
return obj.removeAttribute(prop); return obj.removeAttribute(prop);
if(Reflect.get(obj, prop)==="undefined") if(Reflect.get(obj, prop)==="undefined")
return Reflect.set(obj, prop, ""); return Reflect.set(obj, prop, "");

View File

@ -90,6 +90,7 @@ export function chainableAppend(el){
/** Current namespace for element creation */ /** Current namespace for element creation */
let namespace; let namespace;
import { isInstance, isUndef } from "./helpers.js";
/** /**
* Creates a DOM element with specified tag, attributes and addons * Creates a DOM element with specified tag, attributes and addons
* *
@ -112,7 +113,7 @@ export function createElement(tag, attributes, ...addons){
(scoped===1 ? addons.unshift(...c) : c.forEach(c=> c(el_host)), undefined); (scoped===1 ? addons.unshift(...c) : c.forEach(c=> c(el_host)), undefined);
scope.push({ scope: tag, host }); scope.push({ scope: tag, host });
el= tag(attributes || undefined); el= tag(attributes || undefined);
const is_fragment= el instanceof env.F; const is_fragment= isInstance(el, env.F);
if(el.nodeName==="#comment") break; if(el.nodeName==="#comment") break;
const el_mark= createElement.mark({ const el_mark= createElement.mark({
type: "component", type: "component",
@ -283,7 +284,7 @@ export function assignAttribute(element, key, value){
*/ */
function assignContext(element, _this){ function assignContext(element, _this){
if(assign_context.has(element)) return assign_context.get(element); if(assign_context.has(element)) return assign_context.get(element);
const is_svg= element instanceof env.S; const is_svg= isInstance(element, env.S);
const setRemoveAttr= (is_svg ? setRemoveNS : setRemove).bind(null, element, "Attribute"); const setRemoveAttr= (is_svg ? setRemoveNS : setRemove).bind(null, element, "Attribute");
const s= signals(_this); const s= signals(_this);
return { setRemoveAttr, s }; return { setRemoveAttr, s };
@ -313,11 +314,10 @@ export function classListDeclarative(element, toggle){
* @returns {void} * @returns {void}
*/ */
export function elementAttribute(element, op, key, value){ export function elementAttribute(element, op, key, value){
if(element instanceof env.H) if(isInstance(element, env.H))
return element[op+"Attribute"](key, value); return element[op+"Attribute"](key, value);
return element[op+"AttributeNS"](null, key, value); return element[op+"AttributeNS"](null, key, value);
} }
import { isUndef } from "./helpers.js";
//TODO: add cache? `(Map/Set)<el.tagName+key,isUndef>` //TODO: add cache? `(Map/Set)<el.tagName+key,isUndef>`
/** /**

View File

@ -1,4 +1,5 @@
import { enviroment as env, evc, evd } from './dom-common.js'; import { enviroment as env, evc, evd } from './dom-common.js';
import { isInstance } from "./helpers.js";
/** /**
* Connection changes observer for tracking element connection/disconnection * Connection changes observer for tracking element connection/disconnection
@ -167,9 +168,9 @@ function connectionsChangesObserverConstructor(){
if(store.size > 30)//TODO?: limit if(store.size > 30)//TODO?: limit
await requestIdle(); await requestIdle();
const out= []; const out= [];
if(!(element instanceof Node)) return out; if(!isInstance(element, Node)) return out;
for(const el of store.keys()){ for(const el of store.keys()){
if(el===element || !(el instanceof Node)) continue; if(el===element || !isInstance(el, Node)) continue;
if(element.contains(el)) if(element.contains(el))
out.push(el); out.push(el);
} }

View File

@ -24,6 +24,11 @@ export function typeOf(v){
return Object.prototype.toString.call(v); return Object.prototype.toString.call(v);
} }
export function isInstance(obj, cls){ return obj instanceof cls; }
/** @type {typeof Object.prototype.isPrototypeOf.call} */
export function isProtoFrom(obj, cls){ return Object.prototype.isPrototypeOf.call(cls, obj); }
export function oCreate(proto= null){ return Object.create(proto); }
/** /**
* Handles AbortSignal registration and cleanup * Handles AbortSignal registration and cleanup
* @param {AbortSignal} signal - The AbortSignal to listen to * @param {AbortSignal} signal - The AbortSignal to listen to
@ -31,7 +36,7 @@ export function typeOf(v){
* @returns {Function|undefined|boolean} Cleanup function or undefined if already aborted * @returns {Function|undefined|boolean} Cleanup function or undefined if already aborted
*/ */
export function onAbort(signal, listener){ export function onAbort(signal, listener){
if(!signal || !(signal instanceof AbortSignal)) if(!signal || !isInstance(signal, AbortSignal))
return true; return true;
if(signal.aborted) if(signal.aborted)
return; return;

View File

@ -1,3 +1,4 @@
import { isProtoFrom } from "../helpers.js";
/** /**
* Global signals object with default implementation * Global signals object with default implementation
* @type {Object} * @type {Object}
@ -39,5 +40,5 @@ export function registerReactivity(def, global= true){
* @returns {typeof signals_global} Signals implementation * @returns {typeof signals_global} Signals implementation
*/ */
export function signals(_this){ export function signals(_this){
return signals_global.isPrototypeOf(_this) && _this!==signals_global ? _this : signals_global; return isProtoFrom(_this, signals_global) && _this!==signals_global ? _this : signals_global;
} }

View File

@ -1,6 +1,6 @@
import { queueSignalWrite, mark } from "./helpers.js"; import { queueSignalWrite, mark } from "./helpers.js";
export { mark }; export { mark };
import { hasOwn, Defined } from "../helpers.js"; import { hasOwn, Defined, oCreate, isProtoFrom } from "../helpers.js";
/** /**
* Checks if a value is a signal * Checks if a value is a signal
@ -155,22 +155,8 @@ import { el } from "../dom.js";
import { scope } from "../dom.js"; import { scope } from "../dom.js";
import { on } from "../events.js"; import { on } from "../events.js";
/** Store for memoized values */ export function cache(store= oCreate()){
const storeMemo= new WeakMap(); return (key, fun)=> hasOwn(store, key) ? store[key] : (store[key]= fun());
/**
* Memoizes a function result
*
* @param {string|unknown} key - Cache key (non-strings will be stringified)
* @param {Function} fun - Function to compute value
* @param {keyof storeMemo} [host= fun]
* @returns {unknown} Cached or computed result
*/
export function memo(key, fun, host= fun){
if(typeof key!=="string") key= JSON.stringify(key);
if (!storeMemo.has(host)) storeMemo.set(host, {});
const cache= storeMemo.get(host);
return hasOwn(cache, key) ? cache[key] : (cache[key]= fun());
} }
/** /**
* Creates a reactive DOM element that re-renders when signal changes * Creates a reactive DOM element that re-renders when signal changes
@ -186,15 +172,16 @@ signal.el= function(s, map){
const out= env.D.createDocumentFragment(); const out= env.D.createDocumentFragment();
out.append(mark_start, mark_end); out.append(mark_start, mark_end);
const { current }= scope; const { current }= scope;
let cache_shared= oCreate();
const reRenderReactiveElement= v=> { const reRenderReactiveElement= v=> {
if(!mark_start.parentNode || !mark_end.parentNode) // === `isConnected` or wasnt yet rendered if(!mark_start.parentNode || !mark_end.parentNode) // === `isConnected` or wasnt yet rendered
return removeSignalListener(s, reRenderReactiveElement); return removeSignalListener(s, reRenderReactiveElement);
const cache= {}; // remove unused els from cache const memo= cache(cache_shared);
cache_shared= oCreate();
scope.push(current); scope.push(current);
let els= map(v, function useCache(key, fun){ let els= map(v, function useCache(key, fun){
return (cache[key]= memo(key, fun, reRenderReactiveElement)); return (cache_shared[key]= memo(key, fun));
}); });
storeMemo.set(reRenderReactiveElement, cache);
scope.pop(); scope.pop();
if(!Array.isArray(els)) if(!Array.isArray(els))
els= [ els ]; els= [ els ];
@ -211,6 +198,10 @@ signal.el= function(s, map){
addSignalListener(s, reRenderReactiveElement); addSignalListener(s, reRenderReactiveElement);
removeSignalsFromElements(s, reRenderReactiveElement, mark_start, map); removeSignalsFromElements(s, reRenderReactiveElement, mark_start, map);
reRenderReactiveElement(s()); reRenderReactiveElement(s());
current.host(on.disconnected(()=>
/*! Clears cached elements for reactive element `S.el` */
cache_shared= {}
));
return out; return out;
}; };
/** /**
@ -363,7 +354,7 @@ function create(is_readonly, value, actions){
* Prototype for signal internal objects * Prototype for signal internal objects
* @private * @private
*/ */
const protoSigal= Object.assign(Object.create(null), { const protoSigal= Object.assign(oCreate(), {
/** /**
* Prevents signal propagation * Prevents signal propagation
*/ */