import { Color } from '../common/color.js';
import { FileAccess } from '../common/network.js';
import { URI } from '../common/uri.js';
export type CssFragment = string & { readonly __cssFragment: unique symbol };
function asFragment(raw: string): CssFragment {
return raw as CssFragment;
}
export function asCssValueWithDefault(cssPropertyValue: string | undefined, dflt: string): string {
if (cssPropertyValue !== undefined) {
const variableMatch = cssPropertyValue.match(/^\s*var\((.+)\)$/);
if (variableMatch) {
const varArguments = variableMatch[1].split(',', 2);
if (varArguments.length === 2) {
dflt = asCssValueWithDefault(varArguments[1].trim(), dflt);
}
return `var(${varArguments[0]}, ${dflt})`;
}
return cssPropertyValue;
}
return dflt;
}
export function sizeValue(value: string): CssFragment {
const out = value.replaceAll(/[^\w.%+-]/gi, '');
if (out !== value) {
console.warn(`CSS size ${value} modified to ${out} to be safe for CSS`);
}
return asFragment(out);
}
export function hexColorValue(value: string): CssFragment {
const out = value.replaceAll(/[^[0-9a-fA-F#]]/gi, '');
if (out !== value) {
console.warn(`CSS hex color ${value} modified to ${out} to be safe for CSS`);
}
return asFragment(out);
}
export function identValue(value: string): CssFragment {
const out = value.replaceAll(/[^_\-a-z0-9]/gi, '');
if (out !== value) {
console.warn(`CSS ident value ${value} modified to ${out} to be safe for CSS`);
}
return asFragment(out);
}
export function stringValue(value: string): CssFragment {
return asFragment(`'${value.replaceAll(/'/g, '\\000027')}'`);
}
export function asCSSUrl(uri: URI | null | undefined): CssFragment {
if (!uri) {
return asFragment(`url('')`);
}
return inline`url('${asFragment(CSS.escape(FileAccess.uriToBrowserUri(uri).toString(true)))}')`;
}
export function className(value: string, escapingExpected = false): CssFragment {
const out = CSS.escape(value);
if (!escapingExpected && out !== value) {
console.warn(`CSS class name ${value} modified to ${out} to be safe for CSS`);
}
return asFragment(out);
}
type InlineCssTemplateValue = CssFragment | Color;
export function inline(strings: TemplateStringsArray, ...values: InlineCssTemplateValue[]): CssFragment {
return asFragment(strings.reduce((result, str, i) => {
const value = values[i] || '';
return result + str + value;
}, ''));
}
export class Builder {
private readonly _parts: CssFragment[] = [];
push(...parts: CssFragment[]): void {
this._parts.push(...parts);
}
join(joiner = '\n'): CssFragment {
return asFragment(this._parts.join(joiner));
}
}