Skip to content

Instantly share code, notes, and snippets.

@nightpool
Last active September 29, 2019 16:58
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save nightpool/a9da6bd6942ef82701ec4d4dac29457a to your computer and use it in GitHub Desktop.
Save nightpool/a9da6bd6942ef82701ec4d4dac29457a to your computer and use it in GitHub Desktop.
literally the simplest and dumbest activitypub client i can make
<html>
<body>
<div id="root" style="width: 500px; overflow-wrap: break-word;"></div>
<script>
Object.defineProperty(Object.prototype, 'tap',
{ value: tap, configurable: true, writable: true }
);
const AsyncFunction = (async () => {}).constructor;
const createElement = document.createElement.bind(document);
const outbox = 'https://cybre.space/users/nightpool/outbox';
const init = () => {
resolve('https://cybre.space/users/nightpool');
const page = withPlaceholder(async () => {
const { first } = await resolve(outbox);
return renderPage(await resolve(first));
}, () => spinner().tap(s => s.style = 'display: block; margin: 2em auto'));
document.querySelector('#root').append(page);
}
const renderPage = (page) => {
const items = page.orderedItems || page.items;
return createElement('div').tap(div =>
div.append(...items.map(renderPost))
);
}
const renderPost = (post) =>
createElement('div').tap(div => {
div.style = 'margin: 1em; border: 1px solid black;'
div.append(postHeader(post));
if (post.type === 'Create') {
div.append(renderContent(post.object,
{ showAttribution: false }
));
} else if (post.type === 'Announce') {
div.append(withPlaceholder(reblogTrail(post),
() => spinner().tap(s => s.style = 'display: block; margin: 1em auto')));
post.content && div.append(renderContent(post));
}
});
const postHeader = (post) => createElement('div').tap(div => {
div.style = "margin: 1em;"
div.append(renderUsername(post.attributedTo || post.actor));
if (post.type === 'Announce') {
div.append(createElement('span').tap(s => s.innerText = ' reblogged '));
const rebloggedFrom = resolve(post.object).then(o =>
o.attributedTo || o.actor
);
div.append(renderUsername(rebloggedFrom));
};
});
const reblogTrail = async (post) => {
if (post.type !== 'Announce') {
return [];
}
const reblog = await resolve(post.object);
return reblogTrail(reblog).tap(async trail =>
(await trail).push(renderContent(reblog))
);
}
const renderContent = ({
content, summary,
actor,
attributedTo=actor
}, {
showAttribution = true
} = {}) =>
createElement('div').tap(container => {
container.style.margin = '1em';
if (showAttribution) {
container.append(createElement('div').tap(separator => {
separator.style.display = 'flex';
separator.append(renderUsername(attributedTo));
separator.append(createElement('hr').tap(hr => {
hr.style = 'margin-left: 1em; flex-grow: 1';
}))
}));
}
container.append(createElement('div').tap(contentElement => {
contentElement.innerHTML = content || summary;
}));
});
const renderUsername = (user) =>
withPlaceholder(async () => {
const {
preferredUsername
} = await resolve(await user);
return createElement('strong').tap(attribution => {
attribution.innerText = preferredUsername;
});
}, () => spinner().tap(s => s.style = 'max-height: 1em;'));
const withCors = (url) => `https://cors-anywhere.herokuapp.com/${url}`;
function resolve(objectOrUrl, cors=true) {
if (typeof objectOrUrl === "string") {
if (!resolve.cache[objectOrUrl]) {
const location = cors ? withCors(objectOrUrl) : objectOrUrl;
resolve.cache[objectOrUrl] = fetch(
location,
{headers: {Accept: 'application/activity+json'}}
).then(resp => resp.json());
}
return resolve.cache[objectOrUrl];
} else {
return Promise.resolve(objectOrUrl);
}
}
resolve.cache = {};
function tap(f) {
if (f instanceof AsyncFunction) {
return f(this).then(() => this);
} else {
f(this); return this;
}
};
function withPlaceholder(funcOrPromise, placeholder = spinner) {
const placeholderElement = placeholder();
const promise = funcOrPromise instanceof Promise ?
funcOrPromise : funcOrPromise();
promise.then(element => {
placeholderElement.replaceWith(...([].concat(element)));
}).catch(error => {
placeholderElement.replaceWith(renderError(error));
});
return placeholderElement;
}
function renderError(error) {
console.error(error);
return createElement('span').tap(span => {
span.style = 'color: red; font-weight: bold';
span.innerText = 'error';
})
}
function spinner() {
return createElement('img').tap(img => {
img.alt = 'placeholder spinner';
img.src = "";
})
}
</script>
<script>init();</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment