mirror of
https://github.com/unidoc/unioffice.git
synced 2025-04-27 13:48:54 +08:00

This adds support for extracting a cell's formatted value according to the number format applied to the cell. To do this we need to implement a parser for Excel style format strings and support formatting numbers according to that style. This also enhances the General formatting to be much closer to what Excel normally does. There are likely still a few corner cases where Excel and gooxml differ, but hopefully not too many.
78 lines
2.0 KiB
Go
78 lines
2.0 KiB
Go
// 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 spreadsheet
|
|
|
|
import (
|
|
"baliance.com/gooxml/spreadsheet/formula"
|
|
)
|
|
|
|
func newEvalContext(s *Sheet) *evalContext {
|
|
return &evalContext{s, make(map[string]struct{})}
|
|
}
|
|
|
|
type evalContext struct {
|
|
s *Sheet
|
|
evaluating map[string]struct{}
|
|
}
|
|
|
|
func (e *evalContext) Cell(ref string, ev formula.Evaluator) formula.Result {
|
|
c := e.s.Cell(ref)
|
|
|
|
// if we have a formula, evaluate it
|
|
if c.HasFormula() {
|
|
if _, ok := e.evaluating[ref]; ok {
|
|
// recursively evaluating, so bail out
|
|
return formula.MakeErrorResult("recursion detected during evaluation of " + ref)
|
|
}
|
|
e.evaluating[ref] = struct{}{}
|
|
res := ev.Eval(e, c.GetFormula())
|
|
delete(e.evaluating, ref)
|
|
return res
|
|
}
|
|
|
|
// Evaluating an empty cell returns a zero in Excel
|
|
if c.IsEmpty() {
|
|
return formula.MakeNumberResult(0)
|
|
} else if c.IsNumber() {
|
|
v, _ := c.GetValueAsNumber()
|
|
return formula.MakeNumberResult(v)
|
|
} else if c.IsBool() {
|
|
v, _ := c.GetValueAsBool()
|
|
return formula.MakeBoolResult(v)
|
|
}
|
|
|
|
v, _ := c.GetRawValue()
|
|
return formula.MakeStringResult(v)
|
|
|
|
// TODO: handle this properly
|
|
// return formula.MakeErrorResult()
|
|
}
|
|
|
|
func (e *evalContext) Sheet(name string) formula.Context {
|
|
for _, sheet := range e.s.w.Sheets() {
|
|
if sheet.Name() == name {
|
|
return sheet.FormulaContext()
|
|
}
|
|
}
|
|
return formula.InvalidReferenceContext
|
|
}
|
|
|
|
func (e *evalContext) NamedRange(ref string) formula.Reference {
|
|
for _, dn := range e.s.w.DefinedNames() {
|
|
if dn.Name() == ref {
|
|
return formula.MakeRangeReference(dn.Content())
|
|
}
|
|
}
|
|
for _, tbl := range e.s.w.Tables() {
|
|
if tbl.Name() == ref {
|
|
return formula.MakeRangeReference(tbl.Reference())
|
|
}
|
|
}
|
|
return formula.ReferenceInvalid
|
|
}
|