OLD | NEW |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 /** | 5 /** |
6 * @fileoverview This implements a table control. | 6 * @fileoverview This implements a table control. |
7 */ | 7 */ |
8 | 8 |
9 cr.define('cr.ui', function() { | 9 cr.define('cr.ui', function() { |
10 const TableSelectionModel = cr.ui.table.TableSelectionModel; | 10 const ListSelectionModel = cr.ui.ListSelectionModel; |
11 const ListSelectionController = cr.ui.ListSelectionController; | 11 const ListSelectionController = cr.ui.ListSelectionController; |
12 const ArrayDataModel = cr.ui.ArrayDataModel; | 12 const ArrayDataModel = cr.ui.ArrayDataModel; |
13 const TableColumnModel = cr.ui.table.TableColumnModel; | 13 const TableColumnModel = cr.ui.table.TableColumnModel; |
14 const TableList = cr.ui.table.TableList; | 14 const TableList = cr.ui.table.TableList; |
15 const TableHeader = cr.ui.table.TableHeader; | 15 const TableHeader = cr.ui.table.TableHeader; |
16 | 16 |
17 /** | 17 /** |
18 * Creates a new table element. | 18 * Creates a new table element. |
19 * @param {Object=} opt_propertyBag Optional properties. | 19 * @param {Object=} opt_propertyBag Optional properties. |
20 * @constructor | 20 * @constructor |
21 * @extends {HTMLDivElement} | 21 * @extends {HTMLDivElement} |
22 */ | 22 */ |
23 var Table = cr.ui.define('div'); | 23 var Table = cr.ui.define('div'); |
24 | 24 |
25 Table.prototype = { | 25 Table.prototype = { |
26 __proto__: HTMLDivElement.prototype, | 26 __proto__: HTMLDivElement.prototype, |
27 | 27 |
28 columnModel_: new TableColumnModel([]), | 28 columnModel_: new TableColumnModel([]), |
29 | 29 |
30 /** | 30 /** |
31 * The table data model. | 31 * The table data model. |
32 * | 32 * |
33 * @type {cr.ui.table.TableDataModel} | 33 * @type {cr.ui.ArrayDataModel} |
34 */ | 34 */ |
35 get dataModel() { | 35 get dataModel() { |
36 return this.list_.dataModel; | 36 return this.list_.dataModel; |
37 }, | 37 }, |
38 set dataModel(dataModel) { | 38 set dataModel(dataModel) { |
39 if (this.list_.dataModel != dataModel) { | 39 if (this.list_.dataModel != dataModel) { |
40 this.list_.dataModel = dataModel; | |
41 if (this.list_.dataModel) { | 40 if (this.list_.dataModel) { |
42 this.list_.dataModel.removeEventListener('splice', this.boundRedraw_); | |
43 this.list_.dataModel.removeEventListener('sorted', | 41 this.list_.dataModel.removeEventListener('sorted', |
44 this.boundHandleSorted_); | 42 this.boundHandleSorted_); |
45 } | 43 } |
46 this.list_.dataModel = dataModel; | 44 this.list_.dataModel = dataModel; |
47 this.list_.dataModel.table = this; | |
48 | |
49 | |
50 if (this.list_.dataModel) { | 45 if (this.list_.dataModel) { |
51 this.list_.dataModel.addEventListener('splice', this.boundRedraw_); | |
52 this.list_.dataModel.addEventListener('sorted', | 46 this.list_.dataModel.addEventListener('sorted', |
53 this.boundHandleSorted_); | 47 this.boundHandleSorted_); |
54 } | 48 } |
55 this.header_.redraw(); | 49 this.header_.redraw(); |
56 } | 50 } |
57 }, | 51 }, |
58 | 52 |
59 /** | 53 /** |
60 * The table column model. | 54 * The table column model. |
61 * | 55 * |
62 * @type {cr.ui.table.TableColumnModel} | 56 * @type {cr.ui.table.TableColumnModel} |
63 */ | 57 */ |
64 get columnModel() { | 58 get columnModel() { |
65 return this.columnModel_; | 59 return this.columnModel_; |
66 }, | 60 }, |
67 set columnModel(columnModel) { | 61 set columnModel(columnModel) { |
68 if (this.columnModel_ != columnModel) { | 62 if (this.columnModel_ != columnModel) { |
69 if (this.columnModel_) { | 63 if (this.columnModel_) |
70 this.columnModel_.removeEventListener('change', this.boundRedraw_); | |
71 this.columnModel_.removeEventListener('resize', this.boundResize_); | 64 this.columnModel_.removeEventListener('resize', this.boundResize_); |
72 } | |
73 this.columnModel_ = columnModel; | 65 this.columnModel_ = columnModel; |
74 | 66 |
75 if (this.columnModel_) { | 67 if (this.columnModel_) |
76 this.columnModel_.addEventListener('change', this.boundRedraw_); | |
77 this.columnModel_.addEventListener('resize', this.boundResize_); | 68 this.columnModel_.addEventListener('resize', this.boundResize_); |
78 } | 69 this.list_.invalidate(); |
79 this.redraw(); | 70 this.redraw(); |
80 } | 71 } |
81 }, | 72 }, |
82 | 73 |
83 /** | 74 /** |
84 * The table selection model. | 75 * The table selection model. |
85 * | 76 * |
86 * @type | 77 * @type |
87 * {cr.ui.table.TableSelectionModel|cr.ui.table.TableSingleSelectionModel} | 78 * {cr.ui.ListSelectionModel|cr.ui.table.ListSingleSelectionModel} |
88 */ | 79 */ |
89 get selectionModel() { | 80 get selectionModel() { |
90 return this.list_.selectionModel; | 81 return this.list_.selectionModel; |
91 }, | 82 }, |
92 set selectionModel(selectionModel) { | 83 set selectionModel(selectionModel) { |
93 if (this.list_.selectionModel != selectionModel) { | 84 if (this.list_.selectionModel != selectionModel) { |
94 if (this.dataModel) | 85 if (this.dataModel) |
95 selectionModel.adjust(0, 0, this.dataModel.length); | 86 selectionModel.adjustLength(this.dataModel.length); |
96 this.list_.selectionModel = selectionModel; | 87 this.list_.selectionModel = selectionModel; |
97 this.redraw(); | |
98 } | 88 } |
99 }, | 89 }, |
100 | 90 |
101 /** | 91 /** |
102 * Sets width of the column at the given index. | 92 * Sets width of the column at the given index. |
103 * | 93 * |
104 * @param {number} index The index of the column. | 94 * @param {number} index The index of the column. |
105 * @param {number} Column width. | 95 * @param {number} Column width. |
106 */ | 96 */ |
107 setColumnWidth: function(index, width) { | 97 setColumnWidth: function(index, width) { |
108 this.columnWidths_[index] = width; | 98 this.columnWidths_[index] = width; |
109 }, | 99 }, |
110 | 100 |
111 /** | 101 /** |
112 * Initializes the element. | 102 * Initializes the element. |
113 */ | 103 */ |
114 decorate: function() { | 104 decorate: function() { |
115 this.list_ = this.ownerDocument.createElement('list'); | 105 this.list_ = this.ownerDocument.createElement('list'); |
116 TableList.decorate(this.list_); | 106 TableList.decorate(this.list_); |
117 this.list_.selectionModel = new TableSelectionModel(this); | 107 this.list_.selectionModel = new ListSelectionModel(this); |
118 this.list_.table = this; | 108 this.list_.table = this; |
119 | 109 |
120 this.header_ = this.ownerDocument.createElement('div'); | 110 this.header_ = this.ownerDocument.createElement('div'); |
121 TableHeader.decorate(this.header_); | 111 TableHeader.decorate(this.header_); |
122 this.header_.table = this; | 112 this.header_.table = this; |
123 | 113 |
124 this.classList.add('table'); | 114 this.classList.add('table'); |
125 this.appendChild(this.header_); | 115 this.appendChild(this.header_); |
126 this.appendChild(this.list_); | 116 this.appendChild(this.list_); |
127 this.ownerDocument.defaultView.addEventListener( | 117 this.ownerDocument.defaultView.addEventListener( |
128 'resize', this.header_.updateWidth.bind(this.header_)); | 118 'resize', this.header_.updateWidth.bind(this.header_)); |
129 | 119 |
130 this.boundRedraw_ = this.redraw.bind(this); | |
131 this.boundResize_ = this.resize.bind(this); | 120 this.boundResize_ = this.resize.bind(this); |
132 this.boundHandleSorted_ = this.handleSorted_.bind(this); | 121 this.boundHandleSorted_ = this.handleSorted_.bind(this); |
133 | 122 |
134 // Make table focusable | 123 // Make table focusable |
135 if (!this.hasAttribute('tabindex')) | 124 if (!this.hasAttribute('tabindex')) |
136 this.tabIndex = 0; | 125 this.tabIndex = 0; |
137 this.addEventListener('focus', this.handleElementFocus_, true); | 126 this.addEventListener('focus', this.handleElementFocus_, true); |
138 this.addEventListener('blur', this.handleElementBlur_, true); | 127 this.addEventListener('blur', this.handleElementBlur_, true); |
139 }, | 128 }, |
140 | 129 |
141 /** | 130 /** |
| 131 * Redraws the table. |
| 132 */ |
| 133 redraw: function(index) { |
| 134 this.list_.redraw(); |
| 135 this.header_.redraw(); |
| 136 }, |
| 137 |
| 138 /** |
142 * Resize the table columns. | 139 * Resize the table columns. |
143 */ | 140 */ |
144 resize: function() { | 141 resize: function() { |
145 // We resize columns only instead of full redraw. | 142 // We resize columns only instead of full redraw. |
146 this.list_.resize(); | 143 this.list_.resize(); |
147 this.header_.resize(); | 144 this.header_.resize(); |
148 }, | 145 }, |
149 | 146 |
150 /** | 147 /** |
151 * Ensures that a given index is inside the viewport. | 148 * Ensures that a given index is inside the viewport. |
152 * @param {number} index The index of the item to scroll into view. | 149 * @param {number} index The index of the item to scroll into view. |
153 * @return {boolean} Whether any scrolling was needed. | 150 * @return {boolean} Whether any scrolling was needed. |
154 */ | 151 */ |
155 scrollIndexIntoView: function(i) { | 152 scrollIndexIntoView: function(i) { |
156 this.list_.scrollIndexIntoView(i); | 153 this.list_.scrollIndexIntoView(i); |
157 }, | 154 }, |
158 | 155 |
159 /** | 156 /** |
160 * Find the list item element at the given index. | 157 * Find the list item element at the given index. |
161 * @param {number} index The index of the list item to get. | 158 * @param {number} index The index of the list item to get. |
162 * @return {ListItem} The found list item or null if not found. | 159 * @return {ListItem} The found list item or null if not found. |
163 */ | 160 */ |
164 getListItemByIndex: function(index) { | 161 getListItemByIndex: function(index) { |
165 return this.list_.getListItemByIndex(index); | 162 return this.list_.getListItemByIndex(index); |
166 }, | 163 }, |
167 | 164 |
168 /** | 165 /** |
169 * Redraws the table. | 166 * This handles data model 'sorted' event. |
170 * This forces the list to remove all cached items. | 167 * After sorting we need to redraw header |
| 168 * @param {Event} e The 'sorted' event. |
171 */ | 169 */ |
172 redraw: function() { | 170 handleSorted_: function(e) { |
173 this.list_.startBatchUpdates(); | |
174 if (this.list_.dataModel) { | |
175 for (var i = 0; i < this.list_.dataModel.length; i++) { | |
176 this.list_.redrawItem(i); | |
177 } | |
178 } | |
179 this.list_.endBatchUpdates(); | |
180 this.list_.redraw(); | |
181 this.header_.redraw(); | 171 this.header_.redraw(); |
182 }, | 172 }, |
183 | 173 |
184 /** | 174 /** |
185 * This handles data model 'sorted' event. | |
186 * After sorting we need to | |
187 * - adjust selection | |
188 * - redraw all the items | |
189 * - scroll the list to show selection. | |
190 * @param {Event} e The 'sorted' event. | |
191 */ | |
192 handleSorted_: function(e) { | |
193 var sm = this.list_.selectionModel; | |
194 sm.adjustToReordering(e.sortPermutation); | |
195 | |
196 this.redraw(); | |
197 if (sm.leadIndex != -1) | |
198 this.list_.scrollIndexIntoView(sm.leadIndex) | |
199 }, | |
200 | |
201 /** | |
202 * Sort data by the given column. | 175 * Sort data by the given column. |
203 * @param {number} index The index of the column to sort by. | 176 * @param {number} index The index of the column to sort by. |
204 */ | 177 */ |
205 sort: function(i) { | 178 sort: function(i) { |
206 var cm = this.columnModel_; | 179 var cm = this.columnModel_; |
207 var sortStatus = this.list_.dataModel.sortStatus; | 180 var sortStatus = this.list_.dataModel.sortStatus; |
208 if (sortStatus.field == cm.getId(i)) { | 181 if (sortStatus.field == cm.getId(i)) { |
209 var sortDirection = sortStatus.direction == 'desc' ? 'asc' : 'desc'; | 182 var sortDirection = sortStatus.direction == 'desc' ? 'asc' : 'desc'; |
210 this.list_.dataModel.sort(sortStatus.field, sortDirection); | 183 this.list_.dataModel.sort(sortStatus.field, sortDirection); |
211 } else { | 184 } else { |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
257 * because table contents can contain controls that can be focused, and for | 230 * because table contents can contain controls that can be focused, and for |
258 * some purposes (e.g., styling), the table can still be conceptually focused | 231 * some purposes (e.g., styling), the table can still be conceptually focused |
259 * at that point even though it doesn't actually have the page focus. | 232 * at that point even though it doesn't actually have the page focus. |
260 */ | 233 */ |
261 cr.defineProperty(Table, 'hasElementFocus', cr.PropertyKind.BOOL_ATTR); | 234 cr.defineProperty(Table, 'hasElementFocus', cr.PropertyKind.BOOL_ATTR); |
262 | 235 |
263 return { | 236 return { |
264 Table: Table | 237 Table: Table |
265 }; | 238 }; |
266 }); | 239 }); |
OLD | NEW |