-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
There remains a mountain of bugs and TODOs, but this a big step toward much more substantial vimscript support. Refs #463 Fixes #7136, fixes #7155
- Loading branch information
Showing
21 changed files
with
3,445 additions
and
117 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
import { optWhitespace, Parser, whitespace } from 'parsimmon'; | ||
import { VimState } from '../../state/vimState'; | ||
import { StatusBar } from '../../statusBar'; | ||
import { ExCommand } from '../../vimscript/exCommand'; | ||
import { EvaluationContext } from '../../vimscript/expression/evaluate'; | ||
import { expressionParser } from '../../vimscript/expression/parser'; | ||
import { Expression } from '../../vimscript/expression/types'; | ||
import { displayValue } from '../../vimscript/expression/displayValue'; | ||
|
||
export class EchoCommand extends ExCommand { | ||
public static argParser(echoArgs: { sep: string; error: boolean }): Parser<EchoCommand> { | ||
return optWhitespace | ||
.then(expressionParser.sepBy(whitespace)) | ||
.map((expressions) => new EchoCommand(echoArgs, expressions)); | ||
} | ||
|
||
private sep: string; | ||
private error: boolean; | ||
private expressions: Expression[]; | ||
private constructor(args: { sep: string; error: boolean }, expressions: Expression[]) { | ||
super(); | ||
this.sep = args.sep; | ||
this.error = args.error; | ||
this.expressions = expressions; | ||
} | ||
|
||
public override neovimCapable(): boolean { | ||
return true; | ||
} | ||
|
||
public async execute(vimState: VimState): Promise<void> { | ||
const ctx = new EvaluationContext(); | ||
const values = this.expressions.map((x) => ctx.evaluate(x)); | ||
const message = values.map((v) => displayValue(v)).join(this.sep); | ||
StatusBar.setText(vimState, message, this.error); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,40 @@ | ||
import { optWhitespace, Parser } from 'parsimmon'; | ||
import { VimState } from '../../state/vimState'; | ||
import { ExCommand } from '../../vimscript/exCommand'; | ||
import { expressionParser, functionCallParser } from '../../vimscript/expression/parser'; | ||
import { Expression } from '../../vimscript/expression/types'; | ||
import { EvaluationContext } from '../../vimscript/expression/evaluate'; | ||
|
||
export class EvalCommand extends ExCommand { | ||
public static argParser: Parser<EvalCommand> = optWhitespace | ||
.then(expressionParser) | ||
.map((expression) => new EvalCommand(expression)); | ||
|
||
private expression: Expression; | ||
private constructor(expression: Expression) { | ||
super(); | ||
this.expression = expression; | ||
} | ||
|
||
public async execute(vimState: VimState): Promise<void> { | ||
const ctx = new EvaluationContext(); | ||
ctx.evaluate(this.expression); | ||
} | ||
} | ||
|
||
export class CallCommand extends ExCommand { | ||
public static argParser: Parser<CallCommand> = optWhitespace | ||
.then(functionCallParser) | ||
.map((call) => new CallCommand(call)); | ||
|
||
private expression: Expression; | ||
private constructor(funcCall: Expression) { | ||
super(); | ||
this.expression = funcCall; | ||
} | ||
|
||
public async execute(vimState: VimState): Promise<void> { | ||
const ctx = new EvaluationContext(); | ||
ctx.evaluate(this.expression); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,154 @@ | ||
// eslint-disable-next-line id-denylist | ||
import { alt, optWhitespace, Parser, seq, string, whitespace } from 'parsimmon'; | ||
import { env } from 'process'; | ||
import { VimState } from '../../state/vimState'; | ||
import { StatusBar } from '../../statusBar'; | ||
import { ExCommand } from '../../vimscript/exCommand'; | ||
import { | ||
add, | ||
concat, | ||
divide, | ||
modulo, | ||
multiply, | ||
str, | ||
subtract, | ||
} from '../../vimscript/expression/build'; | ||
import { EvaluationContext } from '../../vimscript/expression/evaluate'; | ||
import { | ||
envVariableParser, | ||
expressionParser, | ||
optionParser, | ||
registerParser, | ||
variableParser, | ||
} from '../../vimscript/expression/parser'; | ||
import { | ||
EnvVariableExpression, | ||
Expression, | ||
OptionExpression, | ||
RegisterExpression, | ||
VariableExpression, | ||
} from '../../vimscript/expression/types'; | ||
import { displayValue } from '../../vimscript/expression/displayValue'; | ||
import { ErrorCode, VimError } from '../../error'; | ||
|
||
export type LetCommandOperation = '=' | '+=' | '-=' | '*=' | '/=' | '%=' | '.=' | '..='; | ||
export type LetCommandVariable = | ||
| VariableExpression | ||
| OptionExpression | ||
| RegisterExpression | ||
| EnvVariableExpression; | ||
export type LetCommandArgs = | ||
| { | ||
operation: LetCommandOperation; | ||
variable: LetCommandVariable; | ||
expression: Expression; | ||
lock: boolean; | ||
} | ||
| { | ||
operation: 'print'; | ||
variables: LetCommandVariable[]; | ||
}; | ||
|
||
const operationParser: Parser<LetCommandOperation> = alt( | ||
string('='), | ||
string('+='), | ||
string('-='), | ||
string('*='), | ||
string('/='), | ||
string('%='), | ||
string('.='), | ||
string('..='), | ||
); | ||
|
||
const letVarParser = alt<LetCommandVariable>( | ||
variableParser, | ||
optionParser, | ||
envVariableParser, | ||
registerParser, | ||
); | ||
|
||
export class LetCommand extends ExCommand { | ||
// TODO: Support unpacking | ||
// TODO: Support indexing | ||
// TODO: Support slicing | ||
public static readonly argParser = (lock: boolean) => | ||
alt<LetCommand>( | ||
// `:let {var} = {expr}` | ||
// `:let {var} += {expr}` | ||
// `:let {var} -= {expr}` | ||
// `:let {var} .= {expr}` | ||
whitespace.then( | ||
seq(letVarParser, operationParser.wrap(optWhitespace, optWhitespace), expressionParser).map( | ||
([variable, operation, expression]) => | ||
new LetCommand({ | ||
operation, | ||
variable, | ||
expression, | ||
lock, | ||
}), | ||
), | ||
), | ||
// `:let` | ||
// `:let {var-name} ...` | ||
optWhitespace | ||
.then(letVarParser.sepBy(whitespace)) | ||
.map((variables) => new LetCommand({ operation: 'print', variables })), | ||
); | ||
|
||
private args: LetCommandArgs; | ||
constructor(args: LetCommandArgs) { | ||
super(); | ||
this.args = args; | ||
} | ||
|
||
async execute(vimState: VimState): Promise<void> { | ||
const context = new EvaluationContext(); | ||
if (this.args.operation === 'print') { | ||
if (this.args.variables.length === 0) { | ||
// TODO | ||
} else { | ||
const variable = this.args.variables[this.args.variables.length - 1]; | ||
const value = context.evaluate(variable); | ||
const prefix = value.type === 'number' ? '#' : value.type === 'funcref' ? '*' : ''; | ||
StatusBar.setText(vimState, `${variable.name} ${prefix}${displayValue(value)}`); | ||
} | ||
} else { | ||
const variable = this.args.variable; | ||
|
||
if (this.args.lock) { | ||
if (this.args.operation !== '=') { | ||
throw VimError.fromCode(ErrorCode.CannotModifyExistingVariable); | ||
} else if (this.args.variable.type !== 'variable') { | ||
// TODO: this error message should vary by type | ||
throw VimError.fromCode(ErrorCode.CannotLockARegister); | ||
} | ||
} | ||
|
||
let value = context.evaluate(this.args.expression); | ||
if (variable.type === 'variable') { | ||
if (this.args.operation === '+=') { | ||
value = context.evaluate(add(variable, value)); | ||
} else if (this.args.operation === '-=') { | ||
value = context.evaluate(subtract(variable, value)); | ||
} else if (this.args.operation === '*=') { | ||
value = context.evaluate(multiply(variable, value)); | ||
} else if (this.args.operation === '/=') { | ||
value = context.evaluate(divide(variable, value)); | ||
} else if (this.args.operation === '%=') { | ||
value = context.evaluate(modulo(variable, value)); | ||
} else if (this.args.operation === '.=') { | ||
value = context.evaluate(concat(variable, value)); | ||
} else if (this.args.operation === '..=') { | ||
value = context.evaluate(concat(variable, value)); | ||
} | ||
context.setVariable(variable, value, this.args.lock); | ||
} else if (variable.type === 'register') { | ||
// TODO | ||
} else if (variable.type === 'option') { | ||
// TODO | ||
} else if (variable.type === 'env_variable') { | ||
value = str(env[variable.name] ?? ''); | ||
} | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.