1 module dlangui.widgets.spreadsheet;
2 
3 import dlangui.core.types;
4 import dlangui.widgets.styles;
5 import dlangui.widgets.widget;
6 import dlangui.widgets.layouts;
7 import dlangui.widgets.controls;
8 import dlangui.widgets.tabs;
9 import dlangui.widgets.editors;
10 import dlangui.widgets.grid;
11 import dlangui.widgets.scrollbar;
12 
13 import std.algorithm : min;
14 
15 /// standard style id for Tab with Up alignment
16 immutable string STYLE_TAB_SHEET_DOWN = "TAB_SHEET_DOWN";
17 /// standard style id for button of Tab with Up alignment
18 immutable string STYLE_TAB_SHEET_DOWN_BUTTON = "TAB_SHEET_DOWN_BUTTON";
19 /// standard style id for button of Tab with Up alignment
20 immutable string STYLE_TAB_SHEET_DOWN_BUTTON_TEXT = "TAB_SHEET_DOWN_BUTTON_TEXT";
21 
22 class SheetTabs : TabControl {
23     /// create with ID parameter
24     this(string ID = null) {
25         super(ID, Align.Bottom);
26         setStyles(STYLE_TAB_SHEET_DOWN, STYLE_TAB_SHEET_DOWN_BUTTON, STYLE_TAB_SHEET_DOWN_BUTTON_TEXT);
27         _moreButton.visibility = Visibility.Gone;
28     }
29 }
30 
31 class SheetEditControl : HorizontalLayout {
32     EditLine _edPosition;
33     EditLine _edText;
34     this(string ID = "sheetEdit") {
35         _edPosition = new EditLine("edPosition");
36         _edText = new EditLine("edText");
37         _edPosition.maxWidth = 100;
38         _edPosition.minWidth = 100;
39         _edText.layoutWidth = FILL_PARENT;
40         addChild(_edPosition);
41         addChild(_edText);
42     }
43 }
44 
45 class SpreadSheetView : StringGridWidget {
46     this(string ID = null) {
47         super(ID);
48         layoutWidth = FILL_PARENT;
49         layoutHeight = FILL_PARENT;
50         defRowHeight = 14;
51         defColumnWidth = 80;
52         styleId = null;
53         backgroundColor = 0xFFFFFF;
54         resize(50, 50);
55         _colWidths[0] = 50;
56         for (int i = 0; i < 26; i++) {
57             dchar[1] t;
58             t[0] = cast(dchar)('A' + i);
59             setColTitle(i, t.dup);
60         }
61         for (int i = 0; i < 50; i++) {
62             dstring label = to!dstring(i + 1);
63             setRowTitle(i, label);
64         }
65     }
66 }
67 
68 class SpreadSheetWidget : WidgetGroupDefaultDrawing, OnScrollHandler, CellSelectedHandler, CellActivatedHandler, ViewScrolledHandler {
69 
70     SheetEditControl _editControl;
71     SheetTabs _tabs;
72 
73     ScrollBar _hScroll1;
74     ScrollBar _hScroll2;
75     ScrollBar _vScroll1;
76     ScrollBar _vScroll2;
77 
78     SpreadSheetView _viewTopLeft;
79     SpreadSheetView _viewTopRight;
80     SpreadSheetView _viewBottomLeft;
81     SpreadSheetView _viewBottomRight;
82 
83     SpreadSheetView[4] _views;
84     ScrollBar[4] _scrollbars;
85 
86     this(string ID = "spreadsheet") {
87         _editControl = new SheetEditControl();
88         _editControl.layoutWidth = FILL_PARENT;
89         _tabs = new SheetTabs();
90         _tabs.layoutWidth = FILL_PARENT;
91         _tabs.addTab("Sheet1", "Sheet1"d);
92         _tabs.addTab("Sheet2", "Sheet2"d);
93         _tabs.addTab("Sheet3", "Sheet3"d);
94         layoutWidth = FILL_PARENT;
95         layoutHeight = FILL_PARENT;
96         backgroundColor = 0xdce2e8;
97         minHeight = 100;
98 
99         _hScroll1 = new ScrollBar("hscroll1", Orientation.Horizontal);
100         _hScroll2 = new ScrollBar("hscroll2", Orientation.Horizontal);
101         _vScroll1 = new ScrollBar("vscroll1", Orientation.Vertical);
102         _vScroll2 = new ScrollBar("vscroll2", Orientation.Vertical);
103 
104         _scrollbars[0] = _hScroll1;
105         _scrollbars[1] = _vScroll1;
106         _scrollbars[2] = _hScroll2;
107         _scrollbars[3] = _vScroll2;
108 
109         _viewTopLeft = new SpreadSheetView("sheetViewTopLeft");
110         _viewTopRight = new SpreadSheetView("sheetViewTopRight");
111         _viewBottomLeft = new SpreadSheetView("sheetViewBottomLeft");
112         _viewBottomRight = new SpreadSheetView("sheetViewBottomRight");
113 
114         _viewTopRight.setColWidth(0, 0);
115         _viewBottomLeft.setRowHeight(0, 0);
116         _viewBottomRight.setRowHeight(0, 0);
117         _viewBottomRight.setColWidth(0, 0);
118 
119         _views[0] = _viewTopLeft;
120         _views[1] = _viewTopRight;
121         _views[2] = _viewBottomLeft;
122         _views[3] = _viewBottomRight;
123 
124         _viewTopLeft.hscrollbar = _hScroll1;
125         _viewTopLeft.vscrollbar = _vScroll1;
126         _viewTopRight.hscrollbar = _hScroll2;
127         _viewTopRight.vscrollbar = _vScroll1;
128         _viewBottomLeft.hscrollbar = _hScroll1;
129         _viewBottomLeft.vscrollbar = _vScroll2;
130         _viewBottomRight.hscrollbar = _hScroll2;
131         _viewBottomRight.vscrollbar = _vScroll2;
132 
133         addChildren([_hScroll1, _vScroll1, _hScroll2, _vScroll2,
134             _viewTopLeft, _viewTopRight, _viewBottomLeft, _viewBottomRight,
135             _editControl, _tabs
136         ]);
137 
138         foreach(sb; _scrollbars)
139             sb.scrollEvent = this;
140         foreach(view; _views) {
141             view.cellSelected = this;
142             view.cellActivated = this;
143             view.viewScrolled = this;
144         }
145     }
146 
147     /// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
148     override void measure(int parentWidth, int parentHeight) { 
149         if (visibility == Visibility.Gone) {
150             return;
151         }
152         _measuredWidth = parentWidth;
153         _measuredHeight = parentHeight;
154         foreach(view; _views)
155             view.measure(parentWidth, parentHeight);
156         foreach(sb; _scrollbars)
157             sb.measure(parentWidth, parentHeight);
158         _editControl.measure(parentWidth, parentHeight);
159         _tabs.measure(parentWidth, parentHeight);
160     }
161     /// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
162     override void layout(Rect rc) {
163         if (visibility == Visibility.Gone) {
164             return;
165         }
166         _pos = rc;
167         _needLayout = false;
168         applyMargins(rc);
169         applyPadding(rc);
170         int editHeight = _editControl.measuredHeight;
171         _editControl.layout(Rect(rc.left, rc.top, rc.right, rc.top + editHeight));
172         rc.top += editHeight;
173         int splitWidth = 4;
174         int splitHeight = 4;
175         int hscrollHeight = _hScroll1.measuredHeight;
176         int vscrollWidth = _vScroll1.measuredWidth;
177         int tabsHeight = _tabs.measuredHeight;
178         int bottomSize = min(hscrollHeight, tabsHeight);
179         int splitx = (rc.width - vscrollWidth - splitWidth) / 2;
180         int splity = (rc.height - hscrollHeight - splitHeight) / 2;
181         _viewTopLeft.layout(Rect(rc.left, rc.top, rc.left + splitx, rc.top + splity));
182         _viewTopRight.layout(Rect(rc.left + splitx + splitWidth, rc.top, rc.right - vscrollWidth, rc.top + splity));
183         _viewBottomLeft.layout(Rect(rc.left, rc.top + splity + splitHeight, rc.left + splitx, rc.bottom - bottomSize));
184         _viewBottomRight.layout(Rect(rc.left + splitx + splitWidth, rc.top + splity + splitHeight, rc.right - vscrollWidth, rc.bottom - bottomSize));
185         int tabsWidth = splitx / 2;
186         _tabs.layout(Rect(rc.left, rc.bottom - bottomSize, rc.left + tabsWidth, rc.bottom - bottomSize + tabsHeight));
187 
188         _hScroll1.layout(Rect(rc.left + tabsWidth + splitWidth, rc.bottom - hscrollHeight, rc.left + splitx, rc.bottom));
189         _hScroll2.layout(Rect(rc.left + splitx + splitWidth, rc.bottom - hscrollHeight, rc.right - vscrollWidth, rc.bottom));
190         _vScroll1.layout(Rect(rc.right - vscrollWidth, rc.top, rc.right, rc.top + splity));
191         _vScroll2.layout(Rect(rc.right - vscrollWidth, rc.top + splity + splitHeight, rc.right, rc.bottom - bottomSize));
192     }
193 
194     /// handle scroll event
195     override bool onScrollEvent(AbstractSlider source, ScrollEvent event) {
196         if (source == _hScroll1) {
197             _viewBottomLeft.onHScroll(event);
198             return _viewTopLeft.onHScroll(event);
199         } else if (source == _hScroll2) {
200             _viewBottomRight.onHScroll(event);
201             return _viewTopRight.onHScroll(event);
202         } else if (source == _vScroll1) {
203             _viewTopRight.onVScroll(event);
204             return _viewTopLeft.onVScroll(event);
205         } else if (source == _vScroll2) {
206             _viewBottomRight.onVScroll(event);
207             return _viewBottomLeft.onVScroll(event);
208         }
209         return true;
210     }
211 
212     /// Callback for handling of cell selection
213     void onCellSelected(GridWidgetBase source, int col, int row) {
214         foreach(view; _views) {
215             if (source != view)
216                 view.selectCell(col + view.headerCols, row + view.headerRows, false, source, false);
217         }
218     }
219 
220     /// Callback for handling of cell double click or Enter key press
221     void onCellActivated(GridWidgetBase source, int col, int row) {
222     }
223 
224     /// Callback for handling of view scroll (top left visible cell change)
225     void onViewScrolled(GridWidgetBase source, int col, int row) {
226         if (source == _viewTopLeft) {
227             _viewTopRight.scrollTo(-1, row, source, false);
228             _viewBottomLeft.scrollTo(col, -1, source, false);
229         } else if (source == _viewTopRight) {
230             _viewTopLeft.scrollTo(-1, row, source, false);
231             _viewBottomRight.scrollTo(col, -1, source, false);
232         } else if (source == _viewBottomLeft) {
233             _viewTopLeft.scrollTo(col, -1, source, false);
234             _viewBottomRight.scrollTo(-1, row, source, false);
235         } else if (source == _viewBottomRight) {
236             _viewTopRight.scrollTo(col, -1, source, false);
237             _viewBottomLeft.scrollTo(-1, row, source, false);
238         }
239     }
240 
241 }