123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317 |
- /* eslint no-bitwise: "off", max-statements: "off", max-lines: "off" */
- // Taken from: https://github.com/walling/unorm/blob/master/lib/unorm.js
- /*
- * UnicodeNormalizer 1.0.0
- * Copyright (c) 2008 Matsuza
- * Dual licensed under the MIT (MIT-LICENSE.txt) and
- * GPL (GPL-LICENSE.txt) licenses.
- * $Date: 2008-06-05 16:44:17 +0200 (Thu, 05 Jun 2008) $
- * $Rev: 13309 $
- */
- "use strict";
- var primitiveSet = require("../../../object/primitive-set")
- , validValue = require("../../../object/valid-value")
- , data = require("./_data");
- var floor = Math.floor
- , forms = primitiveSet("NFC", "NFD", "NFKC", "NFKD")
- , DEFAULT_FEATURE = [null, 0, {}]
- , CACHE_THRESHOLD = 10
- , SBase = 0xac00
- , LBase = 0x1100
- , VBase = 0x1161
- , TBase = 0x11a7
- , LCount = 19
- , VCount = 21
- , TCount = 28
- , NCount = VCount * TCount
- , SCount = LCount * NCount
- , UChar
- , cache = {}
- , cacheCounter = []
- , fromCache
- , fromData
- , fromCpOnly
- , fromRuleBasedJamo
- , fromCpFilter
- , strategies
- , UCharIterator
- , RecursDecompIterator
- , DecompIterator
- , CompIterator
- , createIterator
- , normalize;
- UChar = function (cp, feature) {
- this.codepoint = cp;
- this.feature = feature;
- };
- // Strategies
- (function () {
- for (var i = 0; i <= 0xff; ++i) cacheCounter[i] = 0;
- }());
- fromCache = function (nextStep, cp, needFeature) {
- var ret = cache[cp];
- if (!ret) {
- ret = nextStep(cp, needFeature);
- if (Boolean(ret.feature) && ++cacheCounter[(cp >> 8) & 0xff] > CACHE_THRESHOLD) {
- cache[cp] = ret;
- }
- }
- return ret;
- };
- fromData = function (next, cp) {
- var hash = cp & 0xff00, dunit = UChar.udata[hash] || {}, feature = dunit[cp];
- return feature ? new UChar(cp, feature) : new UChar(cp, DEFAULT_FEATURE);
- };
- fromCpOnly = function (next, cp, needFeature) {
- return needFeature ? next(cp, needFeature) : new UChar(cp, null);
- };
- fromRuleBasedJamo = function (next, cp, needFeature) {
- var char, base, i, arr, SIndex, TIndex, feature, j;
- if (cp < LBase || (LBase + LCount <= cp && cp < SBase) || SBase + SCount < cp) {
- return next(cp, needFeature);
- }
- if (LBase <= cp && cp < LBase + LCount) {
- char = {};
- base = (cp - LBase) * VCount;
- for (i = 0; i < VCount; ++i) {
- char[VBase + i] = SBase + TCount * (i + base);
- }
- arr = new Array(3);
- arr[2] = char;
- return new UChar(cp, arr);
- }
- SIndex = cp - SBase;
- TIndex = SIndex % TCount;
- feature = [];
- if (TIndex === 0) {
- feature[0] = [LBase + floor(SIndex / NCount), VBase + floor(SIndex % NCount / TCount)];
- feature[2] = {};
- for (j = 1; j < TCount; ++j) {
- feature[2][TBase + j] = cp + j;
- }
- } else {
- feature[0] = [SBase + SIndex - TIndex, TBase + TIndex];
- }
- return new UChar(cp, feature);
- };
- fromCpFilter = function (next, cp, needFeature) {
- return cp < 60 || (cp > 13311 && cp < 42607)
- ? new UChar(cp, DEFAULT_FEATURE)
- : next(cp, needFeature);
- };
- strategies = [fromCpFilter, fromCache, fromCpOnly, fromRuleBasedJamo, fromData];
- UChar.fromCharCode = strategies.reduceRight(function (next, strategy) {
- return function (cp, needFeature) {
- return strategy(next, cp, needFeature);
- };
- }, null);
- UChar.isHighSurrogate = function (cp) {
- return cp >= 0xd800 && cp <= 0xdbff;
- };
- UChar.isLowSurrogate = function (cp) {
- return cp >= 0xdc00 && cp <= 0xdfff;
- };
- UChar.prototype.prepFeature = function () {
- if (!this.feature) {
- this.feature = UChar.fromCharCode(this.codepoint, true).feature;
- }
- };
- UChar.prototype.toString = function () {
- var num;
- if (this.codepoint < 0x10000) return String.fromCharCode(this.codepoint);
- num = this.codepoint - 0x10000;
- return String.fromCharCode(floor(num / 0x400) + 0xd800, num % 0x400 + 0xdc00);
- };
- UChar.prototype.getDecomp = function () {
- this.prepFeature();
- return this.feature[0] || null;
- };
- UChar.prototype.isCompatibility = function () {
- this.prepFeature();
- return Boolean(this.feature[1]) && this.feature[1] & (1 << 8);
- };
- UChar.prototype.isExclude = function () {
- this.prepFeature();
- return Boolean(this.feature[1]) && this.feature[1] & (1 << 9);
- };
- UChar.prototype.getCanonicalClass = function () {
- this.prepFeature();
- return this.feature[1] ? this.feature[1] & 0xff : 0;
- };
- UChar.prototype.getComposite = function (following) {
- var cp;
- this.prepFeature();
- if (!this.feature[2]) return null;
- cp = this.feature[2][following.codepoint];
- return cp ? UChar.fromCharCode(cp) : null;
- };
- UCharIterator = function (str) {
- this.str = str;
- this.cursor = 0;
- };
- UCharIterator.prototype.next = function () {
- if (Boolean(this.str) && this.cursor < this.str.length) {
- var cp = this.str.charCodeAt(this.cursor++), d;
- if (
- UChar.isHighSurrogate(cp) &&
- this.cursor < this.str.length &&
- UChar.isLowSurrogate(d = this.str.charCodeAt(this.cursor))
- ) {
- cp = (cp - 0xd800) * 0x400 + (d - 0xdc00) + 0x10000;
- ++this.cursor;
- }
- return UChar.fromCharCode(cp);
- }
- this.str = null;
- return null;
- };
- RecursDecompIterator = function (it, cano) {
- this.it = it;
- this.canonical = cano;
- this.resBuf = [];
- };
- RecursDecompIterator.prototype.next = function () {
- var recursiveDecomp, uchar;
- recursiveDecomp = function (cano, ucharLoc) {
- var decomp = ucharLoc.getDecomp(), ret, i, a, j;
- if (Boolean(decomp) && !(cano && ucharLoc.isCompatibility())) {
- ret = [];
- for (i = 0; i < decomp.length; ++i) {
- a = recursiveDecomp(cano, UChar.fromCharCode(decomp[i]));
- // Ret.concat(a); //<-why does not this work?
- // following block is a workaround.
- for (j = 0; j < a.length; ++j) ret.push(a[j]);
- }
- return ret;
- }
- return [ucharLoc];
- };
- if (this.resBuf.length === 0) {
- uchar = this.it.next();
- if (!uchar) return null;
- this.resBuf = recursiveDecomp(this.canonical, uchar);
- }
- return this.resBuf.shift();
- };
- DecompIterator = function (it) {
- this.it = it;
- this.resBuf = [];
- };
- DecompIterator.prototype.next = function () {
- var cc, uchar, inspt, uchar2, cc2;
- if (this.resBuf.length === 0) {
- do {
- uchar = this.it.next();
- if (!uchar) break;
- cc = uchar.getCanonicalClass();
- inspt = this.resBuf.length;
- if (cc !== 0) {
- for (inspt; inspt > 0; --inspt) {
- uchar2 = this.resBuf[inspt - 1];
- cc2 = uchar2.getCanonicalClass();
- // eslint-disable-next-line max-depth
- if (cc2 <= cc) break;
- }
- }
- this.resBuf.splice(inspt, 0, uchar);
- } while (cc !== 0);
- }
- return this.resBuf.shift();
- };
- CompIterator = function (it) {
- this.it = it;
- this.procBuf = [];
- this.resBuf = [];
- this.lastClass = null;
- };
- CompIterator.prototype.next = function () {
- var uchar, starter, composite, cc;
- while (this.resBuf.length === 0) {
- uchar = this.it.next();
- if (!uchar) {
- this.resBuf = this.procBuf;
- this.procBuf = [];
- break;
- }
- if (this.procBuf.length === 0) {
- this.lastClass = uchar.getCanonicalClass();
- this.procBuf.push(uchar);
- } else {
- starter = this.procBuf[0];
- composite = starter.getComposite(uchar);
- cc = uchar.getCanonicalClass();
- if (Boolean(composite) && (this.lastClass < cc || this.lastClass === 0)) {
- this.procBuf[0] = composite;
- } else {
- if (cc === 0) {
- this.resBuf = this.procBuf;
- this.procBuf = [];
- }
- this.lastClass = cc;
- this.procBuf.push(uchar);
- }
- }
- }
- return this.resBuf.shift();
- };
- createIterator = function (mode, str) {
- switch (mode) {
- case "NFD":
- return new DecompIterator(new RecursDecompIterator(new UCharIterator(str), true));
- case "NFKD":
- return new DecompIterator(new RecursDecompIterator(new UCharIterator(str), false));
- case "NFC":
- return new CompIterator(
- new DecompIterator(new RecursDecompIterator(new UCharIterator(str), true))
- );
- case "NFKC":
- return new CompIterator(
- new DecompIterator(new RecursDecompIterator(new UCharIterator(str), false))
- );
- default:
- throw new Error(mode + " is invalid");
- }
- };
- normalize = function (mode, str) {
- var it = createIterator(mode, str), ret = "", uchar;
- while ((uchar = it.next())) ret += uchar.toString();
- return ret;
- };
- /* Unicode data */
- UChar.udata = data;
- module.exports = function (/* Form*/) {
- var str = String(validValue(this)), form = arguments[0];
- if (form === undefined) form = "NFC";
- else form = String(form);
- if (!forms[form]) throw new RangeError("Invalid normalization form: " + form);
- return normalize(form, str);
- };
|