forked from nostr/nostrweb
profile: show following, contact-list changes and contact events
- added following count in profile header - added contact-list changes events - added new raw event detail view to visulaize event metadata and raw content
parent
027c61e00f
commit
5039b3dece
@ -0,0 +1,96 @@
|
|||||||
|
import {Event, nip19} from 'nostr-tools';
|
||||||
|
import {elem} from './utils/dom';
|
||||||
|
import {dateTime} from './utils/time';
|
||||||
|
import {isPTag, sortByCreatedAt} from './events';
|
||||||
|
import {getViewContent} from './view';
|
||||||
|
import {getMetadata} from './profiles';
|
||||||
|
|
||||||
|
const contactHistoryMap: {
|
||||||
|
[pubkey: string]: Event[]
|
||||||
|
} = {};
|
||||||
|
|
||||||
|
const updateFollowing = (evt: Event) => {
|
||||||
|
const following = getViewContent().querySelector(`[data-following="${evt.pubkey}"]`);
|
||||||
|
if (following) {
|
||||||
|
const count = evt.tags.filter(isPTag).length;
|
||||||
|
const anchor = elem('a', {
|
||||||
|
data: {following: evt.pubkey},
|
||||||
|
href: `/${evt.id}`,
|
||||||
|
title: dateTime.format(new Date(evt.created_at * 1000)),
|
||||||
|
}, `following ${count}`);
|
||||||
|
following.replaceWith(anchor);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setContactList = (evt: Event) => {
|
||||||
|
let contactHistory = contactHistoryMap[evt.pubkey];
|
||||||
|
if (!contactHistory) {
|
||||||
|
contactHistoryMap[evt.pubkey] = [evt];
|
||||||
|
updateFollowing(evt);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (contactHistory.find(({id}) => id === evt.id)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
contactHistory.push(evt);
|
||||||
|
contactHistory.sort(sortByCreatedAt);
|
||||||
|
updateFollowing(contactHistory[0]);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* findChanges
|
||||||
|
* returns added and removed contacts list of P tags, ignores any tag other than 'p'
|
||||||
|
*/
|
||||||
|
const findChanges = (current: Event, previous: Event) => {
|
||||||
|
const previousContacts = previous.tags.join('\n'); // filter for p tags first?
|
||||||
|
const currentContacts = current.tags.join('\n');
|
||||||
|
const addedContacts = current.tags.filter(([tag, pubkey]) => tag === 'p' && !previousContacts.includes(pubkey));
|
||||||
|
const removedContacts = previous.tags.filter(([tag, pubkey]) => tag === 'p' && !currentContacts.includes(pubkey));
|
||||||
|
return [addedContacts, removedContacts];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const updateContactList = (evt: Event) => {
|
||||||
|
const contactHistory = contactHistoryMap[evt.pubkey];
|
||||||
|
if (contactHistory.length === 1) {
|
||||||
|
return [contactHistory[0].tags.filter(isPTag)];
|
||||||
|
}
|
||||||
|
const pos = contactHistory.findIndex(({id}) => id === evt.id);
|
||||||
|
if (evt.id === contactHistory.at(-1)?.id) { // oldest known contact-list update
|
||||||
|
// update existing contact entries
|
||||||
|
contactHistory
|
||||||
|
.slice(0, -1)
|
||||||
|
.forEach((entry, i) => {
|
||||||
|
const previous = contactHistory[i + 1];
|
||||||
|
const [added, removed] = findChanges(entry, previous);
|
||||||
|
const contactNote = getViewContent().querySelector(`[data-contacts="${entry.id}"]`);
|
||||||
|
const updated = getContactUpdateMessage(added, removed);
|
||||||
|
contactNote?.replaceChildren(...updated);
|
||||||
|
});
|
||||||
|
return [evt.tags.filter(isPTag)];
|
||||||
|
}
|
||||||
|
return findChanges(evt, contactHistory[pos + 1]);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getContactUpdateMessage = (
|
||||||
|
addedList: string[][],
|
||||||
|
removedList: string[][],
|
||||||
|
) => {
|
||||||
|
const content = [];
|
||||||
|
// console.log(addedContacts)
|
||||||
|
if (addedList.length && addedList[0]) {
|
||||||
|
const pubkey = addedList[0][1];
|
||||||
|
const {userName} = getMetadata(pubkey);
|
||||||
|
const npub = nip19.npubEncode(pubkey);
|
||||||
|
content.push(
|
||||||
|
'follows ',
|
||||||
|
elem('a', {href: `/${npub}`, data: {profile: pubkey}}, userName),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (addedList.length > 1) {
|
||||||
|
content.push(` (+ ${addedList.length - 1} others)`);
|
||||||
|
}
|
||||||
|
if (removedList?.length > 0) {
|
||||||
|
content.push(elem('small', {}, ` and unfollowed ${removedList.length}`));
|
||||||
|
}
|
||||||
|
return content;
|
||||||
|
};
|
Loading…
Reference in New Issue