- [DEVEXPRESS] Hỗ trợ tìm kiếm highlight không dấu và không khoảng cách trên Gridview Filter
- [C#] Chia sẻ source code phần mềm Image Downloader tải hàng loạt hình ảnh từ danh sách link url
- [C#] Chụp hình và quay video từ camera trên winform
- [C#] Chia sẽ full source code tách file Pdf thành nhiều file với các tùy chọn
- Giới thiệu về Stock Tracker Widget - Công cụ theo dõi cổ phiếu và cảnh báo giá tăng giảm bằng C# và WPF
- [VB.NET] Chia sẻ công cụ nhập số tiền tự động định dạng tiền tệ Việt Nam
- [VB.NET] Hướng dẫn fill dữ liệu từ winform vào Microsoft word
- [VB.NET] Hướng dẫn chọn nhiều dòng trên Datagridview
- Hướng Dẫn Đăng Nhập Nhiều Tài Khoản Zalo Trên Máy Tính Cực Kỳ Đơn Giản
- [C#] Chia sẻ source code phần mềm đếm số trang tập tin file PDF
- [C#] Cách Sử Dụng DeviceId trong C# Để Tạo Khóa Cho Ứng Dụng
- [SQLSERVER] Loại bỏ Restricted User trên database MSSQL
- [C#] Hướng dẫn tạo mã QRcode Style trên winform
- [C#] Hướng dẫn sử dụng temp mail service api trên winform
- [C#] Hướng dẫn tạo mã thanh toán VietQR Pay không sử dụng API trên winform
- [C#] Hướng Dẫn Tạo Windows Service Đơn Giản Bằng Topshelf
- [C#] Chia sẻ source code đọc dữ liệu từ Google Sheet trên winform
- [C#] Chia sẻ source code tạo mã QR MOMO đa năng Winform
- [C#] Chia sẻ source code phần mềm lên lịch tự động chạy ứng dụng Scheduler Task Winform
- [C#] Hướng dẫn download file từ Minio Server Winform
[DEVEXPRESS] Thừa kế control SearchLookupEdit thành MultiSelectGridLookupEdit
Xin chào các bạn, bài viết hôm trước mình đã có hướng dẫn các bạn cách chọn nhiều dòng dữ liệu trên GridLookupEdit, trong bài này mình sẽ tạo control mới thừa kế từ control SearchLookupEdit thành MultiSelectGridLookupEdit.
[DEVEXPRESS] Custom Control SearchLookupEdit to MultiSelectSearchLookupEdit
Trong các công cụ mặc định của Devexpress, nếu các bạn muốn chọn nhiều dòng dữ liệu, các bạn sẽ sử dụng control CheckedComboboxEdit.
Tuy nhiên khi bạn sử dụng công cụ này, thì các bạn không thể hiển thị nhiều cột dữ liệu, không thể filter lọc và tìm kiếm.
Nên bài viết này, mình xây dựng sẵn công cụ để sử dụng cho công việc hay chọn nhiều dòng dữ liệu một cách dễ dàng và nhanh chóng.

Ở công cụ này, các bạn muốn lấy dữ liệu sẽ sử dụng Event MultiValueChanged và giá trị trả về là VALUE thay vì EditValue như GridLookupEdit.
Ngoài các bạn, có thể chỉnh sửa thuộc tính icon cho checkbox chọn nhiều dòng ở properties: MulticheckboxStyle, ở ví dụ trên mình sử dụng hình trái tim.


Và tiếp theo, là khi các bạn chọn nhiều dữ liệu, displaytext sẽ không hiển thị hết các items các bạn đã chọn.
Các bạn có thể chỉnh sửa 2 thuộc tính sau để nó trả về hiển thị là bao nhiêu items vừa được chọn qua 2 thuộc tính:
ShowItemCountMulti = true và TextItemsOutofLabel {0} là số dòng chọn nó hiển thị.

Source code custom control MultiSearchLookupEdit Devexpress c#:
using DevExpress.Data;
using DevExpress.Data.Filtering.Helpers;
using DevExpress.Images;
using DevExpress.Utils;
using DevExpress.Utils.Animation;
using DevExpress.Utils.Extensions;
using DevExpress.Utils.Win;
using DevExpress.XtraEditors;
using DevExpress.XtraEditors.Controls;
using DevExpress.XtraEditors.Drawing;
using DevExpress.XtraEditors.Registrator;
using DevExpress.XtraEditors.Repository;
using DevExpress.XtraEditors.ViewInfo;
using DevExpress.XtraGrid.Columns;
using DevExpress.XtraGrid.Localization;
using DevExpress.XtraGrid.Views.Grid;
using DevExpress.XtraLayout;
using DevExpress.XtraPrinting.Export.Pdf;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Data.SqlTypes;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace multiselectcontrol
{
    [UserRepositoryItem("RegisterMyGridLookUpEdit")]
    public class RepositoryItemMyGridLookUpEdit : RepositoryItemSearchLookUpEdit
    {
        private RepositoryItemCheckEdit repositoryItemCheckEdit;
        public CheckBoxStyle MulticheckBoxStyle { set; get; } = CheckBoxStyle.Default;
        static RepositoryItemMyGridLookUpEdit() { RegisterMyGridLookUpEdit(); }
        public RepositoryItemMyGridLookUpEdit()
        {
        }
        protected override void OnLoaded()
        {
            base.OnLoaded();
            if (IsDesignMode) return;
            GridView gView = this.View;
            gView.OptionsSelection.MultiSelect = true;
            gView.OptionsSelection.MultiSelectMode = GridMultiSelectMode.CheckBoxRowSelect;
            gView.OptionsSelection.CheckBoxSelectorColumnWidth = 40;
            gView.OptionsView.ShowAutoFilterRow = true;
            gView.OptionsBehavior.Editable = false;
            gView.SelectionChanged += GView_SelectionChanged;
            gView.RowCellClick += GView_RowCellClick;
            if (MulticheckBoxStyle != CheckBoxStyle.Default)
            {
                repositoryItemCheckEdit = new RepositoryItemCheckEdit();
                repositoryItemCheckEdit.CheckBoxOptions.Style = MulticheckBoxStyle;
                gView.CustomDrawColumnHeader += GView_CustomDrawColumnHeader;
            }
        }
      
        public void GView_CustomDrawColumnHeader(object sender, ColumnHeaderCustomDrawEventArgs e)
        {
            if (e.Column?.FieldName == "DX$CheckboxSelectorColumn")
            {
                e.Column.ColumnEdit = repositoryItemCheckEdit;
            }
        }
        private void GView_RowCellClick(object sender, RowCellClickEventArgs e)
        {
            GridView view = sender as GridView;
            int i = view.FocusedRowHandle;
            if (view.Columns.Contains(view.FocusedColumn))
            {
                if (view.IsRowSelected(i))
                {
                    view.UnselectRow(i);
                }
                else
                {
                    view.SelectRow(i);
                }
            }
        }
        private void GView_SelectionChanged(object sender, DevExpress.Data.SelectionChangedEventArgs e)
        {
            GridView view = sender as GridView;
            if (view != null)
                view.GetSelectedRows();
        }
        public const string MyGridLookUpEditName = "MultiSelectSearchGridLookUpEdit";
        public override string EditorTypeName { get { return MyGridLookUpEditName; } }
        public static void RegisterMyGridLookUpEdit()
        {
            Image img = null;
            EditorRegistrationInfo.Default.Editors.Add(new EditorClassInfo(MyGridLookUpEditName,
              typeof(MultiSelectLookUpEdit), typeof(RepositoryItemMyGridLookUpEdit),
              typeof(GridLookUpEditBaseViewInfo), new ButtonEditPainter(), true, img));
        }
        public override void Assign(RepositoryItem item)
        {
            BeginUpdate();
            try
            {
                base.Assign(item);
                RepositoryItemMyGridLookUpEdit source =
                    item as RepositoryItemMyGridLookUpEdit;
                if (source == null) return;
            }
            finally
            {
                EndUpdate();
            }
        }
        private void InitializeComponent()
        {
            ((System.ComponentModel.ISupportInitialize)(this)).BeginInit();
            ((System.ComponentModel.ISupportInitialize)(this)).EndInit();
        }
        private IContainer components;
    }
    public class MultiSelectLookUpEdit : SearchLookUpEdit
    {
        public int[] _selectedRows;
        public bool ShowItemCountMulti { set; get; } = false;
        public string TextItemsOutofLabel { set; get; } = "Bạn đã chọn {0}.";
        public string VALUE = "";
        public int[] selectedRows
        {
            set
            {
                this._selectedRows = value;
                if (last_selectedRows == null && _selectedRows == null) { return; }
                if (last_selectedRows == null && _selectedRows != null)
                {
                    this.OnMultiValueChanged(null);
                    return;
                }
                var isDifference = selectedRows.Except(last_selectedRows).Count() != 0;
                if (isDifference)
                {
                    this.OnMultiValueChanged(null);
                    return;
                }
                if (last_selectedRows.Count() != _selectedRows.Count())
                {
                    this.OnMultiValueChanged(null);
                }
            }
            get
            {
                return this._selectedRows;
            }
        }
        public int[] last_selectedRows = null;
        [Category("Action")]
        [Description("Trigger iá trị dữ liệu thay đổi")]
        public event EventHandler MultiValueChanged;
        protected virtual void OnMultiValueChanged(EventArgs e)
        {
            if (MultiValueChanged != null)
                MultiValueChanged(this, e);
        }
        static MultiSelectLookUpEdit()
        {
            RepositoryItemMyGridLookUpEdit.RegisterMyGridLookUpEdit();
        }
        public SvgImageCollection svgImageCollection1;
        public MultiSelectLookUpEdit()
        {
            this.CloseUp += MultiSelectLookUpEdit_CloseUp;
            this.EditValueChanged += MultiSelectLookUpEdit_EditValueChanged;
            this.QueryCloseUp += MultiSelectLookUpEdit_QueryCloseUp;
            this.Popup += MultiSelectLookUpEdit_Popup;
            this.Properties.NullText = "";
            svgImageCollection1 = new SvgImageCollection();
            this.svgImageCollection1.Add("actions_checkcircled", "image://svgimages/icon builder/actions_checkcircled.svg");
            this.svgImageCollection1.Add("actions_zoom", "image://svgimages/icon builder/actions_zoom.svg");
        }
        private void MultiSelectLookUpEdit_Popup(object sender, EventArgs e)
        {
            IPopupControl popupControl = sender as IPopupControl;
            LayoutControl layoutControl = popupControl.PopupWindow.Controls[3].Controls[0] as LayoutControl;
            SimpleButton clearButton = ((LayoutControlItem)layoutControl.Items.FindByName("lciClear")).Control as SimpleButton;
            SimpleButton find = ((LayoutControlItem)layoutControl.Items.FindByName("lciButtonFind")).Control as SimpleButton;
            clearButton.Text = "Đồng ý";
            find.Text = "Tìm kiếm";
            clearButton.ImageOptions.SvgImage = svgImageCollection1[0];
            clearButton.ImageOptions.SvgImageSize = new Size(16, 16);
            find.ImageOptions.SvgImage = svgImageCollection1[1];
            find.ImageOptions.SvgImageSize = new Size(16, 16);
            if (clearButton != null)
            {
                clearButton.Click += ClearButton_Click;
            }
        }
        bool Allowclose = false;
        private void ClearButton_Click(object sender, EventArgs e)
        {
            if (this.IsPopupOpen)
            {
                Allowclose = true;
                this.ClosePopup();
            }
        }
        private void MultiSelectLookUpEdit_QueryCloseUp(object sender, CancelEventArgs e)
        {
            if (!Allowclose)
            {
                e.Cancel = true;
            }
        }
        private void MultiSelectLookUpEdit_EditValueChanged(object sender, EventArgs e)
        {
            this.EditValue = null;
        }
        private void MultiSelectLookUpEdit_CloseUp(object sender, DevExpress.XtraEditors.Controls.CloseUpEventArgs e)
        {
            var repositoryItemMyGridLookUpEdit = this.Properties.View;
            var selected_row = (((DevExpress.XtraEditors.GridLookUpEditBase)(sender)).Properties).View.GetSelectedRows();
            var editValue = string.Join(", ", selected_row.Select(x => repositoryItemMyGridLookUpEdit.GetDataRow(x)[this.Properties.ValueMember].ToString()));
            this.VALUE = editValue;
            selectedRows = selected_row;
            var displayText = string.Join(", ", selectedRows.Select(x => repositoryItemMyGridLookUpEdit.GetDataRow(x)[this.Properties.DisplayMember].ToString()));
            this.Properties.NullText = displayText;
            var width_text = TextRenderer.MeasureText(this.Text, new Font(this.Font.FontFamily, this.Font.Size, this.Font.Style)).Width + 20;
            var width_lookupedit = this.Width;
            if (width_text > width_lookupedit && ShowItemCountMulti)
            {
                if (string.IsNullOrEmpty(TextItemsOutofLabel))
                {
                    TextItemsOutofLabel = "({0}) items.";
                }
                var itemSelected = editValue.Split(',').Length;
                this.Properties.NullText = string.Format(this.TextItemsOutofLabel, itemSelected);
            }
            this.EditValue = null;
            last_selectedRows = (((DevExpress.XtraEditors.GridLookUpEditBase)(sender)).Properties).View.GetSelectedRows();
            Allowclose = false;
        }
        public override string EditorTypeName
        {
            get
            {
                return
                        RepositoryItemMyGridLookUpEdit.MyGridLookUpEditName;
            }
        }
        [DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
        public new RepositoryItemMyGridLookUpEdit Properties
        {
            get { return base.Properties as RepositoryItemMyGridLookUpEdit; }
        }
    }
}
Và source code sử dụng control ở FrmMain:
using DevExpress.Utils.Win;
using DevExpress.XtraEditors;
using DevExpress.XtraEditors.Controls;
using DevExpress.XtraEditors.Repository;
using DevExpress.XtraEditors.ViewInfo;
using DevExpress.XtraGrid.Views.Grid;
using DevExpress.XtraGrid.Views.Grid.ViewInfo;
using DevExpress.XtraLayout;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Data.SqlTypes;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
namespace multiselectcontrol
{
    public partial class Form1 : DevExpress.XtraEditors.XtraForm
    {
        public Form1()
        {
            InitializeComponent();
        }
        private void Form1_Load(object sender, EventArgs e)
        {       
            var data = new DataTable();
            data.Columns.Add("id", typeof(string));
            data.Columns.Add("name", typeof(string));
            data.Columns.Add("address", typeof(string));
            data.Rows.Add("MASV001", "Nguyễn Thảo", "Vũng Tàu");
            data.Rows.Add("MASV002", "Nguyễn Đình Tuyên", "Quảng Bình");
            data.Rows.Add("MASV003", "Nguyễn Phương Nhi", "Kiên Giang");
            data.Rows.Add("MASV004", "Nguyễn Thị Cẩm Tú", "TP.HCM");
            data.Rows.Add("MASV005", "Cái Trí Minh", "Đồng Nai");
            data.Rows.Add("MASV006", "Võ Sơn Băng", "Vĩnh Long");
            multiSelectLookUpEdit1.Properties.DataSource = data;
            multiSelectLookUpEdit1.Properties.ValueMember = "id";
            multiSelectLookUpEdit1.Properties.DisplayMember = "name";         
        }
   
        private void multiSelectLookUpEdit1_MultiValueChanged(object sender, EventArgs e)
        {
            lbl_result.Text = multiSelectLookUpEdit1.VALUE;
        }
     
    }
}
Thanks for watching!

![[DEVEXPRESS] Thừa kế control SearchLookupEdit thành MultiSelectGridLookupEdit](https://laptrinhvb.net/uploads/users/9a8cb514e4428e85fb4ca07588e9103f.png)

![[DEVEXPRESS] AudioGram là gì?  Hướng dẫn vẽ biểu đồ Line AudioGram](https://laptrinhvb.net/uploads/source/image_baiviet/8284c20b891598a9a2792b6c752af763.png)
![[DEVEXPRESS] Tích hợp chức năng Tìm kiếm Search vào CheckedComboboxEdit](https://laptrinhvb.net/uploads/source/new_image_baiviet/search_checkbox_edit.png)
![[DEVEXPRESS] Sử dụng RealTimeSource để đọc dữ liệu theo thời gian thực vào Gridview C#](https://laptrinhvb.net/uploads/source/devexpress/live_data.png)
![[DEVEXPRESS] Hướng dẫn sử dụng HTML Text trong Label, XtraMessageBox](https://laptrinhvb.net/uploads/source/devexpress/XtraMessageBox_HTML_thumb.png)

![[DEVEXPRESS] Hướng dẫn sử dụng layout TileView trong SearchLookupEdit winform](https://laptrinhvb.net/uploads/source/csharp/title_view_searchview_thumb.png)
![[DEVEXPRESS] Hướng dẫn Move up và Move down giữa các dòng trong gridview](https://laptrinhvb.net/uploads/source/image_baiviet/8045cb06b01a6d465c24c92723612857.gif)
![[DEVEXPRESS] Hướng dẫn sử dụng WinExplorer View trong GridView để hiển thị hình ảnh](https://laptrinhvb.net/uploads/source/devexpress/winexplorer_devexpress_thumb.png)
![[DEVEXPRESS] Hướng dẫn tích hợp Popup Menu vào Gallery Control C#](https://laptrinhvb.net/uploads/source/devexpress/popup_menu_thumb.jpg)
![[DEVEXPRESS] Hướng dẫn switch tab trong Fluent Design Form](https://laptrinhvb.net/uploads/source/devexpress/switch_tab_csharp_thumb.png)
![[DEVEXPRESS] Hướng dẫn tạo intro giới thiệu phần mềm sử dụng Guide Adorner trong C#](https://laptrinhvb.net/uploads/source/image_baiviet/20cb084f1b958570fbbc59295e80cd8f.jpg)
![[DEVEXPRESS] Thiết lập mặc định chế độ lọc contain string trên Gridview C#](https://laptrinhvb.net/uploads/source/devexpress/contain_filter_devexpress.png)
![[DEVEXPRESS] Hướng dẫn tải dữ liệu bất đồng bộ Async trên Column Unbound GridView](https://laptrinhvb.net/uploads/source/devexpress/unbound_async_gridview.gif)
![[DEVEXPRESS] Hướng dẫn sử dụng Token Edit làm Attachment File trong Send Mail C#](https://laptrinhvb.net/uploads/source/devexpress/token_attachment_file.gif)
![[DEVEXPRESS] Hướng dẫn thêm icon chênh lệch tăng giảm trên gridview](https://laptrinhvb.net/uploads/source/devexpress/Format_condition_thumb.png)
![[DEVEXPRESS] Hướng dẫn export data từ sqlserver vào file template excel có sẵn C# winform](https://laptrinhvb.net/uploads/source/devexpress/export_excel_template_thumb.png)
![[DEVEXPRESS] Hướng dẫn tạo hiệu ứng transition hiển thị cài đặt nâng cao](https://laptrinhvb.net/uploads/source/devexpress/tramsition_advance_setting.png)
![[DEVEXPRESS] Hướng dẫn sử dụng Alert Control để hiện thị thông báo](https://laptrinhvb.net/uploads/source/devexpress/alertcontrol_thumb.jpg)
![[DEVEXPRESS] Tìm dữ liệu trùng lắp Duplicate và tô màu trên Gridview](https://laptrinhvb.net/uploads/source/devexpress/find_duplicate_value_gridview.gif)

![[DEVEXPRESS] Hướng dẫn lưu và tải lại trạng thái Gridview với GridViewRefreshHelper](https://laptrinhvb.net/uploads/source/web/save_state_gridview_devexpress_thumb.jpg)
![[DEVEXPRESS] Hướng dẫn thiết kế Form Setting Tùy chọn trong Winform](https://laptrinhvb.net/uploads/source/devexpress/form_setting.jpg)

![[DEVEXPRESS] Hướng dẫn sử dụng Flyout Dialog trên lập trình C#, winform](https://laptrinhvb.net/uploads/source/devexpress/flyoutdialog_thumb.jpg)
![[DEVEXPRESS] Thiết kế Dropdown ButtonBarItem trên Form Ribbon](https://laptrinhvb.net/uploads/source/vbnet/dropdown_button_devexpress_thumb.png)
