Your search did not match any results.
Data Grid

Custom Editors

Different editors can be used to edit cell values in grid columns. The default editor depends on the column configuration. The dependency is illustrated in the editorOptions object's description (this object is used to customize the default editor). In this demo, the SelectBox widget is the Status column's default editor, and the editorOptions object is used to specify the widget's itemTemplate.

If the default editor is unsuitable, replace it with a custom editor by specifying an editCellTemplate. In this template, configure the replacement editor's appearance and behavior. In this demo, the default editors in the Owner and Assignees columns are replaced with the DropDownBox and TagBox widgets.

Copy to CodeSandBox
Apply
Reset
<dx-data-grid id="gridContainer" [dataSource]="tasks" [showBorders]="true" (onRowInserted)="$event.component.navigateToRow($event.key)" > <dxo-paging [enabled]="true" [pageSize]="15"></dxo-paging> <dxo-header-filter [visible]="true"></dxo-header-filter> <dxo-search-panel [visible]="true"></dxo-search-panel> <dxo-editing mode="cell" [allowUpdating]="true" [allowAdding]="true"> </dxo-editing> <dxi-column dataField="Owner" [width]="150" [allowSorting]="false" editCellTemplate="singleDropDownBoxEditor" > <dxo-lookup [dataSource]="employees" displayExpr="FullName" valueExpr="ID"> </dxo-lookup> <dxi-validation-rule type="required"></dxi-validation-rule> </dxi-column> <dxi-column dataField="AssignedEmployee" caption="Assignees" [width]="200" [allowSorting]="false" editCellTemplate="tagBoxEditor" [cellTemplate]="cellTemplate" [calculateFilterExpression]="calculateFilterExpression"> <dxo-lookup [dataSource]="employees" valueExpr="ID" displayExpr="FullName"> </dxo-lookup> <dxi-validation-rule type="required"></dxi-validation-rule> </dxi-column> <dxi-column dataField="Subject"> <dxi-validation-rule type="required"></dxi-validation-rule> </dxi-column> <dxi-column dataField="Status" width="200" [editorOptions]="editorOptions" > <dxo-lookup [dataSource]="statuses" displayExpr="name" valueExpr="id"> </dxo-lookup> <dxi-validation-rule type="required"></dxi-validation-rule> </dxi-column> <div *dxTemplate="let cellInfo of 'tagBoxEditor'"> <dx-tag-box [dataSource]="employees" [value]="cellInfo.value" valueExpr="ID" displayExpr="FullName" [showSelectionControls]="true" [maxDisplayedTags]="3" [showMultiTagOnly]="false" applyValueMode="useButtons" [searchEnabled]="true" (onValueChanged)="cellInfo.setValue($event.value)" (onSelectionChanged)="cellInfo.component.updateDimensions()"> </dx-tag-box> </div> <div *dxTemplate="let cellInfo of 'singleDropDownBoxEditor'"> <dx-drop-down-box [dropDownOptions]="dropDownOptions" [dataSource]="employees" [(value)]="cellInfo.value" displayExpr="FullName" valueExpr="ID" contentTemplate="contentTemplate"> <div *dxTemplate="let e of 'contentTemplate'"> <dx-data-grid [dataSource]="employees" [remoteOperations]="true" [height]="250" [selectedRowKeys]="[cellInfo.value]" [focusedRowEnabled]="true" [focusedRowKey]="cellInfo.value" [hoverStateEnabled]="true" (onSelectionChanged)="onSelectionChanged($event.selectedRowKeys, cellInfo, e.component)" > <dxi-column dataField="FullName"></dxi-column> <dxi-column dataField="Title"></dxi-column> <dxi-column dataField="Department"></dxi-column> <dxo-paging [enabled]="true" [pageSize]="10"></dxo-paging> <dxo-scrolling mode="virtual"></dxo-scrolling> <dxo-selection mode="single" ></dxo-selection> </dx-data-grid> </div> </dx-drop-down-box> </div> <div *dxTemplate="let status of 'statusTemplate'"> <div *ngIf="status === null"; else elseBlock> <span>(All)</span> </div> <div dx-template #elseBlock> <img src="images/icons/status-{{status.id}}.png" class="status-icon middle"> <span class="middle">{{status.name}}</span> </div> </div> </dx-data-grid>
import { NgModule, Component, enableProdMode } from '@angular/core'; import { BrowserModule } from '@angular/platform-browser'; import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; import { DxDataGridModule, DxListModule, DxDropDownBoxModule, DxTagBoxModule } from 'devextreme-angular'; import { Service, Status } from './app.service'; import { createStore } from "devextreme-aspnet-data-nojquery"; if(!/localhost/.test(document.location.host)) { enableProdMode(); } @Component({ selector: 'demo-app', templateUrl: 'app/app.component.html', styleUrls: ['app/app.component.css'], providers: [Service] }) export class AppComponent { employees: object; tasks: object; statuses: Status[]; dropDownOptions: object; editorOptions: object; url: string; constructor(service: Service) { this.dropDownOptions = { width: 500 }; this.editorOptions = { itemTemplate: "statusTemplate" } this.url = "https://js.devexpress.com/Demos/Mvc/api/CustomEditors"; this.statuses = service.getStatuses(); this.tasks = createStore({ key: "ID", loadUrl: this.url + "/Tasks", updateUrl: this.url + "/UpdateTask", insertUrl: this.url + "/InsertTask", onBeforeSend: function(method, ajaxOptions) { ajaxOptions.xhrFields = { withCredentials: true }; } }); this.employees = createStore({ key: "ID", loadUrl: this.url + "/Employees", onBeforeSend: function(method, ajaxOptions) { ajaxOptions.xhrFields = { withCredentials: true }; } }); } onSelectionChanged(selectedRowKeys, cellInfo, dropDownBoxComponent) { cellInfo.setValue(selectedRowKeys[0]); if(selectedRowKeys.length > 0) { dropDownBoxComponent.close(); } } calculateFilterExpression(filterValue, selectedFilterOperation, target) { if(target === "search" && typeof(filterValue) === "string") { return [(this as any).dataField, "contains", filterValue]; } return function(data) { return (data.AssignedEmployee || []).indexOf(filterValue) !== -1 } } cellTemplate(container, options) { var noBreakSpace = "\u00A0", text = (options.value || []).map(element => { return options.column.lookup.calculateCellValue(element); }).join(", "); container.textContent = text || noBreakSpace; container.title = text; } } @NgModule({ imports: [ BrowserModule, DxDataGridModule, DxListModule, DxDropDownBoxModule, DxTagBoxModule ], declarations: [AppComponent], bootstrap: [AppComponent] }) export class AppModule { } platformBrowserDynamic().bootstrapModule(AppModule);
.status-icon { height: 16px; width: 16px; display: inline-block; margin-right: 8px; } .middle { vertical-align: middle; }
import { Injectable } from '@angular/core'; export class Status { id: number; name: string; } let statuses: Status[] = [{ "id": 1, "name": "Not Started" }, { "id": 2, "name": "In Progress" }, { "id": 3, "name": "Deferred" }, { "id": 4, "name": "Need Assistance" }, { "id": 5, "name": "Completed" } ]; @Injectable() export class Service { getStatuses() { return statuses; } }
// In real applications, you should not transpile code in the browser. You can see how to create your own application with Angular and DevExtreme here: // https://js.devexpress.com/Documentation/Guide/Angular_Components/Getting_Started/Create_a_DevExtreme_Application/ System.config({ transpiler: 'ts', typescriptOptions: { module: "commonjs", emitDecoratorMetadata: true, experimentalDecorators: true }, meta: { 'typescript': { "exports": "ts" }, 'devextreme-aspnet-data-nojquery': { "esModule": true } }, paths: { 'npm:': 'https://unpkg.com/' }, map: { 'ts': 'npm:plugin-typescript@8.0.0/lib/plugin.js', 'typescript': 'npm:typescript@3.4.5/lib/typescript.js', '@angular/core': 'npm:@angular/core@8.0.0/bundles/core.umd.js', '@angular/common': 'npm:@angular/common@8.0.0/bundles/common.umd.js', '@angular/compiler': 'npm:@angular/compiler@8.0.0/bundles/compiler.umd.js', '@angular/platform-browser': 'npm:@angular/platform-browser@8.0.0/bundles/platform-browser.umd.js', '@angular/platform-browser-dynamic': 'npm:@angular/platform-browser-dynamic@8.0.0/bundles/platform-browser-dynamic.umd.js', '@angular/router': 'npm:@angular/router@8.0.0/bundles/router.umd.js', '@angular/forms': 'npm:@angular/forms@8.0.0/bundles/forms.umd.js', '@angular/common/http': 'npm:@angular/common@8.0.0/bundles/common-http.umd.js', 'tslib': 'npm:tslib/tslib.js', 'rxjs': 'npm:rxjs@6.3.3', 'rxjs/operators': 'npm:rxjs@6.3.3/operators', 'devextreme': 'npm:devextreme@19.2', 'jszip': 'npm:jszip@3.1.3/dist/jszip.min.js', 'quill': 'npm:quill@1.3.7/dist/quill.js', 'devexpress-diagram': 'npm:devexpress-diagram', 'devexpress-gantt': 'npm:devexpress-gantt', 'devextreme-angular': 'npm:devextreme-angular@19.2', 'devextreme-aspnet-data-nojquery': 'npm:devextreme-aspnet-data-nojquery@2.5.1' }, packages: { 'app': { main: './app.component.ts', defaultExtension: 'ts' }, 'devextreme': { defaultExtension: 'js' }, 'rxjs': { main: 'index.js', defaultExtension: 'js' }, 'rxjs/operators': { main: 'index.js', defaultExtension: 'js' }, 'devextreme-angular': { main: 'index.js', defaultExtension: 'js' } } });
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>DevExtreme Demo</title> <meta http-equiv="X-UA-Compatible" content="IE=edge" /> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0" /> <link rel="stylesheet" type="text/css" href="https://cdn3.devexpress.com/jslib/19.2.4/css/dx.common.css" /> <link rel="stylesheet" type="text/css" href="https://cdn3.devexpress.com/jslib/19.2.4/css/dx.light.css" /> <script src="https://unpkg.com/core-js@2.4.1/client/shim.min.js"></script> <script src="https://unpkg.com/zone.js@0.6.25/dist/zone.js"></script> <script src="https://unpkg.com/reflect-metadata@0.1.3/Reflect.js"></script> <script src="https://unpkg.com/systemjs@0.21.3/dist/system.js"></script> <script src="config.js"></script> <script> System.import('app').catch(console.error.bind(console)); </script> </head> <body class="dx-viewport"> <div class="demo-container"> <demo-app>Loading...</demo-app> </div> </body> </html>