summaryrefslogtreecommitdiff
path: root/src/sheet/mod.rs
blob: 11f95c9826132b7d18acceb61d2f4688a0ad329b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
use std::sync::Mutex;

use cell::{Cell, CellRef};
use mlua::prelude::*;
use register::{Register, SheetId};

pub mod cell;
pub mod eval;
pub mod register;

#[derive(Debug, Default)]
pub struct Sheet {
    id: register::SheetId,
    rows: Vec<Vec<Cell>>,
    progress: Mutex<u8>,
}

impl Sheet {
    pub(self) fn new(width: usize, height: usize, id: register::SheetId) -> Self {
        Self {
            id,
            rows: vec![vec![Cell::new_empty(); width]; height],
            progress: Mutex::new(0),
        }
    }

    pub fn set_cell(&mut self, row: usize, column: usize, cell: Cell) {
        if row < self.height() && column < self.width() {
            self.rows[row][column] = cell
        }
    }

    pub fn get_cell<'a>(&'a self, row: usize, column: usize) -> Option<&'a Cell> {
        if let Some(r) = self.rows.get(row) {
            if let Some(_) = r.get(column) {
                return Some(&self.rows[row][column]);
            }
        }

        None
    }

    pub fn get_ref(&self, row: usize, column: usize) -> Option<CellRef> {
        self.get_cell(row, column).map(|cell| {
            unsafe { CellRef::new(self.id, row, column, cell.clone()) }
        })
    }

    pub fn height(&self) -> usize {
        self.rows.len()
    }

    pub fn set_height(&mut self, mut height: usize) {
        height = height.max(1);
        self.rows
            .resize(height, vec![Cell::new_empty(); self.width()])
    }

    pub fn width(&self) -> usize {
        self.rows.get(0).map(|r| r.len()).unwrap_or(1)
    }

    pub fn set_width(&mut self, mut width: usize) {
        width = width.max(1);
        for row in self.rows.iter_mut() {
            row.resize(width, Cell::new_empty());
        }
    }

    pub fn id(&self) -> register::SheetId {
        self.id
    }

    pub fn apply(&mut self, other: Sheet) {
        self.rows = other.rows;
    }

    pub fn progress(&self) -> u8 {
        *self.progress.lock().unwrap()
    }
}

impl Clone for Sheet {
    fn clone(&self) -> Self {
        Sheet {
            id: self.id,
            rows: self.rows.clone(),
            progress: Mutex::new(0),
        }
    }
}

struct SheetLuaRef {
    id: SheetId,
}

impl SheetLuaRef {
    pub fn new(id: SheetId) -> Self {
        Self { id }
    }
}

impl LuaUserData for SheetLuaRef {
    fn add_fields<'lua, F: mlua::UserDataFields<'lua, Self>>(fields: &mut F) {
        fields.add_field_method_get("width", |_, luaref| {
            Ok(Register::get(luaref.id).unwrap().read().unwrap().width())
        });

        fields.add_field_method_get("height", |_, luaref| {
            Ok(Register::get(luaref.id).unwrap().read().unwrap().height())
        });
    }

    fn add_methods<'lua, M: mlua::UserDataMethods<'lua, Self>>(methods: &mut M) {
        methods.add_method_mut("cell", |lua, luaref, (row, column): (usize, usize)| {
            if let Some(rw) = Register::get(luaref.id) {
                let sheet = rw.read().unwrap();
                Ok(sheet.get_ref(row, column).into_lua(lua).unwrap())
            } else {
                Ok(LuaValue::Nil)
            }
        })
    }
}