• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef UI_VIEWS_CONTROLS_TABLE_TABLE_VIEW_VIEWS_H_
6 #define UI_VIEWS_CONTROLS_TABLE_TABLE_VIEW_VIEWS_H_
7 
8 #include <vector>
9 
10 #include "base/memory/scoped_ptr.h"
11 #include "ui/base/models/list_selection_model.h"
12 #include "ui/base/models/table_model.h"
13 #include "ui/base/models/table_model_observer.h"
14 #include "ui/gfx/font.h"
15 #include "ui/views/view.h"
16 #include "ui/views/views_export.h"
17 
18 // A TableView is a view that displays multiple rows with any number of columns.
19 // TableView is driven by a TableModel. The model returns the contents
20 // to display. TableModel also has an Observer which is used to notify
21 // TableView of changes to the model so that the display may be updated
22 // appropriately.
23 //
24 // TableView itself has an observer that is notified when the selection
25 // changes.
26 //
27 // When a table is sorted the model coordinates do not necessarily match the
28 // view coordinates. All table methods are in terms of the model. If you need to
29 // convert to view coordinates use ModelToView().
30 //
31 // Sorting is done by a locale sensitive string sort. You can customize the
32 // sort by way of overriding TableModel::CompareValues().
33 namespace views {
34 
35 struct GroupRange;
36 class TableGrouper;
37 class TableHeader;
38 class TableViewObserver;
39 class TableViewRowBackgroundPainter;
40 class TableViewTestHelper;
41 
42 // The cells in the first column of a table can contain:
43 // - only text
44 // - a small icon (16x16) and some text
45 // - a check box and some text
46 enum TableTypes {
47   TEXT_ONLY = 0,
48   ICON_AND_TEXT,
49 };
50 
51 class VIEWS_EXPORT TableView
52     : public views::View,
53       public ui::TableModelObserver {
54  public:
55   // Used to track a visible column. Useful only for the header.
56   struct VIEWS_EXPORT VisibleColumn {
57     VisibleColumn();
58     ~VisibleColumn();
59 
60     // The column.
61     ui::TableColumn column;
62 
63     // Starting x-coordinate of the column.
64     int x;
65 
66     // Width of the column.
67     int width;
68   };
69 
70   // Describes a sorted column.
71   struct VIEWS_EXPORT SortDescriptor {
SortDescriptorSortDescriptor72     SortDescriptor() : column_id(-1), ascending(true) {}
SortDescriptorSortDescriptor73     SortDescriptor(int column_id, bool ascending)
74         : column_id(column_id),
75           ascending(ascending) {}
76 
77     // ID of the sorted column.
78     int column_id;
79 
80     // Is the sort ascending?
81     bool ascending;
82   };
83 
84   typedef std::vector<SortDescriptor> SortDescriptors;
85 
86   // Creates a new table using the model and columns specified.
87   // The table type applies to the content of the first column (text, icon and
88   // text, checkbox and text).
89   TableView(ui::TableModel* model,
90             const std::vector<ui::TableColumn>& columns,
91             TableTypes table_type,
92             bool single_selection);
93   virtual ~TableView();
94 
95   // Assigns a new model to the table view, detaching the old one if present.
96   // If |model| is NULL, the table view cannot be used after this call. This
97   // should be called in the containing view's destructor to avoid destruction
98   // issues when the model needs to be deleted before the table.
99   void SetModel(ui::TableModel* model);
model()100   ui::TableModel* model() const { return model_; }
101 
102   // Returns a new ScrollView that contains the receiver.
103   View* CreateParentIfNecessary();
104 
105   void SetRowBackgroundPainter(
106       scoped_ptr<TableViewRowBackgroundPainter> painter);
107 
108   // Sets the TableGrouper. TableView does not own |grouper| (common use case is
109   // to have TableModel implement TableGrouper).
110   void SetGrouper(TableGrouper* grouper);
111 
112   // Returns the number of rows in the TableView.
113   int RowCount() const;
114 
115   // Returns the number of selected rows.
116   // TODO(sky): remove this and force callers to use selection_model().
117   int SelectedRowCount();
118 
119   // Selects the specified item, making sure it's visible.
120   void Select(int model_row);
121 
122   // Returns the first selected row in terms of the model.
123   int FirstSelectedRow();
124 
selection_model()125   const ui::ListSelectionModel& selection_model() const {
126     return selection_model_;
127   }
128 
129   // Changes the visibility of the specified column (by id).
130   void SetColumnVisibility(int id, bool is_visible);
131   bool IsColumnVisible(int id) const;
132 
133   // Adds the specified column. |col| is not made visible.
134   void AddColumn(const ui::TableColumn& col);
135 
136   // Returns true if the column with the specified id is known (either visible
137   // or not).
138   bool HasColumn(int id) const;
139 
140   // TODO(sky): rename to set_observer().
SetObserver(TableViewObserver * observer)141   void SetObserver(TableViewObserver* observer) {
142     table_view_observer_ = observer;
143   }
observer()144   TableViewObserver* observer() const { return table_view_observer_; }
145 
visible_columns()146   const std::vector<VisibleColumn>& visible_columns() const {
147     return visible_columns_;
148   }
149 
150   // Sets the width of the column. |index| is in terms of |visible_columns_|.
151   void SetVisibleColumnWidth(int index, int width);
152 
153   // Toggles the sort order of the specified visible column index.
154   void ToggleSortOrder(int visible_column_index);
155 
sort_descriptors()156   const SortDescriptors& sort_descriptors() const { return sort_descriptors_; }
is_sorted()157   bool is_sorted() const { return !sort_descriptors_.empty(); }
158 
159   // Maps from the index in terms of the model to that of the view.
160   int ModelToView(int model_index) const;
161 
162   // Maps from the index in terms of the view to that of the model.
163   int ViewToModel(int view_index) const;
164 
row_height()165   int row_height() const { return row_height_; }
166 
167   // View overrides:
168   virtual void Layout() OVERRIDE;
169   virtual gfx::Size GetPreferredSize() OVERRIDE;
170   virtual bool OnKeyPressed(const ui::KeyEvent& event) OVERRIDE;
171   virtual bool OnMousePressed(const ui::MouseEvent& event) OVERRIDE;
172   virtual void OnGestureEvent(ui::GestureEvent* event) OVERRIDE;
173   virtual bool GetTooltipText(const gfx::Point& p,
174                               string16* tooltip) const OVERRIDE;
175   virtual bool GetTooltipTextOrigin(const gfx::Point& p,
176                                     gfx::Point* loc) const OVERRIDE;
177 
178   // ui::TableModelObserver overrides:
179   virtual void OnModelChanged() OVERRIDE;
180   virtual void OnItemsChanged(int start, int length) OVERRIDE;
181   virtual void OnItemsAdded(int start, int length) OVERRIDE;
182   virtual void OnItemsRemoved(int start, int length) OVERRIDE;
183 
184  protected:
185   // View overrides:
186   virtual gfx::Point GetKeyboardContextMenuLocation() OVERRIDE;
187   virtual void OnPaint(gfx::Canvas* canvas) OVERRIDE;
188   virtual void OnFocus() OVERRIDE;
189   virtual void OnBlur() OVERRIDE;
190 
191  private:
192   friend class TableViewTestHelper;
193   struct GroupSortHelper;
194   struct SortHelper;
195 
196   // Used during painting to determine the range of cells that need to be
197   // painted.
198   // NOTE: the row indices returned by this are in terms of the view and column
199   // indices in terms of |visible_columns_|.
200   struct VIEWS_EXPORT PaintRegion {
201     PaintRegion();
202     ~PaintRegion();
203 
204     int min_row;
205     int max_row;
206     int min_column;
207     int max_column;
208   };
209 
210   // Used by AdvanceSelection() to determine the direction to change the
211   // selection.
212   enum AdvanceDirection {
213     ADVANCE_DECREMENT,
214     ADVANCE_INCREMENT,
215   };
216 
217   // Invoked when the number of rows changes in some way.
218   void NumRowsChanged();
219 
220   // Resets the sort descriptions.
221   void SetSortDescriptors(const SortDescriptors& sort_descriptors);
222 
223   // Does the actual sort and updates the mappings (|view_to_model_| and
224   // |model_to_view_|) appropriately.
225   void SortItemsAndUpdateMapping();
226 
227   // Used to sort the two rows. Returns a value < 0, == 0 or > 0 indicating
228   // whether the row2 comes before row1, row2 is the same as row1 or row1 comes
229   // after row2. This invokes CompareValues on the model with the sorted column.
230   int CompareRows(int model_row1, int model_row2);
231 
232   // Returns the bounds of the specified row.
233   gfx::Rect GetRowBounds(int row) const;
234 
235   // Returns the bounds of the specified cell. |visible_column_index| indexes
236   // into |visible_columns_|.
237   gfx::Rect GetCellBounds(int row, int visible_column_index) const;
238 
239   // Adjusts |bounds| based on where the text should be painted. |bounds| comes
240   // from GetCellBounds() and |visible_column_index| is the corresponding column
241   // (in terms of |visible_columns_|).
242   void AdjustCellBoundsForText(int visible_column_index,
243                                gfx::Rect* bounds) const;
244 
245   // Creates |header_| if necessary.
246   void CreateHeaderIfNecessary();
247 
248   // Updates the |x| and |width| of each of the columns in |visible_columns_|.
249   void UpdateVisibleColumnSizes();
250 
251   // Returns the cells that need to be painted for the specified region.
252   // |bounds| is in terms of |this|.
253   PaintRegion GetPaintRegion(const gfx::Rect& bounds) const;
254 
255   // Returns the bounds that need to be painted based on the clip set on
256   // |canvas|.
257   gfx::Rect GetPaintBounds(gfx::Canvas* canvas) const;
258 
259   // Invokes SchedulePaint() for the selected rows.
260   void SchedulePaintForSelection();
261 
262   // Returns the TableColumn matching the specified id.
263   ui::TableColumn FindColumnByID(int id) const;
264 
265   // Sets the selection to the specified index (in terms of the view).
266   void SelectByViewIndex(int view_index);
267 
268   // Sets the selection model to |new_selection|.
269   void SetSelectionModel(const ui::ListSelectionModel& new_selection);
270 
271   // Advances the selection (from the active index) in the specified direction.
272   void AdvanceSelection(AdvanceDirection direction);
273 
274   // Sets |model| appropriately based on a event.
275   void ConfigureSelectionModelForEvent(const ui::LocatedEvent& event,
276                                        ui::ListSelectionModel* model) const;
277 
278   // Set the selection state of row at |view_index| to |select|, additionally
279   // any other rows in the GroupRange containing |view_index| are updated as
280   // well. This does not change the anchor or active index of |model|.
281   void SelectRowsInRangeFrom(int view_index,
282                              bool select,
283                              ui::ListSelectionModel* model) const;
284 
285   // Returns the range of the specified model index. If a TableGrouper has not
286   // been set this returns a group with a start of |model_index| and length of
287   // 1.
288   GroupRange GetGroupRange(int model_index) const;
289 
290   // Used by both GetTooltipText methods. Returns true if there is a tooltip and
291   // sets |tooltip| and/or |tooltip_origin| as appropriate, each of which may be
292   // NULL.
293   bool GetTooltipImpl(const gfx::Point& location,
294                       string16* tooltip,
295                       gfx::Point* tooltip_origin) const;
296 
297   ui::TableModel* model_;
298 
299   std::vector<ui::TableColumn> columns_;
300 
301   // The set of visible columns. The values of these point to |columns_|. This
302   // may contain a subset of |columns_|.
303   std::vector<VisibleColumn> visible_columns_;
304 
305   // The header. This is only created if more than one column is specified or
306   // the first column has a non-empty title.
307   TableHeader* header_;
308 
309   const TableTypes table_type_;
310 
311   const bool single_selection_;
312 
313   // TODO(sky): rename to observer_.
314   TableViewObserver* table_view_observer_;
315 
316   // The selection, in terms of the model.
317   ui::ListSelectionModel selection_model_;
318 
319   gfx::Font font_;
320 
321   int row_height_;
322 
323   // Width of the ScrollView last time Layout() was invoked. Used to determine
324   // when we should invoke UpdateVisibleColumnSizes().
325   int last_parent_width_;
326 
327   // The width we layout to. This may differ from |last_parent_width_|.
328   int layout_width_;
329 
330   // Current sort.
331   SortDescriptors sort_descriptors_;
332 
333   // Mappings used when sorted.
334   std::vector<int> view_to_model_;
335   std::vector<int> model_to_view_;
336 
337   scoped_ptr<TableViewRowBackgroundPainter> row_background_painter_;
338 
339   TableGrouper* grouper_;
340 
341   // True if in SetVisibleColumnWidth().
342   bool in_set_visible_column_width_;
343 
344   DISALLOW_COPY_AND_ASSIGN(TableView);
345 };
346 
347 }  // namespace views
348 
349 #endif  // UI_VIEWS_CONTROLS_TABLE_TABLE_VIEW_VIEWS_H_
350