Walkthrough / Virtualization in FlexGrid with OData
Virtualization in FlexGrid with OData

In this walkthrough, you will learn how to integrate the DataCollection and DataConnector service components to use data virtualization with an OData service and bind the same to a data-aware control such as FlexGrid for WinForms. DataConnector is a cross-platform, data connectivity library for connecting to popular data sources such as OData and Dynamics 365.

flowchart for odata and flexgrid

Follow the steps below to apply Data Virtualization in FlexGrid for WinForms with OData:

Set up Application

  1. Create a WinForms project.
  2. Drag and drop the FlexGrid control from the toolbox onto the form.
  3. Install the following NuGet packages:
    • C1.AdoNet.OData
    • C1.DataCollection.AdoNet
    • C1.DataCollection.BindingList
  4. To add a NuGet package, right-click the ‘References’ node in the ‘Solution Explorer’ window and select ‘Manage NuGet Packages’.

Create Custom Class

  1. Create a class 'Order' to provide data for binding to the FlexGrid control:
    public class Order : INotifyPropertyChanged, IEditableObject
    {
        // Fields
        private int _orderId, _employeeId, _shipVia;
        private string _customerId, _shipName, _shipAddress, _shipCity,
            _shipRegion, _shipPostalCode, _shipCountry;
        private DateTimeOffset _orderDate, _requiredDate, _shippedDate;
        private decimal _freight;
       
    
        // Properties
    
        public int OrderID
        {
            get => _orderId;
            set
            {
                if (_orderId != value)
                {
                    _orderId = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string CustomerID
        {
            get => _customerId;
            set
            {
                if (_customerId != value)
                {
                    _customerId = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public int EmployeeID
        {
            get => _employeeId;
            set
            {
                if (_employeeId != value)
                {
                    _employeeId = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public DateTimeOffset OrderDate
        {
            get => _orderDate;
            set
            {
                if (_orderDate != value)
                {
                    _orderDate = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public DateTimeOffset RequiredDate
        {
            get => _requiredDate;
            set
            {
                if (_requiredDate != value)
                {
                    _requiredDate = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public DateTimeOffset ShippedDate
        {
            get => _shippedDate;
            set
            {
                if (_shippedDate != value)
                {
                    _shippedDate = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public int ShipVia
        {
            get => _shipVia;
            set
            {
                if (_shipVia != value)
                {
                    _shipVia = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public decimal Freight
        {
            get => _freight;
            set
            {
                if (_freight != value)
                {
                    _freight = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipName
        {
            get => _shipName;
            set
            {
                if (_shipName != value)
                {
                    _shipName = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipAddress
        {
            get => _shipAddress;
            set
            {
                if (_shipAddress != value)
                {
                    _shipAddress = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipCity
        {
            get => _shipCity;
            set
            {
                if (_shipCity != value)
                {
                    _shipCity = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipRegion
        {
            get => _shipRegion;
            set
            {
                if (_shipRegion != value)
                {
                    _shipRegion = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipPostalCode
        {
            get => _shipPostalCode;
            set
            {
                if (_shipPostalCode != value)
                {
                    _shipPostalCode = value;
                    OnPropertyChanged();
                }
            }
        }
    
        public string ShipCountry
        {
            get => _shipCountry;
            set
            {
                if (_shipCountry != value)
                {
                    _shipCountry = value;
                    OnPropertyChanged();
                }
            }
        }
    
        
    
        // INotifyPropertyChanged Members
    
        public event PropertyChangedEventHandler PropertyChanged;
    
        private void OnPropertyChanged([CallerMemberName] string propertyName = "")
        {
            OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
        }
    
        protected void OnPropertyChanged(PropertyChangedEventArgs e)
        {
            PropertyChanged?.Invoke(this, e);
    
        }
    
        // IEditableObject Members
    
        private Order _clone;
    
        public void BeginEdit()
        {
            _clone = (Order)MemberwiseClone();
        }
    
        public void CancelEdit()
        {
            if (_clone != null)
            {
                foreach (var p in GetType().GetRuntimeProperties())
                {
                    if (p.CanRead && p.CanWrite)
                    {
                        p.SetValue(this, p.GetValue(_clone, null), null);
                    }
                }
            }
        }
    
        public void EndEdit()
        {
            _clone = null;
        }
       
    }
    

Bind FlexGrid to C1AdoNetVirtualDataCollection

  1. In the Form_Load event, use Northwind OData Web APIs, and fetch the records from the Orders table. We use the generic class C1AdoNetVirtualDataCollection<T> to get strongly-typed records from the data source.You can use the non-generic class C1AdoNetVirtualDataCollection, which creates an appropriate type for the records at runtime. We wrap our virtual data collection object in C1DataCollectionBindingList to bind it to an instance of FlexGrid.Since the data collection is populated on a different thread, we use the BeginInvoke method of FlexGrid to avoid cross-thread exceptions for changing the DataSource. The PageSize property determines the number of rows requested in each fetch request. Note that in the FlexGrid control, the DrawModeEnum sets whether the control should fire after the OwnerDrawCell event.
    private void Form1_Load(object sender, EventArgs e)
    {
        string connectionString = @"Url=https://services.odata.org/Experimental/Northwind/Northwind.svc/";
        var odataConnection = new C1ODataConnection(connectionString);
        var collectionView = new C1AdoNetVirtualDataCollection<Order>(odataConnection, "Orders");
        collectionView.PageSize = 100;
    
        c1FlexGrid1.BeginInvoke(new MethodInvoker(() =>
        {
            c1FlexGrid1.DataSource = new C1DataCollectionBindingList(collectionView);
            c1FlexGrid1.AllowFiltering = true;
            c1FlexGrid1.AllowSorting = C1.Win.C1FlexGrid.AllowSortingEnum.MultiColumn;
    
        }));
    
        c1FlexGrid1.DrawMode = C1.Win.C1FlexGrid.DrawModeEnum.OwnerDraw;
        c1FlexGrid1.OwnerDrawCell += c1FlexGrid1_OwnerDrawCell;
    }
    
  2. Double click the OwnerDrawCell event, and add the following code:
    private void c1FlexGrid1_OwnerDrawCell(object sender, C1.Win.C1FlexGrid.OwnerDrawCellEventArgs e)
    {
        if (e.Row < c1FlexGrid1.Rows.Fixed)
            return;
    
        if (c1FlexGrid1.Cols[e.Col].DataType == typeof(DateTimeOffset))
        {
            object value = c1FlexGrid1[e.Row, e.Col];
            e.Text = (value != null) ? ((DateTimeOffset)value).DateTime.ToString("d") : string.Empty;
        }
    }
    
  3. Run the project. Now, you can see that the grid initially has 100 rows (configured using the PageSize property). And, as we scroll down, more data is fetched automatically.

    Virtual loading as the data is scrolled down