202 lines
4.4 KiB
V
202 lines
4.4 KiB
V
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
|
// Use of this source code is governed by an MIT license
|
|
// that can be found in the LICENSE file.
|
|
module parser
|
|
|
|
import v.ast
|
|
import v.table
|
|
import v.token
|
|
|
|
fn (mut p Parser) if_expr() ast.IfExpr {
|
|
pos := p.tok.position()
|
|
mut branches := []ast.IfBranch{}
|
|
mut has_else := false
|
|
for p.tok.kind in [.key_if, .key_else] {
|
|
p.inside_if = true
|
|
start_pos := p.tok.position()
|
|
mut comment := ast.Comment{}
|
|
if p.tok.kind == .key_if {
|
|
p.check(.key_if)
|
|
} else {
|
|
// if p.tok.kind == .comment {
|
|
// p.error('place comments inside {}')
|
|
// }
|
|
// comment = p.check_comment()
|
|
p.check(.key_else)
|
|
if p.tok.kind == .key_if {
|
|
p.check(.key_if)
|
|
} else {
|
|
has_else = true
|
|
p.inside_if = false
|
|
end_pos := p.prev_tok.position()
|
|
branches << ast.IfBranch{
|
|
stmts: p.parse_block()
|
|
pos: start_pos.extend(end_pos)
|
|
comment: comment
|
|
}
|
|
break
|
|
}
|
|
}
|
|
mut cond := ast.Expr{}
|
|
mut is_or := false
|
|
// `if x := opt() {`
|
|
if p.peek_tok.kind == .decl_assign {
|
|
is_or = true
|
|
p.open_scope()
|
|
var_pos := p.tok.position()
|
|
var_name := p.check_name()
|
|
p.check(.decl_assign)
|
|
expr := p.expr(0)
|
|
p.scope.register(var_name, ast.Var{
|
|
name: var_name
|
|
expr: expr
|
|
pos: var_pos
|
|
})
|
|
cond = ast.IfGuardExpr{
|
|
var_name: var_name
|
|
expr: expr
|
|
}
|
|
} else {
|
|
cond = p.expr(0)
|
|
}
|
|
end_pos := p.prev_tok.position()
|
|
p.inside_if = false
|
|
stmts := p.parse_block()
|
|
if is_or {
|
|
p.close_scope()
|
|
}
|
|
branches << ast.IfBranch{
|
|
cond: cond
|
|
stmts: stmts
|
|
pos: start_pos.extend(end_pos)
|
|
comment: ast.Comment{}
|
|
}
|
|
if p.tok.kind != .key_else {
|
|
break
|
|
}
|
|
}
|
|
return ast.IfExpr{
|
|
branches: branches
|
|
pos: pos
|
|
has_else: has_else
|
|
}
|
|
}
|
|
|
|
fn (mut p Parser) match_expr() ast.MatchExpr {
|
|
match_first_pos := p.tok.position()
|
|
p.inside_match = true
|
|
p.check(.key_match)
|
|
is_mut := p.tok.kind in [.key_mut, .key_var]
|
|
mut is_sum_type := false
|
|
if is_mut {
|
|
p.next()
|
|
}
|
|
cond_pos := p.tok.position()
|
|
cond := p.expr(0)
|
|
p.inside_match = false
|
|
p.check(.lcbr)
|
|
mut branches := []ast.MatchBranch{}
|
|
for {
|
|
branch_first_pos := p.tok.position()
|
|
comment := p.check_comment() // comment before {}
|
|
mut exprs := []ast.Expr{}
|
|
p.open_scope()
|
|
// final else
|
|
mut is_else := false
|
|
if p.tok.kind == .key_else {
|
|
is_else = true
|
|
p.next()
|
|
} else if p.tok.kind == .name && (p.tok.lit in table.builtin_type_names || (p.tok.lit[0].is_capital() &&
|
|
!p.tok.lit.is_upper()) || p.peek_tok.kind == .dot) {
|
|
// Sum type match
|
|
// if sym.kind == .sum_type {
|
|
// p.warn('is sum')
|
|
// TODO `exprs << ast.Type{...}
|
|
typ := p.parse_type()
|
|
x := ast.Type{
|
|
typ: typ
|
|
}
|
|
mut expr := ast.Expr{}
|
|
expr = x
|
|
exprs << expr
|
|
p.scope.register('it', ast.Var{
|
|
name: 'it'
|
|
typ: typ.to_ptr()
|
|
pos: cond_pos
|
|
is_used: true
|
|
})
|
|
// TODO
|
|
if p.tok.kind == .comma {
|
|
p.next()
|
|
p.parse_type()
|
|
}
|
|
is_sum_type = true
|
|
// Make sure a variable used for the sum type match
|
|
mut var_name := ''
|
|
match cond {
|
|
ast.Ident {
|
|
var_name = it.name
|
|
}
|
|
else {
|
|
// p.error('only variables can be used in sum types matches')
|
|
}
|
|
}
|
|
if var_name != '' {
|
|
// Register a shadow variable with the actual type
|
|
// (this replaces the old `it`)
|
|
// TODO doesn't work right now
|
|
p.scope.register(var_name, ast.Var{
|
|
name: var_name
|
|
typ: typ.to_ptr()
|
|
})
|
|
// println(var_name)
|
|
}
|
|
} else {
|
|
// Expression match
|
|
for {
|
|
p.inside_match_case = true
|
|
expr := p.expr(0)
|
|
p.inside_match_case = false
|
|
exprs << expr
|
|
if p.tok.kind != .comma {
|
|
break
|
|
}
|
|
p.check(.comma)
|
|
}
|
|
}
|
|
branch_last_pos := p.tok.position()
|
|
// p.warn('match block')
|
|
stmts := p.parse_block()
|
|
pos := token.Position{
|
|
line_nr: branch_first_pos.line_nr
|
|
pos: branch_first_pos.pos
|
|
len: branch_last_pos.pos - branch_first_pos.pos + branch_last_pos.len
|
|
}
|
|
branches << ast.MatchBranch{
|
|
exprs: exprs
|
|
stmts: stmts
|
|
pos: pos
|
|
comment: comment
|
|
is_else: is_else
|
|
}
|
|
p.close_scope()
|
|
if p.tok.kind == .rcbr {
|
|
break
|
|
}
|
|
}
|
|
match_last_pos := p.tok.position()
|
|
pos := token.Position{
|
|
line_nr: match_first_pos.line_nr
|
|
pos: match_first_pos.pos
|
|
len: match_last_pos.pos - match_first_pos.pos + match_last_pos.len
|
|
}
|
|
p.check(.rcbr)
|
|
return ast.MatchExpr{
|
|
branches: branches
|
|
cond: cond
|
|
is_sum_type: is_sum_type
|
|
pos: pos
|
|
is_mut: is_mut
|
|
}
|
|
}
|