React TreeList - columns.lookup
Specifies properties of a lookup column.
A lookup column restricts the set of values that can be chosen when a user edits or filters the column. In a lookup column, each cell is a drop-down menu. You can use a lookup column when you need to substitute displayed values with required values. For example, consider that you have two arrays of objects: drivers and buses.
const drivers = [
    { driverID: 1, firstName: "John", lastName: "Smith", busID: 2 },
    { driverID: 2, firstName: "Lizzy", lastName: "Cook", busID: 1 },
    { driverID: 3, firstName: "Brian", lastName: "Hawkins", busID: 3 }
];
const buses = [
    { busID: 1, plates: "123456" },
    { busID: 2, plates: "AB-1234" },
    { busID: 3, plates: "CD-9876" }
];All drivers have the busID field, which refers to a bus. If drivers is the main dataSource, the Bus ID column displays bus IDs, which provides little information to a user. It will be more useful to display bus license plates instead of IDs. For this, the buses array must be set as a lookup dataSource for the Bus ID column. Then, the names of data fields must be assigned to the valueExpr and displayExpr properties. Values from the valueExpr data field will be replaced with values from the displayExpr data field.
jQuery
$(function() {
    $("#treeListContainer").dxTreeList({
        dataSource: drivers,
        // ...
        columns: [{
            dataField: "busID",
            lookup: {
                dataSource: buses,
                valueExpr: "busID",
                displayExpr: "plates"
            }
        }]
    });
});Angular
<dx-tree-list [dataSource]="drivers">
    <dxi-column dataField="busID">
        <dxo-lookup
            [dataSource]="buses"
            valueExpr="busID"
            displayExpr="plates">
        </dxo-lookup>
    </dxi-column>
</dx-tree-list>
import { DxTreeListModule } from "devextreme-angular";
// ...
export class AppComponent {
    drivers = [
        { driverID: 1, firstName: "John", lastName: "Smith", busID: 2 },
        { driverID: 2, firstName: "Lizzy", lastName: "Cook", busID: 1 },
        { driverID: 3, firstName: "Brian", lastName: "Hawkins", busID: 3 }
    ];
    buses = [
        { busID: 1, plates: "123456" },
        { busID: 2, plates: "AB-1234" },
        { busID: 3, plates: "CD-9876" }
    ];
}
@NgModule({
    imports: [
        // ...
        DxTreeListModule
    ],
    // ...
})Vue
<template>
    <DxTreeList ...
        :data-source="drivers">
        <DxColumn data-field="busID">
            <DxLookup
                :data-source="buses"
                value-expr="busID"
                display-expr="plates"
            />
        </DxColumn>
    </DxTreeList>
</template>
<script>
import 'devextreme/dist/css/dx.light.css';
import DxTreeList, {
    DxColumn,
    DxLookup
} from 'devextreme-vue/tree-list';
const drivers = [
    { driverID: 1, firstName: "John", lastName: "Smith", busID: 2 },
    { driverID: 2, firstName: "Lizzy", lastName: "Cook", busID: 1 },
    { driverID: 3, firstName: "Brian", lastName: "Hawkins", busID: 3 }
];
const buses = [
    { busID: 1, plates: "123456" },
    { busID: 2, plates: "AB-1234" },
    { busID: 3, plates: "CD-9876" }
];
export default {
    components: {
        DxTreeList,
        DxColumn,
        DxLookup
    },
    data() {
        return {
            drivers,
            buses
        }
    }
}
</script>React
import React from 'react';
import 'devextreme/dist/css/dx.light.css';
import TreeList, {
    Column,
    Lookup
} from 'devextreme-react/tree-list';
const drivers = [
    { driverID: 1, firstName: "John", lastName: "Smith", busID: 2 },
    { driverID: 2, firstName: "Lizzy", lastName: "Cook", busID: 1 },
    { driverID: 3, firstName: "Brian", lastName: "Hawkins", busID: 3 }
];
const buses = [
    { busID: 1, plates: "123456" },
    { busID: 2, plates: "AB-1234" },
    { busID: 3, plates: "CD-9876" }
];
export default function App() {
    return (
        <TreeList ...
            dataSource={drivers}>
            <Column dataField="busID">
                <Lookup
                    dataSource={buses}
                    valueExpr="busID"
                    displayExpr="plates"
                />
            </Column>
        </TreeList>
    );
}With this code, the Bus ID column contains license plates instead of IDs. Moreover, the user can choose a plate number from the drop-down menu when editing cells or applying a filter to this column.
See Also
allowClearing
Specifies whether to display the Clear button in lookup column cells while they are being edited.
Set this property to true only if your data source accepts null values.
To specify this property based on a condition, set the showClearButton property instead. This is a property of the SelectBox UI component which is used as an editor for lookup column cells. allowClearing is an alias for this property in the TreeList. The following code shows how to set showClearButton in the onEditorPreparing event handler:
jQuery
$(function() {
    $("#treeListContainer").dxTreeList({
        // ...
        onEditorPreparing: function (e) {
            if (/* a condition to set the property's value */) {
                e.editorOptions.showClearButton = true;
            }
        }
    });
});Angular
import { Component } from '@angular/core';
@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.css']
})
export class AppComponent {
    onEditorPreparing(e) {
        if (/* a condition to set the property's value */) {
            e.editorOptions.showClearButton = true;
        }
    }
}
import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { AppComponent } from './app.component';
import { DxTreeListModule } from 'devextreme-angular';
@NgModule({
    declarations: [
        AppComponent
    ],
    imports: [
        BrowserModule,
        DxTreeListModule
    ],
    bootstrap: [AppComponent]
})
export class AppModule { }
<dx-tree-list ...
    (onEditorPreparing)="onEditorPreparing($event)">
    <!-- ... -->
