2023-11-06 10:31:51 +08:00
|
|
|
import AbstractCalculator from './calculator';
|
2023-11-10 14:54:16 +08:00
|
|
|
|
|
|
|
const CALC_UNIT = 'CALC_UNIT';
|
|
|
|
|
2024-04-17 11:02:43 +08:00
|
|
|
const regexp = new RegExp(CALC_UNIT, 'g');
|
|
|
|
|
2023-11-10 14:54:16 +08:00
|
|
|
function unit(value: string | number) {
|
|
|
|
if (typeof value === 'number') {
|
|
|
|
return `${value}${CALC_UNIT}`;
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
2023-11-06 10:31:51 +08:00
|
|
|
|
|
|
|
export default class CSSCalculator extends AbstractCalculator {
|
|
|
|
result: string = '';
|
|
|
|
|
2024-06-05 20:01:19 +08:00
|
|
|
unitlessCssVar: Set<string>;
|
|
|
|
|
2023-11-06 10:31:51 +08:00
|
|
|
lowPriority?: boolean;
|
|
|
|
|
2024-06-05 20:01:19 +08:00
|
|
|
constructor(num: number | string | AbstractCalculator, unitlessCssVar: Set<string>) {
|
2023-11-06 10:31:51 +08:00
|
|
|
super();
|
2024-06-05 20:01:19 +08:00
|
|
|
|
|
|
|
const numType = typeof num;
|
|
|
|
|
|
|
|
this.unitlessCssVar = unitlessCssVar;
|
|
|
|
|
2023-11-06 10:31:51 +08:00
|
|
|
if (num instanceof CSSCalculator) {
|
|
|
|
this.result = `(${num.result})`;
|
2024-06-05 20:01:19 +08:00
|
|
|
} else if (numType === 'number') {
|
|
|
|
this.result = unit(num as number);
|
|
|
|
} else if (numType === 'string') {
|
|
|
|
this.result = num as string;
|
2023-11-06 10:31:51 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-08 14:56:15 +08:00
|
|
|
add(num: number | string | AbstractCalculator): this {
|
2023-11-06 10:31:51 +08:00
|
|
|
if (num instanceof CSSCalculator) {
|
|
|
|
this.result = `${this.result} + ${num.getResult()}`;
|
2023-11-08 14:56:15 +08:00
|
|
|
} else if (typeof num === 'number' || typeof num === 'string') {
|
2023-11-06 10:31:51 +08:00
|
|
|
this.result = `${this.result} + ${unit(num)}`;
|
|
|
|
}
|
|
|
|
this.lowPriority = true;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2023-11-08 14:56:15 +08:00
|
|
|
sub(num: number | string | AbstractCalculator): this {
|
2023-11-06 10:31:51 +08:00
|
|
|
if (num instanceof CSSCalculator) {
|
|
|
|
this.result = `${this.result} - ${num.getResult()}`;
|
2023-11-08 14:56:15 +08:00
|
|
|
} else if (typeof num === 'number' || typeof num === 'string') {
|
2023-11-06 10:31:51 +08:00
|
|
|
this.result = `${this.result} - ${unit(num)}`;
|
|
|
|
}
|
|
|
|
this.lowPriority = true;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2023-11-08 14:56:15 +08:00
|
|
|
mul(num: number | string | AbstractCalculator): this {
|
2023-11-06 10:31:51 +08:00
|
|
|
if (this.lowPriority) {
|
|
|
|
this.result = `(${this.result})`;
|
|
|
|
}
|
|
|
|
if (num instanceof CSSCalculator) {
|
|
|
|
this.result = `${this.result} * ${num.getResult(true)}`;
|
2023-11-08 14:56:15 +08:00
|
|
|
} else if (typeof num === 'number' || typeof num === 'string') {
|
2023-11-06 10:31:51 +08:00
|
|
|
this.result = `${this.result} * ${num}`;
|
|
|
|
}
|
|
|
|
this.lowPriority = false;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2023-11-08 14:56:15 +08:00
|
|
|
div(num: number | string | AbstractCalculator): this {
|
2023-11-06 10:31:51 +08:00
|
|
|
if (this.lowPriority) {
|
|
|
|
this.result = `(${this.result})`;
|
|
|
|
}
|
|
|
|
if (num instanceof CSSCalculator) {
|
|
|
|
this.result = `${this.result} / ${num.getResult(true)}`;
|
2023-11-08 14:56:15 +08:00
|
|
|
} else if (typeof num === 'number' || typeof num === 'string') {
|
2023-11-06 10:31:51 +08:00
|
|
|
this.result = `${this.result} / ${num}`;
|
|
|
|
}
|
|
|
|
this.lowPriority = false;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
getResult(force?: boolean): string {
|
|
|
|
return this.lowPriority || force ? `(${this.result})` : this.result;
|
|
|
|
}
|
|
|
|
|
2023-11-10 14:54:16 +08:00
|
|
|
equal(options?: { unit?: boolean }): string {
|
2024-06-05 20:01:19 +08:00
|
|
|
const { unit: cssUnit } = options || {};
|
|
|
|
|
|
|
|
let mergedUnit: boolean = true;
|
|
|
|
if (typeof cssUnit === 'boolean') {
|
|
|
|
mergedUnit = cssUnit;
|
|
|
|
} else if (Array.from(this.unitlessCssVar).some((cssVar) => this.result.includes(cssVar))) {
|
|
|
|
mergedUnit = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.result = this.result.replace(regexp, mergedUnit ? 'px' : '');
|
2023-11-06 10:31:51 +08:00
|
|
|
if (typeof this.lowPriority !== 'undefined') {
|
|
|
|
return `calc(${this.result})`;
|
|
|
|
}
|
|
|
|
return this.result;
|
|
|
|
}
|
|
|
|
}
|