fabric
Version:
Object model for HTML5 canvas, and SVG-to-canvas parser. Backed by jsdom and node-canvas.
107 lines (96 loc) • 3.16 kB
text/typescript
import { getFabricWindow } from '../env';
/**
* Capitalizes a string
* @param {String} string String to capitalize
* @param {Boolean} [firstLetterOnly] If true only first letter is capitalized
* and other letters stay untouched, if false first letter is capitalized
* and other letters are converted to lowercase.
* @return {String} Capitalized version of a string
*/
export const capitalize = (string: string, firstLetterOnly = false): string =>
`${string.charAt(0).toUpperCase()}${
firstLetterOnly ? string.slice(1) : string.slice(1).toLowerCase()
}`;
/**
* Escapes XML in a string
* @param {String} string String to escape
* @return {String} Escaped version of a string
*/
export const escapeXml = (string: string): string =>
string
.replace(/&/g, '&')
.replace(/"/g, '"')
.replace(/'/g, ''')
.replace(/</g, '<')
.replace(/>/g, '>');
let segmenter: Intl.Segmenter | false;
const getSegmenter = () => {
if (!segmenter) {
segmenter =
'Intl' in getFabricWindow() &&
'Segmenter' in Intl &&
new Intl.Segmenter(undefined, {
granularity: 'grapheme',
});
}
return segmenter;
};
/**
* Divide a string in the user perceived single units
* @param {String} textstring String to escape
* @return {Array} array containing the graphemes
*/
export const graphemeSplit = (textstring: string): string[] => {
segmenter || getSegmenter();
if (segmenter) {
const segments = segmenter.segment(textstring);
return Array.from(segments).map(({ segment }) => segment);
}
//Fallback
return graphemeSplitImpl(textstring);
};
const graphemeSplitImpl = (textstring: string): string[] => {
const graphemes: string[] = [];
for (let i = 0, chr; i < textstring.length; i++) {
if ((chr = getWholeChar(textstring, i)) === false) {
continue;
}
graphemes.push(chr as string);
}
return graphemes;
};
// taken from mdn in the charAt doc page.
const getWholeChar = (str: string, i: number): string | false => {
const code = str.charCodeAt(i);
if (isNaN(code)) {
return ''; // Position not found
}
if (code < 0xd800 || code > 0xdfff) {
return str.charAt(i);
}
// High surrogate (could change last hex to 0xDB7F to treat high private
// surrogates as single characters)
if (0xd800 <= code && code <= 0xdbff) {
if (str.length <= i + 1) {
throw 'High surrogate without following low surrogate';
}
const next = str.charCodeAt(i + 1);
if (0xdc00 > next || next > 0xdfff) {
throw 'High surrogate without following low surrogate';
}
return str.charAt(i) + str.charAt(i + 1);
}
// Low surrogate (0xDC00 <= code && code <= 0xDFFF)
if (i === 0) {
throw 'Low surrogate without preceding high surrogate';
}
const prev = str.charCodeAt(i - 1);
// (could change last hex to 0xDB7F to treat high private
// surrogates as single characters)
if (0xd800 > prev || prev > 0xdbff) {
throw 'Low surrogate without preceding high surrogate';
}
// We can pass over low surrogates now as the second component
// in a pair which we have already processed
return false;
};