-
Notifications
You must be signed in to change notification settings - Fork 25
/
Copy pathminidenticons.js
63 lines (58 loc) · 2.63 KB
/
minidenticons.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
// 9 different colors only for easy distinction (also a sweet spot for collisions)
const COLORS_NB = 9
const DEFAULT_SATURATION = 95
const DEFAULT_LIGHTNESS = 45
const MAGIC_NUMBER = 5
/**
* @type {(str: string) => number}
*/
function simpleHash(str) {
return str.split('')
.reduce((hash, char) => (hash ^ char.charCodeAt(0)) * -MAGIC_NUMBER, MAGIC_NUMBER)
>>> 2 // 32 bit unsigned integer conversion disregarding last 2 bits for better randomness
}
/**
* @type {import('.').minidenticon}
*/
export function minidenticon(seed="", saturation=DEFAULT_SATURATION, lightness=DEFAULT_LIGHTNESS, hashFn=simpleHash) {
const hash = hashFn(seed)
// console.log("%c" + hash.toString(2).padStart(32, "0"), "font-family:monospace") // uncomment to debug
const hue = (hash % COLORS_NB) * (360 / COLORS_NB)
return [...Array(seed ? 25 : 0)].reduce((acc, e, i) =>
// testing the 15 lowest weight bits of the hash
hash & (1 << (i % 15)) ?
acc + `<rect x="${i > 14 ? 7 - ~~(i / 5) : ~~(i / 5)}" y="${i % 5}" width="1" height="1"/>`
: acc,
// xmlns attribute added in case of SVG file generation https://developer.mozilla.org/en-US/docs/Web/SVG/Element/svg#sect1
`<svg viewBox="-1.5 -1.5 8 8" xmlns="http://www.w3.org/2000/svg" fill="hsl(${hue} ${saturation}% ${lightness}%)">`
)
+ '</svg>'
}
/**
* @type {void}
*/
export const minidenticonSvg =
// declared as a pure function to be tree-shaken by the bundler
/*@__PURE__*/globalThis.customElements?.get('minidenticon-svg') ? null :
globalThis.customElements?.define('minidenticon-svg',
class MinidenticonSvg extends HTMLElement {
static observedAttributes = ['username', 'saturation', 'lightness']
// private fields to allow Terser mangling
static #memoized = {}
#isConnected = false
connectedCallback() {
this.#setContent()
this.#isConnected = true
}
// attributeChangedCallback() is called for every observed attribute before connectedCallback()
attributeChangedCallback() { if (this.#isConnected) this.#setContent() }
#setContent() {
const args = MinidenticonSvg.observedAttributes
.map(key => this.getAttribute(key) || undefined)
const memoKey = args.join(',')
this.innerHTML = MinidenticonSvg.#memoized[memoKey] ??=
// @ts-ignore
minidenticon(...args)
}
}
)