Spaces:
Sleeping
Sleeping
function con(b) { | |
if ((b & 0xc0) === 0x80) { | |
return b & 0x3f; | |
} else { | |
throw new Error("invalid UTF-8 encoding"); | |
} | |
} | |
function code(min, n) { | |
if (n < min || (0xd800 <= n && n < 0xe000) || n >= 0x10000) { | |
throw new Error("invalid UTF-8 encoding"); | |
} else { | |
return n; | |
} | |
} | |
export function decode(bytes) { | |
return _decode(bytes) | |
.map(x => String.fromCharCode(x)) | |
.join(""); | |
} | |
function _decode(bytes) { | |
if (bytes.length === 0) { | |
return []; | |
} | |
/** | |
* 1 byte | |
*/ | |
{ | |
const [b1, ...bs] = bytes; | |
if (b1 < 0x80) { | |
return [code(0x0, b1), ..._decode(bs)]; | |
} | |
if (b1 < 0xc0) { | |
throw new Error("invalid UTF-8 encoding"); | |
} | |
} | |
/** | |
* 2 bytes | |
*/ | |
{ | |
const [b1, b2, ...bs] = bytes; | |
if (b1 < 0xe0) { | |
return [code(0x80, ((b1 & 0x1f) << 6) + con(b2)), ..._decode(bs)]; | |
} | |
} | |
/** | |
* 3 bytes | |
*/ | |
{ | |
const [b1, b2, b3, ...bs] = bytes; | |
if (b1 < 0xf0) { | |
return [ | |
code(0x800, ((b1 & 0x0f) << 12) + (con(b2) << 6) + con(b3)), | |
..._decode(bs) | |
]; | |
} | |
} | |
/** | |
* 4 bytes | |
*/ | |
{ | |
const [b1, b2, b3, b4, ...bs] = bytes; | |
if (b1 < 0xf8) { | |
return [ | |
code( | |
0x10000, | |
((((b1 & 0x07) << 18) + con(b2)) << 12) + (con(b3) << 6) + con(b4) | |
), | |
..._decode(bs) | |
]; | |
} | |
} | |
throw new Error("invalid UTF-8 encoding"); | |
} | |