diff options
-rw-r--r-- | documentation/components/components-grid.asciidoc | 519 |
1 files changed, 177 insertions, 342 deletions
diff --git a/documentation/components/components-grid.asciidoc b/documentation/components/components-grid.asciidoc index c379584e01..8221d641d6 100644 --- a/documentation/components/components-grid.asciidoc +++ b/documentation/components/components-grid.asciidoc @@ -7,8 +7,6 @@ layout: page [[components.grid]] = [classname]#Grid# -*_This section has not yet been updated for Vaadin Framework 8_* - ifdef::web[] [.sampler] image:{live-demo-image}[alt="Live Demo", link="http://demo.vaadin.com/sampler/#ui/grids-and-trees/grid"] @@ -41,21 +39,16 @@ Grid is fully themeable with CSS and style names can be set for all grid elements. For data rows and cells, the styles can be generated with a row or cell style generator. -Finally, [classname]#Grid# is designed to be extensible and used just as well -for client-side development - its GWT API is nearly identical to the server-side -API, including data binding. - - [[components.grid.data]] == Binding to Data -[classname]#Grid# is normally used by binding it to a container data source, +[classname]#Grid# is normally used by binding it to a , described in -<<dummy/../../../framework/datamodel/datamodel-overview.asciidoc#datamodel.overview,"Binding Components to Data">>. -By default, it is bound to List of items. You can set the items in the constructor or with +<<dummy/../../../framework/datamodel/datamodel-providers.asciidoc#datamodel.dataproviders,"Showing Many Items in a Listing">>. +By default, it is bound to List of items. You can set the items with the [methodname]#setItems()# method. -For example, if you have a list of beans, you could add to a [classname]#Grid# as follows +For example, if you have a list of beans, you show them in a [classname]#Grid# as follows [source, java] @@ -67,163 +60,160 @@ List<Person> people = Lists.newArrayList( new Person("Johannes Kepler", 1571)); // Create a grid bound to the list -Grid<Person> grid = new Grid<>(people); +Grid<Person> grid = new Grid<>(); +grid.setItems(people); grid.addColumn("Name", Person::getName); grid.addColumn("Year of birth", Person::getBirthYear); layout.addComponent(grid); ---- -In addition to list you can pass items individually: - -[source, java] ----- -grid.setItems(new Person("Nicolaus Copernicus", 1543), - new Person("Galileo Galilei", 1564)); ----- - -Note that you can not use [methodname]#addRow()# to add items if the container -is read-only or has read-only columns, such as generated columns. - - [[components.grid.selection]] == Handling Selection Changes Selection in [classname]#Grid# is handled a bit differently from other selection -components, as it is not an [classname]#AbstractSelect#. Grid supports both -single and multiple selection, defined by the __selection model__. Selection -events can be handled with a [interfacename]#SelectionListener#. +components, as it is not a [classname]#HasValue#. Grid supports +single, multiple, or no-selection, each defined by a specific selection model. Each +selection model has a specific API depending on the type of the selection. -[[components.grid.selection.mode]] -=== Selection Models - -A [classname]#Grid# can be set to be in [literal]#++SINGLE++# (default), -[literal]#++MULTI++#, or [literal]#++NONE++# selection mode, defined in the -[interfacename]#SelectionMode# enum. +For basic usage, switching between the built-in selection models is possible by using the +[method]#setSelectionMode(SelectionMode)#. Possible options are [literal]#++SINGLE++# (default), +[literal]#++MULTI++#, or [literal]#++NONE++#. +Listening to selection changes in any selection model is possible with a [classname]#SelectionListener#, +which provides a generic [classname]#SelectionEvent# which for getting the selected value or values. +Note that the listener is actually attached to the selection model and not the grid, +and will stop getting any events if the selection mode is changed. [source, java] ---- -// Use single-selection mode (default) -grid.setSelectionMode(SelectionMode.SINGLE); ----- - -Empty (null) selection is allowed by default, but can be disabled -with [methodname]#setDeselectAllowed()# in single-selection mode. +Grid<Person> grid = new Grid<>(); -[source, java] ----- -// Pre-select 3rd item from the person list -grid.select(personList.get(2)); +// switch to multiselect mode +grid.setSelectionMode(SelectionMode.MULTI); +grid.addSelectionListener(event -> { + Set<Person> selected = event.getAllSelectedItems(); + Notification.show(selected.size() + " items selected"); +} ---- - -[[components.grid.selection.single]] -=== Handling Selection - -Changes in the selection can be handled with a -[interfacename]#SelectionListener#. You need to implement the -[methodname]#select()# method, which gets a [classname]#SelectionEvent# as -parameter. In addition to selection, you can handle clicks on rows or cells with -a [interfacename]#CellClickListener#. - -You can get the new selection from the selection event with -[methodname]#getSelected()#, which returns a set of items, or more simply -from the grid. - -For example: +Programmatically selecting the value is possible via [methodname]#select(T)#. +In multiselect mode, this will add the given item to the selection. [source, java] ---- -grid.addSelectionListener(selectionEvent -> { - // Get selection from the selection model - Collection<Person> selectedPersons = - selectionEvent.getSelected(); - if (!selectedPersons.isEmpty()) - Notification.show("Selected " + selectedPersons); - else - Notification.show("Nothing selected"); -}); ----- +// in single-select, only one item is selected +grid.select(defaultPerson); -The current selection can be obtained from the [classname]#Grid# object by -[methodname]#getSelectedItem()# or [methodname]#getSelectedItems()#, which return -one (in single-selection mode) or all (in multi-selection mode) selected items. +// switch to multi select, clears selection +grid.setSelectionMode(SelectionMode.MULTI); +// Select items 2-4 +people.subList(2,3).forEach(grid::select); +---- +The current selection can be obtained from the [classname]#Grid# by +[methodname]#getSelectedItems()#, and the returned [classname]#Set# contains either +only one item (in single-selection mode) or several items (in multi-selection mode). [WARNING] ==== -If you change the data source for a grid, it will clear the selection. To keep -the previous selection you must reset the selection afterwards using the -[methodname]#select()# method. +If you change selection mode for a grid, it will clear the selection +and fire a selection event. To keep the previous selection you must +reset the selection afterwards using the [methodname]#select()# method. ==== -[[components.grid.selection.multi]] -=== Multiple Selection +[WARNING] +==== +If you change the grid's items with [methodname]#setItems()# or the used +[classname]#DataProvider#, it will clear the selection and fire a selection event. +To keep the previous selection you must reset the selection afterwards +using the [methodname]#select()# method. +==== -In the multiple selection mode, a user can select multiple items by clicking on -the checkboxes in the leftmost column, or by using the kbd:[Space] to select/deselect the currently focused row. -Space bar is the default key for toggling the selection, but it can be customized. -[[figure.components.grid.selection.multi]] -.Multiple Selection in [classname]#Grid# -image::img/grid-selection-multi.png[width=50%, scaledwidth=75%] +[[components.grid.selection.mode]] +=== Selection Models -You can use [methodname]#select()# to add items to the selection. +For more control over the selection, you can access the used selection model with +[methodname]#getSelectionModel()#. The return type is [classname]#GridSelectionModel# +which has generic selection model API, but you can cast that to the specific selection model type, +typically either [classname]#SingleSelectionModel# or [classname]#MultiSelectionModel#. +The selection model is also returned by the [methodname]#setSelectionMode(SelectionMode)# method. [source, java] ---- -// Grid in multi-selection mode -Grid<Person> grid = Grid<>(personList) -grid.setSelectionMode(SelectionMode.MULTI); - -// Items 2-4 -personList.subList(2,3).forEach(grid::select); +// the default selection model +SingleSelectionModel<Person> defaultModel = + (SingleSelectionModel<Person>) grid.getSelectionModel(); +// Use multi-selection mode +MultiSelectionModel<Person> selectionModel = + (MultiSelectionModel<Person>) grid.setSelectionMode(SelectionMode.MULTI); ---- -The current selection can be read with [methodname]#getSelected()# -in the [classname]#Grid#. +==== Single Selection Model + +By obtaining a reference to the [classname]#SingleSelectionModel#, +you can access more fine grained API for the single-select case. + +The [methodname]#addSingleSelect(SingleSelectionListener)# method provides access to +[classname]#SingleSelectionEvent#, which has some extra API for more convenience. + +In single-select mode, it is possible to control whether the empty (null) selection is allowed. +By default it is enabled, but can be disabled with [methodname]#setDeselectAllowed()#. [source, java] ---- -// Allow deleting the selected items -Button delSelected = new Button("Delete Selected", e -> { - // Delete all selected data items - for (Person person: selection.getSelected()) - personList.remove(person); - - // Disable after deleting - e.getButton().setEnabled(false); +// preselect value +grid.select(defaultItem); - // Reset grid content from the list - grid.setItems(personList); -}); -delSelected.setEnabled(!grid.getSelected().isEmpty()); +SingleSelectionModel<Person> singleSelect = + (SingleSelectionModel<Person>) grid.getSelectionModel(); +// disallow empty selection +singleSelect.setDeselectAllowed(false); ---- -Changes in the selection can be handled with a -[interfacename]#SelectionListener#. The selection event object provides -[methodname]#getAdded()# and [methodname]#getRemoved()# to allow determining the -differences in the selection change. +[[components.grid.selection.multi]] +=== Multi-Selection Model + +In the multi-selection mode, a user can select multiple items by clicking on +the checkboxes in the leftmost column, or by using the kbd:[Space] to select/deselect the currently focused row. +Space bar is the default key for toggling the selection, but it can be customized. + +[[figure.components.grid.selection.multi]] +.Multiple Selection in [classname]#Grid# +image::img/grid-selection-multi.png[width=50%, scaledwidth=75%] + +By obtaining a reference to the [classname]#MultiSelectionModel#, +you can access more fine grained API for the multi-select case. + +The [classname]#MultiSelectionModel# provides [methodname]#addMultiSelectionListener(MultiSelectionListener)# +access to [classname]#MultiSelectionEvent#, which allows to easily access differences in the selection change. [source, java] ---- -// Handle selection changes -grid.addSelectionListener(selection -> { // Java 8 - Notification.show(selection.getAdded().size() + - " items added, " + - selection.getRemoved().size() + - " removed."); +// Grid in multi-selection mode +Grid<Person> grid = Grid<>() +grid.setItems(people); +MultiSelectionModel<Person> selectionModel + = (MultiSelectionModel<Person>) grid.setSelectionMode(SelectionMode.MULTI); + +selectionModel.selectAll(); + +selectionModel.addMultiSelectionListener(event -> { + Notification.show(selection.getAddedSelection().size() + + " items added, " + + selection.getRemovedSelection().size() + + " removed."); // Allow deleting only if there's any selected deleteSelected.setEnabled( - grid.getSelectedRows().size() > 0); -}); + event.getNewSelection().size() > 0); +}; ---- @@ -392,31 +382,23 @@ value. This allows having images, HTML, and buttons in grid cells. image::img/grid-renderers.png[width=75%, scaledwidth=100%] Renderers implement the [interfacename]#Renderer# interface. +Renderers require a specific data type for the column. You set the column renderer in the [classname]#Grid.Column# object as follows: [source, java] ---- - -Column<Integer> bornColumn = grid.addColumn(Person:getBirthYear); - -... -Grid.Column bornColumn = grid.getColumn("born"); -bornColumn.setRenderer(new NumberRenderer("born in %d AD")); +// the type of birthYear is a number +Column<Integer> bornColumn = grid.addColumn(Person:getBirthYear, + new NumberRenderer("born in %d AD")); ---- - -Renderers require a specific data type for the column. To convert to a property -type to a type required by a renderer, you can pass an optional -[interfacename]#Converter# to [methodname]#setRenderer()#, as described later in -this section. A converter can also be used to (pre)format the property values. -The converter is run on the server-side, before sending the values to the -client-side to be rendered with the renderer. - The following renderers are available, as defined in the server-side [package]#com.vaadin.ui.renderers# package: +[classname]#TextRenderer#:: The default renderer, displays plain text as is. Any HTML markup is quoted. + + [classname]#ButtonRenderer#:: Renders the data value as the caption of a button. A [interfacename]#RendererClickListener# can be given to handle the button clicks. -ifdef::web[] + Typically, a button renderer is used to display buttons for operating on a data item, such as edit, view, delete, etc. It is not meaningful to store the button @@ -435,13 +417,12 @@ people.add(new Person("Johannes Kepler", 1571)); Grid<Person> grid = new Grid(people); // Render a button that deletes the data row (item) -grid.addColumn(person -> "Delete" ) - .setRenderer(new ButtonRenderer(clickEvent -> { - people.remove(clickEvent.getValue()); - grid.setItems(people); +grid.addColumn(person -> "Delete", + new ButtonRenderer(clickEvent -> { + people.remove(clickEvent.getValue()); + grid.setItems(people); }); ---- -endif::web[] [classname]#ImageRenderer#:: Renders the cell as an image. The column type must be a [interfacename]#Resource#, as described in @@ -449,7 +430,6 @@ The column type must be a [interfacename]#Resource#, as described in [classname]#ExternalResource# are currently supported for images in [classname]#Grid#. -ifdef::web[] + [source, java] ---- @@ -457,71 +437,36 @@ Column<ThemeResource> imageColumn = grid.addColumn("picture", p -> new ThemeResource("img/"+p.getLastname()+".jpg")); imageColumn.setRenderer(new ImageRenderer()); ---- -+ -You also need to define the row heights so that the images fit there. You can -set it in the theme for all data cells or for the column containing the images. - -+ -For the latter way, first define a CSS style name for grid and the column: - - -+ -[source, java] ----- -grid.setStyleName("gridwithpics128px"); -imageColumn.setCellStyleGenerator(cell -> "imagecol"); ----- -ifdef::web[] -+ -Then, define the style in CSS (Sass): -endif::web[] - - -+ -[source, css] ----- -.gridwithpics128px .imagecol { - height: 128px; - background: black; - text-align: center; -} ----- -endif::web[] [classname]#DateRenderer#:: Formats a column with a [classname]#Date# type using string formatter. The format string is same as for [methodname]#String.format()# in Java API. The date is passed in the parameter index 1, which can be omitted if there is only one format specifier, such as "[literal]#++%tF++#". -ifdef::web[] + [source, java] ---- -Grid.Column<Date> bornColumn = grid.addColumn(person:getBirthDate); -bornColumn.setRenderer( - new DateRenderer("%1$tB %1$te, %1$tY", +Grid.Column<Date> bornColumn = grid.addColumn(person:getBirthDate, + new DateRenderer("%1$tB %1$te, %1$tY", Locale.ENGLISH)); ---- + Optionally, a locale can be given. Otherwise, the default locale (in the component tree) is used. -endif::web[] [classname]#HTMLRenderer#:: Renders the cell as HTML. This allows formatting the cell content, as well as using HTML features such as hyperlinks. -ifdef::web[] + Set the renderer in the [classname]#Grid.Column# object: + [source, java] ---- Column<String> htmlColumn grid.addColumn(person -> - "<a href='" + person.getDetailsUrl() + "' target='_top'>info</a>"); -htmlColumn.setRenderer(new HtmlRenderer()); + "<a href='" + person.getDetailsUrl() + "' target='_top'>info</a>", + new HtmlRenderer()); ---- -endif::web[] [classname]#NumberRenderer#:: Formats column values with a numeric type extending [classname]#Number#: [classname]#Integer#, [classname]#Double#, etc. The format can be specified @@ -529,60 +474,19 @@ either by the subclasses of [classname]#java.text.NumberFormat#, namely [classname]#DecimalFormat# and [classname]#ChoiceFormat#, or by [methodname]#String.format()#. -ifdef::web[] + For example: + [source, java] ---- -// Define some columns -Column<String> nameCol = grid.addColumn(person::getName); -Column<Integer> bornCol = grid.addColumn(person:getBirthYear); -Column<Integer> slettersCol = grid.addColumn("sletters"); -Column<Double> ratingCol = grid.addColumn("rating"); - -// Use decimal format -bornCol.setRenderer(new NumberRenderer( - new DecimalFormat("in #### AD"))); - -// Use textual formatting on numeric ranges -slettersCol.setRenderer(new NumberRenderer( - new ChoiceFormat("0#none|1#one|2#multiple"))); - // Use String.format() formatting -ratingCol.setRenderer(new NumberRenderer( - "%02.4f", Locale.ENGLISH)); - -// Add some data rows -grid.addItems(new Person("Nicolaus Copernicus", 1473, 2, 0.4), - new Person("Galileo Galilei", 1564, 0, 4.2), - new Person("Johannes Kepler", 1571, 1, 2.3)); - +Column<Double> ratingCol = grid.addColumn("rating", + new NumberRenderer("%02.4f", Locale.ENGLISH)); ---- -endif::web[] + [classname]#ProgressBarRenderer#:: Renders a progress bar in a column with a [classname]#Double# type. The value must be between 0.0 and 1.0. -ifdef::web[] -+ -For example: -+ -[source, java] ----- -// Define some columns -Column<String> nameCol = grid.addColumn(person::getName); -Column<Double> ratingCol = grid.addColumn("rating"); -ratingCol.setRenderer(new ProgressBarRenderer()); - -// Add some data rows -grid.addItems(new Person("Nicolaus Copernicus", 0.4), - new Person("Galileo Galilei", 4.2), - new Person("Johannes Kepler", 2.3)); ----- -endif::web[] -[classname]#TextRenderer#:: Displays plain text as is. Any HTML markup is quoted. - - [[components.grid.renderer.custom]] === Custom Renderers @@ -677,10 +581,10 @@ yearsCell.setHtml("<b>Years</b>"); === Components in Header or Footer You can set a component in a header or footer cell with -[methodname]#setComponent()#. Often, this feature is used to allow filtering, as -described in <<components.grid.filtering>>, which also gives an example of the -use. +[methodname]#setComponent()#. Often, this feature is used to allow filtering. +//// +// commented out until filtering is sorted for 8 [[components.grid.filtering]] == Filtering @@ -699,10 +603,11 @@ as follows: [source, java] ---- // Have a list of persons -List<Person> persons = exampleDataSource(); +List<Person> people = getPeople(); // Create a grid bound to it -Grid<Person> grid = new Grid(persons); +Grid<Person> grid = new Grid<>(); +grid.setItems(people); grid.setSelectionMode(SelectionMode.NONE); grid.setWidth("500px"); grid.setHeight("300px"); @@ -730,9 +635,10 @@ for (Column<?> col: grid.getColumns()) { }); cell.setComponent(filterField); - } ---- +//// + [[components.grid.sorting]] == Sorting @@ -773,78 +679,73 @@ grid.sort(Sort.by(cityColumn, SortDirection.ASCENDING) .then(nameColumn, SortDirection.DESCENDING)); ---- + [[components.grid.editing]] -== Editing +== Editing Items Inside Grid Grid supports line-based editing, where double-clicking a row opens the row editor. In the editor, the input fields can be edited, as well as navigated with kbd:[Tab] and kbd:[Shift+Tab] keys. If validation fails, an error is displayed and the user can correct the inputs. -To enable editing, you need to call [methodname]#setEditorEnabled(true)# for the -grid. +The [classname]#Editor# is accessible via [methodname]#getEditor()#, and to enable editing, you need to call [methodname]#setEnabled(true) on it. +The editor is based on [classname]#Binder# which is used to bind the data +to the editor. See <<dummy/../../../framework/datamodel/datamodel-forms.asciidoc#datamodel.forms.beans,"Binding Beans to Forms">> for more information on setting up field components and validation by using [classname]#Binder. +The [classname]#Binder# needs to be set with [methodname]#setBinder# in [classname]#Editor#. [source, java] ---- -Grid<Person> grid = new Grid(persons); -grid.setEditorEnabled(true); ----- +List<Todo> items = Arrays.asList(new Todo("Done task", true), + new Todo("Not done", false)); -Grid supports two row editor modes - buffered and unbuffered. The default mode is -buffered. The mode can be changed with [methodname]#setBuffered(false)# +Grid<Todo> grid = new Grid<>(); -[[components.grid.editing.buffered]] -=== Buffered Mode +TextField taskField = new TextField(); +CheckBox doneField = new CheckBox(); +Binder<Todo> binder = new Binder<>(); -The editor has a [guibutton]#Save# button that commits -the data item to the container data source and closes the editor. The -[guibutton]#Cancel# button discards the changes and exits the editor. +binder.bind(taskField, Todo::getTask, Todo::setTask); +binder.bind(doneField, Todo::isDone, Todo::setDone); -A row under editing is illustrated in <<figure.components.grid.editing>>. +grid.getEditor().setBinder(binder); +grid.getEditor().setEnabled(true); -[[figure.components.grid.editing]] -.Editing a Grid Row -image::img/grid-editor-basic.png[width=50%, scaledwidth=75%] +Column<Todo, String> column = grid + .addColumn(todo -> String.valueOf(todo.isDone())); +column.setWidth(75); +column.setEditorComponent(doneField); -[[components.grid.editing.unbuffered]] -=== Unbuffered Mode - -The editor has no buttons and all changed data is committed directly -to the container. If another row is clicked, the editor for the current row is closed and -a row editor for the clicked row is opened. +grid.addColumn(Todo::getTask).setEditorComponent(taskField); +---- -[[components.grid.editing.fields]] -=== Editor Fields +It is possible to customize the used editor component for each column and row, +by using [methodname]#setEditorComponentGenerator(EditorComponentGenerator)# in +[classname]#Column#. -The editor fields are configured in [classname]#Column#and bound to -the bean data source with a [classname]#Binder#, which -also handles tasks such as validation, as explained later. -To disable editing in a particular column, you can call -[methodname]#setEditorField()# in the [classname]#Column# object with -[parameter]#null# parameter. +[[components.grid.editing.buffered]] +=== Buffered / Unbuffered Mode -In the following example, we configure a field with validation and styling: +Grid supports two editor modes - buffered and unbuffered. The default mode is +buffered. The mode can be changed with [methodname]#setBuffered(false)#. +In the buffered mode, editor has two buttons visible: a [guibutton]#Save# button that commits +the modifications to the bean and closes the editor and a [guibutton]#Cancel# button +discards the changes and exits the editor. -[source, java] ----- -// Create an editor for name -TextField nameEditor = new TextField(); +Editor in buffered mode is illustrated in <<figure.components.grid.editing>>. -// Custom CSS style -nameEditor.addStyleName("nameeditor"); +[[figure.components.grid.editing]] +.Editing a Grid Row +image::img/grid-editor-basic.png[width=50%, scaledwidth=75%] -// Add editor to name column -nameColumn.setEditorField(nameEditor); ----- -Setting an editor field to [parameter]#null# deletes the currently existing -editor field and makes the column non-editable. +In the unbuffered mode, the editor has no buttons and all changed data is committed directly +to the data provider. If another row is clicked, the editor for the current row is closed and +a row editor for the clicked row is opened. -ifdef::web[] [[components.grid.editing.captions]] === Customizing Editor Buttons @@ -854,80 +755,14 @@ set their captions with [methodname]#setEditorSaveCaption()# and In the following example, we demonstrate one way to translate the captions: - [source, java] ---- -// Captions are stored in a resource bundle -ResourceBundle bundle = ResourceBundle.getBundle( - MyAppCaptions.class.getName(), - Locale.forLanguageTag("fi")); // Finnish - // Localize the editor button captions -grid.setEditorSaveCaption( - bundle.getString(MyAppCaptions.SaveKey)); -grid.setEditorCancelCaption( - bundle.getString(MyAppCaptions.CancelKey)); +grid.getEditor().setSaveCaption("Tallenna"); +grid.getEditor().setCancelCaption("Peruuta")); ---- -endif::web[] - -[[components.grid.editing.fieldgroup]] -=== Binding to Data with a Binder - -Data binding to the item under editing is handled with a -[classname]#Binder#, which you need to set with -[methodname]#setEditorFieldGroup#. This is mostly useful when using -special-purpose, such as to enable bean validation. - -For example, assuming that we want to enable bean validation for a bean such as -the following: - -[source, java] ----- -public class Person implements Serializable { - @NotNull - @Size(min=2, max=10) - private String name; - - @Min(1) - @Max(130) - private int age; - ...] ----- - -We can now use a [classname]#BeanBinder# in the [classname]#Grid# as -follows: - -[source, java] ----- -Grid<Person> grid = new Grid(examplePersonList()); -Column<String> nameCol = grid.addColumn(Person::getName); -Column<Integer> ageCol = grid.addColumn(Person::getAge); -grid.setEditorEnabled(true); - -TextField nameEditor = new TextField(); -nameCol.setEditorField(nameEditor); - -// Enable bean validation for the data -BeanBinder<Person> binder = new BeanBinder<>(Person.class); - -// Have some extra validation in a field -binder.addField(nameEditor, "name") - .withValidator(new RegexpValidator( - "^\\p{Alpha}+ \\p{Alpha}+$", - "Need first and last name")); - -grid.setEditorBinder(binder); - ----- - -To use bean validation as in the example above, you need to include an -implementation of the Bean Validation API in the classpath, as described in -<<dummy/../../../framework/datamodel/datamodel-forms.asciidoc#datamodel.forms.beans,"Binding Beans to Forms">>. - - -ifdef::web[] [[components.grid.editing.validation]] === Handling Validation Errors @@ -939,18 +774,18 @@ first error in the editor. .Editing a Grid Row image::img/grid-editor-errors.png[width=50%, scaledwidth=75%] -You can modify the error handling by implementing a custom -[interfacename]#EditorErrorHandler# or by extending the -[classname]#DefaultEditorErrorHandler#. - -endif::web[] +You can modify the error message by implementing a custom +[interfacename]#EditorErrorGenerator# with for the [classname]#Editor#. +//// +// Not supported in 8 [[components.grid.scrolling]] == Programmatic Scrolling You can scroll to first item with [methodname]#scrollToStart()#, to end with [methodname]#scrollToEnd()#, or to a specific row with [methodname]#scrollTo()#. +//// [[components.grid.stylegeneration]] |