Files
einundzwanzig.space/helpers.js
2025-03-03 12:19:38 +01:00

122 lines
3.8 KiB
JavaScript

const { decode, encode } = require('html-entities')
const { mkdirSync, writeFileSync } = require('fs')
const { dirname, join, resolve } = require('path')
const meta = require('./content/meta.json')
const dir = resolve(__dirname, '.')
// configure markdown-it
const transformer = require('jstransformer')
const { _tr: mdTransformer } = transformer(require('jstransformer-markdown-it'))
const config = {
typographer: true,
html: true
}
// monkey-patch render function to pass custom options
const { render: renderMd } = mdTransformer
mdTransformer.render = str => renderMd(str, config)
// constants
const IS_DEV = process.env.NODE_ENV === 'development'
const HOST = IS_DEV ? 'http://localhost:3000' : 'https://einundzwanzig.space'
// replacements
const replacements = str => str && str.replace(/<\/?u>/g, '').replace(meta.tallycoinUrl, meta.shoutoutUrl)
const stripHTML = str => str && encode(decode(str
.replace(/<br(\s?\/?)>/g, '\n')
.replace(/<\/(p|ul|ol)>/g, '\n\n')
.replace(/(<([^>]+)>)/ig, '')
.trim()
.replace(/\n\s*/g, '\n')), { level: 'xml' })
// meetups
const toMeetupMapInfo = m => {
return {
name: m.name,
latLng: [m.latitude, m.longitude],
url: m.url,
city: m.city,
portalUrl: m.portalLink,
websiteUrl: m.websiteUrl,
twitter: m.twitter_username,
event: m.next_event,
style: {
fill: m.name.startsWith('Einundzwanzig') || m.name.includes('Einezwänzg') || m.name.includes('Eenanzwanzeg') || m.name.includes('Yirmibir') ? 'var(--color-accent)' : 'var(--color-neutral-50)'
}
}
}
// slug
const slugify = str => str.toLowerCase()
.replace(/ä/g, 'ae').replace(/ö/g, 'oe').replace(/ü/g, 'ue')
.replace(/\s+/g, '-').replace(/[^\w\-]+/g, '')
.replace(/\-\-+/g, '-').replace(/^-+/, '').replace(/-+$/, '')
const truncate = (str, wordCount) => {
const words = str.trim().split(/\s(?![^\[]*\])/g)
const head = words.splice(0, wordCount).join(' ')
const tail = words.join(' ')
return [head, tail]
}
// participants
const participantsWithAliases = participants => {
const withAliases = {}
Object.entries(participants).forEach(([id, member]) => {
withAliases[id] = member
const aliases = (member.aliases || []).concat(member.name.toLowerCase())
aliases.forEach(alias => {
const aliasId = alias.toLowerCase()
if (!withAliases[aliasId]) withAliases[aliasId] = member
})
})
return withAliases
}
const participantToId = p => p.replace(/\(.*?\)/, '').trim().toLowerCase()
const random = max => Math.floor(Math.random() * Math.floor(max))
const shuffle = arr => { for (let i = arr.length - 1; i > 0; i--) { const j = Math.floor(Math.random() * i); const temp = arr[i]; arr[i] = arr[j]; arr[j] = temp; }; return arr }
const formatDate = date => (new Date(date)).toISOString().replace(/T.*/, '').split('-').reverse().join('.')
const linkTarget = url => url.startsWith('http') ? '_blank' : null
const assetPath = path => {
if (path.startsWith('http')) return path
let revs
try { revs = require('./generated/rev.json') } catch (error) { }
return `${(revs && revs[path]) || path}`
}
const assetUrl = (path, protocol = 'https') => {
if (IS_DEV && !path.startsWith('http')) protocol = 'http'
const base = path.startsWith('http') ? '' : HOST
let url = `${base}${assetPath(path)}`
if (!url.startsWith(`${protocol}:`)) url = url.replace(/^.*:/, `${protocol}:`)
return url
}
const write = (name, data) => {
const dst = join(dir, name)
mkdirSync(dirname(dst), { recursive: true })
writeFileSync(dst, data)
}
const writeJSON = (name, data) => write(`generated/${name}.json`, JSON.stringify(data, null, 2))
module.exports = {
markdown: mdTransformer.render,
random,
shuffle,
assetUrl,
assetPath,
formatDate,
linkTarget,
replacements,
slugify,
stripHTML,
truncate,
participantsWithAliases,
participantToId,
toMeetupMapInfo,
write,
writeJSON
}