@theia/core
Version:
Theia is a cloud & desktop IDE framework implemented in TypeScript.
109 lines (98 loc) • 4.11 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2017 TypeFox and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************
import { injectable } from 'inversify';
import { isObject, isString } from '../common';
export interface LabelIcon {
name: string;
animation?: string;
}
export namespace LabelIcon {
export function is(val: unknown): val is LabelIcon {
return isObject<LabelIcon>(val) && isString(val.name);
}
}
export type LabelPart = string | LabelIcon;
()
export class LabelParser {
/**
* Returns an array with parts of the given text.
* These parts are of type LabelPart which can be either a string or a LabelIcon.
* For splitting up the giving text the parser follows this rule:
* The text gets parsed for the following pattern: $(iconName~iconAnimation).
* If the parser finds such pattern a new LabelIcon object
* { name: 'iconName', animation: 'iconAnimation'} is added to the returned array.
* iconName can be for instance the name of an icon of e.g. FontAwesome and the (optional) iconAnimation
* the name of an animation class which must be supported by the particular icon toolkit.
*
* Every string before, between or after such icon patterns gets also added to the array
* before, between or after the related LabelIcon.
*
* @param text - the label text to parse
*/
parse(text: string): LabelPart[] {
const parserArray: LabelPart[] = [];
let arrPointer = 0;
let potentialIcon = '';
for (let idx = 0; idx < text.length; idx++) {
const char = text.charAt(idx);
parserArray[arrPointer] = parserArray[arrPointer] || '';
if (potentialIcon === '') {
if (char === '$') {
potentialIcon += char;
} else {
parserArray[arrPointer] += char;
}
} else if (potentialIcon === '$') {
if (char === '(') {
potentialIcon += char;
} else {
parserArray[arrPointer] += potentialIcon + char;
potentialIcon = '';
}
} else {
if (char === ')') {
const iconClassArr = potentialIcon.substring(2, potentialIcon.length).split('~');
if (parserArray[arrPointer] !== '') {
arrPointer++;
}
parserArray[arrPointer] = { name: iconClassArr[0], animation: iconClassArr[1] };
arrPointer++;
potentialIcon = '';
} else {
potentialIcon += char;
}
}
}
if (potentialIcon !== '') {
parserArray[arrPointer] += potentialIcon;
}
return parserArray;
}
/**
* Strips icon specifiers from the given `text`, leaving only a
* space-separated concatenation of the non-icon segments.
*
* @param text text to be stripped of icon specifiers
* @returns the `text` with icon specifiers stripped out
*/
stripIcons(text: string): string {
return this.parse(text)
.filter(item => !LabelIcon.is(item))
.map(s => (s as string).trim())
.filter(s => s.length)
.join(' ');
}
}