feed: refactoring and deterministic chronological feed order (maybe)

chronological order: Getting there..

cleanup: removed debounce code for now

keep also recommend server events in the dom map, not sure if just
everything should be in the map.
pull/2/head
OFF0 2 years ago
parent 549bfed7dd
commit 09bb691f6c
Signed by: offbyn
GPG Key ID: 94A2F643C51F37FA

@ -4,21 +4,21 @@ import {elem} from './domutil.js';
const pool = relayPool(); const pool = relayPool();
pool.addRelay('wss://nostr.x1ddos.ch', {read: true, write: true}); pool.addRelay('wss://nostr.x1ddos.ch', {read: true, write: true});
// pool.addRelay('wss://nostr.bitcoiner.social/', {read: true, write: true}); // pool.addRelay('wss://nostr.bitcoiner.social/', {read: true, write: true});
// pool.addRelay('wss://nostr.openchain.fr', {read: true, write: true}); pool.addRelay('wss://nostr.openchain.fr', {read: true, write: true});
// pool.addRelay('wss://relay.nostr.info', {read: true, write: true}); // pool.addRelay('wss://relay.nostr.info', {read: true, write: true});
// pool.addRelay('wss://relay.damus.io', {read: true, write: true}); // pool.addRelay('wss://relay.damus.io', {read: true, write: true});
// read only // read only
// pool.addRelay('wss://nostr.rocks', {read: true, write: false}); // pool.addRelay('wss://nostr.rocks', {read: true, write: false});
// pool.addRelay('wss://nostr-relay.wlvs.space', {read: true, write: false}); // pool.addRelay('wss://nostr-relay.wlvs.space', {read: true, write: false});
const dateTime = new Intl.DateTimeFormat(navigator.language, { const dateTime = new Intl.DateTimeFormat('de-ch' /* navigator.language */, {
dateStyle: 'short', dateStyle: 'short',
timeStyle: 'short', timeStyle: 'medium',
}); });
let max = 0; let max = 0;
function onEvent(evt, relay) { function onEvent(evt, relay) {
if (max++ >= 23) { if (max++ >= 2123) {
return subscription.unsub(); return subscription.unsub();
} }
switch (evt.kind) { switch (evt.kind) {
@ -29,7 +29,7 @@ function onEvent(evt, relay) {
handleTextNote(evt, relay); handleTextNote(evt, relay);
break; break;
case 2: case 2:
feedContainer.prepend(renderRecommendServer(evt, relay)); handleRecommendServer(evt, relay);
break; break;
case 3: case 3:
updateContactList(evt, relay); updateContactList(evt, relay);
@ -39,21 +39,20 @@ function onEvent(evt, relay) {
} }
} }
const pubkey = localStorage.getItem('pub_key') // const pubkey = localStorage.getItem('pub_key')
console.log({pubkey})
const subscription = pool.sub({ const subscription = pool.sub({
cb: onEvent, cb: onEvent,
filter: { filter: {
authors: [ // authors: [
// '52155da703585f25053830ac39863c80ea6adc57b360472c16c566a412d2bc38', // quark // '52155da703585f25053830ac39863c80ea6adc57b360472c16c566a412d2bc38', // quark
// 'a6057742e73ff93b89587c27a74edf2cdab86904291416e90dc98af1c5f70cfa', // mosc // 'a6057742e73ff93b89587c27a74edf2cdab86904291416e90dc98af1c5f70cfa', // mosc
'3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d', // fiatjaf // '3bf0c63fcb93463407af97a5e5ee64fa883d107ef9e558472c4eb9aaaefa459d', // fiatjaf
'52155da703585f25053830ac39863c80ea6adc57b360472c16c566a412d2bc38' // x1ddos // '52155da703585f25053830ac39863c80ea6adc57b360472c16c566a412d2bc38', // x1ddos
// pubkey, // me // // pubkey, // me
// '32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245' // jb55 // '32e1827635450ebb3c5a7d12c1f8e7b2b514439ac10a67eef3d9fd9c5c68e245', // jb55
], // ],
limit: 100, limit: 2000,
} }
}); });
@ -62,35 +61,33 @@ const replyList = [];
const eventRelayMap = {}; const eventRelayMap = {};
const hasEventTag = tag => tag[0] === 'e'; const hasEventTag = tag => tag[0] === 'e';
let debounceRenderFeedTimer;
function handleTextNote(evt, relay) { function handleTextNote(evt, relay) {
if (eventRelayMap[evt.id]) { if (eventRelayMap[evt.id]) {
eventRelayMap[evt.id] = [relay, ...(eventRelayMap[evt.id])]; eventRelayMap[evt.id] = [relay, ...(eventRelayMap[evt.id])];
} else { } else {
eventRelayMap[evt.id] = [relay]; eventRelayMap[evt.id] = [relay];
(evt.tags.some(hasEventTag) ? replyList : textNoteList).push(evt); (evt.tags.some(hasEventTag) ? replyList : textNoteList).push(evt);
clearTimeout(debounceRenderFeedTimer); renderFeed()
debounceRenderFeedTimer = setTimeout(renderFeed, 200);
} }
} }
// feed // feed
const feedContainer = document.querySelector('#feed'); const feedContainer = document.querySelector('#feed');
const feedDomMap = {}; const feedDomMap = {};
const sortByCreatedAt = (evt1, evt2) => {
let debounceDebugMessageTimer;
function renderFeed() {
const sortedFeeds = textNoteList.sort((evt1, evt2) => {
if (evt1.created_at === evt2.created_at) { if (evt1.created_at === evt2.created_at) {
console.log('OMG exactly at the same time', evt1, evt2); //console.log('OMG exactly at the same time', evt1, evt2);
} }
return evt1.created_at > evt2.created_at ? -1 : 1; return evt1.created_at > evt2.created_at ? -1 : 1;
}); };
let debounceDebugMessageTimer;
function renderFeed() {
const sortedFeeds = textNoteList.sort(sortByCreatedAt).reverse();
clearTimeout(debounceDebugMessageTimer); clearTimeout(debounceDebugMessageTimer);
debounceDebugMessageTimer = setTimeout(() => { debounceDebugMessageTimer = setTimeout(() => {
console.log(`${sortedFeeds.map(e => dateTime.format(e.created_at * 1000)).join('\n')}`) console.log(`${sortedFeeds.map(e => dateTime.format(e.created_at * 1000)).join('\n')}`)
}, 1000); }, 2000);
sortedFeeds.forEach((textNoteEvent, i) => { sortedFeeds.forEach((textNoteEvent, i) => {
if (feedDomMap[textNoteEvent.id]) { if (feedDomMap[textNoteEvent.id]) {
// TODO check eventRelayMap if event was published to different relays // TODO check eventRelayMap if event was published to different relays
@ -99,9 +96,14 @@ function renderFeed() {
const art = renderTextNote(textNoteEvent, eventRelayMap[textNoteEvent.id]); const art = renderTextNote(textNoteEvent, eventRelayMap[textNoteEvent.id]);
feedDomMap[textNoteEvent.id] = art; feedDomMap[textNoteEvent.id] = art;
if (i === 0) { if (i === 0) {
feedContainer.prepend(art); feedContainer.append(art);
} else { } else {
feedDomMap[sortedFeeds[i - 1].id].after(art); feedDomMap[sortedFeeds[i - 1].id].before(art);
console.log(
dateTime.format(sortedFeeds[i + 1].created_at * 1000),
' > ',
dateTime.format(textNoteEvent.created_at * 1000),
)
// feedContainer.insertBefore(art, feedDomMap[sortedFeeds[i - 1].id]); // feedContainer.insertBefore(art, feedDomMap[sortedFeeds[i - 1].id]);
//feedDomMap[sortedFeeds[i - 1].id].after(art); //feedDomMap[sortedFeeds[i - 1].id].after(art);
} }
@ -124,6 +126,28 @@ function renderTextNote(evt, relay) {
return rendernArticle([img, body], isReply && {className: 'mbox-reply'}); return rendernArticle([img, body], isReply && {className: 'mbox-reply'});
} }
function handleRecommendServer(evt, relay) {
if (feedDomMap[evt.id]) {
// TODO event might also be published to different relays
return;
}
const art = renderRecommendServer(evt, relay);
if (textNoteList.length < 2) {
console.log('prob does happen often')
feedContainer.append(art);
return;
}
const closestTextNotes = textNoteList.sort((evt1, evt2) => {
if (Math.abs(evt1.created_at - evt.created_at) < Math.abs(evt2.created_at - evt.created_at)) {
return -1;
} else {
return 1;
}
});
feedDomMap[closestTextNotes[0].id].after(art);
feedDomMap[evt.id] = art;
}
function renderRecommendServer(evt, relay) { function renderRecommendServer(evt, relay) {
const [host, img, time, userName] = getMetadata(evt, relay); const [host, img, time, userName] = getMetadata(evt, relay);
const body = elem('div', {className: 'mbox-body', title: dateTime.format(time)}, [ const body = elem('div', {className: 'mbox-body', title: dateTime.format(time)}, [
@ -139,7 +163,6 @@ function renderProfile(userName, host) {
]); ]);
} }
function rendernArticle(content, props) { function rendernArticle(content, props) {
const className = ['mbox', props?.className].join(' '); const className = ['mbox', props?.className].join(' ');
return elem('article', {...props, className}, content); return elem('article', {...props, className}, content);

Loading…
Cancel
Save