2026-02-09 22:15:33 -06:00
|
|
|
import {Command, ParseContext, StrTerm, unwrapString, wrapString} from "./command.js";
|
2025-11-11 21:37:32 -06:00
|
|
|
import {LexInput} from "../lexer.js";
|
2026-02-09 18:09:47 -06:00
|
|
|
import {StrVM} from "../vm.js";
|
|
|
|
|
import {Awaitable} from "../../util/types.js";
|
2025-11-11 21:37:32 -06:00
|
|
|
|
|
|
|
|
export type EncloseData = {
|
|
|
|
|
left?: StrTerm,
|
|
|
|
|
right?: StrTerm,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export class Enclose extends Command<EncloseData> {
|
|
|
|
|
attemptParse(context: ParseContext): EncloseData {
|
|
|
|
|
return {
|
|
|
|
|
left: context.popOptionalTerm(),
|
|
|
|
|
right: context.popOptionalTerm(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getDisplayName(): string {
|
|
|
|
|
return 'enclose'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
isParseCandidate(token: LexInput): boolean {
|
|
|
|
|
return this.isKeyword(token, 'enclose')
|
|
|
|
|
}
|
2026-02-09 18:09:47 -06:00
|
|
|
|
|
|
|
|
execute(vm: StrVM, data: EncloseData): Awaitable<StrVM> {
|
2026-02-09 22:15:33 -06:00
|
|
|
return vm.tapInPlace(ctx => {
|
2026-02-09 18:09:47 -06:00
|
|
|
const [left, right] = this.determineSurroundingStrings(
|
|
|
|
|
data.left ? ctx.resolveString(data.left) : undefined,
|
|
|
|
|
data.right ? ctx.resolveString(data.right) : undefined,
|
|
|
|
|
)
|
|
|
|
|
|
2026-02-09 22:15:33 -06:00
|
|
|
return ctx.replaceSubjectAsString(sub => `${left}${sub}${right}`)
|
2026-02-09 18:09:47 -06:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private determineSurroundingStrings(left?: string, right?: string): [string, string] {
|
|
|
|
|
if ( !left ) {
|
|
|
|
|
left = '('
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( !right ) {
|
|
|
|
|
right = ({
|
|
|
|
|
'(': ')',
|
|
|
|
|
'[': ']',
|
|
|
|
|
'{': '}',
|
|
|
|
|
'<': '>',
|
|
|
|
|
})[left] ?? left
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [left, right]
|
|
|
|
|
}
|
2025-11-11 21:37:32 -06:00
|
|
|
}
|