import type { Nat } from "."; import type { _AddDigit } from "./internal/_AddDigit"; import type { _CompareStrLength } from "./internal/_CompareStrLength"; import type { Digit } from ".."; import type { Arg0, Arg1, Fn } from "../../HKT"; import type { Reverse } from "../../Str/Reverse"; import type { GT } from "../../typeclass/Ord"; import type { _StrToNum } from "../internal/_StrToNum"; /** * Add two {@link Nat}s. * * Sig: `(n: Nat, m: Nat) => Nat` */ export type Add = _StrToNum>; /** * Add two string representations of {@link Nat}s. * @private */ type SAdd = _CompareStrLength extends GT ? Reverse<_SAdd, Reverse>> : Reverse<_SAdd, Reverse>>; type _SAdd< N extends string, M extends string, Carry extends 0 | 1 = 0, Result extends string = "", > = N extends `${infer F extends Digit}${infer R}` ? M extends `${infer G extends Digit}${infer S}` ? _AddDigit extends infer D extends Nat ? `${D}` extends `1${infer D extends Digit}` ? _SAdd : D}`> : D extends Digit ? [D, Carry] extends [9, 1] ? _SAdd : _SAdd : D}`> : never : never : Carry extends 1 ? `${Result}${_SAdd}` : `${Result}${N}` : Carry extends 1 ? `${Result}1` : Result; /** * [Fn] Add two {@link Nat}s. * * Sig: `(n: Nat, m: Nat) => Nat` */ export default interface AddFn extends Fn<[Nat, Nat], Nat> { def: () => Add, Arg1>; }