1 Star 2 Fork 1

wx-fork / unioffice

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
range.go 3.25 KB
一键复制 编辑 原始数据 按行查看 历史
Todd 提交于 2017-09-20 19:04 . formula: compatibility & bug fixes
// Copyright 2017 Baliance. All rights reserved.
//
// Use of this source code is governed by the terms of the Affero GNU General
// Public License version 3.0 as published by the Free Software Foundation and
// appearing in the file LICENSE included in the packaging of this file. A
// commercial license can be purchased by contacting sales@baliance.com.
package formula
import (
"fmt"
"strconv"
"strings"
)
// Range is a range expression that when evaluated returns a list of Results.
type Range struct {
from, to Expression
}
// NewRange constructs a new range.
func NewRange(from, to Expression) Expression {
return Range{from, to}
}
// Eval evaluates the range returning a list of results or an error.
func (r Range) Eval(ctx Context, ev Evaluator) Result {
from := r.from.Reference(ctx, ev)
to := r.to.Reference(ctx, ev)
if from.Type == ReferenceTypeCell && to.Type == ReferenceTypeCell {
return resultFromCellRange(ctx, ev, from.Value, to.Value)
}
return MakeErrorResult("invalid range " + from.Value + " to " + to.Value)
}
func (r Range) Reference(ctx Context, ev Evaluator) Reference {
from := r.from.Reference(ctx, ev)
to := r.to.Reference(ctx, ev)
if from.Type == ReferenceTypeCell && to.Type == ReferenceTypeCell {
return MakeRangeReference(from.Value + ":" + to.Value)
}
return ReferenceInvalid
}
// TODO: move these somewhere to remove duplication
func ParseCellReference(s string) (col string, row uint32, err error) {
s = strings.Replace(s, "$", "", -1)
split := -1
lfor:
for i := 0; i < len(s); i++ {
switch {
case s[i] >= '0' && s[i] <= '9':
split = i
break lfor
}
}
switch split {
case 0:
return col, row, fmt.Errorf("no letter prefix in %s", s)
case -1:
return col, row, fmt.Errorf("no digits in %s", s)
}
col = s[0:split]
r64, err := strconv.ParseUint(s[split:], 10, 32)
row = uint32(r64)
return col, row, err
}
// ColumnToIndex maps a column to a zero based index (e.g. A = 0, B = 1, AA = 26)
func ColumnToIndex(col string) uint32 {
col = strings.ToUpper(col)
res := uint32(0)
for _, c := range col {
res *= 26
res += uint32(c - 'A' + 1)
}
return res - 1
}
// IndexToColumn maps a column number to a coumn name (e.g. 0 = A, 1 = B, 26 = AA)
func IndexToColumn(col uint32) string {
col++
res := []byte{}
for col > 26 {
res = append(res, byte('A'+(col-1)%26))
col /= 26
}
res = append(res, byte('A'+(col-1)%26))
// reverse it
for i := 0; i < len(res)/2; i++ {
res[i], res[len(res)-i-1] = res[len(res)-i-1], res[i]
}
return string(res)
}
func resultFromCellRange(ctx Context, ev Evaluator, from, to string) Result {
fc, fr, fe := ParseCellReference(from)
tc, tr, te := ParseCellReference(to)
if fe != nil {
return MakeErrorResult("unable to parse range " + from)
}
if te != nil {
return MakeErrorResult("unable to parse range " + to)
}
bc := ColumnToIndex(fc)
ec := ColumnToIndex(tc)
arr := [][]Result{}
for r := fr; r <= tr; r++ {
args := []Result{}
for c := bc; c <= ec; c++ {
res := ctx.Cell(fmt.Sprintf("%s%d", IndexToColumn(c), r), ev)
args = append(args, res)
}
arr = append(arr, args)
}
// for a single row, just return a list
if len(arr) == 1 {
// single cell result
if len(arr[0]) == 1 {
return arr[0][0]
}
return MakeListResult(arr[0])
}
return MakeArrayResult(arr)
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/wx-fork/unioffice.git
git@gitee.com:wx-fork/unioffice.git
wx-fork
unioffice
unioffice
v0.4.0

搜索帮助

344bd9b3 5694891 D2dac590 5694891