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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
|
/*
* Copyright 2000-2013 Vaadin Ltd.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package com.vaadin.ui.components.grid;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import com.vaadin.data.Container;
import com.vaadin.data.Container.PropertySetChangeEvent;
import com.vaadin.data.Container.PropertySetChangeListener;
import com.vaadin.data.Container.PropertySetChangeNotifier;
import com.vaadin.data.RpcDataProviderExtension;
import com.vaadin.server.KeyMapper;
import com.vaadin.shared.ui.grid.ColumnGroupRowState;
import com.vaadin.shared.ui.grid.GridColumnState;
import com.vaadin.shared.ui.grid.GridState;
import com.vaadin.ui.AbstractComponent;
/**
* Data grid component
*
* <h3>Lazy loading</h3> TODO To be revised when the data data source
* implementation has been don.
*
* <h3>Columns</h3> The grid columns are based on the property ids of the
* underlying data source. Each property id represents one column in the grid.
* To retrive a column in the grid you can use {@link Grid#getColumn(Object)}
* with the property id of the column. A grid column contains properties like
* the width, the footer and header captions of the column.
*
* <h3>Auxiliary headers and footers</h3> TODO To be revised when column
* grouping is implemented.
*
* @since 7.2
* @author Vaadin Ltd
*/
public class Grid extends AbstractComponent {
/**
* The data source attached to the grid
*/
private Container.Indexed datasource;
/**
* Property id to column instance mapping
*/
private final Map<Object, GridColumn> columns = new HashMap<Object, GridColumn>();
/**
* Key generator for column server-to-client communication
*/
private final KeyMapper<Object> columnKeys = new KeyMapper<Object>();
/**
* The column groups added to the grid
*/
private final List<ColumnGroupRow> columnGroupRows = new ArrayList<ColumnGroupRow>();
/**
* Property listener for listening to changes in data source properties.
*/
private final PropertySetChangeListener propertyListener = new PropertySetChangeListener() {
@Override
public void containerPropertySetChange(PropertySetChangeEvent event) {
Collection<?> properties = new HashSet<Object>(event.getContainer()
.getContainerPropertyIds());
// Cleanup columns that are no longer in grid
List<Object> removedColumns = new LinkedList<Object>();
for (Object columnId : columns.keySet()) {
if (!properties.contains(columnId)) {
removedColumns.add(columnId);
}
}
for (Object columnId : removedColumns) {
GridColumn column = columns.remove(columnId);
columnKeys.remove(columnId);
getState().columns.remove(column.getState());
}
// Add new columns
for (Object propertyId : properties) {
if (!columns.containsKey(propertyId)) {
appendColumn(propertyId);
}
}
}
};
private RpcDataProviderExtension datasourceExtension;
/**
* Creates a new Grid using the given datasource.
*
* @param datasource
* the data source for the grid
*/
public Grid(Container.Indexed datasource) {
setContainerDatasource(datasource);
}
/**
* Sets the grid data source.
*
* @param container
* The container data source. Cannot be null.
* @throws IllegalArgumentException
* if the data source is null
*/
public void setContainerDatasource(Container.Indexed container) {
if (container == null) {
throw new IllegalArgumentException(
"Cannot set the datasource to null");
}
if (datasource == container) {
return;
}
// Remove old listener
if (datasource instanceof PropertySetChangeNotifier) {
((PropertySetChangeNotifier) datasource)
.removePropertySetChangeListener(propertyListener);
}
if (datasourceExtension != null) {
removeExtension(datasourceExtension);
}
datasource = container;
datasourceExtension = new RpcDataProviderExtension(container);
datasourceExtension.extend(this);
// Listen to changes in properties and remove columns if needed
if (datasource instanceof PropertySetChangeNotifier) {
((PropertySetChangeNotifier) datasource)
.addPropertySetChangeListener(propertyListener);
}
getState().columns.clear();
// Add columns
for (Object propertyId : datasource.getContainerPropertyIds()) {
if (!columns.containsKey(propertyId)) {
GridColumn column = appendColumn(propertyId);
// Add by default property id as column header
column.setHeaderCaption(String.valueOf(propertyId));
}
}
}
/**
* Returns the grid data source.
*
* @return the container data source of the grid
*/
public Container.Indexed getContainerDatasource() {
return datasource;
}
/**
* Returns a column based on the property id
*
* @param propertyId
* the property id of the column
* @return the column or <code>null</code> if not found
*/
public GridColumn getColumn(Object propertyId) {
return columns.get(propertyId);
}
/**
* Sets the header rows visible.
*
* @param visible
* <code>true</code> if the header rows should be visible
*/
public void setColumnHeadersVisible(boolean visible) {
getState().columnHeadersVisible = visible;
}
/**
* Are the header rows visible?
*
* @return <code>true</code> if the headers of the columns are visible
*/
public boolean isColumnHeadersVisible() {
return getState(false).columnHeadersVisible;
}
/**
* Sets the footer rows visible.
*
* @param visible
* <code>true</code> if the footer rows should be visible
*/
public void setColumnFootersVisible(boolean visible) {
getState().columnFootersVisible = visible;
}
/**
* Are the footer rows visible.
*
* @return <code>true</code> if the footer rows should be visible
*/
public boolean isColumnFootersVisible() {
return getState(false).columnFootersVisible;
}
/**
* <p>
* Adds a new column group to the grid.
*
* <p>
* Column group rows are rendered in the header and footer of the grid.
* Column group rows are made up of column groups which groups together
* columns for adding a common auxiliary header or footer for the columns.
* </p>
* </p>
*
* <p>
* Example usage:
*
* <pre>
* // Add a new column group row to the grid
* ColumnGroupRow row = grid.addColumnGroupRow();
*
* // Group "Column1" and "Column2" together to form a header in the row
* ColumnGroup column12 = row.addGroup("Column1", "Column2");
*
* // Set a common header for "Column1" and "Column2"
* column12.setHeader("Column 1&2");
* </pre>
*
* </p>
*
* @return a column group instance you can use to add column groups
*/
public ColumnGroupRow addColumnGroupRow() {
ColumnGroupRowState state = new ColumnGroupRowState();
ColumnGroupRow row = new ColumnGroupRow(this, state, columnKeys);
columnGroupRows.add(row);
getState().columnGroupRows.add(state);
return row;
}
/**
* Adds a new column group to the grid at a specific index
*
* @param rowIndex
* the index of the row
* @return a column group instance you can use to add column groups
*/
public ColumnGroupRow addColumnGroupRow(int rowIndex) {
ColumnGroupRowState state = new ColumnGroupRowState();
ColumnGroupRow row = new ColumnGroupRow(this, state, columnKeys);
columnGroupRows.add(rowIndex, row);
getState().columnGroupRows.add(rowIndex, state);
return row;
}
/**
* Removes a column group.
*
* @param row
* the row to remove
*/
public void removeColumnGroupRow(ColumnGroupRow row) {
columnGroupRows.remove(row);
getState().columnGroupRows.remove(row.getState());
}
/**
* Gets the column group rows.
*
* @return an unmodifiable list of column group rows
*/
public List<ColumnGroupRow> getColumnGroupRows() {
return Collections.unmodifiableList(new ArrayList<ColumnGroupRow>(
columnGroupRows));
}
/**
* Used internally by the {@link Grid} to get a {@link GridColumn} by
* referencing its generated state id. Also used by {@link GridColumn} to
* verify if it has been detached from the {@link Grid}.
*
* @param columnId
* the client id generated for the column when the column is
* added to the grid
* @return the column with the id or <code>null</code> if not found
*/
GridColumn getColumnByColumnId(String columnId) {
Object propertyId = getPropertyIdByColumnId(columnId);
return getColumn(propertyId);
}
/**
* Used internally by the {@link Grid} to get a property id by referencing
* the columns generated state id.
*
* @param columnId
* The state id of the column
* @return The column instance or null if not found
*/
Object getPropertyIdByColumnId(String columnId) {
return columnKeys.get(columnId);
}
@Override
protected GridState getState() {
return (GridState) super.getState();
}
@Override
protected GridState getState(boolean markAsDirty) {
return (GridState) super.getState(markAsDirty);
}
/**
* Creates a new column based on a property id and appends it as the last
* column.
*
* @param datasourcePropertyId
* The property id of a property in the datasource
*/
private GridColumn appendColumn(Object datasourcePropertyId) {
if (datasourcePropertyId == null) {
throw new IllegalArgumentException("Property id cannot be null");
}
assert datasource.getContainerPropertyIds().contains(
datasourcePropertyId) : "Datasource should contain the property id";
GridColumnState columnState = new GridColumnState();
columnState.id = columnKeys.key(datasourcePropertyId);
getState().columns.add(columnState);
GridColumn column = new GridColumn(this, columnState);
columns.put(datasourcePropertyId, column);
return column;
}
}
|