Compare commits
4 Commits
feature/da
...
dynamic-ar
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bf894d0045 | ||
|
|
3a399fc91a | ||
|
|
71a2bb2dca | ||
|
|
889845b948 |
7
Makefile
7
Makefile
@@ -31,7 +31,12 @@ clean: remove-artifacts
|
|||||||
rm -r -f base/target
|
rm -r -f base/target
|
||||||
rm -r -f xlsx/target
|
rm -r -f xlsx/target
|
||||||
rm -r -f bindings/python/target
|
rm -r -f bindings/python/target
|
||||||
rm -r -f bindings/wasm/targets
|
rm -r -f bindings/wasm/target
|
||||||
|
rm -r -f bindings/wasm/pkg
|
||||||
|
rm -r -f webapp/IronCalc/node_modules
|
||||||
|
rm -r -f webapp/IronCalc/dist
|
||||||
|
rm -r -f webapp/app.ironcalc.com/frontend/node_modules
|
||||||
|
rm -r -f webapp/app.ironcalc.com/frontend/dist
|
||||||
rm -f cargo-test-*
|
rm -f cargo-test-*
|
||||||
rm -f base/cargo-test-*
|
rm -f base/cargo-test-*
|
||||||
rm -f xlsx/cargo-test-*
|
rm -f xlsx/cargo-test-*
|
||||||
|
|||||||
61
base/CALC.md
Normal file
61
base/CALC.md
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
# Evaluation Strategy
|
||||||
|
|
||||||
|
|
||||||
|
We have a list of the spill cells:
|
||||||
|
|
||||||
|
```
|
||||||
|
// Checks if the array starting at cell will cover cells whose values
|
||||||
|
// has been requested
|
||||||
|
def CheckSpill(cell, array):
|
||||||
|
for c in cell+array:
|
||||||
|
support CellHasBeenRequested(c):
|
||||||
|
if support is not empty:
|
||||||
|
return support
|
||||||
|
return []
|
||||||
|
|
||||||
|
// Fills cells with the result (an array)
|
||||||
|
def FillCells(cell, result):
|
||||||
|
|
||||||
|
|
||||||
|
def EvaluateNodeInContext(node, context):
|
||||||
|
match node:
|
||||||
|
case OP(left, right, op):
|
||||||
|
l = EvaluateNodeInContext(left, context)?
|
||||||
|
r = EvaluateNodeInContext(left, context)?
|
||||||
|
return op(l, r)
|
||||||
|
case FUNCTION(args, fn):
|
||||||
|
...
|
||||||
|
case CELL(cell):
|
||||||
|
EvaluateCell(cell)
|
||||||
|
case RANGE(start, end):
|
||||||
|
...
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def EvaluateCell(cell):
|
||||||
|
if IsCellEvaluating(cell):
|
||||||
|
return CIRC
|
||||||
|
MarkEvaluating(cell)
|
||||||
|
result = EvaluateNodeInContext(cell.formula, cell)
|
||||||
|
if isSpill(result):
|
||||||
|
CheckSpill(cell, array)?
|
||||||
|
FillCells(result)
|
||||||
|
|
||||||
|
|
||||||
|
def EvaluateWorkbook():
|
||||||
|
spill_cells = [cell_1, ...., cell_n];
|
||||||
|
|
||||||
|
|
||||||
|
for cell in spill_cells:
|
||||||
|
result = evaluate(cell)
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# When updating a cell value
|
||||||
|
|
||||||
|
If it was a spill cell we nee
|
||||||
@@ -22,7 +22,7 @@ impl Model {
|
|||||||
.cell(row, column)
|
.cell(row, column)
|
||||||
.and_then(|c| c.get_formula())
|
.and_then(|c| c.get_formula())
|
||||||
{
|
{
|
||||||
let node = &self.parsed_formulas[sheet as usize][f as usize].clone();
|
let node = &self.parsed_formulas[sheet as usize][f as usize].0.clone();
|
||||||
let cell_reference = CellReferenceRC {
|
let cell_reference = CellReferenceRC {
|
||||||
sheet: self.workbook.worksheets[sheet as usize].get_name(),
|
sheet: self.workbook.worksheets[sheet as usize].get_name(),
|
||||||
row,
|
row,
|
||||||
|
|||||||
@@ -77,8 +77,6 @@ impl Model {
|
|||||||
match to_f64(&node) {
|
match to_f64(&node) {
|
||||||
Ok(f2) => match op(f1, f2) {
|
Ok(f2) => match op(f1, f2) {
|
||||||
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
||||||
Err(Error::DIV) => data_row.push(ArrayNode::Error(Error::DIV)),
|
|
||||||
Err(Error::VALUE) => data_row.push(ArrayNode::Error(Error::VALUE)),
|
|
||||||
Err(e) => data_row.push(ArrayNode::Error(e)),
|
Err(e) => data_row.push(ArrayNode::Error(e)),
|
||||||
},
|
},
|
||||||
Err(err) => data_row.push(ArrayNode::Error(err)),
|
Err(err) => data_row.push(ArrayNode::Error(err)),
|
||||||
@@ -100,8 +98,6 @@ impl Model {
|
|||||||
match to_f64(&node) {
|
match to_f64(&node) {
|
||||||
Ok(f1) => match op(f1, f2) {
|
Ok(f1) => match op(f1, f2) {
|
||||||
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
||||||
Err(Error::DIV) => data_row.push(ArrayNode::Error(Error::DIV)),
|
|
||||||
Err(Error::VALUE) => data_row.push(ArrayNode::Error(Error::VALUE)),
|
|
||||||
Err(e) => data_row.push(ArrayNode::Error(e)),
|
Err(e) => data_row.push(ArrayNode::Error(e)),
|
||||||
},
|
},
|
||||||
Err(err) => data_row.push(ArrayNode::Error(err)),
|
Err(err) => data_row.push(ArrayNode::Error(err)),
|
||||||
@@ -137,10 +133,6 @@ impl Model {
|
|||||||
(Some(v1), Some(v2)) => match (to_f64(v1), to_f64(v2)) {
|
(Some(v1), Some(v2)) => match (to_f64(v1), to_f64(v2)) {
|
||||||
(Ok(f1), Ok(f2)) => match op(f1, f2) {
|
(Ok(f1), Ok(f2)) => match op(f1, f2) {
|
||||||
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
Ok(x) => data_row.push(ArrayNode::Number(x)),
|
||||||
Err(Error::DIV) => data_row.push(ArrayNode::Error(Error::DIV)),
|
|
||||||
Err(Error::VALUE) => {
|
|
||||||
data_row.push(ArrayNode::Error(Error::VALUE))
|
|
||||||
}
|
|
||||||
Err(e) => data_row.push(ArrayNode::Error(e)),
|
Err(e) => data_row.push(ArrayNode::Error(e)),
|
||||||
},
|
},
|
||||||
(Err(e), _) | (_, Err(e)) => data_row.push(ArrayNode::Error(e)),
|
(Err(e), _) | (_, Err(e)) => data_row.push(ArrayNode::Error(e)),
|
||||||
|
|||||||
100
base/src/cell.rs
100
base/src/cell.rs
@@ -64,12 +64,50 @@ impl Cell {
|
|||||||
/// Returns the formula of a cell if any.
|
/// Returns the formula of a cell if any.
|
||||||
pub fn get_formula(&self) -> Option<i32> {
|
pub fn get_formula(&self) -> Option<i32> {
|
||||||
match self {
|
match self {
|
||||||
Cell::CellFormula { f, .. } => Some(*f),
|
Cell::CellFormula { f, .. }
|
||||||
Cell::CellFormulaBoolean { f, .. } => Some(*f),
|
| Cell::CellFormulaBoolean { f, .. }
|
||||||
Cell::CellFormulaNumber { f, .. } => Some(*f),
|
| Cell::CellFormulaNumber { f, .. }
|
||||||
Cell::CellFormulaString { f, .. } => Some(*f),
|
| Cell::CellFormulaString { f, .. }
|
||||||
Cell::CellFormulaError { f, .. } => Some(*f),
|
| Cell::CellFormulaError { f, .. }
|
||||||
_ => None,
|
| Cell::DynamicCellFormula { f, .. }
|
||||||
|
| Cell::DynamicCellFormulaBoolean { f, .. }
|
||||||
|
| Cell::DynamicCellFormulaNumber { f, .. }
|
||||||
|
| Cell::DynamicCellFormulaString { f, .. }
|
||||||
|
| Cell::DynamicCellFormulaError { f, .. } => Some(*f),
|
||||||
|
Cell::EmptyCell { .. }
|
||||||
|
| Cell::BooleanCell { .. }
|
||||||
|
| Cell::NumberCell { .. }
|
||||||
|
| Cell::ErrorCell { .. }
|
||||||
|
| Cell::SharedString { .. }
|
||||||
|
| Cell::SpillNumberCell { .. }
|
||||||
|
| Cell::SpillBooleanCell { .. }
|
||||||
|
| Cell::SpillErrorCell { .. }
|
||||||
|
| Cell::SpillStringCell { .. } => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the dynamic range of a cell if any.
|
||||||
|
pub fn get_dynamic_range(&self) -> Option<(i32, i32)> {
|
||||||
|
match self {
|
||||||
|
Cell::DynamicCellFormula { r, .. } => Some(*r),
|
||||||
|
Cell::DynamicCellFormulaBoolean { r, .. } => Some(*r),
|
||||||
|
Cell::DynamicCellFormulaNumber { r, .. } => Some(*r),
|
||||||
|
Cell::DynamicCellFormulaString { r, .. } => Some(*r),
|
||||||
|
Cell::DynamicCellFormulaError { r, .. } => Some(*r),
|
||||||
|
Cell::EmptyCell { .. }
|
||||||
|
| Cell::BooleanCell { .. }
|
||||||
|
| Cell::NumberCell { .. }
|
||||||
|
| Cell::ErrorCell { .. }
|
||||||
|
| Cell::SharedString { .. }
|
||||||
|
| Cell::CellFormula { .. }
|
||||||
|
| Cell::CellFormulaBoolean { .. }
|
||||||
|
| Cell::CellFormulaNumber { .. }
|
||||||
|
| Cell::CellFormulaString { .. }
|
||||||
|
| Cell::CellFormulaError { .. }
|
||||||
|
| Cell::SpillNumberCell { .. }
|
||||||
|
| Cell::SpillBooleanCell { .. }
|
||||||
|
| Cell::SpillErrorCell { .. }
|
||||||
|
| Cell::SpillStringCell { .. } => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -89,6 +127,15 @@ impl Cell {
|
|||||||
Cell::CellFormulaNumber { s, .. } => *s = style,
|
Cell::CellFormulaNumber { s, .. } => *s = style,
|
||||||
Cell::CellFormulaString { s, .. } => *s = style,
|
Cell::CellFormulaString { s, .. } => *s = style,
|
||||||
Cell::CellFormulaError { s, .. } => *s = style,
|
Cell::CellFormulaError { s, .. } => *s = style,
|
||||||
|
Cell::SpillBooleanCell { s, .. } => *s = style,
|
||||||
|
Cell::SpillNumberCell { s, .. } => *s = style,
|
||||||
|
Cell::SpillStringCell { s, .. } => *s = style,
|
||||||
|
Cell::SpillErrorCell { s, .. } => *s = style,
|
||||||
|
Cell::DynamicCellFormula { s, .. } => *s = style,
|
||||||
|
Cell::DynamicCellFormulaBoolean { s, .. } => *s = style,
|
||||||
|
Cell::DynamicCellFormulaNumber { s, .. } => *s = style,
|
||||||
|
Cell::DynamicCellFormulaString { s, .. } => *s = style,
|
||||||
|
Cell::DynamicCellFormulaError { s, .. } => *s = style,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -104,6 +151,15 @@ impl Cell {
|
|||||||
Cell::CellFormulaNumber { s, .. } => *s,
|
Cell::CellFormulaNumber { s, .. } => *s,
|
||||||
Cell::CellFormulaString { s, .. } => *s,
|
Cell::CellFormulaString { s, .. } => *s,
|
||||||
Cell::CellFormulaError { s, .. } => *s,
|
Cell::CellFormulaError { s, .. } => *s,
|
||||||
|
Cell::SpillBooleanCell { s, .. } => *s,
|
||||||
|
Cell::SpillNumberCell { s, .. } => *s,
|
||||||
|
Cell::SpillStringCell { s, .. } => *s,
|
||||||
|
Cell::SpillErrorCell { s, .. } => *s,
|
||||||
|
Cell::DynamicCellFormula { s, .. } => *s,
|
||||||
|
Cell::DynamicCellFormulaBoolean { s, .. } => *s,
|
||||||
|
Cell::DynamicCellFormulaNumber { s, .. } => *s,
|
||||||
|
Cell::DynamicCellFormulaString { s, .. } => *s,
|
||||||
|
Cell::DynamicCellFormulaError { s, .. } => *s,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,6 +175,15 @@ impl Cell {
|
|||||||
Cell::CellFormulaNumber { .. } => CellType::Number,
|
Cell::CellFormulaNumber { .. } => CellType::Number,
|
||||||
Cell::CellFormulaString { .. } => CellType::Text,
|
Cell::CellFormulaString { .. } => CellType::Text,
|
||||||
Cell::CellFormulaError { .. } => CellType::ErrorValue,
|
Cell::CellFormulaError { .. } => CellType::ErrorValue,
|
||||||
|
Cell::SpillBooleanCell { .. } => CellType::LogicalValue,
|
||||||
|
Cell::SpillNumberCell { .. } => CellType::Number,
|
||||||
|
Cell::SpillStringCell { .. } => CellType::Text,
|
||||||
|
Cell::SpillErrorCell { .. } => CellType::ErrorValue,
|
||||||
|
Cell::DynamicCellFormula { .. } => CellType::Number,
|
||||||
|
Cell::DynamicCellFormulaBoolean { .. } => CellType::LogicalValue,
|
||||||
|
Cell::DynamicCellFormulaNumber { .. } => CellType::Number,
|
||||||
|
Cell::DynamicCellFormulaString { .. } => CellType::Text,
|
||||||
|
Cell::DynamicCellFormulaError { .. } => CellType::ErrorValue,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,7 +201,7 @@ impl Cell {
|
|||||||
Cell::EmptyCell { .. } => CellValue::None,
|
Cell::EmptyCell { .. } => CellValue::None,
|
||||||
Cell::BooleanCell { v, s: _ } => CellValue::Boolean(*v),
|
Cell::BooleanCell { v, s: _ } => CellValue::Boolean(*v),
|
||||||
Cell::NumberCell { v, s: _ } => CellValue::Number(*v),
|
Cell::NumberCell { v, s: _ } => CellValue::Number(*v),
|
||||||
Cell::ErrorCell { ei, .. } => {
|
Cell::ErrorCell { ei, .. } | Cell::SpillErrorCell { ei, .. } => {
|
||||||
let v = ei.to_localized_error_string(language);
|
let v = ei.to_localized_error_string(language);
|
||||||
CellValue::String(v)
|
CellValue::String(v)
|
||||||
}
|
}
|
||||||
@@ -148,14 +213,25 @@ impl Cell {
|
|||||||
};
|
};
|
||||||
CellValue::String(v)
|
CellValue::String(v)
|
||||||
}
|
}
|
||||||
Cell::CellFormula { .. } => CellValue::String("#ERROR!".to_string()),
|
Cell::DynamicCellFormula { .. } | Cell::CellFormula { .. } => {
|
||||||
Cell::CellFormulaBoolean { v, .. } => CellValue::Boolean(*v),
|
CellValue::String("#ERROR!".to_string())
|
||||||
Cell::CellFormulaNumber { v, .. } => CellValue::Number(*v),
|
}
|
||||||
Cell::CellFormulaString { v, .. } => CellValue::String(v.clone()),
|
Cell::DynamicCellFormulaBoolean { v, .. } | Cell::CellFormulaBoolean { v, .. } => {
|
||||||
Cell::CellFormulaError { ei, .. } => {
|
CellValue::Boolean(*v)
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaNumber { v, .. } | Cell::CellFormulaNumber { v, .. } => {
|
||||||
|
CellValue::Number(*v)
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaString { v, .. } | Cell::CellFormulaString { v, .. } => {
|
||||||
|
CellValue::String(v.clone())
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaError { ei, .. } | Cell::CellFormulaError { ei, .. } => {
|
||||||
let v = ei.to_localized_error_string(language);
|
let v = ei.to_localized_error_string(language);
|
||||||
CellValue::String(v)
|
CellValue::String(v)
|
||||||
}
|
}
|
||||||
|
Cell::SpillBooleanCell { v, .. } => CellValue::Boolean(*v),
|
||||||
|
Cell::SpillNumberCell { v, .. } => CellValue::Number(*v),
|
||||||
|
Cell::SpillStringCell { v, .. } => CellValue::String(v.clone()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -186,7 +186,8 @@ pub fn add_implicit_intersection(node: &mut Node, add: bool) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) enum StaticResult {
|
#[derive(Clone)]
|
||||||
|
pub enum StaticResult {
|
||||||
Scalar,
|
Scalar,
|
||||||
Array(i32, i32),
|
Array(i32, i32),
|
||||||
Range(i32, i32),
|
Range(i32, i32),
|
||||||
@@ -222,7 +223,7 @@ fn static_analysis_op_nodes(left: &Node, right: &Node) -> StaticResult {
|
|||||||
// * Array(a, b) if we know it will be an a x b array.
|
// * Array(a, b) if we know it will be an a x b array.
|
||||||
// * Range(a, b) if we know it will be a a x b range.
|
// * Range(a, b) if we know it will be a a x b range.
|
||||||
// * Unknown if we cannot guaranty either
|
// * Unknown if we cannot guaranty either
|
||||||
fn run_static_analysis_on_node(node: &Node) -> StaticResult {
|
pub(crate) fn run_static_analysis_on_node(node: &Node) -> StaticResult {
|
||||||
match node {
|
match node {
|
||||||
Node::BooleanKind(_)
|
Node::BooleanKind(_)
|
||||||
| Node::NumberKind(_)
|
| Node::NumberKind(_)
|
||||||
|
|||||||
@@ -96,7 +96,7 @@ impl Model {
|
|||||||
|
|
||||||
match cell.get_formula() {
|
match cell.get_formula() {
|
||||||
Some(f) => {
|
Some(f) => {
|
||||||
let node = &self.parsed_formulas[sheet_index as usize][f as usize];
|
let node = &self.parsed_formulas[sheet_index as usize][f as usize].0;
|
||||||
matches!(
|
matches!(
|
||||||
node,
|
node,
|
||||||
Node::FunctionKind {
|
Node::FunctionKind {
|
||||||
|
|||||||
@@ -11,8 +11,9 @@ use crate::{
|
|||||||
lexer::LexerMode,
|
lexer::LexerMode,
|
||||||
parser::{
|
parser::{
|
||||||
move_formula::{move_formula, MoveContext},
|
move_formula::{move_formula, MoveContext},
|
||||||
|
static_analysis::{run_static_analysis_on_node, StaticResult},
|
||||||
stringify::{rename_defined_name_in_node, to_rc_format, to_string},
|
stringify::{rename_defined_name_in_node, to_rc_format, to_string},
|
||||||
Node, Parser,
|
ArrayNode, Node, Parser,
|
||||||
},
|
},
|
||||||
token::{get_error_by_name, Error, OpCompare, OpProduct, OpSum, OpUnary},
|
token::{get_error_by_name, Error, OpCompare, OpProduct, OpSum, OpUnary},
|
||||||
types::*,
|
types::*,
|
||||||
@@ -83,6 +84,24 @@ pub(crate) enum ParsedDefinedName {
|
|||||||
InvalidDefinedNameFormula,
|
InvalidDefinedNameFormula,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A support node is either a cell or a range of cells
|
||||||
|
pub(crate) enum SupportNode {
|
||||||
|
/// (sheet, row, column)
|
||||||
|
Cell((u32, i32, i32)),
|
||||||
|
/// (sheet, row, column, height, width)
|
||||||
|
Range((u32, i32, i32, u32, u32))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The state of the computation
|
||||||
|
pub(crate) enum EvaluationState {
|
||||||
|
/// the model is ready for a new evaluation
|
||||||
|
Ready,
|
||||||
|
/// the model is evaluating cells that might spill
|
||||||
|
EvaluatingSpills,
|
||||||
|
/// the model is evaluating cells normally
|
||||||
|
Evaluating
|
||||||
|
}
|
||||||
|
|
||||||
/// A dynamical IronCalc model.
|
/// A dynamical IronCalc model.
|
||||||
///
|
///
|
||||||
/// Its is composed of a `Workbook`. Everything else are dynamical quantities:
|
/// Its is composed of a `Workbook`. Everything else are dynamical quantities:
|
||||||
@@ -99,15 +118,13 @@ pub struct Model {
|
|||||||
/// A Rust internal representation of an Excel workbook
|
/// A Rust internal representation of an Excel workbook
|
||||||
pub workbook: Workbook,
|
pub workbook: Workbook,
|
||||||
/// A list of parsed formulas
|
/// A list of parsed formulas
|
||||||
pub parsed_formulas: Vec<Vec<Node>>,
|
pub parsed_formulas: Vec<Vec<(Node, StaticResult)>>,
|
||||||
/// A list of parsed defined names
|
/// A list of parsed defined names
|
||||||
pub(crate) parsed_defined_names: HashMap<(Option<u32>, String), ParsedDefinedName>,
|
pub(crate) parsed_defined_names: HashMap<(Option<u32>, String), ParsedDefinedName>,
|
||||||
/// An optimization to lookup strings faster
|
/// An optimization to lookup strings faster
|
||||||
pub(crate) shared_strings: HashMap<String, usize>,
|
pub(crate) shared_strings: HashMap<String, usize>,
|
||||||
/// An instance of the parser
|
/// An instance of the parser
|
||||||
pub(crate) parser: Parser,
|
pub(crate) parser: Parser,
|
||||||
/// The list of cells with formulas that are evaluated or being evaluated
|
|
||||||
pub(crate) cells: HashMap<(u32, i32, i32), CellState>,
|
|
||||||
/// The locale of the model
|
/// The locale of the model
|
||||||
pub(crate) locale: Locale,
|
pub(crate) locale: Locale,
|
||||||
/// The language used
|
/// The language used
|
||||||
@@ -116,6 +133,16 @@ pub struct Model {
|
|||||||
pub(crate) tz: Tz,
|
pub(crate) tz: Tz,
|
||||||
/// The view id. A view consists of a selected sheet and ranges.
|
/// The view id. A view consists of a selected sheet and ranges.
|
||||||
pub(crate) view_id: u32,
|
pub(crate) view_id: u32,
|
||||||
|
/// ** Runtime ***
|
||||||
|
/// The list of cells with formulas that are evaluated or being evaluated
|
||||||
|
pub(crate) cells: HashMap<(u32, i32, i32), CellState>,
|
||||||
|
/// The support graph. For a given cell (sheet, row, column) the list of cells and ranges that were requested
|
||||||
|
pub(crate) support_graph: HashMap<(u32, i32, i32), Vec<SupportNode>>,
|
||||||
|
/// If the model is in a switch state then spill cells in the indices should be switched and recalculation redone
|
||||||
|
pub(crate) switch_cells: Option<(i32, i32)>,
|
||||||
|
/// Stack of cells being evaluated
|
||||||
|
pub(crate) stack: Vec<(u32, i32, i32)>,
|
||||||
|
pub(crate) state: EvaluationState,
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: Maybe this should be the same as CellReference
|
// FIXME: Maybe this should be the same as CellReference
|
||||||
@@ -522,14 +549,203 @@ impl Model {
|
|||||||
}
|
}
|
||||||
Ok(format!("{}!{}{}", sheet.name, column, cell_reference.row))
|
Ok(format!("{}!{}{}", sheet.name, column, cell_reference.row))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sets sheet, target_row, target_column, (width, height), &v
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
fn set_spill_cell_with_formula_value(
|
||||||
|
&mut self,
|
||||||
|
sheet: u32,
|
||||||
|
row: i32,
|
||||||
|
column: i32,
|
||||||
|
r: (i32, i32),
|
||||||
|
v: &CalcResult,
|
||||||
|
s: i32,
|
||||||
|
f: i32,
|
||||||
|
) -> Result<(), String> {
|
||||||
|
let new_cell = match v {
|
||||||
|
CalcResult::EmptyCell => Cell::DynamicCellFormulaNumber {
|
||||||
|
f,
|
||||||
|
v: 0.0,
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
a: false,
|
||||||
|
},
|
||||||
|
CalcResult::String(v) => Cell::DynamicCellFormulaString {
|
||||||
|
f,
|
||||||
|
v: v.clone(),
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
a: false,
|
||||||
|
},
|
||||||
|
CalcResult::Number(v) => Cell::DynamicCellFormulaNumber {
|
||||||
|
v: *v,
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
f,
|
||||||
|
a: false,
|
||||||
|
},
|
||||||
|
CalcResult::Boolean(b) => Cell::DynamicCellFormulaBoolean {
|
||||||
|
v: *b,
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
f,
|
||||||
|
a: false,
|
||||||
|
},
|
||||||
|
CalcResult::Error { error, .. } => Cell::DynamicCellFormulaError {
|
||||||
|
ei: error.clone(),
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
f,
|
||||||
|
a: false,
|
||||||
|
o: "".to_string(),
|
||||||
|
m: "".to_string(),
|
||||||
|
},
|
||||||
|
|
||||||
|
// These cannot happen
|
||||||
|
// FIXME: Maybe the type of get_cell_value should be different
|
||||||
|
CalcResult::Range { .. } | CalcResult::EmptyArg | CalcResult::Array(_) => {
|
||||||
|
Cell::DynamicCellFormulaError {
|
||||||
|
ei: Error::ERROR,
|
||||||
|
s,
|
||||||
|
r,
|
||||||
|
f,
|
||||||
|
a: false,
|
||||||
|
o: "".to_string(),
|
||||||
|
m: "".to_string(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let sheet_data = &mut self.workbook.worksheet_mut(sheet)?.sheet_data;
|
||||||
|
|
||||||
|
match sheet_data.get_mut(&row) {
|
||||||
|
Some(column_data) => match column_data.get(&column) {
|
||||||
|
Some(_cell) => {
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
let mut column_data = HashMap::new();
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
sheet_data.insert(row, column_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets a cell with a "spill" value
|
||||||
|
fn set_spill_cell_with_value(
|
||||||
|
&mut self,
|
||||||
|
sheet: u32,
|
||||||
|
row: i32,
|
||||||
|
column: i32,
|
||||||
|
m: (i32, i32),
|
||||||
|
v: &CalcResult,
|
||||||
|
) -> Result<(), String> {
|
||||||
|
let style_index = self.get_cell_style_index(sheet, row, column)?;
|
||||||
|
let new_style_index = if self.workbook.styles.style_is_quote_prefix(style_index) {
|
||||||
|
self.workbook
|
||||||
|
.styles
|
||||||
|
.get_style_without_quote_prefix(style_index)?
|
||||||
|
} else {
|
||||||
|
style_index
|
||||||
|
};
|
||||||
|
let new_cell = match v {
|
||||||
|
CalcResult::EmptyCell => Cell::SpillNumberCell {
|
||||||
|
v: 0.0,
|
||||||
|
s: style_index,
|
||||||
|
m,
|
||||||
|
},
|
||||||
|
CalcResult::String(s) => Cell::SpillStringCell {
|
||||||
|
v: s.clone(),
|
||||||
|
s: new_style_index,
|
||||||
|
m,
|
||||||
|
},
|
||||||
|
CalcResult::Number(f) => Cell::SpillNumberCell {
|
||||||
|
v: *f,
|
||||||
|
s: new_style_index,
|
||||||
|
m,
|
||||||
|
},
|
||||||
|
CalcResult::Boolean(b) => Cell::SpillBooleanCell {
|
||||||
|
v: *b,
|
||||||
|
s: new_style_index,
|
||||||
|
m,
|
||||||
|
},
|
||||||
|
CalcResult::Error { error, .. } => Cell::SpillErrorCell {
|
||||||
|
ei: error.clone(),
|
||||||
|
s: style_index,
|
||||||
|
m,
|
||||||
|
},
|
||||||
|
|
||||||
|
// These cannot happen
|
||||||
|
// FIXME: Maybe the type of get_cell_value should be different
|
||||||
|
CalcResult::Range { .. } | CalcResult::EmptyArg | CalcResult::Array(_) => {
|
||||||
|
Cell::SpillErrorCell {
|
||||||
|
ei: Error::ERROR,
|
||||||
|
s: style_index,
|
||||||
|
m,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let sheet_data = &mut self.workbook.worksheet_mut(sheet)?.sheet_data;
|
||||||
|
|
||||||
|
match sheet_data.get_mut(&row) {
|
||||||
|
Some(column_data) => match column_data.get(&column) {
|
||||||
|
Some(_cell) => {
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
let mut column_data = HashMap::new();
|
||||||
|
column_data.insert(column, new_cell);
|
||||||
|
sheet_data.insert(row, column_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns `None` if no cell has called this cell, otherwise returns the dependent cell
|
||||||
|
fn get_support_cell(&self, sheet: u32, row: i32, column: i32) -> Result<Option<&Cell>, String> {
|
||||||
|
self.workbook.supporting_cells.get(&(sheet, row, column)).map(|c| Some(c)).ok_or_else(|| "Cell not found".into())
|
||||||
|
}
|
||||||
|
|
||||||
/// Sets `result` in the cell given by `sheet` sheet index, row and column
|
/// Sets `result` in the cell given by `sheet` sheet index, row and column
|
||||||
/// Note that will panic if the cell does not exist
|
/// Note that will panic if the cell does not exist
|
||||||
/// It will do nothing if the cell does not have a formula
|
/// It will do nothing if the cell does not have a formula
|
||||||
|
/// If the cell is an array or a range it will check if it is possible to spill to other cells
|
||||||
|
/// if it is not it will return an error.
|
||||||
|
/// Then it will check if any of the cells has been requested before.
|
||||||
#[allow(clippy::expect_used)]
|
#[allow(clippy::expect_used)]
|
||||||
fn set_cell_value(&mut self, cell_reference: CellReferenceIndex, result: &CalcResult) {
|
fn set_cell_value(
|
||||||
|
&mut self,
|
||||||
|
cell_reference: CellReferenceIndex,
|
||||||
|
result: &CalcResult,
|
||||||
|
) -> Result<(), String> {
|
||||||
let CellReferenceIndex { sheet, column, row } = cell_reference;
|
let CellReferenceIndex { sheet, column, row } = cell_reference;
|
||||||
let cell = &self.workbook.worksheets[sheet as usize].sheet_data[&row][&column];
|
let cell = self
|
||||||
|
.workbook
|
||||||
|
.worksheet(sheet)?
|
||||||
|
.cell(row, column)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
let s = cell.get_style();
|
let s = cell.get_style();
|
||||||
|
// If the cell is a dynamic cell we need to delete all the cells in the range
|
||||||
|
if let Some((width, height)) = cell.get_dynamic_range() {
|
||||||
|
for r in row..row + height {
|
||||||
|
for c in column..column + width {
|
||||||
|
// skip the "mother" cell
|
||||||
|
if r == row && c == column {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
self.cell_clear_contents(sheet, r, c)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
if let Some(f) = cell.get_formula() {
|
if let Some(f) = cell.get_formula() {
|
||||||
match result {
|
match result {
|
||||||
CalcResult::Number(value) => {
|
CalcResult::Number(value) => {
|
||||||
@@ -594,19 +810,145 @@ impl Model {
|
|||||||
ei: error.clone(),
|
ei: error.clone(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
CalcResult::EmptyCell | CalcResult::EmptyArg => {
|
||||||
|
*self.workbook.worksheets[sheet as usize]
|
||||||
|
.sheet_data
|
||||||
|
.get_mut(&row)
|
||||||
|
.expect("expected a row")
|
||||||
|
.get_mut(&column)
|
||||||
|
.expect("expected a column") = Cell::CellFormulaNumber { f, s, v: 0.0 };
|
||||||
|
}
|
||||||
CalcResult::Range { left, right } => {
|
CalcResult::Range { left, right } => {
|
||||||
if left.sheet == right.sheet
|
if left.sheet == right.sheet
|
||||||
&& left.row == right.row
|
&& left.row == right.row
|
||||||
&& left.column == right.column
|
&& left.column == right.column
|
||||||
{
|
{
|
||||||
let intersection_cell = CellReferenceIndex {
|
// There is only one cell
|
||||||
|
let single_cell = CellReferenceIndex {
|
||||||
sheet: left.sheet,
|
sheet: left.sheet,
|
||||||
column: left.column,
|
column: left.column,
|
||||||
row: left.row,
|
row: left.row,
|
||||||
};
|
};
|
||||||
let v = self.evaluate_cell(intersection_cell);
|
let v = self.evaluate_cell(single_cell);
|
||||||
self.set_cell_value(cell_reference, &v);
|
self.set_cell_value(cell_reference, &v)?;
|
||||||
} else {
|
} else {
|
||||||
|
// We need to check if all the cells are empty, otherwise we mark the cell as #SPILL!
|
||||||
|
let mut all_empty = true;
|
||||||
|
for r in row..=row + right.row - left.row {
|
||||||
|
for c in column..=column + right.column - left.column {
|
||||||
|
if r == row && c == column {
|
||||||
|
// skip the "mother" cell
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if !self.is_empty_cell(sheet, r, c).unwrap_or(false) {
|
||||||
|
all_empty = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if let Some(support) = self.get_support_cell(sheet, r, c) {
|
||||||
|
all_empty = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !all_empty {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !all_empty {
|
||||||
|
let o = match self.cell_reference_to_string(&cell_reference) {
|
||||||
|
Ok(s) => s,
|
||||||
|
Err(_) => "".to_string(),
|
||||||
|
};
|
||||||
|
*self.workbook.worksheets[sheet as usize]
|
||||||
|
.sheet_data
|
||||||
|
.get_mut(&row)
|
||||||
|
.expect("expected a row")
|
||||||
|
.get_mut(&column)
|
||||||
|
.expect("expected a column") = Cell::DynamicCellFormulaError {
|
||||||
|
f,
|
||||||
|
s,
|
||||||
|
o,
|
||||||
|
m: "Result would spill to non empty cells".to_string(),
|
||||||
|
ei: Error::SPILL,
|
||||||
|
r: (1, 1),
|
||||||
|
a: false,
|
||||||
|
};
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
// evaluate all the cells in that range
|
||||||
|
for r in left.row..=right.row {
|
||||||
|
for c in left.column..=right.column {
|
||||||
|
let cell_reference = CellReferenceIndex {
|
||||||
|
sheet: left.sheet,
|
||||||
|
row: r,
|
||||||
|
column: c,
|
||||||
|
};
|
||||||
|
// FIXME: We ned to return an error
|
||||||
|
self.evaluate_cell(cell_reference);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// now write the result in the target
|
||||||
|
for r in left.row..=right.row {
|
||||||
|
let row_delta = r - left.row;
|
||||||
|
for c in left.column..=right.column {
|
||||||
|
let column_delta = c - left.column;
|
||||||
|
// We need to put whatever is in (left.sheet, r, c) in
|
||||||
|
// (sheet, row + row_delta, column + column_delta)
|
||||||
|
// But we need to preserve the style
|
||||||
|
let target_row = row + row_delta;
|
||||||
|
let target_column = column + column_delta;
|
||||||
|
let cell = self
|
||||||
|
.workbook
|
||||||
|
.worksheet(left.sheet)?
|
||||||
|
.cell(r, c)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
|
let cell_reference = CellReferenceIndex {
|
||||||
|
sheet: left.sheet,
|
||||||
|
row: r,
|
||||||
|
column: c,
|
||||||
|
};
|
||||||
|
let v = self.get_cell_value(&cell, cell_reference);
|
||||||
|
if row == target_row && column == target_column {
|
||||||
|
// let cell_reference = CellReferenceIndex { sheet, row, column };
|
||||||
|
// self.set_cell_value(cell_reference, &v);
|
||||||
|
self.set_spill_cell_with_formula_value(
|
||||||
|
sheet,
|
||||||
|
target_row,
|
||||||
|
target_column,
|
||||||
|
(right.column - left.column + 1, right.row - left.row + 1),
|
||||||
|
&v,
|
||||||
|
s,
|
||||||
|
f,
|
||||||
|
)?;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
self.set_spill_cell_with_value(
|
||||||
|
sheet,
|
||||||
|
target_row,
|
||||||
|
target_column,
|
||||||
|
(row, column),
|
||||||
|
&v,
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
CalcResult::Array(array) => {
|
||||||
|
let width = array[0].len() as i32;
|
||||||
|
let height = array.len() as i32;
|
||||||
|
// First we check that we don't spill:
|
||||||
|
let mut all_empty = true;
|
||||||
|
for r in row..row + height {
|
||||||
|
for c in column..column + width {
|
||||||
|
if r == row && c == column {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if !self.is_empty_cell(sheet, r, c).unwrap_or(false) {
|
||||||
|
all_empty = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !all_empty {
|
||||||
let o = match self.cell_reference_to_string(&cell_reference) {
|
let o = match self.cell_reference_to_string(&cell_reference) {
|
||||||
Ok(s) => s,
|
Ok(s) => s,
|
||||||
Err(_) => "".to_string(),
|
Err(_) => "".to_string(),
|
||||||
@@ -620,57 +962,65 @@ impl Model {
|
|||||||
f,
|
f,
|
||||||
s,
|
s,
|
||||||
o,
|
o,
|
||||||
m: "Implicit Intersection not implemented".to_string(),
|
m: "Result would spill to non empty cells".to_string(),
|
||||||
ei: Error::NIMPL,
|
ei: Error::SPILL,
|
||||||
};
|
};
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
let mut target_row = row;
|
||||||
|
for data_row in array {
|
||||||
|
let mut target_column = column;
|
||||||
|
for value in data_row {
|
||||||
|
if row == target_row && column == target_column {
|
||||||
|
// This is the root cell of the dynamic array
|
||||||
|
let cell_reference = CellReferenceIndex { sheet, row, column };
|
||||||
|
let v = match value {
|
||||||
|
ArrayNode::Boolean(b) => CalcResult::Boolean(*b),
|
||||||
|
ArrayNode::Number(f) => CalcResult::Number(*f),
|
||||||
|
ArrayNode::String(s) => CalcResult::String(s.clone()),
|
||||||
|
ArrayNode::Error(error) => CalcResult::new_error(
|
||||||
|
error.clone(),
|
||||||
|
cell_reference,
|
||||||
|
error.to_localized_error_string(&self.language),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
self.set_spill_cell_with_formula_value(
|
||||||
|
sheet,
|
||||||
|
target_row,
|
||||||
|
target_column,
|
||||||
|
(width, height),
|
||||||
|
&v,
|
||||||
|
s,
|
||||||
|
f,
|
||||||
|
)?;
|
||||||
|
target_column += 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let v = match value {
|
||||||
|
ArrayNode::Boolean(b) => CalcResult::Boolean(*b),
|
||||||
|
ArrayNode::Number(f) => CalcResult::Number(*f),
|
||||||
|
ArrayNode::String(s) => CalcResult::String(s.clone()),
|
||||||
|
ArrayNode::Error(error) => CalcResult::new_error(
|
||||||
|
error.clone(),
|
||||||
|
cell_reference,
|
||||||
|
error.to_localized_error_string(&self.language),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
self.set_spill_cell_with_value(
|
||||||
|
sheet,
|
||||||
|
target_row,
|
||||||
|
target_column,
|
||||||
|
(row, column),
|
||||||
|
&v,
|
||||||
|
)?;
|
||||||
|
target_column += 1;
|
||||||
|
}
|
||||||
|
target_row += 1;
|
||||||
}
|
}
|
||||||
// if let Some(intersection_cell) = implicit_intersection(&cell_reference, &range)
|
|
||||||
// {
|
|
||||||
// let v = self.evaluate_cell(intersection_cell);
|
|
||||||
// self.set_cell_value(cell_reference, &v);
|
|
||||||
// } else {
|
|
||||||
// let o = match self.cell_reference_to_string(&cell_reference) {
|
|
||||||
// Ok(s) => s,
|
|
||||||
// Err(_) => "".to_string(),
|
|
||||||
// };
|
|
||||||
// *self.workbook.worksheets[sheet as usize]
|
|
||||||
// .sheet_data
|
|
||||||
// .get_mut(&row)
|
|
||||||
// .expect("expected a row")
|
|
||||||
// .get_mut(&column)
|
|
||||||
// .expect("expected a column") = Cell::CellFormulaError {
|
|
||||||
// f,
|
|
||||||
// s,
|
|
||||||
// o,
|
|
||||||
// m: "Invalid reference".to_string(),
|
|
||||||
// ei: Error::VALUE,
|
|
||||||
// };
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
CalcResult::EmptyCell | CalcResult::EmptyArg => {
|
|
||||||
*self.workbook.worksheets[sheet as usize]
|
|
||||||
.sheet_data
|
|
||||||
.get_mut(&row)
|
|
||||||
.expect("expected a row")
|
|
||||||
.get_mut(&column)
|
|
||||||
.expect("expected a column") = Cell::CellFormulaNumber { f, s, v: 0.0 };
|
|
||||||
}
|
|
||||||
CalcResult::Array(_) => {
|
|
||||||
*self.workbook.worksheets[sheet as usize]
|
|
||||||
.sheet_data
|
|
||||||
.get_mut(&row)
|
|
||||||
.expect("expected a row")
|
|
||||||
.get_mut(&column)
|
|
||||||
.expect("expected a column") = Cell::CellFormulaError {
|
|
||||||
f,
|
|
||||||
s,
|
|
||||||
o: "".to_string(),
|
|
||||||
m: "Arrays not supported yet".to_string(),
|
|
||||||
ei: Error::NIMPL,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the color of the sheet tab.
|
/// Sets the color of the sheet tab.
|
||||||
@@ -714,16 +1064,18 @@ impl Model {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EmptyCell, Boolean, Number, String, Error
|
||||||
fn get_cell_value(&self, cell: &Cell, cell_reference: CellReferenceIndex) -> CalcResult {
|
fn get_cell_value(&self, cell: &Cell, cell_reference: CellReferenceIndex) -> CalcResult {
|
||||||
use Cell::*;
|
use Cell::*;
|
||||||
match cell {
|
match cell {
|
||||||
EmptyCell { .. } => CalcResult::EmptyCell,
|
EmptyCell { .. } => CalcResult::EmptyCell,
|
||||||
BooleanCell { v, .. } => CalcResult::Boolean(*v),
|
BooleanCell { v, .. } | SpillBooleanCell { v, .. } => CalcResult::Boolean(*v),
|
||||||
NumberCell { v, .. } => CalcResult::Number(*v),
|
NumberCell { v, .. } | SpillNumberCell { v, .. } => CalcResult::Number(*v),
|
||||||
ErrorCell { ei, .. } => {
|
ErrorCell { ei, .. } | SpillErrorCell { ei, .. } => {
|
||||||
let message = ei.to_localized_error_string(&self.language);
|
let message = ei.to_localized_error_string(&self.language);
|
||||||
CalcResult::new_error(ei.clone(), cell_reference, message)
|
CalcResult::new_error(ei.clone(), cell_reference, message)
|
||||||
}
|
}
|
||||||
|
SpillStringCell { v, .. } => CalcResult::String(v.clone()),
|
||||||
SharedString { si, .. } => {
|
SharedString { si, .. } => {
|
||||||
if let Some(s) = self.workbook.shared_strings.get(*si as usize) {
|
if let Some(s) = self.workbook.shared_strings.get(*si as usize) {
|
||||||
CalcResult::String(s.clone())
|
CalcResult::String(s.clone())
|
||||||
@@ -732,15 +1084,21 @@ impl Model {
|
|||||||
CalcResult::new_error(Error::ERROR, cell_reference, message)
|
CalcResult::new_error(Error::ERROR, cell_reference, message)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
CellFormula { .. } => CalcResult::Error {
|
DynamicCellFormula { .. } | CellFormula { .. } => CalcResult::Error {
|
||||||
error: Error::ERROR,
|
error: Error::ERROR,
|
||||||
origin: cell_reference,
|
origin: cell_reference,
|
||||||
message: "Unevaluated formula".to_string(),
|
message: "Unevaluated formula".to_string(),
|
||||||
},
|
},
|
||||||
CellFormulaBoolean { v, .. } => CalcResult::Boolean(*v),
|
DynamicCellFormulaBoolean { v, .. } | CellFormulaBoolean { v, .. } => {
|
||||||
CellFormulaNumber { v, .. } => CalcResult::Number(*v),
|
CalcResult::Boolean(*v)
|
||||||
CellFormulaString { v, .. } => CalcResult::String(v.clone()),
|
}
|
||||||
CellFormulaError { ei, o, m, .. } => {
|
DynamicCellFormulaNumber { v, .. } | CellFormulaNumber { v, .. } => {
|
||||||
|
CalcResult::Number(*v)
|
||||||
|
}
|
||||||
|
DynamicCellFormulaString { v, .. } | CellFormulaString { v, .. } => {
|
||||||
|
CalcResult::String(v.clone())
|
||||||
|
}
|
||||||
|
DynamicCellFormulaError { ei, o, m, .. } | CellFormulaError { ei, o, m, .. } => {
|
||||||
if let Some(cell_reference) = self.parse_reference(o) {
|
if let Some(cell_reference) = self.parse_reference(o) {
|
||||||
CalcResult::new_error(ei.clone(), cell_reference, m.clone())
|
CalcResult::new_error(ei.clone(), cell_reference, m.clone())
|
||||||
} else {
|
} else {
|
||||||
@@ -772,6 +1130,8 @@ impl Model {
|
|||||||
self.workbook.worksheet(sheet)?.is_empty_cell(row, column)
|
self.workbook.worksheet(sheet)?.is_empty_cell(row, column)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Evaluates the cell. After the evaluation is done puts the value in the cell and other cells if it spills.
|
||||||
|
/// If when writing a spill cell encounter a cell whose value has been requested marks the model as "dirty"
|
||||||
pub(crate) fn evaluate_cell(&mut self, cell_reference: CellReferenceIndex) -> CalcResult {
|
pub(crate) fn evaluate_cell(&mut self, cell_reference: CellReferenceIndex) -> CalcResult {
|
||||||
let row_data = match self.workbook.worksheets[cell_reference.sheet as usize]
|
let row_data = match self.workbook.worksheets[cell_reference.sheet as usize]
|
||||||
.sheet_data
|
.sheet_data
|
||||||
@@ -810,9 +1170,10 @@ impl Model {
|
|||||||
self.cells.insert(key, CellState::Evaluating);
|
self.cells.insert(key, CellState::Evaluating);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let node = &self.parsed_formulas[cell_reference.sheet as usize][f as usize].clone();
|
let (node, _static_result) =
|
||||||
let result = self.evaluate_node_in_context(node, cell_reference);
|
&self.parsed_formulas[cell_reference.sheet as usize][f as usize];
|
||||||
self.set_cell_value(cell_reference, &result);
|
let result = self.evaluate_node_in_context(&node.clone(), cell_reference);
|
||||||
|
let _ = self.set_cell_value(cell_reference, &result);
|
||||||
// mark cell as evaluated
|
// mark cell as evaluated
|
||||||
self.cells.insert(key, CellState::Evaluated);
|
self.cells.insert(key, CellState::Evaluated);
|
||||||
result
|
result
|
||||||
@@ -922,6 +1283,10 @@ impl Model {
|
|||||||
locale,
|
locale,
|
||||||
tz,
|
tz,
|
||||||
view_id: 0,
|
view_id: 0,
|
||||||
|
support_graph: HashMap::new(),
|
||||||
|
switch_cells: None,
|
||||||
|
stack: Vec::new(),
|
||||||
|
state: EvaluationState::Ready,
|
||||||
};
|
};
|
||||||
|
|
||||||
model.parse_formulas();
|
model.parse_formulas();
|
||||||
@@ -1100,7 +1465,8 @@ impl Model {
|
|||||||
Some(cell) => match cell.get_formula() {
|
Some(cell) => match cell.get_formula() {
|
||||||
None => cell.get_text(&self.workbook.shared_strings, &self.language),
|
None => cell.get_text(&self.workbook.shared_strings, &self.language),
|
||||||
Some(i) => {
|
Some(i) => {
|
||||||
let formula = &self.parsed_formulas[sheet as usize][i as usize];
|
let (formula, static_result) =
|
||||||
|
&self.parsed_formulas[sheet as usize][i as usize];
|
||||||
let cell_ref = CellReferenceRC {
|
let cell_ref = CellReferenceRC {
|
||||||
sheet: self.workbook.worksheets[sheet as usize].get_name(),
|
sheet: self.workbook.worksheets[sheet as usize].get_name(),
|
||||||
row: target_row,
|
row: target_row,
|
||||||
@@ -1203,7 +1569,8 @@ impl Model {
|
|||||||
.get(sheet as usize)
|
.get(sheet as usize)
|
||||||
.ok_or("missing sheet")?
|
.ok_or("missing sheet")?
|
||||||
.get(formula_index as usize)
|
.get(formula_index as usize)
|
||||||
.ok_or("missing formula")?;
|
.ok_or("missing formula")?
|
||||||
|
.0;
|
||||||
let cell_ref = CellReferenceRC {
|
let cell_ref = CellReferenceRC {
|
||||||
sheet: worksheet.get_name(),
|
sheet: worksheet.get_name(),
|
||||||
row,
|
row,
|
||||||
@@ -1437,6 +1804,25 @@ impl Model {
|
|||||||
column: i32,
|
column: i32,
|
||||||
value: String,
|
value: String,
|
||||||
) -> Result<(), String> {
|
) -> Result<(), String> {
|
||||||
|
// We need to check if the cell is part of a dynamic array
|
||||||
|
let cell = self
|
||||||
|
.workbook
|
||||||
|
.worksheet(sheet)?
|
||||||
|
.cell(row, column)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
|
// If the cell is a dynamic cell we need to delete all the cells in the range
|
||||||
|
if let Some((width, height)) = cell.get_dynamic_range() {
|
||||||
|
for r in row..row + height {
|
||||||
|
for c in column..column + width {
|
||||||
|
// skip the "mother" cell
|
||||||
|
if r == row && c == column {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
self.cell_clear_contents(sheet, r, c)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
// If value starts with "'" then we force the style to be quote_prefix
|
// If value starts with "'" then we force the style to be quote_prefix
|
||||||
let style_index = self.get_cell_style_index(sheet, row, column)?;
|
let style_index = self.get_cell_style_index(sheet, row, column)?;
|
||||||
if let Some(new_value) = value.strip_prefix('\'') {
|
if let Some(new_value) = value.strip_prefix('\'') {
|
||||||
@@ -1462,8 +1848,9 @@ impl Model {
|
|||||||
self.set_cell_with_formula(sheet, row, column, formula, new_style_index)?;
|
self.set_cell_with_formula(sheet, row, column, formula, new_style_index)?;
|
||||||
// Update the style if needed
|
// Update the style if needed
|
||||||
let cell = CellReferenceIndex { sheet, row, column };
|
let cell = CellReferenceIndex { sheet, row, column };
|
||||||
let parsed_formula = &self.parsed_formulas[sheet as usize][formula_index as usize];
|
let (parsed_formula, static_result) =
|
||||||
if let Some(units) = self.compute_node_units(parsed_formula, &cell) {
|
self.parsed_formulas[sheet as usize][formula_index as usize].clone();
|
||||||
|
if let Some(units) = self.compute_node_units(&parsed_formula, &cell) {
|
||||||
let new_style_index = self
|
let new_style_index = self
|
||||||
.workbook
|
.workbook
|
||||||
.styles
|
.styles
|
||||||
@@ -1471,6 +1858,14 @@ impl Model {
|
|||||||
let style = self.workbook.styles.get_style(new_style_index)?;
|
let style = self.workbook.styles.get_style(new_style_index)?;
|
||||||
self.set_cell_style(sheet, row, column, &style)?
|
self.set_cell_style(sheet, row, column, &style)?
|
||||||
}
|
}
|
||||||
|
match static_result {
|
||||||
|
StaticResult::Scalar => {}
|
||||||
|
StaticResult::Array(_, _)
|
||||||
|
| StaticResult::Range(_, _)
|
||||||
|
| StaticResult::Unknown => {
|
||||||
|
self.workbook.spill_cells.push((sheet, row, column));
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// The list of currencies is '$', '€' and the local currency
|
// The list of currencies is '$', '€' and the local currency
|
||||||
let mut currencies = vec!["$", "€"];
|
let mut currencies = vec!["$", "€"];
|
||||||
@@ -1544,6 +1939,7 @@ impl Model {
|
|||||||
_ => parsed_formula = new_parsed_formula,
|
_ => parsed_formula = new_parsed_formula,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
let static_result = run_static_analysis_on_node(&parsed_formula);
|
||||||
|
|
||||||
let s = to_rc_format(&parsed_formula);
|
let s = to_rc_format(&parsed_formula);
|
||||||
let mut formula_index: i32 = -1;
|
let mut formula_index: i32 = -1;
|
||||||
@@ -1552,7 +1948,7 @@ impl Model {
|
|||||||
}
|
}
|
||||||
if formula_index == -1 {
|
if formula_index == -1 {
|
||||||
shared_formulas.push(s);
|
shared_formulas.push(s);
|
||||||
self.parsed_formulas[sheet as usize].push(parsed_formula);
|
self.parsed_formulas[sheet as usize].push((parsed_formula, static_result));
|
||||||
formula_index = (shared_formulas.len() as i32) - 1;
|
formula_index = (shared_formulas.len() as i32) - 1;
|
||||||
}
|
}
|
||||||
worksheet.set_cell_with_formula(row, column, formula_index, style)?;
|
worksheet.set_cell_with_formula(row, column, formula_index, style)?;
|
||||||
@@ -1747,7 +2143,7 @@ impl Model {
|
|||||||
};
|
};
|
||||||
match cell.get_formula() {
|
match cell.get_formula() {
|
||||||
Some(formula_index) => {
|
Some(formula_index) => {
|
||||||
let formula = &self.parsed_formulas[sheet as usize][formula_index as usize];
|
let formula = &self.parsed_formulas[sheet as usize][formula_index as usize].0;
|
||||||
let cell_ref = CellReferenceRC {
|
let cell_ref = CellReferenceRC {
|
||||||
sheet: worksheet.get_name(),
|
sheet: worksheet.get_name(),
|
||||||
row,
|
row,
|
||||||
@@ -1783,9 +2179,34 @@ impl Model {
|
|||||||
|
|
||||||
/// Evaluates the model with a top-down recursive algorithm
|
/// Evaluates the model with a top-down recursive algorithm
|
||||||
pub fn evaluate(&mut self) {
|
pub fn evaluate(&mut self) {
|
||||||
// clear all computation artifacts
|
// We first evaluate all the cells that might spill to other cells
|
||||||
self.cells.clear();
|
let mut spills_computed = false;
|
||||||
|
self.state = EvaluationState::EvaluatingSpills;
|
||||||
|
while !spills_computed {
|
||||||
|
spills_computed = true;
|
||||||
|
// clear all computation artifacts
|
||||||
|
self.cells.clear();
|
||||||
|
// Evaluate all the cells that might spill
|
||||||
|
let spill_cells = self.workbook.spill_cells.clone();
|
||||||
|
for (sheet, row, column) in spill_cells {
|
||||||
|
self.evaluate_cell(CellReferenceIndex { sheet, row, column });
|
||||||
|
if self.switch_cells.is_some() {
|
||||||
|
spills_computed = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if let Some((index1, index2)) = self.switch_cells {
|
||||||
|
spills_computed = false;
|
||||||
|
// switch the cells indices in the spill_cells
|
||||||
|
let cell1 = self.workbook.spill_cells[index1 as usize];
|
||||||
|
let cell2 = self.workbook.spill_cells[index2 as usize];
|
||||||
|
self.workbook.spill_cells[index1 as usize] = cell2;
|
||||||
|
self.workbook.spill_cells[index2 as usize] = cell1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.state = EvaluationState::Evaluating;
|
||||||
|
|
||||||
|
// Now we compute all the rest
|
||||||
let cells = self.get_all_cells();
|
let cells = self.get_all_cells();
|
||||||
|
|
||||||
for cell in cells {
|
for cell in cells {
|
||||||
@@ -1795,6 +2216,7 @@ impl Model {
|
|||||||
column: cell.column,
|
column: cell.column,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
self.state = EvaluationState::Ready;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Removes the content of the cell but leaves the style.
|
/// Removes the content of the cell but leaves the style.
|
||||||
@@ -1818,9 +2240,22 @@ impl Model {
|
|||||||
/// # }
|
/// # }
|
||||||
/// ```
|
/// ```
|
||||||
pub fn cell_clear_contents(&mut self, sheet: u32, row: i32, column: i32) -> Result<(), String> {
|
pub fn cell_clear_contents(&mut self, sheet: u32, row: i32, column: i32) -> Result<(), String> {
|
||||||
self.workbook
|
// If it has a spill formula we need to delete the contents of all the spilled cells
|
||||||
.worksheet_mut(sheet)?
|
let worksheet = self.workbook.worksheet_mut(sheet)?;
|
||||||
.cell_clear_contents(row, column)?;
|
if let Some(cell) = worksheet.cell(row, column) {
|
||||||
|
if let Some((width, height)) = cell.get_dynamic_range() {
|
||||||
|
for r in row..row + height {
|
||||||
|
for c in column..column + width {
|
||||||
|
if row == r && column == c {
|
||||||
|
// we skip the root cell
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
worksheet.cell_clear_contents(r, c)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
worksheet.cell_clear_contents(row, column)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1845,6 +2280,18 @@ impl Model {
|
|||||||
/// # }
|
/// # }
|
||||||
pub fn cell_clear_all(&mut self, sheet: u32, row: i32, column: i32) -> Result<(), String> {
|
pub fn cell_clear_all(&mut self, sheet: u32, row: i32, column: i32) -> Result<(), String> {
|
||||||
let worksheet = self.workbook.worksheet_mut(sheet)?;
|
let worksheet = self.workbook.worksheet_mut(sheet)?;
|
||||||
|
// Delete the contents of spilled cells if any
|
||||||
|
if let Some(cell) = worksheet.cell(row, column) {
|
||||||
|
if let Some((width, height)) = cell.get_dynamic_range() {
|
||||||
|
for r in row..row + height {
|
||||||
|
for c in column..column + width {
|
||||||
|
if row == r && c == column {
|
||||||
|
worksheet.cell_clear_contents(r, c)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let sheet_data = &mut worksheet.sheet_data;
|
let sheet_data = &mut worksheet.sheet_data;
|
||||||
if let Some(row_data) = sheet_data.get_mut(&row) {
|
if let Some(row_data) = sheet_data.get_mut(&row) {
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ use crate::{
|
|||||||
expressions::{
|
expressions::{
|
||||||
lexer::LexerMode,
|
lexer::LexerMode,
|
||||||
parser::{
|
parser::{
|
||||||
|
static_analysis::run_static_analysis_on_node,
|
||||||
stringify::{rename_sheet_in_node, to_rc_format, to_string},
|
stringify::{rename_sheet_in_node, to_rc_format, to_string},
|
||||||
Parser,
|
Parser,
|
||||||
},
|
},
|
||||||
@@ -15,7 +16,7 @@ use crate::{
|
|||||||
},
|
},
|
||||||
language::get_language,
|
language::get_language,
|
||||||
locale::get_locale,
|
locale::get_locale,
|
||||||
model::{get_milliseconds_since_epoch, Model, ParsedDefinedName},
|
model::{get_milliseconds_since_epoch, EvaluationState, Model, ParsedDefinedName},
|
||||||
types::{
|
types::{
|
||||||
DefinedName, Metadata, SheetState, Workbook, WorkbookSettings, WorkbookView, Worksheet,
|
DefinedName, Metadata, SheetState, Workbook, WorkbookSettings, WorkbookView, Worksheet,
|
||||||
WorksheetView,
|
WorksheetView,
|
||||||
@@ -94,7 +95,8 @@ impl Model {
|
|||||||
let mut parse_formula = Vec::new();
|
let mut parse_formula = Vec::new();
|
||||||
for formula in shared_formulas {
|
for formula in shared_formulas {
|
||||||
let t = self.parser.parse(formula, &cell_reference);
|
let t = self.parser.parse(formula, &cell_reference);
|
||||||
parse_formula.push(t);
|
let static_result = run_static_analysis_on_node(&t);
|
||||||
|
parse_formula.push((t, static_result));
|
||||||
}
|
}
|
||||||
self.parsed_formulas.push(parse_formula);
|
self.parsed_formulas.push(parse_formula);
|
||||||
}
|
}
|
||||||
@@ -405,6 +407,7 @@ impl Model {
|
|||||||
},
|
},
|
||||||
tables: HashMap::new(),
|
tables: HashMap::new(),
|
||||||
views,
|
views,
|
||||||
|
spill_cells: Vec::new(),
|
||||||
};
|
};
|
||||||
let parsed_formulas = Vec::new();
|
let parsed_formulas = Vec::new();
|
||||||
let worksheets = &workbook.worksheets;
|
let worksheets = &workbook.worksheets;
|
||||||
@@ -427,6 +430,10 @@ impl Model {
|
|||||||
language,
|
language,
|
||||||
tz,
|
tz,
|
||||||
view_id: 0,
|
view_id: 0,
|
||||||
|
support_graph: HashMap::new(),
|
||||||
|
switch_cells: None,
|
||||||
|
stack: Vec::new(),
|
||||||
|
state: EvaluationState::Ready,
|
||||||
};
|
};
|
||||||
model.parse_formulas();
|
model.parse_formulas();
|
||||||
Ok(model)
|
Ok(model)
|
||||||
|
|||||||
@@ -52,6 +52,7 @@ mod test_fn_offset;
|
|||||||
mod test_number_format;
|
mod test_number_format;
|
||||||
|
|
||||||
mod test_arrays;
|
mod test_arrays;
|
||||||
|
mod test_dynamic_arrays;
|
||||||
mod test_escape_quotes;
|
mod test_escape_quotes;
|
||||||
mod test_extend;
|
mod test_extend;
|
||||||
mod test_fn_fv;
|
mod test_fn_fv;
|
||||||
|
|||||||
50
base/src/test/test_dynamic_arrays.rs
Normal file
50
base/src/test/test_dynamic_arrays.rs
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
#![allow(clippy::unwrap_used)]
|
||||||
|
|
||||||
|
use crate::test::util::new_empty_model;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn they_spill() {
|
||||||
|
let mut model = new_empty_model();
|
||||||
|
model._set("A1", "42");
|
||||||
|
model._set("A2", "5");
|
||||||
|
model._set("A3", "7");
|
||||||
|
|
||||||
|
model._set("B1", "=A1:A3");
|
||||||
|
|
||||||
|
model.evaluate();
|
||||||
|
|
||||||
|
assert_eq!(model._get_text("B1"), *"42");
|
||||||
|
assert_eq!(model._get_text("B2"), *"5");
|
||||||
|
assert_eq!(model._get_text("B3"), *"7");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn spill_error() {
|
||||||
|
let mut model = new_empty_model();
|
||||||
|
model._set("A1", "42");
|
||||||
|
model._set("A2", "5");
|
||||||
|
model._set("A3", "7");
|
||||||
|
|
||||||
|
model._set("B1", "=A1:A3");
|
||||||
|
model._set("B2", "4");
|
||||||
|
|
||||||
|
model.evaluate();
|
||||||
|
|
||||||
|
assert_eq!(model._get_text("B1"), *"#SPILL!");
|
||||||
|
assert_eq!(model._get_text("B2"), *"4");
|
||||||
|
assert_eq!(model._get_text("B3"), *"");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn second_evaluation() {
|
||||||
|
let mut model = new_empty_model();
|
||||||
|
model._set("C3", "={1,2,3}");
|
||||||
|
model.evaluate();
|
||||||
|
|
||||||
|
assert_eq!(model._get_text("D3"), "2");
|
||||||
|
|
||||||
|
model._set("D8", "23");
|
||||||
|
model.evaluate();
|
||||||
|
|
||||||
|
assert_eq!(model._get_text("D3"), "2");
|
||||||
|
}
|
||||||
@@ -91,12 +91,12 @@ fn fn_or_xor() {
|
|||||||
model._set("A10", &format!("={func}(X99:Z99"));
|
model._set("A10", &format!("={func}(X99:Z99"));
|
||||||
|
|
||||||
// Reference to cell with reference to empty range
|
// Reference to cell with reference to empty range
|
||||||
model._set("B11", "=X99:Z99");
|
model._set("B11", "=@X99:Z99");
|
||||||
model._set("A11", &format!("={func}(B11)"));
|
model._set("A11", &format!("={func}(B11)"));
|
||||||
|
|
||||||
// Reference to cell with non-empty range
|
// Reference to cell with non-empty range
|
||||||
model._set("X12", "1");
|
model._set("X12", "1");
|
||||||
model._set("B12", "=X12:Z12");
|
model._set("B12", "=@X12:Z12");
|
||||||
model._set("A12", &format!("={func}(B12)"));
|
model._set("A12", &format!("={func}(B12)"));
|
||||||
|
|
||||||
// Reference to text cell
|
// Reference to text cell
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
use crate::test::util::new_empty_model;
|
use crate::test::util::new_empty_model;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn simple_colum() {
|
fn simple_column() {
|
||||||
let mut model = new_empty_model();
|
let mut model = new_empty_model();
|
||||||
// We populate cells A1 to A3
|
// We populate cells A1 to A3
|
||||||
model._set("A1", "1");
|
model._set("A1", "1");
|
||||||
@@ -30,7 +30,7 @@ fn return_of_array_is_n_impl() {
|
|||||||
|
|
||||||
model.evaluate();
|
model.evaluate();
|
||||||
|
|
||||||
assert_eq!(model._get_text("C2"), "#N/IMPL!".to_string());
|
assert_eq!(model._get_text("C2"), "1".to_string());
|
||||||
assert_eq!(model._get_text("D2"), "1.89188842".to_string());
|
assert_eq!(model._get_text("D2"), "1.89188842".to_string());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,8 +6,10 @@ mod test_border;
|
|||||||
mod test_clear_cells;
|
mod test_clear_cells;
|
||||||
mod test_column_style;
|
mod test_column_style;
|
||||||
mod test_defined_names;
|
mod test_defined_names;
|
||||||
|
mod test_delete_evaluates;
|
||||||
mod test_delete_row_column_formatting;
|
mod test_delete_row_column_formatting;
|
||||||
mod test_diff_queue;
|
mod test_diff_queue;
|
||||||
|
mod test_dynamic_array;
|
||||||
mod test_evaluation;
|
mod test_evaluation;
|
||||||
mod test_general;
|
mod test_general;
|
||||||
mod test_grid_lines;
|
mod test_grid_lines;
|
||||||
|
|||||||
47
base/src/test/user_model/test_delete_evaluates.rs
Normal file
47
base/src/test/user_model/test_delete_evaluates.rs
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
#![allow(clippy::unwrap_used)]
|
||||||
|
|
||||||
|
use crate::{expressions::types::Area, UserModel};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn clear_cell_contents_evaluates() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "42").unwrap();
|
||||||
|
model.set_user_input(0, 1, 2, "=A1").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("42".to_string())
|
||||||
|
);
|
||||||
|
model
|
||||||
|
.range_clear_contents(&Area {
|
||||||
|
sheet: 0,
|
||||||
|
row: 1,
|
||||||
|
column: 1,
|
||||||
|
width: 1,
|
||||||
|
height: 1,
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(model.get_formatted_cell_value(0, 1, 2), Ok("0".to_string()));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn clear_cell_all_evaluates() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "42").unwrap();
|
||||||
|
model.set_user_input(0, 1, 2, "=A1").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("42".to_string())
|
||||||
|
);
|
||||||
|
model
|
||||||
|
.range_clear_all(&Area {
|
||||||
|
sheet: 0,
|
||||||
|
row: 1,
|
||||||
|
column: 1,
|
||||||
|
width: 1,
|
||||||
|
height: 1,
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
assert_eq!(model.get_formatted_cell_value(0, 1, 2), Ok("0".to_string()));
|
||||||
|
}
|
||||||
130
base/src/test/user_model/test_dynamic_array.rs
Normal file
130
base/src/test/user_model/test_dynamic_array.rs
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
#![allow(clippy::unwrap_used)]
|
||||||
|
|
||||||
|
use crate::{expressions::types::Area, UserModel};
|
||||||
|
|
||||||
|
// Tests basic behavour.
|
||||||
|
#[test]
|
||||||
|
fn basic() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
// We put a value by the dynamic array to check the border conditions
|
||||||
|
model.set_user_input(0, 2, 1, "22").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "={34,35,3}").unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 1),
|
||||||
|
Ok("34".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test that overwriting a dynamic array with a single value dissolves the array
|
||||||
|
#[test]
|
||||||
|
fn sett_user_input_mother() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "={34,35,3}").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("35".to_string())
|
||||||
|
);
|
||||||
|
model.set_user_input(0, 1, 1, "123").unwrap();
|
||||||
|
assert_eq!(model.get_formatted_cell_value(0, 1, 2), Ok("".to_string()));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn set_user_input_sibling() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "={43,55,34}").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("55".to_string())
|
||||||
|
);
|
||||||
|
// This does nothing
|
||||||
|
model.set_user_input(0, 1, 2, "123").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("55".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn basic_undo_redo() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
model.set_user_input(0, 1, 1, "={34,35,3}").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("35".to_string())
|
||||||
|
);
|
||||||
|
model.undo().unwrap();
|
||||||
|
assert_eq!(model.get_formatted_cell_value(0, 1, 2), Ok("".to_string()));
|
||||||
|
model.redo().unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 1, 2),
|
||||||
|
Ok("35".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn mixed_spills() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
// D9 => ={1,2,3}
|
||||||
|
model.set_user_input(0, 9, 4, "={34,35,3}").unwrap();
|
||||||
|
// F6 => ={1;2;3;4}
|
||||||
|
model.set_user_input(0, 6, 6, "={1;2;3;4}").unwrap();
|
||||||
|
|
||||||
|
// F6 should be #SPILL!
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 6, 6),
|
||||||
|
Ok("#SPILL!".to_string())
|
||||||
|
);
|
||||||
|
|
||||||
|
// We delete D9
|
||||||
|
model
|
||||||
|
.range_clear_contents(&Area {
|
||||||
|
sheet: 0,
|
||||||
|
row: 9,
|
||||||
|
column: 4,
|
||||||
|
width: 1,
|
||||||
|
height: 1,
|
||||||
|
})
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
// F6 should be 1
|
||||||
|
assert_eq!(model.get_formatted_cell_value(0, 6, 6), Ok("1".to_string()));
|
||||||
|
|
||||||
|
// Now we undo that
|
||||||
|
model.undo().unwrap();
|
||||||
|
// F6 should be #SPILL!
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 6, 6),
|
||||||
|
Ok("#SPILL!".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn spill_order_d9_f6() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
// D9 => ={1,2,3}
|
||||||
|
model.set_user_input(0, 9, 4, "={34,35,3}").unwrap();
|
||||||
|
// F6 => ={1;2;3;4}
|
||||||
|
model.set_user_input(0, 6, 6, "={1;2;3;4}").unwrap();
|
||||||
|
|
||||||
|
// F6 should be #SPILL!
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 6, 6),
|
||||||
|
Ok("#SPILL!".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn spill_order_f6_d9() {
|
||||||
|
let mut model = UserModel::new_empty("model", "en", "UTC").unwrap();
|
||||||
|
// F6 => ={1;2;3;4}
|
||||||
|
model.set_user_input(0, 6, 6, "={1;2;3;4}").unwrap();
|
||||||
|
// D9 => ={1,2,3}
|
||||||
|
model.set_user_input(0, 9, 4, "={34,35,3}").unwrap();
|
||||||
|
|
||||||
|
// D9 should be #SPILL!
|
||||||
|
assert_eq!(
|
||||||
|
model.get_formatted_cell_value(0, 9, 4),
|
||||||
|
Ok("#SPILL!".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -51,6 +51,8 @@ pub struct Workbook {
|
|||||||
pub metadata: Metadata,
|
pub metadata: Metadata,
|
||||||
pub tables: HashMap<String, Table>,
|
pub tables: HashMap<String, Table>,
|
||||||
pub views: HashMap<u32, WorkbookView>,
|
pub views: HashMap<u32, WorkbookView>,
|
||||||
|
/// The list of cells that spill in the order of evaluation
|
||||||
|
pub spill_cells: Vec<(u32, i32, i32)>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A defined name. The `sheet_id` is the sheet index in case the name is local
|
/// A defined name. The `sheet_id` is the sheet index in case the name is local
|
||||||
@@ -159,17 +161,17 @@ pub enum CellType {
|
|||||||
CompoundData = 128,
|
CompoundData = 128,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Cell types
|
||||||
|
/// s is always the style index of the cell
|
||||||
#[derive(Encode, Decode, Debug, Clone, PartialEq)]
|
#[derive(Encode, Decode, Debug, Clone, PartialEq)]
|
||||||
pub enum Cell {
|
pub enum Cell {
|
||||||
EmptyCell {
|
EmptyCell {
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
|
|
||||||
BooleanCell {
|
BooleanCell {
|
||||||
v: bool,
|
v: bool,
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
|
|
||||||
NumberCell {
|
NumberCell {
|
||||||
v: f64,
|
v: f64,
|
||||||
s: i32,
|
s: i32,
|
||||||
@@ -181,6 +183,7 @@ pub enum Cell {
|
|||||||
},
|
},
|
||||||
// Always a shared string
|
// Always a shared string
|
||||||
SharedString {
|
SharedString {
|
||||||
|
// string index
|
||||||
si: i32,
|
si: i32,
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
@@ -189,13 +192,11 @@ pub enum Cell {
|
|||||||
f: i32,
|
f: i32,
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
|
|
||||||
CellFormulaBoolean {
|
CellFormulaBoolean {
|
||||||
f: i32,
|
f: i32,
|
||||||
v: bool,
|
v: bool,
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
|
|
||||||
CellFormulaNumber {
|
CellFormulaNumber {
|
||||||
f: i32,
|
f: i32,
|
||||||
v: f64,
|
v: f64,
|
||||||
@@ -207,9 +208,9 @@ pub enum Cell {
|
|||||||
v: String,
|
v: String,
|
||||||
s: i32,
|
s: i32,
|
||||||
},
|
},
|
||||||
|
|
||||||
CellFormulaError {
|
CellFormulaError {
|
||||||
f: i32,
|
f: i32,
|
||||||
|
// error index
|
||||||
ei: Error,
|
ei: Error,
|
||||||
s: i32,
|
s: i32,
|
||||||
// Origin: Sheet3!C4
|
// Origin: Sheet3!C4
|
||||||
@@ -217,7 +218,81 @@ pub enum Cell {
|
|||||||
// Error Message: "Not implemented function"
|
// Error Message: "Not implemented function"
|
||||||
m: String,
|
m: String,
|
||||||
},
|
},
|
||||||
// TODO: Array formulas
|
// All Spill/dynamic cells have a boolean, a for array, if true it is an array formula
|
||||||
|
// Spill cells point to a mother cell (row, column)
|
||||||
|
SpillNumberCell {
|
||||||
|
v: f64,
|
||||||
|
s: i32,
|
||||||
|
// mother cell (row, column)
|
||||||
|
m: (i32, i32),
|
||||||
|
},
|
||||||
|
SpillBooleanCell {
|
||||||
|
v: bool,
|
||||||
|
s: i32,
|
||||||
|
// mother cell (row, column)
|
||||||
|
m: (i32, i32),
|
||||||
|
},
|
||||||
|
SpillErrorCell {
|
||||||
|
ei: Error,
|
||||||
|
s: i32,
|
||||||
|
// mother cell (row, column)
|
||||||
|
m: (i32, i32),
|
||||||
|
},
|
||||||
|
SpillStringCell {
|
||||||
|
v: String,
|
||||||
|
s: i32,
|
||||||
|
// mother cell (row, column)
|
||||||
|
m: (i32, i32),
|
||||||
|
},
|
||||||
|
// Dynamic cell formulas have a range (width, height)
|
||||||
|
DynamicCellFormula {
|
||||||
|
f: i32,
|
||||||
|
s: i32,
|
||||||
|
// range of the formula (width, height)
|
||||||
|
r: (i32, i32),
|
||||||
|
// true if the formula is a CSE formula
|
||||||
|
a: bool,
|
||||||
|
},
|
||||||
|
DynamicCellFormulaBoolean {
|
||||||
|
f: i32,
|
||||||
|
v: bool,
|
||||||
|
s: i32,
|
||||||
|
// range of the formula (width, height)
|
||||||
|
r: (i32, i32),
|
||||||
|
// true if the formula is a CSE formula
|
||||||
|
a: bool,
|
||||||
|
},
|
||||||
|
DynamicCellFormulaNumber {
|
||||||
|
f: i32,
|
||||||
|
v: f64,
|
||||||
|
s: i32,
|
||||||
|
// range of the formula (width, height)
|
||||||
|
r: (i32, i32),
|
||||||
|
// true if the formula is a CSE formula
|
||||||
|
a: bool,
|
||||||
|
},
|
||||||
|
DynamicCellFormulaString {
|
||||||
|
f: i32,
|
||||||
|
v: String,
|
||||||
|
s: i32,
|
||||||
|
// range of the formula (width, height)
|
||||||
|
r: (i32, i32),
|
||||||
|
// true if the formula is a CSE formula
|
||||||
|
a: bool,
|
||||||
|
},
|
||||||
|
DynamicCellFormulaError {
|
||||||
|
f: i32,
|
||||||
|
ei: Error,
|
||||||
|
s: i32,
|
||||||
|
// Cell origin of the error
|
||||||
|
o: String,
|
||||||
|
// Error message in text
|
||||||
|
m: String,
|
||||||
|
// range of the formula (width, height)
|
||||||
|
r: (i32, i32),
|
||||||
|
// true if the formula is a CSE formula
|
||||||
|
a: bool,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Cell {
|
impl Default for Cell {
|
||||||
|
|||||||
@@ -24,6 +24,18 @@ use crate::user_model::history::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use super::border_utils::is_max_border;
|
use super::border_utils::is_max_border;
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub enum CellArrayStructure {
|
||||||
|
// It s just a single cell
|
||||||
|
SingleCell,
|
||||||
|
// It is part o a dynamic array
|
||||||
|
// (mother_row, mother_column, width, height)
|
||||||
|
DynamicChild(i32, i32, i32, i32),
|
||||||
|
// Mother of a dynamic array (width, height)
|
||||||
|
DynamicMother(i32, i32),
|
||||||
|
}
|
||||||
|
|
||||||
/// Data for the clipboard
|
/// Data for the clipboard
|
||||||
pub type ClipboardData = HashMap<i32, HashMap<i32, ClipboardCell>>;
|
pub type ClipboardData = HashMap<i32, HashMap<i32, ClipboardCell>>;
|
||||||
|
|
||||||
@@ -627,6 +639,7 @@ impl UserModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.push_diff_list(diff_list);
|
self.push_diff_list(diff_list);
|
||||||
|
self.evaluate_if_not_paused();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -656,6 +669,7 @@ impl UserModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.push_diff_list(diff_list);
|
self.push_diff_list(diff_list);
|
||||||
|
self.evaluate_if_not_paused();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1741,6 +1755,65 @@ impl UserModel {
|
|||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the geometric structure of a cell
|
||||||
|
pub fn get_cell_array_structure(
|
||||||
|
&self,
|
||||||
|
sheet: u32,
|
||||||
|
row: i32,
|
||||||
|
column: i32,
|
||||||
|
) -> Result<CellArrayStructure, String> {
|
||||||
|
let cell = self
|
||||||
|
.model
|
||||||
|
.workbook
|
||||||
|
.worksheet(sheet)?
|
||||||
|
.cell(row, column)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
|
match cell {
|
||||||
|
Cell::EmptyCell { .. }
|
||||||
|
| Cell::BooleanCell { .. }
|
||||||
|
| Cell::NumberCell { .. }
|
||||||
|
| Cell::ErrorCell { .. }
|
||||||
|
| Cell::SharedString { .. }
|
||||||
|
| Cell::CellFormula { .. }
|
||||||
|
| Cell::CellFormulaBoolean { .. }
|
||||||
|
| Cell::CellFormulaNumber { .. }
|
||||||
|
| Cell::CellFormulaString { .. }
|
||||||
|
| Cell::CellFormulaError { .. } => Ok(CellArrayStructure::SingleCell),
|
||||||
|
Cell::SpillNumberCell { m, .. }
|
||||||
|
| Cell::SpillBooleanCell { m, .. }
|
||||||
|
| Cell::SpillErrorCell { m, .. }
|
||||||
|
| Cell::SpillStringCell { m, .. } => {
|
||||||
|
let (m_row, m_column) = m;
|
||||||
|
let m_cell = self
|
||||||
|
.model
|
||||||
|
.workbook
|
||||||
|
.worksheet(sheet)?
|
||||||
|
.cell(m_row, m_column)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
|
let (width, height) = match m_cell {
|
||||||
|
Cell::DynamicCellFormula { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaBoolean { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaNumber { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaString { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaError { r, .. } => (r.0, r.1),
|
||||||
|
_ => return Err("Invalid structure".to_string()),
|
||||||
|
};
|
||||||
|
Ok(CellArrayStructure::DynamicChild(
|
||||||
|
m_row, m_column, width, height,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormula { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaBoolean { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaNumber { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaString { r, .. }
|
||||||
|
| Cell::DynamicCellFormulaError { r, .. } => {
|
||||||
|
Ok(CellArrayStructure::DynamicMother(r.0, r.1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns a copy of the selected area
|
/// Returns a copy of the selected area
|
||||||
pub fn copy_to_clipboard(&self) -> Result<Clipboard, String> {
|
pub fn copy_to_clipboard(&self) -> Result<Clipboard, String> {
|
||||||
let selected_area = self.get_selected_view();
|
let selected_area = self.get_selected_view();
|
||||||
@@ -2043,6 +2116,24 @@ impl UserModel {
|
|||||||
old_value,
|
old_value,
|
||||||
} => {
|
} => {
|
||||||
needs_evaluation = true;
|
needs_evaluation = true;
|
||||||
|
let cell = self
|
||||||
|
.model
|
||||||
|
.workbook
|
||||||
|
.worksheet(*sheet)?
|
||||||
|
.cell(*row, *column)
|
||||||
|
.cloned()
|
||||||
|
.unwrap_or_default();
|
||||||
|
if let Some((width, height)) = cell.get_dynamic_range() {
|
||||||
|
for r in *row..*row + height {
|
||||||
|
for c in *column..*column + width {
|
||||||
|
// skip the "mother" cell
|
||||||
|
if r == *row && c == *column {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
self.model.cell_clear_contents(*sheet, r, c)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
match *old_value.clone() {
|
match *old_value.clone() {
|
||||||
Some(value) => {
|
Some(value) => {
|
||||||
self.model
|
self.model
|
||||||
|
|||||||
@@ -5,18 +5,21 @@ use bitcode::{Decode, Encode};
|
|||||||
use crate::types::{Cell, Col, Row, SheetState, Style, Worksheet};
|
use crate::types::{Cell, Col, Row, SheetState, Style, Worksheet};
|
||||||
|
|
||||||
#[derive(Clone, Encode, Decode)]
|
#[derive(Clone, Encode, Decode)]
|
||||||
|
#[cfg_attr(debug_assertions, derive(Debug))]
|
||||||
pub(crate) struct RowData {
|
pub(crate) struct RowData {
|
||||||
pub(crate) row: Option<Row>,
|
pub(crate) row: Option<Row>,
|
||||||
pub(crate) data: HashMap<i32, Cell>,
|
pub(crate) data: HashMap<i32, Cell>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Encode, Decode)]
|
#[derive(Clone, Encode, Decode)]
|
||||||
|
#[cfg_attr(debug_assertions, derive(Debug))]
|
||||||
pub(crate) struct ColumnData {
|
pub(crate) struct ColumnData {
|
||||||
pub(crate) column: Option<Col>,
|
pub(crate) column: Option<Col>,
|
||||||
pub(crate) data: HashMap<i32, Cell>,
|
pub(crate) data: HashMap<i32, Cell>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Encode, Decode)]
|
#[derive(Clone, Encode, Decode)]
|
||||||
|
#[cfg_attr(debug_assertions, derive(Debug))]
|
||||||
pub(crate) enum Diff {
|
pub(crate) enum Diff {
|
||||||
// Cell diffs
|
// Cell diffs
|
||||||
SetCellValue {
|
SetCellValue {
|
||||||
|
|||||||
@@ -766,4 +766,21 @@ impl Model {
|
|||||||
.get_first_non_empty_in_row_after_column(sheet, row, column)
|
.get_first_non_empty_in_row_after_column(sheet, row, column)
|
||||||
.map_err(to_js_error)
|
.map_err(to_js_error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen(
|
||||||
|
js_name = "getCellArrayStructure",
|
||||||
|
unchecked_return_type = "CellArrayStructure"
|
||||||
|
)]
|
||||||
|
pub fn get_cell_array_structure(
|
||||||
|
&self,
|
||||||
|
sheet: u32,
|
||||||
|
row: i32,
|
||||||
|
column: i32,
|
||||||
|
) -> Result<JsValue, JsError> {
|
||||||
|
let cell_structure = self
|
||||||
|
.model
|
||||||
|
.get_cell_array_structure(sheet, row, column)
|
||||||
|
.map_err(|e| to_js_error(e.to_string()))?;
|
||||||
|
serde_wasm_bindgen::to_value(&cell_structure).map_err(JsError::from)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -109,6 +109,11 @@ export interface MarkedToken {
|
|||||||
end: number;
|
end: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type CellArrayStructure =
|
||||||
|
| "SingleCell"
|
||||||
|
| { DynamicChild: [number, number, number, number] }
|
||||||
|
| { DynamicMother: [number, number] };
|
||||||
|
|
||||||
export interface WorksheetProperties {
|
export interface WorksheetProperties {
|
||||||
name: string;
|
name: string;
|
||||||
color: string;
|
color: string;
|
||||||
@@ -216,7 +221,7 @@ export interface SelectedView {
|
|||||||
// };
|
// };
|
||||||
|
|
||||||
// type ClipboardData = Record<string, Record <string, ClipboardCell>>;
|
// type ClipboardData = Record<string, Record <string, ClipboardCell>>;
|
||||||
type ClipboardData = Map<number, Map <number, ClipboardCell>>;
|
type ClipboardData = Map<number, Map<number, ClipboardCell>>;
|
||||||
|
|
||||||
export interface ClipboardCell {
|
export interface ClipboardCell {
|
||||||
text: string;
|
text: string;
|
||||||
@@ -233,4 +238,4 @@ export interface DefinedName {
|
|||||||
name: string;
|
name: string;
|
||||||
scope?: number;
|
scope?: number;
|
||||||
formula: string;
|
formula: string;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import type { WorkbookState } from "../workbookState";
|
|||||||
type FormulaBarProps = {
|
type FormulaBarProps = {
|
||||||
cellAddress: string;
|
cellAddress: string;
|
||||||
formulaValue: string;
|
formulaValue: string;
|
||||||
|
isPartOfArray: boolean;
|
||||||
model: Model;
|
model: Model;
|
||||||
workbookState: WorkbookState;
|
workbookState: WorkbookState;
|
||||||
onChange: () => void;
|
onChange: () => void;
|
||||||
@@ -23,6 +24,7 @@ function FormulaBar(properties: FormulaBarProps) {
|
|||||||
const {
|
const {
|
||||||
cellAddress,
|
cellAddress,
|
||||||
formulaValue,
|
formulaValue,
|
||||||
|
isPartOfArray,
|
||||||
model,
|
model,
|
||||||
onChange,
|
onChange,
|
||||||
onTextUpdated,
|
onTextUpdated,
|
||||||
@@ -62,6 +64,9 @@ function FormulaBar(properties: FormulaBarProps) {
|
|||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}}
|
}}
|
||||||
|
sx={{
|
||||||
|
color: isPartOfArray ? "grey" : "black",
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<Editor
|
<Editor
|
||||||
originalText={formulaValue}
|
originalText={formulaValue}
|
||||||
|
|||||||
@@ -362,6 +362,19 @@ const Workbook = (props: { model: Model; workbookState: WorkbookState }) => {
|
|||||||
return model.getCellContent(sheet, row, column);
|
return model.getCellContent(sheet, row, column);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// returns true if it is either single cell or the root cell of an array
|
||||||
|
const isRootCellOfArray = () => {
|
||||||
|
const { sheet, row, column } = model.getSelectedView();
|
||||||
|
const r = model.getCellArrayStructure(sheet, row, column);
|
||||||
|
if (r === "SingleCell") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if ("DynamicMother" in r) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
const getCellStyle = useCallback(() => {
|
const getCellStyle = useCallback(() => {
|
||||||
const { sheet, row, column } = model.getSelectedView();
|
const { sheet, row, column } = model.getSelectedView();
|
||||||
return model.getCellStyle(sheet, row, column);
|
return model.getCellStyle(sheet, row, column);
|
||||||
@@ -705,6 +718,7 @@ const Workbook = (props: { model: Model; workbookState: WorkbookState }) => {
|
|||||||
}}
|
}}
|
||||||
model={model}
|
model={model}
|
||||||
workbookState={workbookState}
|
workbookState={workbookState}
|
||||||
|
isPartOfArray={isRootCellOfArray()}
|
||||||
/>
|
/>
|
||||||
<Worksheet
|
<Worksheet
|
||||||
model={model}
|
model={model}
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import {
|
|||||||
LAST_COLUMN,
|
LAST_COLUMN,
|
||||||
LAST_ROW,
|
LAST_ROW,
|
||||||
ROW_HEIGH_SCALE,
|
ROW_HEIGH_SCALE,
|
||||||
|
cellArrayStructureColor,
|
||||||
outlineBackgroundColor,
|
outlineBackgroundColor,
|
||||||
outlineColor,
|
outlineColor,
|
||||||
} from "../WorksheetCanvas/constants";
|
} from "../WorksheetCanvas/constants";
|
||||||
@@ -59,6 +60,7 @@ const Worksheet = forwardRef(
|
|||||||
const spacerElement = useRef<HTMLDivElement>(null);
|
const spacerElement = useRef<HTMLDivElement>(null);
|
||||||
const cellOutline = useRef<HTMLDivElement>(null);
|
const cellOutline = useRef<HTMLDivElement>(null);
|
||||||
const areaOutline = useRef<HTMLDivElement>(null);
|
const areaOutline = useRef<HTMLDivElement>(null);
|
||||||
|
const cellArrayStructure = useRef<HTMLDivElement>(null);
|
||||||
const extendToOutline = useRef<HTMLDivElement>(null);
|
const extendToOutline = useRef<HTMLDivElement>(null);
|
||||||
const columnResizeGuide = useRef<HTMLDivElement>(null);
|
const columnResizeGuide = useRef<HTMLDivElement>(null);
|
||||||
const rowResizeGuide = useRef<HTMLDivElement>(null);
|
const rowResizeGuide = useRef<HTMLDivElement>(null);
|
||||||
@@ -85,6 +87,7 @@ const Worksheet = forwardRef(
|
|||||||
|
|
||||||
const outline = cellOutline.current;
|
const outline = cellOutline.current;
|
||||||
const area = areaOutline.current;
|
const area = areaOutline.current;
|
||||||
|
const arrayStructure = cellArrayStructure.current;
|
||||||
const extendTo = extendToOutline.current;
|
const extendTo = extendToOutline.current;
|
||||||
const editor = editorElement.current;
|
const editor = editorElement.current;
|
||||||
|
|
||||||
@@ -98,7 +101,8 @@ const Worksheet = forwardRef(
|
|||||||
!area ||
|
!area ||
|
||||||
!extendTo ||
|
!extendTo ||
|
||||||
!scrollElement.current ||
|
!scrollElement.current ||
|
||||||
!editor
|
!editor ||
|
||||||
|
!arrayStructure
|
||||||
)
|
)
|
||||||
return;
|
return;
|
||||||
// FIXME: This two need to be computed.
|
// FIXME: This two need to be computed.
|
||||||
@@ -115,6 +119,7 @@ const Worksheet = forwardRef(
|
|||||||
rowGuide: rowGuideRef,
|
rowGuide: rowGuideRef,
|
||||||
columnHeaders: columnHeadersRef,
|
columnHeaders: columnHeadersRef,
|
||||||
cellOutline: outline,
|
cellOutline: outline,
|
||||||
|
cellArrayStructure: arrayStructure,
|
||||||
areaOutline: area,
|
areaOutline: area,
|
||||||
extendToOutline: extendTo,
|
extendToOutline: extendTo,
|
||||||
editor: editor,
|
editor: editor,
|
||||||
@@ -329,6 +334,7 @@ const Worksheet = forwardRef(
|
|||||||
/>
|
/>
|
||||||
</EditorWrapper>
|
</EditorWrapper>
|
||||||
<AreaOutline ref={areaOutline} />
|
<AreaOutline ref={areaOutline} />
|
||||||
|
<CellArrayStructure ref={cellArrayStructure} />
|
||||||
<ExtendToOutline ref={extendToOutline} />
|
<ExtendToOutline ref={extendToOutline} />
|
||||||
<ColumnResizeGuide ref={columnResizeGuide} />
|
<ColumnResizeGuide ref={columnResizeGuide} />
|
||||||
<RowResizeGuide ref={rowResizeGuide} />
|
<RowResizeGuide ref={rowResizeGuide} />
|
||||||
@@ -514,6 +520,12 @@ const AreaOutline = styled("div")`
|
|||||||
background-color: ${outlineBackgroundColor};
|
background-color: ${outlineBackgroundColor};
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const CellArrayStructure = styled("div")`
|
||||||
|
position: absolute;
|
||||||
|
border: 1px solid ${cellArrayStructureColor};
|
||||||
|
border-radius: 3px;
|
||||||
|
`;
|
||||||
|
|
||||||
const CellOutline = styled("div")`
|
const CellOutline = styled("div")`
|
||||||
position: absolute;
|
position: absolute;
|
||||||
border: 2px solid ${outlineColor};
|
border: 2px solid ${outlineColor};
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ export const defaultTextColor = "#2E414D";
|
|||||||
|
|
||||||
export const outlineColor = "#F2994A";
|
export const outlineColor = "#F2994A";
|
||||||
export const outlineBackgroundColor = "#F2994A1A";
|
export const outlineBackgroundColor = "#F2994A1A";
|
||||||
|
export const cellArrayStructureColor = "#64BDFDA1";
|
||||||
|
|
||||||
export const LAST_COLUMN = 16_384;
|
export const LAST_COLUMN = 16_384;
|
||||||
export const LAST_ROW = 1_048_576;
|
export const LAST_ROW = 1_048_576;
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ export interface CanvasSettings {
|
|||||||
canvas: HTMLCanvasElement;
|
canvas: HTMLCanvasElement;
|
||||||
cellOutline: HTMLDivElement;
|
cellOutline: HTMLDivElement;
|
||||||
areaOutline: HTMLDivElement;
|
areaOutline: HTMLDivElement;
|
||||||
|
cellArrayStructure: HTMLDivElement;
|
||||||
extendToOutline: HTMLDivElement;
|
extendToOutline: HTMLDivElement;
|
||||||
columnGuide: HTMLDivElement;
|
columnGuide: HTMLDivElement;
|
||||||
rowGuide: HTMLDivElement;
|
rowGuide: HTMLDivElement;
|
||||||
@@ -90,6 +91,8 @@ export default class WorksheetCanvas {
|
|||||||
|
|
||||||
cellOutlineHandle: HTMLDivElement;
|
cellOutlineHandle: HTMLDivElement;
|
||||||
|
|
||||||
|
cellArrayStructure: HTMLDivElement;
|
||||||
|
|
||||||
extendToOutline: HTMLDivElement;
|
extendToOutline: HTMLDivElement;
|
||||||
|
|
||||||
workbookState: WorkbookState;
|
workbookState: WorkbookState;
|
||||||
@@ -124,6 +127,7 @@ export default class WorksheetCanvas {
|
|||||||
this.refresh = options.refresh;
|
this.refresh = options.refresh;
|
||||||
|
|
||||||
this.cellOutline = options.elements.cellOutline;
|
this.cellOutline = options.elements.cellOutline;
|
||||||
|
this.cellArrayStructure = options.elements.cellArrayStructure;
|
||||||
this.areaOutline = options.elements.areaOutline;
|
this.areaOutline = options.elements.areaOutline;
|
||||||
this.extendToOutline = options.elements.extendToOutline;
|
this.extendToOutline = options.elements.extendToOutline;
|
||||||
this.rowGuide = options.elements.rowGuide;
|
this.rowGuide = options.elements.rowGuide;
|
||||||
@@ -1515,16 +1519,20 @@ export default class WorksheetCanvas {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private drawCellOutline(): void {
|
private drawCellOutline(): void {
|
||||||
const { cellOutline, areaOutline, cellOutlineHandle } = this;
|
const { cellArrayStructure, cellOutline, areaOutline, cellOutlineHandle } =
|
||||||
|
this;
|
||||||
if (this.workbookState.getEditingCell()) {
|
if (this.workbookState.getEditingCell()) {
|
||||||
cellOutline.style.visibility = "hidden";
|
cellOutline.style.visibility = "hidden";
|
||||||
cellOutlineHandle.style.visibility = "hidden";
|
cellOutlineHandle.style.visibility = "hidden";
|
||||||
areaOutline.style.visibility = "hidden";
|
areaOutline.style.visibility = "hidden";
|
||||||
|
cellArrayStructure.style.visibility = "hidden";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
cellOutline.style.visibility = "visible";
|
cellOutline.style.visibility = "visible";
|
||||||
cellOutlineHandle.style.visibility = "visible";
|
cellOutlineHandle.style.visibility = "visible";
|
||||||
areaOutline.style.visibility = "visible";
|
areaOutline.style.visibility = "visible";
|
||||||
|
// This one is hidden by default
|
||||||
|
cellArrayStructure.style.visibility = "hidden";
|
||||||
|
|
||||||
const [selectedSheet, selectedRow, selectedColumn] =
|
const [selectedSheet, selectedRow, selectedColumn] =
|
||||||
this.model.getSelectedCell();
|
this.model.getSelectedCell();
|
||||||
@@ -1580,6 +1588,34 @@ export default class WorksheetCanvas {
|
|||||||
[handleX, handleY] = this.getCoordinatesByCell(rowStart, columnStart);
|
[handleX, handleY] = this.getCoordinatesByCell(rowStart, columnStart);
|
||||||
handleX += this.getColumnWidth(selectedSheet, columnStart);
|
handleX += this.getColumnWidth(selectedSheet, columnStart);
|
||||||
handleY += this.getRowHeight(selectedSheet, rowStart);
|
handleY += this.getRowHeight(selectedSheet, rowStart);
|
||||||
|
// we draw the array structure if needed only in this case
|
||||||
|
const arrayStructure = this.model.getCellArrayStructure(
|
||||||
|
selectedSheet,
|
||||||
|
selectedRow,
|
||||||
|
selectedColumn,
|
||||||
|
);
|
||||||
|
let array = null;
|
||||||
|
if (arrayStructure === "SingleCell") {
|
||||||
|
// nothing to see here
|
||||||
|
} else if ("DynamicMother" in arrayStructure) {
|
||||||
|
cellArrayStructure.style.visibility = "visible";
|
||||||
|
const [arrayWidth, arrayHeight] = arrayStructure.DynamicMother;
|
||||||
|
array = [selectedRow, selectedColumn, arrayWidth, arrayHeight];
|
||||||
|
} else {
|
||||||
|
cellArrayStructure.style.visibility = "visible";
|
||||||
|
array = arrayStructure.DynamicChild;
|
||||||
|
}
|
||||||
|
if (array !== null) {
|
||||||
|
const [arrayX, arrayY] = this.getCoordinatesByCell(array[0], array[1]);
|
||||||
|
const [arrayX1, arrayY1] = this.getCoordinatesByCell(
|
||||||
|
array[0] + array[3],
|
||||||
|
array[1] + array[2],
|
||||||
|
);
|
||||||
|
cellArrayStructure.style.left = `${arrayX}px`;
|
||||||
|
cellArrayStructure.style.top = `${arrayY}px`;
|
||||||
|
cellArrayStructure.style.width = `${arrayX1 - arrayX}px`;
|
||||||
|
cellArrayStructure.style.height = `${arrayY1 - arrayY}px`;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
areaOutline.style.visibility = "visible";
|
areaOutline.style.visibility = "visible";
|
||||||
cellOutlineHandle.style.visibility = "visible";
|
cellOutlineHandle.style.visibility = "visible";
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import "./App.css";
|
|||||||
import styled from "@emotion/styled";
|
import styled from "@emotion/styled";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { FileBar } from "./components/FileBar";
|
import { FileBar } from "./components/FileBar";
|
||||||
import LeftDrawer from "./components/LeftDrawer/LeftDrawer";
|
|
||||||
import {
|
import {
|
||||||
get_documentation_model,
|
get_documentation_model,
|
||||||
get_model,
|
get_model,
|
||||||
@@ -10,10 +9,7 @@ import {
|
|||||||
} from "./components/rpc";
|
} from "./components/rpc";
|
||||||
import {
|
import {
|
||||||
createNewModel,
|
createNewModel,
|
||||||
deleteModelByUuid,
|
|
||||||
deleteSelectedModel,
|
deleteSelectedModel,
|
||||||
// getModelsMetadata,
|
|
||||||
// getSelectedUuid,
|
|
||||||
loadModelFromStorageOrCreate,
|
loadModelFromStorageOrCreate,
|
||||||
saveModelToStorage,
|
saveModelToStorage,
|
||||||
saveSelectedModelInStorage,
|
saveSelectedModelInStorage,
|
||||||
@@ -25,7 +21,6 @@ import { IronCalc, IronCalcIcon, Model, init } from "@ironcalc/workbook";
|
|||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [model, setModel] = useState<Model | null>(null);
|
const [model, setModel] = useState<Model | null>(null);
|
||||||
const [isDrawerOpen, setIsDrawerOpen] = useState(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
async function start() {
|
async function start() {
|
||||||
@@ -84,80 +79,48 @@ function App() {
|
|||||||
// We could use context for model, but the problem is that it should initialized to null.
|
// We could use context for model, but the problem is that it should initialized to null.
|
||||||
// Passing the property down makes sure it is always defined.
|
// Passing the property down makes sure it is always defined.
|
||||||
|
|
||||||
// Handlers for model changes that also update our models state
|
|
||||||
const handleNewModel = () => {
|
|
||||||
const newModel = createNewModel();
|
|
||||||
setModel(newModel);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSetModel = (uuid: string) => {
|
|
||||||
const newModel = selectModelFromStorage(uuid);
|
|
||||||
if (newModel) {
|
|
||||||
setModel(newModel);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDeleteModel = () => {
|
|
||||||
const newModel = deleteSelectedModel();
|
|
||||||
if (newModel) {
|
|
||||||
setModel(newModel);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDeleteModelByUuid = (uuid: string) => {
|
|
||||||
const newModel = deleteModelByUuid(uuid);
|
|
||||||
if (newModel) {
|
|
||||||
setModel(newModel);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AppContainer>
|
<Wrapper>
|
||||||
<LeftDrawer
|
<FileBar
|
||||||
open={isDrawerOpen}
|
model={model}
|
||||||
onClose={() => setIsDrawerOpen(false)}
|
onModelUpload={async (arrayBuffer: ArrayBuffer, fileName: string) => {
|
||||||
newModel={handleNewModel}
|
const blob = await uploadFile(arrayBuffer, fileName);
|
||||||
setModel={handleSetModel}
|
|
||||||
onDelete={handleDeleteModelByUuid}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<MainContent isDrawerOpen={isDrawerOpen}>
|
const bytes = new Uint8Array(await blob.arrayBuffer());
|
||||||
<FileBar
|
const newModel = Model.from_bytes(bytes);
|
||||||
model={model}
|
saveModelToStorage(newModel);
|
||||||
onModelUpload={async (arrayBuffer: ArrayBuffer, fileName: string) => {
|
|
||||||
const blob = await uploadFile(arrayBuffer, fileName);
|
setModel(newModel);
|
||||||
const bytes = new Uint8Array(await blob.arrayBuffer());
|
}}
|
||||||
const newModel = Model.from_bytes(bytes);
|
newModel={() => {
|
||||||
saveModelToStorage(newModel);
|
setModel(createNewModel());
|
||||||
|
}}
|
||||||
|
setModel={(uuid: string) => {
|
||||||
|
const newModel = selectModelFromStorage(uuid);
|
||||||
|
if (newModel) {
|
||||||
setModel(newModel);
|
setModel(newModel);
|
||||||
}}
|
}
|
||||||
newModel={handleNewModel}
|
}}
|
||||||
setModel={handleSetModel}
|
onDelete={() => {
|
||||||
onDelete={handleDeleteModel}
|
const newModel = deleteSelectedModel();
|
||||||
isDrawerOpen={isDrawerOpen}
|
if (newModel) {
|
||||||
setIsDrawerOpen={setIsDrawerOpen}
|
setModel(newModel);
|
||||||
/>
|
}
|
||||||
<IronCalc model={model} />
|
}}
|
||||||
</MainContent>
|
/>
|
||||||
</AppContainer>
|
<IronCalc model={model} />
|
||||||
|
</Wrapper>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const AppContainer = styled("div")`
|
const Wrapper = styled("div")`
|
||||||
display: flex;
|
margin: 0px;
|
||||||
|
padding: 0px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
position: relative;
|
|
||||||
overflow: hidden;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const MainContent = styled("div")<{ isDrawerOpen: boolean }>`
|
|
||||||
margin-left: ${({ isDrawerOpen }) => (isDrawerOpen ? "0px" : "-264px")};
|
|
||||||
transition: margin-left 0.3s ease;
|
|
||||||
width: ${({ isDrawerOpen }) =>
|
|
||||||
isDrawerOpen ? "calc(100% - 264px)" : "100%"};
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
position: absolute;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const Loading = styled("div")`
|
const Loading = styled("div")`
|
||||||
|
|||||||
@@ -12,9 +12,19 @@ function DeleteWorkbookDialog(properties: DeleteWorkbookDialogProperties) {
|
|||||||
const deleteButtonRef = useRef<HTMLButtonElement>(null);
|
const deleteButtonRef = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
const root = document.getElementById("root");
|
||||||
|
if (root) {
|
||||||
|
root.style.filter = "blur(2px)";
|
||||||
|
}
|
||||||
if (deleteButtonRef.current) {
|
if (deleteButtonRef.current) {
|
||||||
deleteButtonRef.current.focus();
|
deleteButtonRef.current.focus();
|
||||||
}
|
}
|
||||||
|
return () => {
|
||||||
|
const root = document.getElementById("root");
|
||||||
|
if (root) {
|
||||||
|
root.style.filter = "none";
|
||||||
|
}
|
||||||
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import styled from "@emotion/styled";
|
import styled from "@emotion/styled";
|
||||||
import type { Model } from "@ironcalc/workbook";
|
import type { Model } from "@ironcalc/workbook";
|
||||||
import { Button, IconButton } from "@mui/material";
|
import { IronCalcIcon, IronCalcLogo } from "@ironcalc/workbook";
|
||||||
import { PanelLeftClose, PanelLeftOpen } from "lucide-react";
|
|
||||||
import { useLayoutEffect, useRef, useState } from "react";
|
import { useLayoutEffect, useRef, useState } from "react";
|
||||||
import { DesktopMenu, MobileMenu } from "./FileMenu";
|
import { FileMenu } from "./FileMenu";
|
||||||
|
import { HelpMenu } from "./HelpMenu";
|
||||||
import { ShareButton } from "./ShareButton";
|
import { ShareButton } from "./ShareButton";
|
||||||
import ShareWorkbookDialog from "./ShareWorkbookDialog";
|
import ShareWorkbookDialog from "./ShareWorkbookDialog";
|
||||||
import { WorkbookTitle } from "./WorkbookTitle";
|
import { WorkbookTitle } from "./WorkbookTitle";
|
||||||
@@ -30,8 +30,6 @@ export function FileBar(properties: {
|
|||||||
setModel: (key: string) => void;
|
setModel: (key: string) => void;
|
||||||
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
||||||
onDelete: () => void;
|
onDelete: () => void;
|
||||||
isDrawerOpen: boolean;
|
|
||||||
setIsDrawerOpen: (open: boolean) => void;
|
|
||||||
}) {
|
}) {
|
||||||
const [isDialogOpen, setIsDialogOpen] = useState(false);
|
const [isDialogOpen, setIsDialogOpen] = useState(false);
|
||||||
const spacerRef = useRef<HTMLDivElement>(null);
|
const spacerRef = useRef<HTMLDivElement>(null);
|
||||||
@@ -47,49 +45,24 @@ export function FileBar(properties: {
|
|||||||
}
|
}
|
||||||
}, [width]);
|
}, [width]);
|
||||||
|
|
||||||
// Common handler functions for both menu types
|
|
||||||
const handleDownload = async () => {
|
|
||||||
const model = properties.model;
|
|
||||||
const bytes = model.toBytes();
|
|
||||||
const fileName = model.getName();
|
|
||||||
await downloadModel(bytes, fileName);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<FileBarWrapper>
|
<FileBarWrapper>
|
||||||
<DrawerButton
|
<StyledDesktopLogo />
|
||||||
$isDrawerOpen={properties.isDrawerOpen}
|
<StyledIronCalcIcon />
|
||||||
onClick={() => properties.setIsDrawerOpen(!properties.isDrawerOpen)}
|
<Divider />
|
||||||
disableRipple
|
<FileMenu
|
||||||
title="Toggle sidebar"
|
newModel={properties.newModel}
|
||||||
>
|
setModel={properties.setModel}
|
||||||
{properties.isDrawerOpen ? <PanelLeftClose /> : <PanelLeftOpen />}
|
onModelUpload={properties.onModelUpload}
|
||||||
</DrawerButton>
|
onDownload={async () => {
|
||||||
<DesktopButtonsWrapper>
|
const model = properties.model;
|
||||||
<DesktopMenu
|
const bytes = model.toBytes();
|
||||||
newModel={properties.newModel}
|
const fileName = model.getName();
|
||||||
setModel={properties.setModel}
|
await downloadModel(bytes, fileName);
|
||||||
onModelUpload={properties.onModelUpload}
|
}}
|
||||||
onDownload={handleDownload}
|
onDelete={properties.onDelete}
|
||||||
onDelete={properties.onDelete}
|
/>
|
||||||
/>
|
<HelpMenu />
|
||||||
<FileBarButton
|
|
||||||
disableRipple
|
|
||||||
onClick={() => window.open("https://docs.ironcalc.com", "_blank")}
|
|
||||||
>
|
|
||||||
Help
|
|
||||||
</FileBarButton>
|
|
||||||
</DesktopButtonsWrapper>
|
|
||||||
<MobileButtonsWrapper>
|
|
||||||
<MobileMenu
|
|
||||||
newModel={properties.newModel}
|
|
||||||
setModel={properties.setModel}
|
|
||||||
onModelUpload={properties.onModelUpload}
|
|
||||||
onDownload={handleDownload}
|
|
||||||
onDelete={properties.onDelete}
|
|
||||||
/>
|
|
||||||
</MobileButtonsWrapper>
|
|
||||||
<Spacer ref={spacerRef} />
|
|
||||||
<WorkbookTitleWrapper>
|
<WorkbookTitleWrapper>
|
||||||
<WorkbookTitle
|
<WorkbookTitle
|
||||||
name={properties.model.getName()}
|
name={properties.model.getName()}
|
||||||
@@ -115,8 +88,12 @@ export function FileBar(properties: {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We want the workbook title to be exactly an the center of the page,
|
||||||
|
// so we need an absolute position
|
||||||
const WorkbookTitleWrapper = styled("div")`
|
const WorkbookTitleWrapper = styled("div")`
|
||||||
position: relative;
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
`;
|
`;
|
||||||
|
|
||||||
// The "Spacer" component occupies as much space as possible between the menu and the share button
|
// The "Spacer" component occupies as much space as possible between the menu and the share button
|
||||||
@@ -124,79 +101,38 @@ const Spacer = styled("div")`
|
|||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const DrawerButton = styled(IconButton)<{ $isDrawerOpen: boolean }>`
|
const StyledDesktopLogo = styled(IronCalcLogo)`
|
||||||
margin-left: 8px;
|
width: 120px;
|
||||||
height: 32px;
|
margin-left: 12px;
|
||||||
width: 32px;
|
@media (max-width: 769px) {
|
||||||
padding: 8px;
|
display: none;
|
||||||
border-radius: 4px;
|
|
||||||
cursor: ${(props) => (props.$isDrawerOpen ? "w-resize" : "e-resize")};
|
|
||||||
svg {
|
|
||||||
stroke-width: 2px;
|
|
||||||
stroke: #757575;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
}
|
}
|
||||||
&:hover {
|
`;
|
||||||
background-color: #f2f2f2;
|
|
||||||
}
|
const StyledIronCalcIcon = styled(IronCalcIcon)`
|
||||||
&:active {
|
width: 36px;
|
||||||
background-color: #e0e0e0;
|
margin-left: 10px;
|
||||||
|
@media (min-width: 769px) {
|
||||||
|
display: none;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const Divider = styled("div")`
|
||||||
|
margin: 0px 8px 0px 16px;
|
||||||
|
height: 12px;
|
||||||
|
border-left: 1px solid #e0e0e0;
|
||||||
|
`;
|
||||||
|
|
||||||
// The container must be relative positioned so we can position the title absolutely
|
// The container must be relative positioned so we can position the title absolutely
|
||||||
const FileBarWrapper = styled("div")`
|
const FileBarWrapper = styled("div")`
|
||||||
position: relative;
|
position: relative;
|
||||||
height: 60px;
|
height: 60px;
|
||||||
min-height: 60px;
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
border-bottom: 1px solid #e0e0e0;
|
border-bottom: 1px solid #e0e0e0;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
box-sizing: border-box;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const DesktopButtonsWrapper = styled("div")`
|
|
||||||
display: flex;
|
|
||||||
gap: 4px;
|
|
||||||
margin-left: 8px;
|
|
||||||
@media (max-width: 600px) {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const MobileButtonsWrapper = styled("div")`
|
|
||||||
display: flex;
|
|
||||||
gap: 4px;
|
|
||||||
@media (min-width: 601px) {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
@media (max-width: 600px) {
|
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const FileBarButton = styled(Button)`
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
align-items: center;
|
|
||||||
font-size: 12px;
|
|
||||||
height: 32px;
|
|
||||||
width: auto;
|
|
||||||
padding: 4px 8px;
|
|
||||||
font-weight: 400;
|
|
||||||
min-width: 0px;
|
|
||||||
text-transform: capitalize;
|
|
||||||
color: #333333;
|
|
||||||
&:hover {
|
|
||||||
background-color: #f2f2f2;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: #e0e0e0;
|
|
||||||
}
|
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const DialogContainer = styled("div")`
|
const DialogContainer = styled("div")`
|
||||||
|
|||||||
@@ -1,165 +1,93 @@
|
|||||||
import styled from "@emotion/styled";
|
import styled from "@emotion/styled";
|
||||||
import { Button, IconButton, Menu, MenuItem, Modal } from "@mui/material";
|
import { Menu, MenuItem, Modal } from "@mui/material";
|
||||||
import {
|
import { Check, FileDown, FileUp, Plus, Trash2 } from "lucide-react";
|
||||||
ChevronRight,
|
|
||||||
EllipsisVertical,
|
|
||||||
FileDown,
|
|
||||||
FileUp,
|
|
||||||
Plus,
|
|
||||||
Trash2,
|
|
||||||
} from "lucide-react";
|
|
||||||
import { useRef, useState } from "react";
|
import { useRef, useState } from "react";
|
||||||
import DeleteWorkbookDialog from "./DeleteWorkbookDialog";
|
import DeleteWorkbookDialog from "./DeleteWorkbookDialog";
|
||||||
import UploadFileDialog from "./UploadFileDialog";
|
import UploadFileDialog from "./UploadFileDialog";
|
||||||
import { getModelsMetadata, getSelectedUuid } from "./storage";
|
import { getModelsMetadata, getSelectedUuid } from "./storage";
|
||||||
|
|
||||||
export function DesktopMenu(props: {
|
|
||||||
newModel: () => void;
|
|
||||||
setModel: (key: string) => void;
|
|
||||||
onDownload: () => void;
|
|
||||||
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
|
||||||
onDelete: () => void;
|
|
||||||
}) {
|
|
||||||
const [isFileMenuOpen, setFileMenuOpen] = useState(false);
|
|
||||||
const anchorElement = useRef<HTMLButtonElement>(
|
|
||||||
null as unknown as HTMLButtonElement,
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<FileBarButton
|
|
||||||
onClick={(): void => setFileMenuOpen(!isFileMenuOpen)}
|
|
||||||
ref={anchorElement}
|
|
||||||
disableRipple
|
|
||||||
isOpen={isFileMenuOpen}
|
|
||||||
>
|
|
||||||
File
|
|
||||||
</FileBarButton>
|
|
||||||
<FileMenu
|
|
||||||
newModel={props.newModel}
|
|
||||||
setModel={props.setModel}
|
|
||||||
onDownload={props.onDownload}
|
|
||||||
onModelUpload={props.onModelUpload}
|
|
||||||
onDelete={props.onDelete}
|
|
||||||
isFileMenuOpen={isFileMenuOpen}
|
|
||||||
setFileMenuOpen={setFileMenuOpen}
|
|
||||||
setMobileMenuOpen={() => {}}
|
|
||||||
anchorElement={anchorElement}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function MobileMenu(props: {
|
|
||||||
newModel: () => void;
|
|
||||||
setModel: (key: string) => void;
|
|
||||||
onDownload: () => void;
|
|
||||||
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
|
||||||
onDelete: () => void;
|
|
||||||
}) {
|
|
||||||
const [isMobileMenuOpen, setMobileMenuOpen] = useState(false);
|
|
||||||
const [isFileMenuOpen, setFileMenuOpen] = useState(false);
|
|
||||||
const anchorElement = useRef<HTMLButtonElement>(
|
|
||||||
null as unknown as HTMLButtonElement,
|
|
||||||
);
|
|
||||||
const [fileMenuAnchorEl, setFileMenuAnchorEl] = useState<HTMLElement | null>(
|
|
||||||
null,
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<MenuButton
|
|
||||||
onClick={(): void => setMobileMenuOpen(true)}
|
|
||||||
ref={anchorElement}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<EllipsisVertical />
|
|
||||||
</MenuButton>
|
|
||||||
<StyledMenu
|
|
||||||
open={isMobileMenuOpen}
|
|
||||||
onClose={(): void => setMobileMenuOpen(false)}
|
|
||||||
anchorEl={anchorElement.current}
|
|
||||||
>
|
|
||||||
<MenuItemWrapper
|
|
||||||
onClick={(event) => {
|
|
||||||
setFileMenuOpen(true);
|
|
||||||
setFileMenuAnchorEl(event.currentTarget);
|
|
||||||
}}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<MenuItemText>File</MenuItemText>
|
|
||||||
<ChevronRight />
|
|
||||||
</MenuItemWrapper>
|
|
||||||
<MenuDivider />
|
|
||||||
<MenuItemWrapper
|
|
||||||
onClick={() => {
|
|
||||||
window.open("https://docs.ironcalc.com", "_blank");
|
|
||||||
setMobileMenuOpen(false);
|
|
||||||
}}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<MenuItemText>Help</MenuItemText>
|
|
||||||
</MenuItemWrapper>
|
|
||||||
</StyledMenu>
|
|
||||||
<FileMenu
|
|
||||||
newModel={props.newModel}
|
|
||||||
setModel={props.setModel}
|
|
||||||
onDownload={props.onDownload}
|
|
||||||
onModelUpload={props.onModelUpload}
|
|
||||||
onDelete={props.onDelete}
|
|
||||||
isFileMenuOpen={isFileMenuOpen}
|
|
||||||
setFileMenuOpen={setFileMenuOpen}
|
|
||||||
setMobileMenuOpen={setMobileMenuOpen}
|
|
||||||
anchorElement={anchorElement}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function FileMenu(props: {
|
export function FileMenu(props: {
|
||||||
newModel: () => void;
|
newModel: () => void;
|
||||||
setModel: (key: string) => void;
|
setModel: (key: string) => void;
|
||||||
onDownload: () => void;
|
onDownload: () => void;
|
||||||
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
onModelUpload: (blob: ArrayBuffer, fileName: string) => Promise<void>;
|
||||||
onDelete: () => void;
|
onDelete: () => void;
|
||||||
isFileMenuOpen: boolean;
|
|
||||||
setFileMenuOpen: (open: boolean) => void;
|
|
||||||
setMobileMenuOpen: (open: boolean) => void;
|
|
||||||
anchorElement: React.RefObject<HTMLButtonElement>;
|
|
||||||
}) {
|
}) {
|
||||||
|
const [isMenuOpen, setMenuOpen] = useState(false);
|
||||||
const [isImportMenuOpen, setImportMenuOpen] = useState(false);
|
const [isImportMenuOpen, setImportMenuOpen] = useState(false);
|
||||||
|
const anchorElement = useRef<HTMLButtonElement>(null);
|
||||||
const models = getModelsMetadata();
|
const models = getModelsMetadata();
|
||||||
|
const uuids = Object.keys(models);
|
||||||
const selectedUuid = getSelectedUuid();
|
const selectedUuid = getSelectedUuid();
|
||||||
const [isDeleteDialogOpen, setDeleteDialogOpen] = useState(false);
|
const [isDeleteDialogOpen, setDeleteDialogOpen] = useState(false);
|
||||||
|
|
||||||
|
const elements = [];
|
||||||
|
for (const uuid of uuids) {
|
||||||
|
elements.push(
|
||||||
|
<MenuItemWrapper
|
||||||
|
key={uuid}
|
||||||
|
onClick={() => {
|
||||||
|
props.setModel(uuid);
|
||||||
|
setMenuOpen(false);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<CheckIndicator>
|
||||||
|
{uuid === selectedUuid ? (
|
||||||
|
<StyledIcon>
|
||||||
|
<Check />
|
||||||
|
</StyledIcon>
|
||||||
|
) : (
|
||||||
|
""
|
||||||
|
)}
|
||||||
|
</CheckIndicator>
|
||||||
|
<MenuItemText
|
||||||
|
style={{
|
||||||
|
maxWidth: "240px",
|
||||||
|
overflow: "hidden",
|
||||||
|
textOverflow: "ellipsis",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{models[uuid]}
|
||||||
|
</MenuItemText>
|
||||||
|
</MenuItemWrapper>,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<StyledMenu
|
<FileMenuWrapper
|
||||||
open={props.isFileMenuOpen}
|
type="button"
|
||||||
onClose={(): void => props.setFileMenuOpen(false)}
|
id="file-menu-button"
|
||||||
anchorEl={props.anchorElement.current}
|
onClick={(): void => setMenuOpen(true)}
|
||||||
anchorOrigin={{
|
ref={anchorElement}
|
||||||
vertical: "bottom",
|
$isActive={isMenuOpen}
|
||||||
horizontal: "left",
|
aria-haspopup="true"
|
||||||
|
>
|
||||||
|
File
|
||||||
|
</FileMenuWrapper>
|
||||||
|
<Menu
|
||||||
|
open={isMenuOpen}
|
||||||
|
onClose={(): void => setMenuOpen(false)}
|
||||||
|
anchorEl={anchorElement.current}
|
||||||
|
autoFocus={false}
|
||||||
|
disableRestoreFocus={true}
|
||||||
|
sx={{
|
||||||
|
"& .MuiPaper-root": { borderRadius: "8px", padding: "4px 0px" },
|
||||||
|
"& .MuiList-root": { padding: "0" },
|
||||||
|
transform: "translate(-4px, 4px)",
|
||||||
}}
|
}}
|
||||||
transformOrigin={{
|
slotProps={{
|
||||||
vertical: "top",
|
list: {
|
||||||
horizontal: "left",
|
"aria-labelledby": "file-menu-button",
|
||||||
}}
|
tabIndex: -1,
|
||||||
// To prevent closing parent menu when interacting with submenu
|
},
|
||||||
onMouseLeave={() => {
|
|
||||||
if (!isImportMenuOpen && !isDeleteDialogOpen) {
|
|
||||||
props.setFileMenuOpen(false);
|
|
||||||
}
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<MenuItemWrapper
|
<MenuItemWrapper
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
props.newModel();
|
props.newModel();
|
||||||
props.setFileMenuOpen(false);
|
setMenuOpen(false);
|
||||||
props.setMobileMenuOpen(false);
|
|
||||||
}}
|
}}
|
||||||
disableRipple
|
|
||||||
>
|
>
|
||||||
<StyledIcon>
|
<StyledIcon>
|
||||||
<Plus />
|
<Plus />
|
||||||
@@ -169,41 +97,34 @@ export function FileMenu(props: {
|
|||||||
<MenuItemWrapper
|
<MenuItemWrapper
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setImportMenuOpen(true);
|
setImportMenuOpen(true);
|
||||||
props.setFileMenuOpen(false);
|
setMenuOpen(false);
|
||||||
props.setMobileMenuOpen(false);
|
|
||||||
}}
|
}}
|
||||||
disableRipple
|
|
||||||
>
|
>
|
||||||
<StyledIcon>
|
<StyledIcon>
|
||||||
<FileUp />
|
<FileUp />
|
||||||
</StyledIcon>
|
</StyledIcon>
|
||||||
<MenuItemText>Import</MenuItemText>
|
<MenuItemText>Import</MenuItemText>
|
||||||
</MenuItemWrapper>
|
</MenuItemWrapper>
|
||||||
<MenuItemWrapper
|
<MenuItemWrapper onClick={props.onDownload}>
|
||||||
onClick={() => {
|
<StyledIcon>
|
||||||
props.onDownload();
|
<FileDown />
|
||||||
props.setMobileMenuOpen(false);
|
</StyledIcon>
|
||||||
}}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<StyledFileDown />
|
|
||||||
<MenuItemText>Download (.xlsx)</MenuItemText>
|
<MenuItemText>Download (.xlsx)</MenuItemText>
|
||||||
</MenuItemWrapper>
|
</MenuItemWrapper>
|
||||||
<MenuDivider />
|
|
||||||
<MenuItemWrapper
|
<MenuItemWrapper
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setDeleteDialogOpen(true);
|
setDeleteDialogOpen(true);
|
||||||
props.setFileMenuOpen(false);
|
setMenuOpen(false);
|
||||||
props.setMobileMenuOpen(false);
|
|
||||||
}}
|
}}
|
||||||
disableRipple
|
|
||||||
>
|
>
|
||||||
<StyledIcon>
|
<StyledIcon>
|
||||||
<Trash2 />
|
<Trash2 />
|
||||||
</StyledIcon>
|
</StyledIcon>
|
||||||
<MenuItemText>Delete workbook</MenuItemText>
|
<MenuItemText>Delete workbook</MenuItemText>
|
||||||
</MenuItemWrapper>
|
</MenuItemWrapper>
|
||||||
</StyledMenu>
|
<MenuDivider />
|
||||||
|
{elements}
|
||||||
|
</Menu>
|
||||||
<Modal
|
<Modal
|
||||||
open={isImportMenuOpen}
|
open={isImportMenuOpen}
|
||||||
onClose={() => {
|
onClose={() => {
|
||||||
@@ -228,7 +149,7 @@ export function FileMenu(props: {
|
|||||||
<DeleteWorkbookDialog
|
<DeleteWorkbookDialog
|
||||||
onClose={() => setDeleteDialogOpen(false)}
|
onClose={() => setDeleteDialogOpen(false)}
|
||||||
onConfirm={props.onDelete}
|
onConfirm={props.onDelete}
|
||||||
workbookName={selectedUuid ? models[selectedUuid]?.name || "" : ""}
|
workbookName={selectedUuid ? models[selectedUuid] : ""}
|
||||||
/>
|
/>
|
||||||
</Modal>
|
</Modal>
|
||||||
</>
|
</>
|
||||||
@@ -246,55 +167,7 @@ const StyledIcon = styled.div`
|
|||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const MenuButton = styled(IconButton)`
|
const MenuDivider = styled.div`
|
||||||
height: 32px;
|
|
||||||
width: 32px;
|
|
||||||
padding: 8px;
|
|
||||||
border-radius: 4px;
|
|
||||||
svg {
|
|
||||||
stroke-width: 2px;
|
|
||||||
stroke: #757575;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
}
|
|
||||||
&:hover {
|
|
||||||
background-color: #f2f2f2;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: #e0e0e0;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const FileBarButton = styled(Button)<{ isOpen: boolean }>`
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
align-items: center;
|
|
||||||
font-size: 12px;
|
|
||||||
height: 32px;
|
|
||||||
width: auto;
|
|
||||||
padding: 4px 8px;
|
|
||||||
font-weight: 400;
|
|
||||||
min-width: 0px;
|
|
||||||
text-transform: capitalize;
|
|
||||||
color: #333333;
|
|
||||||
background-color: ${({ isOpen }) => (isOpen ? "#f2f2f2" : "none")};
|
|
||||||
&:hover {
|
|
||||||
background-color: #f2f2f2;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background-color: #e0e0e0;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const StyledFileDown = styled(FileDown)`
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
color: #333333;
|
|
||||||
padding-right: 10px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
|
|
||||||
const MenuDivider = styled("div")`
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin: auto;
|
margin: auto;
|
||||||
margin-top: 4px;
|
margin-top: 4px;
|
||||||
@@ -305,7 +178,6 @@ const MenuDivider = styled("div")`
|
|||||||
const MenuItemText = styled.div`
|
const MenuItemText = styled.div`
|
||||||
color: #000;
|
color: #000;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
flex-grow: 1;
|
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const MenuItemWrapper = styled(MenuItem)`
|
const MenuItemWrapper = styled(MenuItem)`
|
||||||
@@ -318,19 +190,26 @@ const MenuItemWrapper = styled(MenuItem)`
|
|||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
height: 32px;
|
height: 32px;
|
||||||
min-height: 32px;
|
`;
|
||||||
svg {
|
|
||||||
width: 16px;
|
const FileMenuWrapper = styled.button<{ $isActive: boolean }>`
|
||||||
height: 16px;
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 12px;
|
||||||
|
font-family: Inter;
|
||||||
|
padding: 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
background-color: ${(props) => (props.$isActive ? "#e6e6e6" : "transparent")};
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
&:hover {
|
||||||
|
background-color: #f2f2f2;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const StyledMenu = styled(Menu)`
|
const CheckIndicator = styled.span`
|
||||||
.MuiPaper-root {
|
display: flex;
|
||||||
border-radius: 8px;
|
justify-content: center;
|
||||||
padding: 4px 0px;
|
min-width: 26px;
|
||||||
},
|
|
||||||
.MuiList-root {
|
|
||||||
padding: 0;
|
|
||||||
},
|
|
||||||
`;
|
`;
|
||||||
|
|||||||
@@ -1,29 +0,0 @@
|
|||||||
import styled from "@emotion/styled";
|
|
||||||
import WorkbookList from "./WorkbookList";
|
|
||||||
|
|
||||||
interface DrawerContentProps {
|
|
||||||
setModel: (key: string) => void;
|
|
||||||
onDelete: (uuid: string) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
function DrawerContent(props: DrawerContentProps) {
|
|
||||||
const { setModel, onDelete } = props;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ContentContainer>
|
|
||||||
<WorkbookList setModel={setModel} onDelete={onDelete} />
|
|
||||||
</ContentContainer>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const ContentContainer = styled("div")`
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 4px;
|
|
||||||
padding: 16px 12px;
|
|
||||||
height: 100%;
|
|
||||||
overflow: scroll;
|
|
||||||
font-size: 12px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
export default DrawerContent;
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
import styled from "@emotion/styled";
|
|
||||||
import { BookOpen } from "lucide-react";
|
|
||||||
|
|
||||||
function DrawerFooter() {
|
|
||||||
return (
|
|
||||||
<StyledDrawerFooter>
|
|
||||||
<FooterLink
|
|
||||||
href="https://docs.ironcalc.com/"
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener noreferrer"
|
|
||||||
>
|
|
||||||
<OpenBookIcon>
|
|
||||||
<BookOpen />
|
|
||||||
</OpenBookIcon>
|
|
||||||
<FooterLinkText>Documentation</FooterLinkText>
|
|
||||||
</FooterLink>
|
|
||||||
</StyledDrawerFooter>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const StyledDrawerFooter = styled("div")`
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 12px;
|
|
||||||
justify-content: space-between;
|
|
||||||
max-height: 60px;
|
|
||||||
height: 60px;
|
|
||||||
border-top: 1px solid #e0e0e0;
|
|
||||||
box-sizing: border-box;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const FooterLink = styled("a")`
|
|
||||||
display: flex;
|
|
||||||
gap: 8px;
|
|
||||||
justify-content: flex-start;
|
|
||||||
font-size: 14px;
|
|
||||||
width: 100%;
|
|
||||||
min-width: 172px;
|
|
||||||
border-radius: 8px;
|
|
||||||
padding: 8px 4px 8px 8px;
|
|
||||||
transition: gap 0.5s;
|
|
||||||
background-color: transparent;
|
|
||||||
color: #000;
|
|
||||||
text-decoration: none;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background-color: #e0e0e0 !important;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const OpenBookIcon = styled("div")`
|
|
||||||
height: 16px;
|
|
||||||
width: 16px;
|
|
||||||
svg {
|
|
||||||
height: 16px;
|
|
||||||
width: 16px;
|
|
||||||
stroke: #9e9e9e;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const FooterLinkText = styled("div")`
|
|
||||||
color: #000;
|
|
||||||
font-size: 12px;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 240px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
`;
|
|
||||||
|
|
||||||
export default DrawerFooter;
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
import styled from "@emotion/styled";
|
|
||||||
import { IronCalcLogo } from "@ironcalc/workbook";
|
|
||||||
import { IconButton } from "@mui/material";
|
|
||||||
import { Plus } from "lucide-react";
|
|
||||||
|
|
||||||
interface DrawerHeaderProps {
|
|
||||||
onNewModel: () => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
function DrawerHeader({ onNewModel }: DrawerHeaderProps) {
|
|
||||||
return (
|
|
||||||
<HeaderContainer>
|
|
||||||
<StyledDesktopLogo />
|
|
||||||
<AddButton onClick={onNewModel} title="New workbook">
|
|
||||||
<PlusIcon />
|
|
||||||
</AddButton>
|
|
||||||
</HeaderContainer>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const HeaderContainer = styled("div")`
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 12px 8px 12px 16px;
|
|
||||||
justify-content: space-between;
|
|
||||||
max-height: 60px;
|
|
||||||
min-height: 60px;
|
|
||||||
border-bottom: 1px solid #e0e0e0;
|
|
||||||
box-sizing: border-box;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const StyledDesktopLogo = styled(IronCalcLogo)`
|
|
||||||
width: 120px;
|
|
||||||
height: 28px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const AddButton = styled(IconButton)`
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
cursor: pointer;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
padding: 8px;
|
|
||||||
height: 32px;
|
|
||||||
width: 32px;
|
|
||||||
border-radius: 4px;
|
|
||||||
margin-left: 10px;
|
|
||||||
color: #333333;
|
|
||||||
stroke-width: 2px;
|
|
||||||
&:hover {
|
|
||||||
background-color: #e0e0e0;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const PlusIcon = styled(Plus)`
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
export default DrawerHeader;
|
|
||||||
@@ -1,51 +0,0 @@
|
|||||||
import styled from "@emotion/styled";
|
|
||||||
import { Drawer } from "@mui/material";
|
|
||||||
import DrawerContent from "./DrawerContent";
|
|
||||||
import DrawerFooter from "./DrawerFooter";
|
|
||||||
import DrawerHeader from "./DrawerHeader";
|
|
||||||
|
|
||||||
interface LeftDrawerProps {
|
|
||||||
open: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
newModel: () => void;
|
|
||||||
setModel: (key: string) => void;
|
|
||||||
onDelete: (uuid: string) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
function LeftDrawer({
|
|
||||||
open,
|
|
||||||
onClose,
|
|
||||||
newModel,
|
|
||||||
setModel,
|
|
||||||
onDelete,
|
|
||||||
}: LeftDrawerProps) {
|
|
||||||
return (
|
|
||||||
<DrawerWrapper
|
|
||||||
variant="persistent"
|
|
||||||
anchor="left"
|
|
||||||
open={open}
|
|
||||||
onClose={onClose}
|
|
||||||
>
|
|
||||||
<DrawerHeader onNewModel={newModel} />
|
|
||||||
<DrawerContent setModel={setModel} onDelete={onDelete} />
|
|
||||||
|
|
||||||
<DrawerFooter />
|
|
||||||
</DrawerWrapper>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const DrawerWrapper = styled(Drawer)`
|
|
||||||
width: 264px;
|
|
||||||
height: 100%;
|
|
||||||
flex-shrink: 0;
|
|
||||||
font-family: "Inter", sans-serif;
|
|
||||||
|
|
||||||
.MuiDrawer-paper {
|
|
||||||
width: 264px;
|
|
||||||
background-color: #f5f5f5;
|
|
||||||
overflow: hidden;
|
|
||||||
border-right: 1px solid #e0e0e0;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export default LeftDrawer;
|
|
||||||
@@ -1,379 +0,0 @@
|
|||||||
import styled from "@emotion/styled";
|
|
||||||
import { Menu, MenuItem, Modal } from "@mui/material";
|
|
||||||
import {
|
|
||||||
EllipsisVertical,
|
|
||||||
FileDown,
|
|
||||||
FileSpreadsheet,
|
|
||||||
Trash2,
|
|
||||||
} from "lucide-react";
|
|
||||||
import type React from "react";
|
|
||||||
import { useEffect, useState } from "react";
|
|
||||||
import DeleteWorkbookDialog from "../DeleteWorkbookDialog";
|
|
||||||
import { downloadModel } from "../rpc";
|
|
||||||
import {
|
|
||||||
getModelsMetadata,
|
|
||||||
getSelectedUuid,
|
|
||||||
selectModelFromStorage,
|
|
||||||
} from "../storage";
|
|
||||||
|
|
||||||
interface WorkbookListProps {
|
|
||||||
setModel: (key: string) => void;
|
|
||||||
onDelete: (uuid: string) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
function WorkbookList({ setModel, onDelete }: WorkbookListProps) {
|
|
||||||
const [menuAnchorEl, setMenuAnchorEl] = useState<null | HTMLElement>(null);
|
|
||||||
const [selectedWorkbookUuid, setSelectedWorkbookUuid] = useState<
|
|
||||||
string | null
|
|
||||||
>(null);
|
|
||||||
const [isDeleteDialogOpen, setIsDeleteDialogOpen] = useState(false);
|
|
||||||
const [workbookToDelete, setWorkbookToDelete] = useState<string | null>(null);
|
|
||||||
const [intendedSelection, setIntendedSelection] = useState<string | null>(
|
|
||||||
null,
|
|
||||||
);
|
|
||||||
|
|
||||||
const selectedUuid = getSelectedUuid();
|
|
||||||
|
|
||||||
// Clear intended selection when selectedUuid changes from outside
|
|
||||||
useEffect(() => {
|
|
||||||
if (intendedSelection && selectedUuid === intendedSelection) {
|
|
||||||
setIntendedSelection(null);
|
|
||||||
}
|
|
||||||
}, [selectedUuid, intendedSelection]);
|
|
||||||
|
|
||||||
const handleMenuOpen = (
|
|
||||||
event: React.MouseEvent<HTMLButtonElement>,
|
|
||||||
uuid: string,
|
|
||||||
) => {
|
|
||||||
console.log("Menu open", uuid);
|
|
||||||
event.stopPropagation();
|
|
||||||
setSelectedWorkbookUuid(uuid);
|
|
||||||
setMenuAnchorEl(event.currentTarget);
|
|
||||||
setIntendedSelection(uuid);
|
|
||||||
setModel(uuid);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleMenuClose = () => {
|
|
||||||
console.log(
|
|
||||||
"Menu closing, selectedWorkbookUuid:",
|
|
||||||
selectedWorkbookUuid,
|
|
||||||
"intendedSelection:",
|
|
||||||
intendedSelection,
|
|
||||||
);
|
|
||||||
setMenuAnchorEl(null);
|
|
||||||
// If we have an intended selection, make sure it's still selected
|
|
||||||
if (intendedSelection && intendedSelection !== selectedUuid) {
|
|
||||||
console.log("Re-selecting intended workbook:", intendedSelection);
|
|
||||||
setModel(intendedSelection);
|
|
||||||
}
|
|
||||||
// Don't reset selectedWorkbookUuid here - we want to keep track of which workbook was selected
|
|
||||||
// The selectedWorkbookUuid will be used for download/delete operations
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDeleteClick = (uuid: string) => {
|
|
||||||
console.log("Delete workbook:", uuid);
|
|
||||||
setWorkbookToDelete(uuid);
|
|
||||||
setIsDeleteDialogOpen(true);
|
|
||||||
setIntendedSelection(null);
|
|
||||||
handleMenuClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDeleteConfirm = () => {
|
|
||||||
if (workbookToDelete) {
|
|
||||||
onDelete(workbookToDelete);
|
|
||||||
setWorkbookToDelete(null);
|
|
||||||
}
|
|
||||||
setIsDeleteDialogOpen(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDeleteCancel = () => {
|
|
||||||
setWorkbookToDelete(null);
|
|
||||||
setIsDeleteDialogOpen(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleDownload = async (uuid: string) => {
|
|
||||||
try {
|
|
||||||
const model = selectModelFromStorage(uuid);
|
|
||||||
if (model) {
|
|
||||||
const bytes = model.toBytes();
|
|
||||||
const fileName = model.getName();
|
|
||||||
await downloadModel(bytes, fileName);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Failed to download workbook:", error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Group workbooks by creation date
|
|
||||||
const groupWorkbooks = () => {
|
|
||||||
const now = Date.now();
|
|
||||||
const millisecondsInDay = 24 * 60 * 60 * 1000;
|
|
||||||
const millisecondsIn30Days = 30 * millisecondsInDay;
|
|
||||||
|
|
||||||
const modelsCreatedToday = [];
|
|
||||||
const modelsCreatedThisMonth = [];
|
|
||||||
const olderModels = [];
|
|
||||||
const modelsMetadata = getModelsMetadata();
|
|
||||||
|
|
||||||
for (const uuid in modelsMetadata) {
|
|
||||||
const createdAt = modelsMetadata[uuid].createdAt;
|
|
||||||
const age = now - createdAt;
|
|
||||||
|
|
||||||
if (age < millisecondsInDay) {
|
|
||||||
modelsCreatedToday.push(uuid);
|
|
||||||
} else if (age < millisecondsIn30Days) {
|
|
||||||
modelsCreatedThisMonth.push(uuid);
|
|
||||||
} else {
|
|
||||||
olderModels.push(uuid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sort each group by creation timestamp (newest first)
|
|
||||||
const sortByNewest = (uuids: string[]) =>
|
|
||||||
uuids.sort(
|
|
||||||
(a, b) => modelsMetadata[b].createdAt - modelsMetadata[a].createdAt,
|
|
||||||
);
|
|
||||||
|
|
||||||
return {
|
|
||||||
modelsCreatedToday: sortByNewest(modelsCreatedToday),
|
|
||||||
modelsCreatedThisMonth: sortByNewest(modelsCreatedThisMonth),
|
|
||||||
olderModels: sortByNewest(olderModels),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const { modelsCreatedToday, modelsCreatedThisMonth, olderModels } =
|
|
||||||
groupWorkbooks();
|
|
||||||
|
|
||||||
const renderWorkbookItem = (uuid: string) => {
|
|
||||||
const isMenuOpen = menuAnchorEl !== null && selectedWorkbookUuid === uuid;
|
|
||||||
const isAnyMenuOpen = menuAnchorEl !== null;
|
|
||||||
const models = getModelsMetadata();
|
|
||||||
return (
|
|
||||||
<WorkbookListItem
|
|
||||||
key={uuid}
|
|
||||||
onClick={() => {
|
|
||||||
// Prevent clicking on list items when any menu is open
|
|
||||||
if (isAnyMenuOpen) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setModel(uuid);
|
|
||||||
}}
|
|
||||||
selected={uuid === selectedUuid}
|
|
||||||
disableRipple
|
|
||||||
style={{ pointerEvents: isAnyMenuOpen ? "none" : "auto" }}
|
|
||||||
>
|
|
||||||
<StorageIndicator>
|
|
||||||
<FileSpreadsheet />
|
|
||||||
</StorageIndicator>
|
|
||||||
<WorkbookListText>{models[uuid].name}</WorkbookListText>
|
|
||||||
<EllipsisButton
|
|
||||||
onClick={(e) => handleMenuOpen(e, uuid)}
|
|
||||||
isOpen={isMenuOpen}
|
|
||||||
onMouseDown={(e) => e.stopPropagation()}
|
|
||||||
style={{ pointerEvents: "auto" }}
|
|
||||||
>
|
|
||||||
<EllipsisVertical />
|
|
||||||
</EllipsisButton>
|
|
||||||
</WorkbookListItem>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderSection = (title: string, uuids: string[]) => {
|
|
||||||
if (uuids.length === 0) return null;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<SectionContainer key={title}>
|
|
||||||
<SectionTitle>{title}</SectionTitle>
|
|
||||||
{uuids.map(renderWorkbookItem)}
|
|
||||||
</SectionContainer>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const models = getModelsMetadata();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{renderSection("Today", modelsCreatedToday)}
|
|
||||||
{renderSection("Last 30 Days", modelsCreatedThisMonth)}
|
|
||||||
{renderSection("Older", olderModels)}
|
|
||||||
|
|
||||||
<StyledMenu
|
|
||||||
anchorEl={menuAnchorEl}
|
|
||||||
open={Boolean(menuAnchorEl)}
|
|
||||||
onClose={handleMenuClose}
|
|
||||||
MenuListProps={{
|
|
||||||
dense: true,
|
|
||||||
}}
|
|
||||||
anchorOrigin={{
|
|
||||||
vertical: "bottom",
|
|
||||||
horizontal: "right",
|
|
||||||
}}
|
|
||||||
transformOrigin={{
|
|
||||||
vertical: "top",
|
|
||||||
horizontal: "right",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<MenuItemWrapper
|
|
||||||
onClick={() => {
|
|
||||||
console.log(
|
|
||||||
"Download clicked, selectedWorkbookUuid:",
|
|
||||||
selectedWorkbookUuid,
|
|
||||||
);
|
|
||||||
if (selectedWorkbookUuid) {
|
|
||||||
handleDownload(selectedWorkbookUuid);
|
|
||||||
}
|
|
||||||
setIntendedSelection(null);
|
|
||||||
handleMenuClose();
|
|
||||||
}}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<FileDown />
|
|
||||||
Download (.xlsx)
|
|
||||||
</MenuItemWrapper>
|
|
||||||
<MenuDivider />
|
|
||||||
<MenuItemWrapper
|
|
||||||
selected={false}
|
|
||||||
onClick={() => {
|
|
||||||
if (selectedWorkbookUuid) {
|
|
||||||
handleDeleteClick(selectedWorkbookUuid);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
disableRipple
|
|
||||||
>
|
|
||||||
<Trash2 size={16} />
|
|
||||||
Delete workbook
|
|
||||||
</MenuItemWrapper>
|
|
||||||
</StyledMenu>
|
|
||||||
|
|
||||||
<Modal
|
|
||||||
open={isDeleteDialogOpen}
|
|
||||||
onClose={handleDeleteCancel}
|
|
||||||
aria-labelledby="delete-dialog-title"
|
|
||||||
aria-describedby="delete-dialog-description"
|
|
||||||
>
|
|
||||||
<DeleteWorkbookDialog
|
|
||||||
onClose={handleDeleteCancel}
|
|
||||||
onConfirm={handleDeleteConfirm}
|
|
||||||
workbookName={workbookToDelete ? models[workbookToDelete].name : ""}
|
|
||||||
/>
|
|
||||||
</Modal>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const StorageIndicator = styled("div")`
|
|
||||||
height: 16px;
|
|
||||||
width: 16px;
|
|
||||||
svg {
|
|
||||||
height: 16px;
|
|
||||||
width: 16px;
|
|
||||||
stroke: #9e9e9e;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const EllipsisButton = styled("button")<{ isOpen: boolean }>`
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
cursor: pointer;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
padding: 4px;
|
|
||||||
height: 24px;
|
|
||||||
width: 24px;
|
|
||||||
border-radius: 4px;
|
|
||||||
color: #333333;
|
|
||||||
stroke-width: 2px;
|
|
||||||
background-color: ${({ isOpen }) => (isOpen ? "#E0E0E0" : "none")};
|
|
||||||
opacity: ${({ isOpen }) => (isOpen ? "1" : "0.5")};
|
|
||||||
transition: opacity 0.3s, background-color 0.3s;
|
|
||||||
&:hover {
|
|
||||||
background: none;
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
&:active {
|
|
||||||
background: #bdbdbd;
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const WorkbookListItem = styled(MenuItem)<{ selected: boolean }>`
|
|
||||||
display: flex;
|
|
||||||
gap: 8px;
|
|
||||||
justify-content: flex-start;
|
|
||||||
font-size: 14px;
|
|
||||||
width: 100%;
|
|
||||||
min-width: 172px;
|
|
||||||
border-radius: 8px;
|
|
||||||
padding: 8px 4px 8px 8px;
|
|
||||||
height: 32px;
|
|
||||||
min-height: 32px;
|
|
||||||
transition: gap 0.5s;
|
|
||||||
background-color: ${({ selected }) =>
|
|
||||||
selected ? "#e0e0e0 !important" : "transparent"};
|
|
||||||
|
|
||||||
/* Prevent hover effects when menu is open */
|
|
||||||
&:hover {
|
|
||||||
background-color: ${({ selected }) =>
|
|
||||||
selected ? "#e0e0e0 !important" : "transparent"};
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const WorkbookListText = styled("div")`
|
|
||||||
color: #000;
|
|
||||||
font-size: 12px;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 240px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const StyledMenu = styled(Menu)`
|
|
||||||
.MuiPaper-root {
|
|
||||||
border-radius: 8px;
|
|
||||||
padding: 4px 0px;
|
|
||||||
box-shadow: 0px 2px 4px rgba(0, 0, 0, 0.01);
|
|
||||||
},
|
|
||||||
.MuiList-root {
|
|
||||||
padding: 0;
|
|
||||||
},
|
|
||||||
`;
|
|
||||||
|
|
||||||
const MenuDivider = styled("div")`
|
|
||||||
width: 100%;
|
|
||||||
margin: auto;
|
|
||||||
margin-top: 4px;
|
|
||||||
margin-bottom: 4px;
|
|
||||||
border-top: 1px solid #eeeeee;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const MenuItemWrapper = styled(MenuItem)`
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-start;
|
|
||||||
font-size: 12px;
|
|
||||||
width: calc(100% - 8px);
|
|
||||||
min-width: 140px;
|
|
||||||
margin: 0px 4px;
|
|
||||||
border-radius: 4px;
|
|
||||||
padding: 8px;
|
|
||||||
height: 32px;
|
|
||||||
gap: 8px;
|
|
||||||
svg {
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const SectionContainer = styled("div")`
|
|
||||||
margin-bottom: 16px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
const SectionTitle = styled("div")`
|
|
||||||
font-weight: 600;
|
|
||||||
color: #9e9e9e;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
padding: 0px 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
`;
|
|
||||||
|
|
||||||
export default WorkbookList;
|
|
||||||
@@ -72,10 +72,10 @@ export function WorkbookTitle(properties: {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const Container = styled("div")`
|
const Container = styled("div")`
|
||||||
text-align: left;
|
text-align: center;
|
||||||
padding: 6px 4px;
|
padding: 8px;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
font-weight: 600;
|
font-weight: 700;
|
||||||
font-family: Inter;
|
font-family: Inter;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
@@ -108,7 +108,7 @@ const TitleInput = styled("input")`
|
|||||||
background-color: #f2f2f2;
|
background-color: #f2f2f2;
|
||||||
}
|
}
|
||||||
&:focus {
|
&:focus {
|
||||||
outline: 1px solid grey;
|
border: 1px solid grey;
|
||||||
}
|
}
|
||||||
font-weight: inherit;
|
font-weight: inherit;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { base64ToBytes, bytesToBase64 } from "./util";
|
|||||||
|
|
||||||
const MAX_WORKBOOKS = 50;
|
const MAX_WORKBOOKS = 50;
|
||||||
|
|
||||||
type ModelsMetadata = Record<string, { name: string; createdAt: number }>;
|
type ModelsMetadata = Record<string, string>;
|
||||||
|
|
||||||
export function updateNameSelectedWorkbook(model: Model, newName: string) {
|
export function updateNameSelectedWorkbook(model: Model, newName: string) {
|
||||||
const uuid = localStorage.getItem("selected");
|
const uuid = localStorage.getItem("selected");
|
||||||
@@ -12,11 +12,7 @@ export function updateNameSelectedWorkbook(model: Model, newName: string) {
|
|||||||
if (modelsJson) {
|
if (modelsJson) {
|
||||||
try {
|
try {
|
||||||
const models = JSON.parse(modelsJson);
|
const models = JSON.parse(modelsJson);
|
||||||
if (models[uuid]) {
|
models[uuid] = newName;
|
||||||
models[uuid].name = newName;
|
|
||||||
} else {
|
|
||||||
models[uuid] = { name: newName, createdAt: Date.now() };
|
|
||||||
}
|
|
||||||
localStorage.setItem("models", JSON.stringify(models));
|
localStorage.setItem("models", JSON.stringify(models));
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.warn("Failed saving new name");
|
console.warn("Failed saving new name");
|
||||||
@@ -32,26 +28,7 @@ export function getModelsMetadata(): ModelsMetadata {
|
|||||||
if (!modelsJson) {
|
if (!modelsJson) {
|
||||||
modelsJson = "{}";
|
modelsJson = "{}";
|
||||||
}
|
}
|
||||||
const models = JSON.parse(modelsJson);
|
return JSON.parse(modelsJson);
|
||||||
|
|
||||||
// Migrate old format to new format
|
|
||||||
const migratedModels: ModelsMetadata = {};
|
|
||||||
for (const [uuid, value] of Object.entries(models)) {
|
|
||||||
if (typeof value === "string") {
|
|
||||||
// Old format: just the name string
|
|
||||||
migratedModels[uuid] = { name: value, createdAt: Date.now() };
|
|
||||||
} else if (typeof value === "object" && value !== null && "name" in value) {
|
|
||||||
// New format: object with name and createdAt
|
|
||||||
migratedModels[uuid] = value as { name: string; createdAt: number };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save migrated data back to localStorage
|
|
||||||
if (JSON.stringify(models) !== JSON.stringify(migratedModels)) {
|
|
||||||
localStorage.setItem("models", JSON.stringify(migratedModels));
|
|
||||||
}
|
|
||||||
|
|
||||||
return migratedModels;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pick a different name Workbook{N} where N = 1, 2, 3
|
// Pick a different name Workbook{N} where N = 1, 2, 3
|
||||||
@@ -71,14 +48,14 @@ function getNewName(existingNames: string[]): string {
|
|||||||
|
|
||||||
export function createNewModel(): Model {
|
export function createNewModel(): Model {
|
||||||
const models = getModelsMetadata();
|
const models = getModelsMetadata();
|
||||||
const name = getNewName(Object.values(models).map((m) => m.name));
|
const name = getNewName(Object.values(models));
|
||||||
|
|
||||||
const model = new Model(name, "en", "UTC");
|
const model = new Model(name, "en", "UTC");
|
||||||
const uuid = crypto.randomUUID();
|
const uuid = crypto.randomUUID();
|
||||||
localStorage.setItem("selected", uuid);
|
localStorage.setItem("selected", uuid);
|
||||||
localStorage.setItem(uuid, bytesToBase64(model.toBytes()));
|
localStorage.setItem(uuid, bytesToBase64(model.toBytes()));
|
||||||
|
|
||||||
models[uuid] = { name, createdAt: Date.now() };
|
models[uuid] = name;
|
||||||
localStorage.setItem("models", JSON.stringify(models));
|
localStorage.setItem("models", JSON.stringify(models));
|
||||||
return model;
|
return model;
|
||||||
}
|
}
|
||||||
@@ -118,7 +95,7 @@ export function saveModelToStorage(model: Model) {
|
|||||||
modelsJson = "{}";
|
modelsJson = "{}";
|
||||||
}
|
}
|
||||||
const models = JSON.parse(modelsJson);
|
const models = JSON.parse(modelsJson);
|
||||||
models[uuid] = { name: model.getName(), createdAt: Date.now() };
|
models[uuid] = model.getName();
|
||||||
localStorage.setItem("models", JSON.stringify(models));
|
localStorage.setItem("models", JSON.stringify(models));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -150,37 +127,3 @@ export function deleteSelectedModel(): Model | null {
|
|||||||
}
|
}
|
||||||
return selectModelFromStorage(uuids[0]);
|
return selectModelFromStorage(uuids[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function deleteModelByUuid(uuid: string): Model | null {
|
|
||||||
localStorage.removeItem(uuid);
|
|
||||||
const metadata = getModelsMetadata();
|
|
||||||
delete metadata[uuid];
|
|
||||||
localStorage.setItem("models", JSON.stringify(metadata));
|
|
||||||
|
|
||||||
// If this was the selected model, we need to select a different one
|
|
||||||
const selectedUuid = localStorage.getItem("selected");
|
|
||||||
if (selectedUuid === uuid) {
|
|
||||||
const uuids = Object.keys(metadata);
|
|
||||||
if (uuids.length === 0) {
|
|
||||||
return createNewModel();
|
|
||||||
}
|
|
||||||
// Find the newest workbook by creation timestamp
|
|
||||||
const newestUuid = uuids.reduce((newest, current) => {
|
|
||||||
const newestTime = metadata[newest]?.createdAt || 0;
|
|
||||||
const currentTime = metadata[current]?.createdAt || 0;
|
|
||||||
return currentTime > newestTime ? current : newest;
|
|
||||||
});
|
|
||||||
return selectModelFromStorage(newestUuid);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If it wasn't the selected model, return the currently selected model
|
|
||||||
if (selectedUuid) {
|
|
||||||
const modelBytesString = localStorage.getItem(selectedUuid);
|
|
||||||
if (modelBytesString) {
|
|
||||||
return Model.from_bytes(base64ToBytes(modelBytesString));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fallback to creating a new model if no valid selected model
|
|
||||||
return createNewModel();
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ use itertools::Itertools;
|
|||||||
|
|
||||||
use ironcalc_base::{
|
use ironcalc_base::{
|
||||||
expressions::{
|
expressions::{
|
||||||
parser::{stringify::to_excel_string, Node},
|
parser::{static_analysis::StaticResult, stringify::to_excel_string, Node},
|
||||||
types::CellReferenceRC,
|
types::CellReferenceRC,
|
||||||
utils::number_to_column,
|
utils::number_to_column,
|
||||||
},
|
},
|
||||||
@@ -56,7 +56,7 @@ fn get_formula_attribute(
|
|||||||
|
|
||||||
pub(crate) fn get_worksheet_xml(
|
pub(crate) fn get_worksheet_xml(
|
||||||
worksheet: &Worksheet,
|
worksheet: &Worksheet,
|
||||||
parsed_formulas: &[Node],
|
parsed_formulas: &[(Node, StaticResult)],
|
||||||
dimension: &str,
|
dimension: &str,
|
||||||
is_sheet_selected: bool,
|
is_sheet_selected: bool,
|
||||||
) -> String {
|
) -> String {
|
||||||
@@ -104,7 +104,7 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
row_data_str.push(format!("<c r=\"{cell_name}\"{style}/>"));
|
row_data_str.push(format!("<c r=\"{cell_name}\"{style}/>"));
|
||||||
}
|
}
|
||||||
Cell::BooleanCell { v, s } => {
|
Cell::SpillBooleanCell { v, s, .. } | Cell::BooleanCell { v, s } => {
|
||||||
// <c r="A8" t="b" s="1">
|
// <c r="A8" t="b" s="1">
|
||||||
// <v>1</v>
|
// <v>1</v>
|
||||||
// </c>
|
// </c>
|
||||||
@@ -114,7 +114,7 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
"<c r=\"{cell_name}\" t=\"b\"{style}><v>{b}</v></c>"
|
"<c r=\"{cell_name}\" t=\"b\"{style}><v>{b}</v></c>"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
Cell::NumberCell { v, s } => {
|
Cell::SpillNumberCell { v, s, .. } | Cell::NumberCell { v, s } => {
|
||||||
// Normally the type number is left out. Example:
|
// Normally the type number is left out. Example:
|
||||||
// <c r="C6" s="1">
|
// <c r="C6" s="1">
|
||||||
// <v>3</v>
|
// <v>3</v>
|
||||||
@@ -122,7 +122,7 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
row_data_str.push(format!("<c r=\"{cell_name}\"{style}><v>{v}</v></c>"));
|
row_data_str.push(format!("<c r=\"{cell_name}\"{style}><v>{v}</v></c>"));
|
||||||
}
|
}
|
||||||
Cell::ErrorCell { ei, s } => {
|
Cell::SpillErrorCell { ei, s, .. } | Cell::ErrorCell { ei, s } => {
|
||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
row_data_str.push(format!(
|
row_data_str.push(format!(
|
||||||
"<c r=\"{cell_name}\" t=\"e\"{style}><v>{ei}</v></c>"
|
"<c r=\"{cell_name}\" t=\"e\"{style}><v>{ei}</v></c>"
|
||||||
@@ -153,7 +153,7 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
worksheet.get_name(),
|
worksheet.get_name(),
|
||||||
*row_index,
|
*row_index,
|
||||||
*column_index,
|
*column_index,
|
||||||
&parsed_formulas[*f as usize],
|
&parsed_formulas[*f as usize].0,
|
||||||
);
|
);
|
||||||
|
|
||||||
let b = i32::from(*v);
|
let b = i32::from(*v);
|
||||||
@@ -172,7 +172,7 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
worksheet.get_name(),
|
worksheet.get_name(),
|
||||||
*row_index,
|
*row_index,
|
||||||
*column_index,
|
*column_index,
|
||||||
&parsed_formulas[*f as usize],
|
&parsed_formulas[*f as usize].0,
|
||||||
);
|
);
|
||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
|
|
||||||
@@ -189,14 +189,14 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
worksheet.get_name(),
|
worksheet.get_name(),
|
||||||
*row_index,
|
*row_index,
|
||||||
*column_index,
|
*column_index,
|
||||||
&parsed_formulas[*f as usize],
|
&parsed_formulas[*f as usize].0,
|
||||||
);
|
);
|
||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
let escaped_v = escape_xml(v);
|
let escaped_v = escape_xml(v);
|
||||||
|
|
||||||
row_data_str.push(format!(
|
row_data_str.push(format!(
|
||||||
"<c r=\"{cell_name}\" t=\"str\"{style}><f>{formula}</f><v>{escaped_v}</v></c>"
|
"<c r=\"{cell_name}\" t=\"str\"{style}><f>{formula}</f><v>{escaped_v}</v></c>"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
Cell::CellFormulaError {
|
Cell::CellFormulaError {
|
||||||
f,
|
f,
|
||||||
@@ -213,13 +213,135 @@ pub(crate) fn get_worksheet_xml(
|
|||||||
worksheet.get_name(),
|
worksheet.get_name(),
|
||||||
*row_index,
|
*row_index,
|
||||||
*column_index,
|
*column_index,
|
||||||
&parsed_formulas[*f as usize],
|
&parsed_formulas[*f as usize].0,
|
||||||
);
|
);
|
||||||
let style = get_cell_style_attribute(*s);
|
let style = get_cell_style_attribute(*s);
|
||||||
row_data_str.push(format!(
|
row_data_str.push(format!(
|
||||||
"<c r=\"{cell_name}\" t=\"e\"{style}><f>{formula}</f><v>{ei}</v></c>"
|
"<c r=\"{cell_name}\" t=\"e\"{style}><f>{formula}</f><v>{ei}</v></c>"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
Cell::SpillStringCell { v, s, .. } => {
|
||||||
|
// inline string
|
||||||
|
// <c r="A1" t="str">
|
||||||
|
let style = get_cell_style_attribute(*s);
|
||||||
|
let escaped_v = escape_xml(v);
|
||||||
|
row_data_str.push(format!(
|
||||||
|
"<c r=\"{cell_name}\" t=\"str\"{style}><v>{escaped_v}</v></c>"
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormula { .. } => {
|
||||||
|
panic!("Model needs to be evaluated before saving!");
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaBoolean { f, v, s, r, a: _ } => {
|
||||||
|
// <c r="A1" s="3" cm="1">
|
||||||
|
// <f t="array" ref="A1:A10">A1:A10</f>
|
||||||
|
// <v>1</v>
|
||||||
|
// </c>
|
||||||
|
let style = get_cell_style_attribute(*s);
|
||||||
|
let range = format!(
|
||||||
|
"{}{}:{}{}",
|
||||||
|
column_name,
|
||||||
|
row_index,
|
||||||
|
number_to_column(r.0 + column_index).unwrap(),
|
||||||
|
r.1 + row_index
|
||||||
|
);
|
||||||
|
|
||||||
|
let formula = get_formula_attribute(
|
||||||
|
worksheet.get_name(),
|
||||||
|
*row_index,
|
||||||
|
*column_index,
|
||||||
|
&parsed_formulas[*f as usize].0,
|
||||||
|
);
|
||||||
|
|
||||||
|
let b = i32::from(*v);
|
||||||
|
row_data_str.push(format!(
|
||||||
|
r#"<c r="{cell_name}" t="b" s="{style}" cm="1"><f t="array" ref="{range}">{formula}</f><v>{b}</v></c>"#
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaNumber { f, v, s, r, a: _ } => {
|
||||||
|
// <c r="C4" s="3" cm="1">
|
||||||
|
// <f t="array" ref="C4:C10">C4:C10</f>
|
||||||
|
// <v>123</v>
|
||||||
|
// </c>
|
||||||
|
let style = get_cell_style_attribute(*s);
|
||||||
|
let range = format!(
|
||||||
|
"{}{}:{}{}",
|
||||||
|
column_name,
|
||||||
|
row_index,
|
||||||
|
number_to_column(r.0 + column_index).unwrap(),
|
||||||
|
r.1 + row_index
|
||||||
|
);
|
||||||
|
|
||||||
|
let formula = get_formula_attribute(
|
||||||
|
worksheet.get_name(),
|
||||||
|
*row_index,
|
||||||
|
*column_index,
|
||||||
|
&parsed_formulas[*f as usize].0,
|
||||||
|
);
|
||||||
|
|
||||||
|
row_data_str.push(format!(
|
||||||
|
r#"<c r="{cell_name}" s="{style}" cm="1"><f t="array" ref="{range}">{formula}</f><v>{v}</v></c>"#
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaString { f, v, s, r, a: _ } => {
|
||||||
|
// <c r="C6" t="str" s="5" cm="1">
|
||||||
|
// <f t="array" ref="C6:C10">C6:C10</f>
|
||||||
|
// <v>Hello world!</v>
|
||||||
|
// </c>
|
||||||
|
let style = get_cell_style_attribute(*s);
|
||||||
|
let range = format!(
|
||||||
|
"{}{}:{}{}",
|
||||||
|
column_name,
|
||||||
|
row_index,
|
||||||
|
number_to_column(r.0 + column_index).unwrap(),
|
||||||
|
r.1 + row_index
|
||||||
|
);
|
||||||
|
|
||||||
|
let formula = get_formula_attribute(
|
||||||
|
worksheet.get_name(),
|
||||||
|
*row_index,
|
||||||
|
*column_index,
|
||||||
|
&parsed_formulas[*f as usize].0,
|
||||||
|
);
|
||||||
|
let escaped_v = escape_xml(v);
|
||||||
|
|
||||||
|
row_data_str.push(format!(
|
||||||
|
r#"<c r="{cell_name}" t="str" s="{style}" cm="1"><f t="array" ref="{range}">{formula}</f><v>{escaped_v}</v></c>"#
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Cell::DynamicCellFormulaError {
|
||||||
|
f,
|
||||||
|
ei,
|
||||||
|
s,
|
||||||
|
o: _,
|
||||||
|
m: _,
|
||||||
|
r,
|
||||||
|
a: _,
|
||||||
|
} => {
|
||||||
|
// <c r="C6" t="e" s="4" cm="1">
|
||||||
|
// <f t="array" ref="C6:C10">C6:C10</f>
|
||||||
|
// <v>#DIV/0!</v>
|
||||||
|
// </c>
|
||||||
|
let style = get_cell_style_attribute(*s);
|
||||||
|
let range = format!(
|
||||||
|
"{}{}:{}{}",
|
||||||
|
column_name,
|
||||||
|
row_index,
|
||||||
|
number_to_column(r.0 + column_index).unwrap(),
|
||||||
|
r.1 + row_index
|
||||||
|
);
|
||||||
|
|
||||||
|
let formula = get_formula_attribute(
|
||||||
|
worksheet.get_name(),
|
||||||
|
*row_index,
|
||||||
|
*column_index,
|
||||||
|
&parsed_formulas[*f as usize].0,
|
||||||
|
);
|
||||||
|
|
||||||
|
row_data_str.push(format!(
|
||||||
|
r#"<c r="{cell_name}" t="e" s="{style}" cm="1"><f t="array" ref="{range}">{formula}</f><v>{ei}</v></c>"#
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let row_style_str = match row_style_dict.get(row_index) {
|
let row_style_str = match row_style_dict.get(row_index) {
|
||||||
|
|||||||
@@ -303,13 +303,15 @@ fn from_a1_to_rc(
|
|||||||
context: String,
|
context: String,
|
||||||
tables: HashMap<String, Table>,
|
tables: HashMap<String, Table>,
|
||||||
defined_names: Vec<DefinedNameS>,
|
defined_names: Vec<DefinedNameS>,
|
||||||
|
is_array: bool,
|
||||||
) -> Result<String, XlsxError> {
|
) -> Result<String, XlsxError> {
|
||||||
let mut parser = Parser::new(worksheets.to_owned(), defined_names, tables);
|
let mut parser = Parser::new(worksheets.to_owned(), defined_names, tables);
|
||||||
let cell_reference =
|
let cell_reference =
|
||||||
parse_reference(&context).map_err(|error| XlsxError::Xml(error.to_string()))?;
|
parse_reference(&context).map_err(|error| XlsxError::Xml(error.to_string()))?;
|
||||||
let mut t = parser.parse(&formula, &cell_reference);
|
let mut t = parser.parse(&formula, &cell_reference);
|
||||||
add_implicit_intersection(&mut t, true);
|
if !is_array {
|
||||||
|
add_implicit_intersection(&mut t, true);
|
||||||
|
}
|
||||||
Ok(to_rc_format(&t))
|
Ok(to_rc_format(&t))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -837,6 +839,7 @@ pub(super) fn load_sheet<R: Read + std::io::Seek>(
|
|||||||
};
|
};
|
||||||
|
|
||||||
let cell_metadata = cell.attribute("cm");
|
let cell_metadata = cell.attribute("cm");
|
||||||
|
let is_dynamic_array = cell_metadata == Some("1");
|
||||||
|
|
||||||
// type, the default type being "n" for number
|
// type, the default type being "n" for number
|
||||||
// If the cell does not have a value is an empty cell
|
// If the cell does not have a value is an empty cell
|
||||||
@@ -903,6 +906,7 @@ pub(super) fn load_sheet<R: Read + std::io::Seek>(
|
|||||||
context,
|
context,
|
||||||
tables.clone(),
|
tables.clone(),
|
||||||
defined_names.clone(),
|
defined_names.clone(),
|
||||||
|
is_dynamic_array,
|
||||||
)?;
|
)?;
|
||||||
match index_map.get(&si) {
|
match index_map.get(&si) {
|
||||||
Some(index) => {
|
Some(index) => {
|
||||||
@@ -951,7 +955,6 @@ pub(super) fn load_sheet<R: Read + std::io::Seek>(
|
|||||||
return Err(XlsxError::NotImplemented("data table formulas".to_string()));
|
return Err(XlsxError::NotImplemented("data table formulas".to_string()));
|
||||||
}
|
}
|
||||||
"array" | "normal" => {
|
"array" | "normal" => {
|
||||||
let is_dynamic_array = cell_metadata == Some("1");
|
|
||||||
if formula_type == "array" && !is_dynamic_array {
|
if formula_type == "array" && !is_dynamic_array {
|
||||||
// Dynamic formulas in Excel are formulas of type array with the cm=1, those we support.
|
// Dynamic formulas in Excel are formulas of type array with the cm=1, those we support.
|
||||||
// On the other hand the old CSE formulas or array formulas are not supported in IronCalc for the time being
|
// On the other hand the old CSE formulas or array formulas are not supported in IronCalc for the time being
|
||||||
@@ -966,6 +969,7 @@ pub(super) fn load_sheet<R: Read + std::io::Seek>(
|
|||||||
context,
|
context,
|
||||||
tables.clone(),
|
tables.clone(),
|
||||||
defined_names.clone(),
|
defined_names.clone(),
|
||||||
|
is_dynamic_array,
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
match get_formula_index(&formula, &shared_formulas) {
|
match get_formula_index(&formula, &shared_formulas) {
|
||||||
|
|||||||
BIN
xlsx/tests/calc_tests/simple_spill.xlsx
Normal file
BIN
xlsx/tests/calc_tests/simple_spill.xlsx
Normal file
Binary file not shown.
Reference in New Issue
Block a user