</dx-tree-list>Vue
<template>
    <DxTreeList ...
        :on-editor-preparing="onEditorPreparing">
        <!-- ... -->
    </DxTreeList>
</template>
<script>
import 'devextreme/dist/css/dx.light.css';
import DxTreeList from 'devextreme-vue/tree-list';
export default {
    components: {
        DxTreeList
    },
    // ...
    methods: {
        onEditorPreparing(e) {
            if (/* a condition to set the property's value */) {
                e.editorOptions.showClearButton = true;
            }
        }
    }
}
</script>React
import React from 'react';
import 'devextreme/dist/css/dx.light.css';
import TreeList from 'devextreme-react/tree-list';
class App extends React.Component {
    onEditorPreparing(e) {
        if (/* a condition to set the property's value */) {
            e.editorOptions.showClearButton = true;
        }
    }
    render() {
        return (
            <TreeList ...
                onEditorPreparing={this.onEditorPreparing}>
                {/* ... */}
            </TreeList>
        );
    }
}
export default App;dataSource
Specifies the data source for the lookup column.
An array of objects or primitives, a store instance, or a DataSource configuration.
This property accepts one of the following:
- Array of objects or primitives 
 A JavaScript array that contains plain objects or primitives.
- DataSource configuration object 
 A DataSource configuration object. You can find code for this example in the following help topic: Bind a Lookup Column to a Custom Data Source. For more information about the DataSource and DevExtreme Data Layer, refer to the Data Layer article.IMPORTANTDataSource instances are not supported.
- Store instance 
 An ArrayStore, LocalStore, ODataStore, or CustomStore instance.
- Function 
 A function that returns one of the above.NOTE- When the component is initialized, it calls the dataSource function with an empty options object. To avoid execution errors, check that options properties are defined before accessing them. The following code snippet demonstrates how to check if options.data is defined before using the property in a filter expression: - jQueryindex.js- $("#treeListContainer").dxTreeList({ columns: [{ lookup: { dataSource(options) { return { store: dataStore, filter: options.data ? ['StateID', '=', options.data.StateID] : null, }; }, } }] })- Angularapp.component.htmlapp.component.ts- <dx-tree-list ... > <dxi-column ... > <dxo-lookup [dataSource]="lookupDataSource" ... ></dxo-lookup> </dxi-column> </dx-tree-list>- export class AppComponent { lookupDataSource = (options) => { return { store: this.dataStore, filter: options.data ? ['StateID', '=', options.data.StateID] : null, } } }- VueApp.vue- <script setup> import { DxTreeList, DxColumn, DxLookup } from 'devextreme-vue/tree-list'; const lookupDataSource = (options) => { return { store: dataStore, filter: options.data ? ['StateID', '=', options.data.StateID] : null, } } </script> <template> <DxTreeList ... > <DxColumn ... > <DxLookup :dataSource="lookupDataSource" ... /> </DxColumn> </DxTreeList> </template>- ReactApp.js- import { TreeList } from 'devextreme-react'; import { Column, Lookup } from 'devextreme-react/tree-list'; function App() { function lookupDataSource(options) { return { store: dataStore, filter: options.data ? ['StateID', '=', options.data.StateID] : null, } } return ( <> <TreeList ... > <Column ... > <Lookup dataSource={lookupDataSource} ... /> </Column> </TreeList> </> ) }
