/* Copyright 2014 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ function optimizeCMap(data) { let i = 1; while (i < data.body.length) { if (data.body[i - 1].type === data.body[i].type) { data.body[i - 1].items = data.body[i - 1].items.concat( data.body[i].items ); data.body.splice(i, 1); } else { i++; } } // split into groups with different lengths i = 0; while (i < data.body.length) { const item = data.body[i]; const keys = Object.keys(item.items[0]).filter(function (val) { return typeof item.items[0][val] === "string"; }); let j = 1; while (j < item.items.length) { let different = false; for (let q = 0; q < keys.length && !different; q++) { different = item.items[j - 1][keys[q]].length !== item.items[j][keys[q]].length; } if (different) { break; } j++; } if (j < item.items.length) { data.body.splice(i + 1, 0, { type: item.type, items: item.items.splice(j, item.items.length - j), }); } i++; } // find sequences of single char ranges i = 0; while (i < data.body.length) { const item = data.body[i]; if (item.type === 3 || item.type === 5) { let j = 0; while (j < item.items.length) { const q = j; while ( j < item.items.length && item.items[j].start === item.items[j].end ) { j++; } if (j - q >= 9) { if (j < item.items.length) { data.body.splice(i + 1, 0, { type: item.type, items: item.items.splice(j, item.items.length - j), }); } if (q > 0) { data.body.splice(i + 1, 0, { type: item.type - 1, items: item.items.splice(q, j - q).map(function (val) { return { char: val.start, code: val.code }; }), }); i++; } else { item.type -= 1; item.items = item.items.map(function (val) { return { char: val.start, code: val.code }; }); } continue; } j++; } } i++; } // find sequences of increasing code/ranges order i = 0; while (i < data.body.length) { const item = data.body[i]; if (item.type >= 2 && item.type <= 5) { let j = 1; const startProp = item.type === 2 || item.type === 4 ? "char" : "start"; const endProp = item.type === 2 || item.type === 4 ? "char" : "end"; while (j < item.items.length) { const q = j - 1; while ( j < item.items.length && incHex(item.items[j - 1][endProp]) === item.items[j][startProp] ) { j++; } if (j - q >= 9) { if (j < item.items.length) { data.body.splice(i + 1, 0, { type: item.type, items: item.items.splice(j, item.items.length - j), }); } if (q > 0) { data.body.splice(i + 1, 0, { type: item.type, items: item.items.splice(q, j - q), sequence: true, }); i++; } else { item.sequence = true; } continue; } j++; } } i++; } // split non-sequences two groups where codes are close i = 0; while (i < data.body.length) { const item = data.body[i]; if (!item.sequence && (item.type === 2 || item.type === 3)) { const subitems = item.items; const codes = subitems.map(function (val) { return val.code; }); codes.sort(function (a, b) { return a - b; }); const maxDistance = 100, minItems = 10, itemsPerBucket = 50; if (subitems.length > minItems && codes.at(-1) - codes[0] > maxDistance) { const gapsCount = Math.max(2, (subitems.length / itemsPerBucket) | 0); const gaps = []; for (let q = 0; q < gapsCount; q++) { gaps.push({ length: 0 }); } for (let j = 1; j < codes.length; j++) { const gapLength = codes[j] - codes[j - 1]; let q = 0; while (q < gaps.length && gaps[q].length > gapLength) { q++; } if (q >= gaps.length) { continue; } let q0 = q; while (q < gaps.length) { if (gaps[q].length < gaps[q0].length) { q0 = q; } q++; } gaps[q0] = { length: gapLength, boundary: codes[j] }; } const groups = gaps .filter(function (g) { return g.length >= maxDistance; }) .map(function (g) { return g.boundary; }); groups.sort(function (a, b) { return a - b; }); if (groups.length > 1) { const buckets = [(item.items = [])]; for (let j = 0; j < groups.length; j++) { const newItem = { type: item.type, items: [] }; buckets.push(newItem.items); i++; data.body.splice(i, 0, newItem); } for (const subitem of subitems) { const { code } = subitem; let q = 0; while (q < groups.length && groups[q] <= code) { q++; } buckets[q].push(subitem); } } } } i++; } } function incHex(a) { let c = 1, s = ""; for (let i = a.length - 1; i >= 0; i--) { c += parseInt(a[i], 16); if (c >= 16) { s = "0" + s; c = 1; } else { s = c.toString(16) + s; c = 0; } } return s; } export { optimizeCMap };