mirror of
https://github.com/autismjs/monorepo.git
synced 2026-01-10 05:08:07 -05:00
works for now :)
This commit is contained in:
@@ -14,7 +14,6 @@ export class CustomElement extends HTMLElement implements ICustomElement {
|
||||
css: string;
|
||||
html: string;
|
||||
#tree?: VNode;
|
||||
#approot?: any;
|
||||
#lastAttrUpdated = 0;
|
||||
#attrUpdateTimeout: any;
|
||||
|
||||
@@ -34,14 +33,14 @@ export class CustomElement extends HTMLElement implements ICustomElement {
|
||||
return h('div');
|
||||
}
|
||||
|
||||
refresh() {
|
||||
if (this.shadowRoot) {
|
||||
this.#tree = this.render();
|
||||
this.shadowRoot.innerHTML = '';
|
||||
this.shadowRoot.appendChild(html(`<style>${this.css}</style>`));
|
||||
this.#tree.append(this.shadowRoot);
|
||||
}
|
||||
}
|
||||
create = () => {
|
||||
if (!this.shadowRoot) return;
|
||||
|
||||
this.#tree = this.render();
|
||||
this.shadowRoot.innerHTML = '';
|
||||
this.shadowRoot.appendChild(html(`<style>${this.css}</style>`));
|
||||
this.#tree.append(this.shadowRoot);
|
||||
};
|
||||
|
||||
get state() {
|
||||
return Array.from(this.attributes).reduce(
|
||||
@@ -55,8 +54,8 @@ export class CustomElement extends HTMLElement implements ICustomElement {
|
||||
|
||||
async connectedCallback() {
|
||||
this.attachShadow({ mode: 'open' });
|
||||
this.refresh();
|
||||
await this.onmount();
|
||||
this.create();
|
||||
}
|
||||
|
||||
attributeChangedCallback(key: string, ov: string, nv: string) {
|
||||
@@ -73,7 +72,7 @@ export class CustomElement extends HTMLElement implements ICustomElement {
|
||||
this.#attrUpdateTimeout = null;
|
||||
}
|
||||
this.#lastAttrUpdated = now;
|
||||
this.refresh();
|
||||
await this.update();
|
||||
};
|
||||
|
||||
if (timeSince > wait) {
|
||||
@@ -95,10 +94,13 @@ export class CustomElement extends HTMLElement implements ICustomElement {
|
||||
if (!this.shadowRoot) return;
|
||||
|
||||
if (!this.#tree) {
|
||||
this.refresh();
|
||||
this.create();
|
||||
} else if (this.#tree) {
|
||||
await this.onupdate();
|
||||
this.#tree.update();
|
||||
|
||||
const oldTree = this.#tree;
|
||||
const newTree = this.render();
|
||||
newTree.patch(oldTree.el);
|
||||
this.onupdated();
|
||||
}
|
||||
};
|
||||
@@ -134,8 +136,6 @@ export class VNode {
|
||||
style?: CSSStyleDeclaration;
|
||||
parentNode?: VNode;
|
||||
content?: string;
|
||||
#patches: (() => any)[] = [];
|
||||
#updates: (() => any)[] = [];
|
||||
#el?: any;
|
||||
|
||||
constructor(options: VNodeProps) {
|
||||
@@ -146,7 +146,6 @@ export class VNode {
|
||||
this.style = options.style;
|
||||
this.content = options.content;
|
||||
this.children = options.children || [];
|
||||
this.#updates = [];
|
||||
|
||||
for (const node of this.children) {
|
||||
node.parentNode = this;
|
||||
@@ -165,21 +164,67 @@ export class VNode {
|
||||
return this;
|
||||
}
|
||||
|
||||
addPatch(patchFn: () => void) {
|
||||
this.#patches.push(patchFn);
|
||||
}
|
||||
|
||||
append(root: ShadowRoot) {
|
||||
return new Promise(() => {
|
||||
const frag = this.createElement();
|
||||
root.append(frag);
|
||||
return frag;
|
||||
});
|
||||
const frag = this.createElement();
|
||||
root.append(frag);
|
||||
return frag;
|
||||
}
|
||||
|
||||
update() {
|
||||
for (const update of this.#updates) {
|
||||
update();
|
||||
patch(rootElement: Element) {
|
||||
const frag = this.createElement();
|
||||
|
||||
const lastEl = rootElement;
|
||||
const newEl = frag.children[0];
|
||||
this._patchOne(lastEl, newEl);
|
||||
}
|
||||
|
||||
_patchOne(lastEl: Element, newEl: Element) {
|
||||
if (lastEl.tagName !== newEl.tagName) {
|
||||
lastEl.replaceWith(newEl);
|
||||
return;
|
||||
}
|
||||
|
||||
if (newEl.attributes.length) {
|
||||
for (const attr of Array.from(newEl.attributes)) {
|
||||
if (lastEl.getAttribute(attr.name) !== attr.value) {
|
||||
lastEl?.setAttribute(attr.name, attr.value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (newEl.classList.length) {
|
||||
for (const name of Array.from(newEl.classList)) {
|
||||
if (!lastEl.classList.contains(name)) {
|
||||
lastEl?.classList.add(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (lastEl.classList.length) {
|
||||
for (const name of Array.from(newEl.classList)) {
|
||||
if (!newEl.classList.contains(name)) {
|
||||
lastEl?.classList.remove(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (lastEl.tagName === 'TEXT' && lastEl.textContent !== newEl.textContent) {
|
||||
lastEl.textContent = newEl.textContent;
|
||||
}
|
||||
|
||||
const maxlength = Math.max(newEl.children.length, lastEl.children.length);
|
||||
|
||||
for (let i = 0; i < maxlength; i++) {
|
||||
const lastChild = lastEl.children[i];
|
||||
const newChild = newEl.children[i];
|
||||
|
||||
if (lastChild && newChild) {
|
||||
this._patchOne(lastChild, newChild);
|
||||
} else if (!lastChild && newChild) {
|
||||
lastEl.append(newChild);
|
||||
} else if (lastChild && !newChild) {
|
||||
lastEl.removeChild(lastChild);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -216,16 +261,6 @@ export class VNode {
|
||||
el.appendChild(childFrag);
|
||||
}
|
||||
|
||||
if (this.#patches.length) {
|
||||
const patch = () => {
|
||||
this.#patches.forEach((fn) => {
|
||||
fn();
|
||||
});
|
||||
};
|
||||
|
||||
this.rootNode.#updates.push(patch);
|
||||
}
|
||||
|
||||
this.#el = el;
|
||||
|
||||
return frag;
|
||||
@@ -252,8 +287,6 @@ export const h = (
|
||||
const id = name.match(/(?<=[#*])([^.#\[\]]*)+?(?=[(#.\s\[)*])?/g);
|
||||
const attributes = name.match(/(?<=[\[])([^.#\[\]]*)+?(?=[\]*])?/g) || [];
|
||||
|
||||
const patches: (() => { from: VNode | VNode[]; to: VNode | VNode[] })[] = [];
|
||||
|
||||
let options: VNodeProps = {
|
||||
tagName: tagName![0],
|
||||
classList: classList || [],
|
||||
@@ -286,20 +319,6 @@ export const h = (
|
||||
...options,
|
||||
});
|
||||
|
||||
for (const patch of patches) {
|
||||
vnode.addPatch(() => {
|
||||
if (!vnode.el) return;
|
||||
|
||||
const { from, to } = patch();
|
||||
|
||||
if (Array.isArray(to)) {
|
||||
vnode.el.append(...to.map((n) => n.createElement()));
|
||||
} else if (!Array.isArray(from)) {
|
||||
from.el.replaceWith(to.createElement());
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
return vnode;
|
||||
|
||||
function reduceChildren(
|
||||
@@ -342,16 +361,6 @@ export const h = (
|
||||
retNodes = [nodes];
|
||||
}
|
||||
|
||||
const oldNodes = nodes;
|
||||
|
||||
patches.push(() => {
|
||||
const newNodes = nodeOrText();
|
||||
return {
|
||||
from: oldNodes,
|
||||
to: newNodes,
|
||||
};
|
||||
});
|
||||
|
||||
return retNodes as VNode[];
|
||||
}
|
||||
|
||||
|
||||
@@ -19,25 +19,26 @@ export default class ProfileImage extends CustomElement {
|
||||
|
||||
attributeChangedCallback(key: string, ov: string, nv: string) {
|
||||
super.attributeChangedCallback(key, ov, nv);
|
||||
// console.log('changing profile-image attr', key, nv);
|
||||
this.update();
|
||||
}
|
||||
|
||||
async onmount() {
|
||||
const { creator } = this.state;
|
||||
const store = getStore();
|
||||
const node: NodeStore = store.get('node');
|
||||
const user = await node.node.db.db.getProfile(creator || '');
|
||||
const user = await node.getUser(this.state.creator || '');
|
||||
|
||||
let url =
|
||||
'';
|
||||
if (user.profileImageUrl) {
|
||||
url = user.profileImageUrl;
|
||||
} else if (creator) {
|
||||
url = 'data:image/svg+xml;utf8,' + minidenticon(creator, 50, 50);
|
||||
}
|
||||
user?.subscribe((u) => {
|
||||
let url =
|
||||
'';
|
||||
if (u.profileImageUrl) {
|
||||
url = u.profileImageUrl;
|
||||
} else if (this.state.creator) {
|
||||
url =
|
||||
'data:image/svg+xml;utf8,' + minidenticon(this.state.creator, 50, 50);
|
||||
}
|
||||
|
||||
// console.log('setting src', url);
|
||||
this.setAttribute('src', url);
|
||||
this.setAttribute('src', url);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { CustomElement, h, register, xh } from '../../../lib/ui.ts';
|
||||
import { CustomElement, h, register } from '../../../lib/ui.ts';
|
||||
import { getStore } from '../../state';
|
||||
import { default as NodeState } from '../../state/node.ts';
|
||||
import '../../components/Post';
|
||||
|
||||
@@ -12,6 +12,7 @@ export default class Node extends Store {
|
||||
|
||||
$globalPosts = new Observable<string[]>([]);
|
||||
$posts = new ObservableMap<string, Post>();
|
||||
$users = new ObservableMap<string, any>();
|
||||
|
||||
constructor(options?: StateOptions) {
|
||||
super(options);
|
||||
@@ -20,7 +21,7 @@ export default class Node extends Store {
|
||||
bootstrap: [
|
||||
'/ip4/127.0.0.1/tcp/64029/ws/p2p/12D3KooWG1yjigfXNQtfkLC8TbzoSreeDx8User1Q5wo49MUK1NB',
|
||||
'/ip4/192.168.86.30/tcp/64029/ws/p2p/12D3KooWG1yjigfXNQtfkLC8TbzoSreeDx8User1Q5wo49MUK1NB',
|
||||
'/ip4/192.168.86.24/tcp/64029/ws/p2p/12D3KooWG1yjigfXNQtfkLC8TbzoSreeDx8User1Q5wo49MUK1NB'
|
||||
'/ip4/192.168.86.24/tcp/64029/ws/p2p/12D3KooWG1yjigfXNQtfkLC8TbzoSreeDx8User1Q5wo49MUK1NB',
|
||||
],
|
||||
});
|
||||
this.node = node;
|
||||
@@ -63,4 +64,10 @@ export default class Node extends Store {
|
||||
getPost = async (hash: string): Promise<Post | null> => {
|
||||
return this.node.db.db.getMessage(hash) as Promise<Post | null>;
|
||||
};
|
||||
|
||||
getUser = async (creator = '') => {
|
||||
const profile = await this.node.db.db.getProfile(creator);
|
||||
this.$users.set(creator, profile);
|
||||
return this.$users.get(creator);
|
||||
};
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user