If the lookup data source contains objects, specify the valueExpr and displayExpr properties in addition to the dataSource.
displayExpr
Specifies the data source field whose values must be displayed.
A row's data.
The displayed value.
This property accepts a string - the name of the data field that provides displayed values, or a function that returns the displayed value.
- Values in a lookup column are sorted by the valueExpr field. Implement the column's calculateSortValue function to sort by the displayExpr field instead: - jQueryindex.js- $(function() { $("#treeListContainer").dxTreeList({ columns: [{ // ... lookup: { // ... }, calculateSortValue: function (data) { const value = this.calculateCellValue(data); return this.lookup.calculateCellValue(value); } }] }); });- Angularapp.component.htmlapp.component.tsapp.module.ts- <dx-tree-list ... > <dxi-column ... [calculateSortValue]="calculateSortValue"> <dxo-lookup ... ></dxo-lookup> </dxi-column> </dx-tree-list>- import { Component } from '@angular/core'; @Component({ selector: 'app-root', templateUrl: './app.component.html', styleUrls: ['./app.component.css'] }) export class AppComponent { calculateSortValue (data) { const column = this as any; const value = column.calculateCellValue(data); return column.lookup.calculateCellValue(value); } }- import { BrowserModule } from '@angular/platform-browser'; import { NgModule } from '@angular/core'; import { AppComponent } from './app.component'; import { DxTreeListModule } from 'devextreme-angular'; @NgModule({ declarations: [ AppComponent ], imports: [ BrowserModule, DxTreeListModule ], providers: [ ], bootstrap: [AppComponent] }) export class AppModule { }- VueApp.vue- <template> <DxTreeList ... > <DxColumn ... :calculate-sort-value="calculateSortValue"> <DxLookup ... /> </DxColumn> </DxTreeList> </template> <script setup> import 'devextreme/dist/css/dx.light.css'; import DxTreeList, { DxColumn, DxLookup } from 'devextreme-vue/tree-list'; const calculateSortValue = (data) => { const column = this; const value = column.calculateCellValue(data); return column.lookup.calculateCellValue(value); } </script>- ReactApp.js- import React from 'react'; import 'devextreme/dist/css/dx.light.css'; import TreeList, { Column, Lookup } from 'devextreme-react/tree-list'; function calculateSortValue (data) { const column = this; const value = column.calculateCellValue(data); return column.lookup.calculateCellValue(value); } export default function App() { return ( <TreeList ... > <Column ... calculateSortValue={calculateSortValue}> <Lookup ... /> </Column> </TreeList> ); }
- When headerFilter and columns.lookup are both enabled, ensure data is not - nullbefore accessing its properties in the displayExpr method of lookup columns:JavaScript- displayExpr: (data) => { return data ? `${data.ID} ${data.Name}` : '' }- The - nullvalue adds a "(Blanks)" item to column header filters. To avoid displaying this "(Blanks)" item, implement DataSource.postProcess in the columns.headerFilter.dataSource method of lookup columns:- jQueryindex.js- $(function() { $("#treeListContainer").dxTreeList({ columns: [{ lookup: { // ... }, headerFilter: { dataSource: function (options) { options.dataSource.postProcess = function (results) { return results.filter((item) => item.value !== null) } } } }] }) })- Angularapp.component.htmlapp.component.ts- <dx-tree-list ... > <dxo-header-filter [visible]="true"></dxo-header-filter> <dxi-column ... > <dxo-header-filter ... [dataSource]="calculateDataSource" ></dxo-header-filter> <dxo-lookup ... ></dxo-lookup> </dxi-column> </dx-tree-list>- export class AppComponent { calculateDataSource (options) { options.dataSource.postProcess = function (results) { return results.filter((item) => item.value !== null) } } }- VueApp.vue- <template> <DxTreeList ... > <DxHeaderFilter :visible="true" /> <DxColumn ... > <DxHeaderFilter ... :data-source="calculateDataSource" /> <DxLookup ... /> </DxColumn> </DxTreeList> </template> <script setup> import DxTreeList, { DxColumn, DxLookup, DxHeaderFilter } from 'devextreme-vue/tree-list'; const calculateDataSource = (options) => { options.dataSource.postProcess = function (results) { return results.filter((item) => item.value !== null) } } </script>- ReactApp.js- import TreeList, { Column, Lookup, HeaderFilter } from 'devextreme-react/tree-list'; function calculateDataSource(options) { options.dataSource.postProcess = function (results) { return results.filter((item) => item.value !== null) } } export default function App() { return ( <TreeList ... > <HeaderFilter visible={true} /> <Column ... <HeaderFilter ... dataSource={calculateDataSource} /> <Lookup ... /> </Column> </TreeList> ); }
valueExpr
Specifies the data field whose values should be replaced with values from the displayExpr field.
Values from this data field must have the same type as the column's dataField values.
If you have technical questions, please create a support ticket in the DevExpress Support Center.