adaptive-expressions
Version:
Common Expression Language
69 lines (63 loc) • 2.34 kB
text/typescript
/**
* @module adaptive-expressions
*/
/**
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License.
*/
import { Expression } from '../expression';
import { EvaluateExpressionDelegate, ExpressionEvaluator } from '../expressionEvaluator';
import { ExpressionType } from '../expressionType';
import { FunctionUtils } from '../functionUtils';
import { ReturnType } from '../returnType';
/**
* Return the result from adding two or more numbers (pure number case) or concatting two or more strings (other case).
*/
export class Add extends ExpressionEvaluator {
/**
* Initializes a new instance of the [Add](xref:adaptive-expressions.Add) class.
*/
constructor() {
super(ExpressionType.Add, Add.evaluator(), ReturnType.String | ReturnType.Number, Add.validator);
}
/**
* @private
*/
private static evaluator(): EvaluateExpressionDelegate {
return FunctionUtils.applySequenceWithError((args: any[]): any => {
let value: any;
let error: string;
const stringConcat = !FunctionUtils.isNumber(args[0]) || !FunctionUtils.isNumber(args[1]);
if (
(args[0] == null && FunctionUtils.isNumber(args[1])) ||
(args[1] == null && FunctionUtils.isNumber(args[0]))
) {
error = "Operator '+' or add cannot be applied to operands of type 'number' and null object.";
} else if (stringConcat) {
if (args[0] == null && args[1] == null) {
value = '';
} else if (args[0] == null) {
value = args[1].toString();
} else if (args[1] == null) {
value = args[0].toString();
} else {
value = args[0].toString() + args[1].toString();
}
} else {
value = args[0] + args[1];
}
return { value, error };
}, FunctionUtils.verifyNumberOrStringOrNull);
}
/**
* @private
*/
private static validator(expression: Expression): void {
FunctionUtils.validateArityAndAnyType(
expression,
2,
Number.MAX_SAFE_INTEGER,
ReturnType.String | ReturnType.Number,
);
}
}