Files: c5cabd1c1390883a88e7ce23acc23075309f39bd / render.js
11862 bytesRaw
1 | var path = require('path'); |
2 | var pull = require("pull-stream"); |
3 | var marked = require("ssb-marked"); |
4 | var htime = require("human-time"); |
5 | var emojis = require("emoji-named-characters"); |
6 | var cat = require("pull-cat"); |
7 | var h = require('hyperscript'); |
8 | |
9 | var emojiDir = path.join(require.resolve("emoji-named-characters"), "../pngs"); |
10 | |
11 | exports.wrapPage = wrapPage; |
12 | exports.MdRenderer = MdRenderer; |
13 | exports.renderEmoji = renderEmoji; |
14 | exports.formatMsgs = formatMsgs; |
15 | exports.renderThread = renderThread; |
16 | exports.renderAbout = renderAbout; |
17 | |
18 | function MdRenderer(opts) { |
19 | marked.Renderer.call(this, {}); |
20 | this.opts = opts; |
21 | } |
22 | |
23 | MdRenderer.prototype = new marked.Renderer(); |
24 | |
25 | MdRenderer.prototype.urltransform = function(href) { |
26 | if (!href) return false; |
27 | switch (href[0]) { |
28 | case "#": |
29 | return this.opts.base + "channel/" + href.slice(1); |
30 | case "%": |
31 | return this.opts.msg_base + encodeURIComponent(href); |
32 | case "@": |
33 | href = this.opts.mentions[href.substr(1)] || href; |
34 | return this.opts.feed_base + encodeURIComponent(href); |
35 | case "&": |
36 | return this.opts.blob_base + encodeURIComponent(href); |
37 | } |
38 | if (href.indexOf("javascript:") === 0) return false; |
39 | return href; |
40 | }; |
41 | |
42 | MdRenderer.prototype.image = function(href, title, text) { |
43 | return h('img', |
44 | { src: this.opts.img_base + href, |
45 | alt: text, |
46 | title: title |
47 | }).outerHTML; |
48 | }; |
49 | |
50 | function renderEmoji(emoji) { |
51 | var opts = this.renderer.opts; |
52 | var mentions = opts.mentions; |
53 | var url = mentions[emoji] |
54 | ? opts.blob_base + encodeURIComponent(mentions[emoji]) |
55 | : emoji in emojis && opts.emoji_base + escape(emoji) + '.png'; |
56 | return url |
57 | ? h('img.ssb-emoji', |
58 | { src: url, |
59 | alt: ':' + escape(emoji) + ':', |
60 | title: ':' + escape(emoji) + ':', |
61 | height: 16, width: 16 |
62 | }).outerHTML |
63 | : ":" + emoji + ":"; |
64 | } |
65 | |
66 | function escape(str) { |
67 | return String(str) |
68 | .replace(/&/g, "&") |
69 | .replace(/</g, "<") |
70 | .replace(/>/g, ">") |
71 | .replace(/"/g, """); |
72 | } |
73 | |
74 | function formatMsgs(id, ext, opts) { |
75 | switch (ext || "html") { |
76 | case "html": |
77 | return pull(renderThread(opts), wrapPage(id)); |
78 | case "js": |
79 | return pull(renderThread(opts), wrapJSEmbed(opts)); |
80 | case "json": |
81 | return wrapJSON(); |
82 | default: |
83 | return null; |
84 | } |
85 | } |
86 | |
87 | function wrap(before, after) { |
88 | return function(read) { |
89 | return cat([pull.once(before), read, pull.once(after)]); |
90 | }; |
91 | } |
92 | |
93 | function callToAction() { |
94 | return h('a.call-to-action', |
95 | { href: 'https://www.scuttlebutt.nz' }, |
96 | 'Join Scuttlebutt now').outerHTML; |
97 | } |
98 | |
99 | function toolTipTop() { |
100 | return h('span.top-tip', |
101 | 'You are reading content from ', |
102 | h('a', { href: 'https://www.scuttlebutt.nz' }, |
103 | 'Scuttlebutt')).outerHTML; |
104 | } |
105 | |
106 | function renderAbout(opts, about, showAllHTML = "") { |
107 | var figCaption = h('figcaption'); |
108 | figCaption.innerHTML = 'Feed of ' + about.name + '<br>' + |
109 | (about.description != undefined ? |
110 | marked(about.description, opts.marked) : ''); |
111 | return pull( |
112 | pull.map(renderMsg.bind(this, opts)), |
113 | wrap(toolTipTop() + '<main>' + |
114 | h('article', |
115 | h('header', |
116 | h('figure', |
117 | h('img', |
118 | { src: opts.img_base + about.image, |
119 | height: 200, |
120 | width: 200 |
121 | }), |
122 | figCaption) |
123 | )).outerHTML, |
124 | showAllHTML + '</main>' + callToAction()) |
125 | ); |
126 | } |
127 | |
128 | function renderThread(opts, showAllHTML = "") { |
129 | return pull( |
130 | pull.map(renderMsg.bind(this, opts)), |
131 | wrap(toolTipTop() + '<main>', |
132 | showAllHTML + '</main>' + callToAction()) |
133 | ); |
134 | } |
135 | |
136 | function wrapPage(id) { |
137 | return wrap( |
138 | "<!doctype html><html><head>" + |
139 | "<meta charset=utf-8>" + |
140 | "<title>" + |
141 | id + " | ssb-viewer" + |
142 | "</title>" + |
143 | '<meta name=viewport content="width=device-width,initial-scale=1">' + |
144 | styles + |
145 | "</head><body>", |
146 | "</body></html>" |
147 | ); |
148 | } |
149 | |
150 | var styles = ` |
151 | <style> |
152 | html { background-color: #f1f3f5; } |
153 | body { |
154 | color: #212529; |
155 | font-family: "Helvetica Neue", "Calibri Light", Roboto, sans-serif; |
156 | -webkit-font-smoothing: antialiased; |
157 | -moz-osx-font-smoothing: grayscale; |
158 | letter-spacing: 0.02em; |
159 | padding-top: 30px; |
160 | padding-bottom: 50px; |
161 | } |
162 | a { color: #364fc7; } |
163 | |
164 | .top-tip, .top-tip a { |
165 | color: #868e96; |
166 | } |
167 | .top-tip { |
168 | text-align: center; |
169 | display: block; |
170 | margin-bottom: 10px; |
171 | font-size: 14px; |
172 | } |
173 | main { margin: 0 auto; max-width: 40rem; } |
174 | main article:first-child { border-radius: 3px 3px 0 0; } |
175 | main article:last-child { border-radius: 0 0 3px 3px; } |
176 | article { |
177 | background-color: white; |
178 | padding: 20px; |
179 | box-shadow: 0 1px 3px #949494; |
180 | position: relative; |
181 | } |
182 | .top-right { position: absolute; top: 20px; right: 20px; } |
183 | article > header { margin-bottom: 20px; } |
184 | article > header > figure { |
185 | margin: 0; display: flex; |
186 | } |
187 | article > header > figure > img { |
188 | border-radius: 2px; margin-right: 10px; |
189 | } |
190 | article > header > figure > figcaption { |
191 | display: flex; flex-direction: column; justify-content: space-around; |
192 | } |
193 | .ssb-avatar-name { font-size: 1.2em; font-weight: bold; } |
194 | time a { color: #868e96; } |
195 | .ssb-avatar-name, time a { |
196 | text-decoration: none; |
197 | } |
198 | .ssb-avatar-name:hover, time:hover a { |
199 | text-decoration: underline; |
200 | } |
201 | section p { line-height: 1.45em; } |
202 | section p img { |
203 | max-width: 100%; |
204 | max-height: 50vh; |
205 | margin: 0 auto; |
206 | } |
207 | .status { |
208 | font-style: italic; |
209 | } |
210 | |
211 | code { |
212 | display: inline; |
213 | padding: 2px 5px; |
214 | font-weight: 600; |
215 | background-color: #e9ecef; |
216 | border-radius: 3px; |
217 | color: #495057; |
218 | } |
219 | blockquote { |
220 | padding-left: 1.2em; |
221 | margin: 0; |
222 | color: #868e96; |
223 | border-left: 5px solid #ced4da; |
224 | } |
225 | pre { |
226 | background-color: #212529; |
227 | color: #ced4da; |
228 | font-weight: bold; |
229 | padding: 5px; |
230 | border-radius: 3px; |
231 | position: relative; |
232 | } |
233 | pre::before { |
234 | content: "METADATA"; |
235 | position: absolute; |
236 | top: -7px; |
237 | left: 0px; |
238 | background-color: #212529; |
239 | padding: 2px 4px 0; |
240 | border-radius: 2px; |
241 | font-family: "Helvetica Neue", "Calibri Light", Roboto, sans-serif; |
242 | font-size: 9px; |
243 | } |
244 | .call-to-action { |
245 | display: block; |
246 | margin: 0 auto; |
247 | width: 13em; |
248 | text-align: center; |
249 | text-decoration: none; |
250 | margin-top: 20px; |
251 | margin-bottom: 60px; |
252 | background-color: #5c7cfa; |
253 | padding: 15px 0; |
254 | color: #edf2ff; |
255 | border-radius: 3px; |
256 | border-bottom: 3px solid #3b5bdb; |
257 | } |
258 | .call-to-action:hover { |
259 | background-color: #748ffc; |
260 | border-bottom: 3px solid #4c6ef5; |
261 | } |
262 | </style> |
263 | `; |
264 | |
265 | function wrapJSON() { |
266 | var first = true; |
267 | return pull(pull.map(JSON.stringify), join(","), wrap("[", "]")); |
268 | } |
269 | |
270 | function wrapJSEmbed(opts) { |
271 | return pull( |
272 | wrap('<link rel=stylesheet href="' + opts.base + 'static/base.css">', ""), |
273 | pull.map(docWrite), |
274 | opts.base_token && rewriteBase(new RegExp(opts.base_token, "g")) |
275 | ); |
276 | } |
277 | |
278 | function rewriteBase(token) { |
279 | // detect the origin of the script and rewrite the js/html to use it |
280 | return pull( |
281 | replace(token, '" + SSB_VIEWER_ORIGIN + "/'), |
282 | wrap( |
283 | "var SSB_VIEWER_ORIGIN = (function () {" + |
284 | 'var scripts = document.getElementsByTagName("script")\n' + |
285 | "var script = scripts[scripts.length-1]\n" + |
286 | "if (!script) return location.origin\n" + |
287 | 'return script.src.replace(/\\/%.*$/, "")\n' + |
288 | "}())\n", |
289 | "" |
290 | ) |
291 | ); |
292 | } |
293 | |
294 | function join(delim) { |
295 | var first = true; |
296 | return pull.map(function(val) { |
297 | if (!first) return delim + String(val); |
298 | first = false; |
299 | return val; |
300 | }); |
301 | } |
302 | |
303 | function replace(re, rep) { |
304 | return pull.map(function(val) { |
305 | return String(val).replace(re, rep); |
306 | }); |
307 | } |
308 | |
309 | function docWrite(str) { |
310 | return "document.write(" + JSON.stringify(str) + ")\n"; |
311 | } |
312 | |
313 | function renderMsg(opts, msg) { |
314 | var c = msg.value.content || {}; |
315 | var name = encodeURIComponent(msg.key); |
316 | return h('article#' + name, |
317 | h('header', |
318 | h('figure', |
319 | h('img', { alt: '', |
320 | src: opts.img_base + msg.author.image, |
321 | height: 50, width: 50 }), |
322 | h('figcaption', |
323 | h('a.ssb-avatar-name', |
324 | { href: opts.base + escape(msg.value.author) }, |
325 | msg.author.name), |
326 | msgTimestamp(msg, opts.base + name)))), |
327 | render(opts, c)).outerHTML; |
328 | } |
329 | |
330 | function msgTimestamp(msg, link) { |
331 | var date = new Date(msg.value.timestamp); |
332 | var isoStr = date.toISOString(); |
333 | return h('time.ssb-timestamp', |
334 | { datetime: isoStr }, |
335 | h('a', |
336 | { href: link, |
337 | title: isoStr }, |
338 | formatDate(date))); |
339 | } |
340 | |
341 | function formatDate(date) { |
342 | return htime(date); |
343 | } |
344 | |
345 | function render(opts, c) { |
346 | var base = opts.base; |
347 | if (c.type === "post") { |
348 | var channel = c.channel |
349 | ? h('div.top-right', |
350 | h('a', |
351 | { href: base + 'channel/' + c.channel }, |
352 | '#' + c.channel)) |
353 | : ""; |
354 | return [channel, renderPost(opts, c)]; |
355 | } else if (c.type == "vote" && c.vote.expression == "Dig") { |
356 | var channel = c.channel |
357 | ? [' in ', |
358 | h('a', |
359 | { href: base + 'channel/' + c.channel }, |
360 | '#' + c.channel)] |
361 | : ""; |
362 | var linkedText = "this"; |
363 | if (typeof c.vote.linkedText != "undefined") |
364 | linkedText = c.vote.linkedText.substring(0, 75); |
365 | return h('span.status', |
366 | ['Liked ', |
367 | h('a', { href: base + c.vote.link }, linkedText), |
368 | channel]); |
369 | } else if (c.type == "vote") { |
370 | var linkedText = "this"; |
371 | if (typeof c.vote.linkedText != "undefined") |
372 | linkedText = c.vote.linkedText.substring(0, 75); |
373 | return h('span.status', |
374 | ['Voted ', |
375 | h('a', { href: base + c.vote.link }, linkedText)]); |
376 | } else if (c.type == "contact" && c.following) { |
377 | var name = c.contact; |
378 | if (typeof c.contactAbout != "undefined") |
379 | name = c.contactAbout.name; |
380 | return h('span.status', |
381 | ['Followed ', |
382 | h('a', { href: base + c.contact }, name)]); |
383 | } else if (c.type == "contact" && !c.following) { |
384 | var name = c.contact; |
385 | if (typeof c.contactAbout != "undefined") |
386 | name = c.contactAbout.name; |
387 | return h('span.status', |
388 | ['Unfollowed ', |
389 | h('a', { href: base + c.contact }, name)]); |
390 | } else if (typeof c == "string") { |
391 | return h('span.status', 'Wrote something private') |
392 | } |
393 | else if (c.type == "about") { |
394 | return [h('span.status', 'Changed something in about'), |
395 | renderDefault(c)]; |
396 | } |
397 | else if (c.type == "issue") { |
398 | return [h('span.status', |
399 | "Created a git issue" + |
400 | (c.repoName != undefined ? " in repo " + c.repoName : ""), |
401 | renderPost(opts, c))]; |
402 | } |
403 | else if (c.type == "git-update") { |
404 | return h('span.status', |
405 | "Did a git update " + |
406 | (c.repoName != undefined ? " in repo " + c.repoName : "") + |
407 | '<br>' + |
408 | (c.commits != undefined ? |
409 | c.commits.map(com => { return "-" +com.title; }).join('<br>') : "")); |
410 | } |
411 | else if (c.type == "ssb-dns") { |
412 | return [h('span.status', 'Updated DNS'), renderDefault(c)]; |
413 | } |
414 | else if (c.type == "pub") { |
415 | return h('span.status', 'Connected to the pub ' + c.address.host); |
416 | } |
417 | else if (c.type == "channel" && c.subscribed) |
418 | return h('span.status', |
419 | 'Subscribed to channel ', |
420 | h('a', |
421 | { href: base + 'channel/' + c.channel }, |
422 | '#' + c.channel)); |
423 | else if (c.type == "channel" && !c.subscribed) |
424 | return h('span.status', |
425 | 'Unsubscribed from channel ', |
426 | h('a', |
427 | { href: base + 'channel/' + c.channel }, |
428 | '#' + c.channel)) |
429 | else return renderDefault(c); |
430 | } |
431 | |
432 | function renderPost(opts, c) { |
433 | opts.mentions = {}; |
434 | if (Array.isArray(c.mentions)) { |
435 | c.mentions.forEach(function (link) { |
436 | if (link && link.name && link.link) |
437 | opts.mentions[link.name] = link.link; |
438 | }); |
439 | } |
440 | var s = h('section'); |
441 | s.innerHTML = marked(String(c.text), opts.marked); |
442 | return s; |
443 | } |
444 | |
445 | function renderDefault(c) { |
446 | return h('pre', JSON.stringify(c, 0, 2)); |
447 | } |
448 |
Built with git-ssb-web