fix profile iamge

This commit is contained in:
tsukino
2023-12-24 03:58:06 -08:00
parent 7f9e615097
commit 6357c588f1
6 changed files with 118 additions and 44 deletions

View File

@@ -11,18 +11,32 @@ interface CustomElementConstructor {
interface ICustomElement extends HTMLElement {
state: any;
render(): void;
render: () => VTree;
onmount(): Promise<void>;
onupdate(): Promise<void>;
onupdated(): Promise<void>;
}
export class CustomElement extends HTMLElement implements ICustomElement {
css: string;
html: string;
#tree?: VTree;
#approot?: any;
onmount(): Promise<void> {
return Promise.resolve();
}
onupdate(): Promise<void> {
return Promise.resolve();
}
onupdated(): Promise<void> {
return Promise.resolve();
}
render(): VTree {
return hpx``;
return hx`<div></div>`;
}
get state() {
@@ -35,24 +49,34 @@ export class CustomElement extends HTMLElement implements ICustomElement {
);
}
connectedCallback() {
async connectedCallback() {
this.attachShadow({ mode: 'open' });
await this.onmount();
this.patch();
}
patch = () => {
if (!this.#approot) {
this.#tree = this.render();
this.#approot = createElement(this.#tree as VText);
this.shadowRoot?.appendChild(html(`<style>${this.css}</style>`));
this.shadowRoot?.appendChild(this.#approot);
} else if (this.#tree) {
const newTree = this.render();
const patches = diff(this.#tree!, newTree);
this.#approot = patch(this.#approot, patches);
this.#tree = newTree;
}
};
async attributeChangedCallback() {
this.patch();
}
patch = () =>
requestAnimationFrame(async () => {
await this.onupdate();
if (!this.#approot) {
this.#tree = this.render();
this.#approot = createElement(this.#tree as VText);
this.shadowRoot?.appendChild(html(`<style>${this.css}</style>`));
this.shadowRoot?.appendChild(this.#approot);
} else if (this.#tree) {
const newTree = this.render();
const patches = diff(this.#tree!, newTree);
this.#approot = patch(this.#approot, patches);
this.#tree = newTree;
}
this.onupdated();
});
}
export function html(htmlString: string) {
@@ -69,6 +93,51 @@ export function register(name: string, el: CustomElementConstructor) {
window.customElements.define(name, el);
}
type VNodeProps = {
tagName: string;
classList: string[];
id?: string;
attributes: Map<string, string>;
children?: VNode[];
};
export class VNode {
#id?: string;
#tagName: string;
#classList: string[] = [];
#attributes = new Map<string, string>();
#children: VNode[] = [];
constructor(options: VNodeProps) {
this.#tagName = options.tagName;
this.#classList = options.classList;
this.#id = options.id;
this.#attributes = options.attributes;
this.#children = options.children || [];
}
}
export const $ = (name: string) => {
const tagName = name.match(/^[^.|#|\[]*/g);
const classList = name.match(/(?<=[.*])([^.#\[\]]*)+?(?=[(#.\s\[)*])?/g);
const id = name.match(/(?<=[#*])([^.#\[\]]*)+?(?=[(#.\s\[)*])?/g);
const attributes = name.match(/(?<=[\[])([^.#\[\]]*)+?(?=[\]*])?/g) || [];
const vnode = new VNode({
tagName: tagName![0],
classList: classList || [],
id: id ? id[0] : undefined,
attributes: new Map(
attributes.map((attr) => {
const [key, value] = attr.split('=');
return [key, value || ''];
}),
),
});
return vnode;
};
export const Q = (root: ShadowRoot | Element | null) => {
if (!root) return root;

View File

@@ -24,9 +24,11 @@ export default class Post extends CustomElement {
render() {
const { creator, name, handle, createat, content } = this.state;
if (!creator) return hx`<div></div>`;
return hx`
<div class="post">
<profile-image address="${creator}"></profile-image>
${hx`<profile-image id="${creator}" creator="${creator}"></profile-image>`}
<div class="top">
<div class="creator">${name}</div>
<div class="userId">${handle}</div>

View File

@@ -7,41 +7,34 @@ import { minidenticon } from 'minidenticons';
export default class ProfileImage extends CustomElement {
static get observedAttributes() {
return ['address'];
return ['creator', 'src'];
}
css = css.toString();
render() {
const { src } = this.state;
return hx`
<img src="${src}" />
`;
}
async connectedCallback() {
super.connectedCallback();
this.setAttribute('src', await this.getSrc());
}
async getSrc() {
const { address } = this.state;
async onmount() {
const { id: creator } = this.state;
const store = getStore();
const node: NodeStore = store.get('node');
const user = await node.node.db.db.getProfile(address || '');
if (user.profileImageUrl) {
return user.profileImageUrl;
} else if (address) {
return encodeURIComponent(minidenticon(address, 50, 50));
} else {
return '';
}
}
const user = await node.node.db.db.getProfile(creator || '');
attributeChangedCallback(key: string, oldVal: string, newVal: string) {
console.log(key);
this.patch();
// this.setAttribute('src', await this.getSrc());
let url =
'';
if (user.profileImageUrl) {
url = user.profileImageUrl;
} else if (creator) {
url = 'data:image/svg+xml;utf8,' + minidenticon(creator, 50, 50);
}
this.setAttribute('src', url);
}
}

View File

@@ -2,6 +2,7 @@ import { MessageType, Post, PostSubtype, ProofType } from '@autismjs/message';
import { ECDSA } from '../../crypto/src';
import { getStore } from './state';
import App from './pages/App';
import { $ } from '../lib/ui.ts';
const ecdsa = new ECDSA();
console.log('ecdsa', ecdsa);
@@ -18,6 +19,7 @@ console.log('post', p.json);
(async () => {
const state = getStore();
console.log($('button#submit-btn[type=text].button.button-primary'));
console.log('state', state);
document.body.append(new App());

View File

@@ -3,13 +3,19 @@ import { getStore } from '../../state';
import { default as NodeState } from '../../state/node.ts';
import '../../components/Post';
import css from './index.scss';
import * as console from 'console';
export default class App extends CustomElement {
css = css.toString();
async connectedCallback() {
super.connectedCallback();
this.subscribeToPosts();
async onupdate(): Promise<void> {
await super.onupdate();
console.time('app-container');
}
async onupdated(): Promise<void> {
await super.onupdated();
console.timeEnd('app-container');
}
render() {
@@ -32,7 +38,7 @@ export default class App extends CustomElement {
`;
}
subscribeToPosts() {
async onmount() {
const state = getStore();
const node = state.get<NodeState>('node');
node.$globalPosts.subscribe(this.patch);

View File

@@ -18,7 +18,9 @@ export default class Node extends Store {
const node = new Autism({
bootstrap: [
'/ip4/127.0.0.1/tcp/57575/ws/p2p/12D3KooWSoKnYV5idyrrJt3T6WM4eB6wcu58zUuy5bj7cEZNLwdm',
'/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'
],
});
this.node = node;