Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# - Windows Forms의 데이터 바인딩 지원(DataBinding, DataSource)

WPF의 MVVM에 가려져 잘 알려지진 않았지만, Windows Form도 나름대로의 DataBinding 기능이 있습니다. 게다가 Control 타입 수준에서 제공하고 있기 때문에,

Control.DataBindings Property
; https://learn.microsoft.com/en-us/dotnet/api/system.windows.forms.control.databindings

WPF처럼 Windows Forms 역시 내부 구조 자체에서 MVVM과 유사한 기능을 이미 구현하고 있는 것입니다. 실제로 간단하게 테스트를 해볼까요? 다음과 같이 타입을 하나 만들고,

public class MyTitle
{
    public string Title { get; set; } = "test2";
}

이것을 TextBox를 하나 담고 있는 Windows Forms에서 다음과 같은 식으로 바인딩할 수 있습니다.

public partial class Form1 : Form
{
    MyTitle title = new MyTitle();

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        var myTitleBinding = new BindingSource();
        myTitleBinding.DataSource = title;

        textBox1.DataBindings.Add(new Binding("Text", myTitleBinding, "Title", true, DataSourceUpdateMode.Never));
    }
}

보시면, (TextBox 타입인) "textBox1"의 DataBindings에 MyTitle 개체를 DataSource로 담고 있는 바인딩을 연결하고 있습니다. 따라서 위의 프로그램을 실행하면 textBox1은 실행 시에 "title" 인스턴스의 "Title" 속성으로부터 값을 가져와 textBox1.Text 속성에 자동으로 값을 설정합니다. 결국 화면에는 텍스트 상자에 "test2"라는 글자가 보이게 됩니다.

당연히, title.Title 속성 값이 변경되면 자동으로 textBox1에서 그 값을 가져오는 기능도 있습니다. 이를 위해서는 값이 바뀌는 것을 인지하기 위해 MyTitle 스스로 값이 바뀌었음을 알려야 합니다.

public class MyTitle : INotifyPropertyChanged
{
    string title = "test2";

    public string Title
    {
        get { return title; }
        set
        {
            if (title == value)
            {
                return;
            }

            title = value;
            var propArg = new PropertyChangedEventArgs(nameof(Title));
            PropertyChanged?.Invoke(this, propArg);
        }
    }

    public event PropertyChangedEventHandler? PropertyChanged;
}

그리고, DataBindings의 Binding에서도 OnPropertyChanged 상태에서 값을 업데이트하겠다는 설정을 하면,

textBox1.DataBindings.Add(new Binding("Text", myTitleBinding, "Title", true, DataSourceUpdateMode.OnPropertyChanged));

이후 MyTitle 인스턴스의 값이 바뀔 때마다

private void button1_Click(object sender, EventArgs e)
{
    this.title.Title = DateTime.Now.ToString(); // 자동으로 textBox1.Text의 값도 바뀜
}

TextBox의 글자가 함께 바뀌게 됩니다. 사실상 기능면으로 보면 WPF와 다를 바 없고, 단지 WPF가 XML로 데이터 바인딩을 지정하는 기능이 더 있다는 정도가 되겠습니다.




DataGridView도, Control로부터 상속받았기 때문에 DataBindings 속성을 그대로 가지고 있습니다. 하지만, 이것 외에도 데이터를 지정할 수 있는 방법을 몇 개 더 제공하고 있습니다.

우선, DataGridView 스스로 칼럼을 지정하고 값을 설정할 수 있습니다.

private void Form1_Load(object sender, EventArgs e)
{
    this.dataGridView1.Columns.Add(new DataGridViewColumn(new DataGridViewTextBoxCell()) { HeaderText = "idx"});
    this.dataGridView1.Columns.Add(new DataGridViewColumn(new DataGridViewTextBoxCell()) { HeaderText = "value" });

    this.dataGridView1.Rows.Add(1, "test1");
    this.dataGridView1.Rows.Add(2, "test2");
}

winform_databindings_1.png

이런 경우, 값을 변경하는 것도 단순히 Rows 속성을 통하면 됩니다.

private void button1_Click(object sender, EventArgs e)
{
    this.dataGridView1.Rows.Add("3", "test3"); // 값을 추가하고,
    this.dataGridView1.Rows.RemoveAt(0); // 값을 삭제하고.
}

또 다른 방법으로는, DataGridView에서 직접 제공하는 DataSource 속성을 이용해,

DataGridView.DataSource Property
; https://learn.microsoft.com/en-us/dotnet/api/system.windows.forms.datagridview.datasource

사용자가 만든 개체를,

public class MyData
{
    public int Age { get; set; }
    public string Name { get; set; } = "";
}

직접 목록으로 전달할 수 있습니다.

MyData m1 = new MyData { Age = 30, Name = "Kevin" };
MyData m2 = new MyData { Age = 31, Name = "Winnie" };

List data = new List { m1, m2 };
// 또는, MyData[] data = new MyData[] { m1, m2 };

this.dataGridView1.DataSource = data;

그럼, GridDataView는 해당 타입을 Reflection으로 접근해 읽기 가능한 공용 (필드가 아닌) 속성을 열거해 자동으로 칼럼을 구성해 값을 보여줍니다.

winform_databindings_2.png

그런데, 이런 경우에는 후에 data 목록의 값을 변경해도 GridDataView는 그 변화를 인지하지 못합니다.

private void button1_Click(object sender, EventArgs e)
{
    List<MyData>? data = this.dataGridView1.DataSource as List<MyData>;
    data?.Add(new MyData { Age = 32, Name = "Cooper" }); // 값을 추가해도 DataGridView에는 변화 없음!
}

당연하겠죠? 이런 경우 이 글의 처음 예제에서 INotifyPropertyChanged를 구현했던 것처럼, 목록 역시 그 변화를 알리는 IBindingList.ListChanged 이벤트를 제공해야 합니다. 이를 위해 직접 IBindingList를 구현한 사용자 정의 목록 타입을 만들어도 되지만, 마이크로소프트는 이를 위한 목적으로 이미 BindingList 타입을 제공하고 있으니,

BindingList
; https://learn.microsoft.com/en-us/dotnet/api/system.componentmodel.bindinglist-1

이것을 이용해 구현하시면 됩니다.

private void Form1_Load(object sender, EventArgs e)
{
    MyData m1 = new MyData { Age = 30, Name = "Kevin" };
    MyData m2 = new MyData { Age = 31, Name = "Winnie" };

    BindingList<MyData> data = new BindingList<MyData>() { m1, m2 };
    this.dataGridView1.DataSource = data;
}

private void button1_Click(object sender, EventArgs e)
{
    BindingList<MyData>? data = this.dataGridView1.DataSource as BindingList<MyData>;
    data?.Add(new MyData { Age = 32, Name = "Cooper" }); // 추가된 값이 DataGridView에도 반영됨
}

혹은, Control.DataBindings가 그랬던 것처럼 BindingSource를 이용해 경유하는 것도 가능합니다.

private void Form1_Load(object sender, EventArgs e)
{
    MyData m1 = new MyData { Age = 30, Name = "Kevin" };
    MyData m2 = new MyData { Age = 31, Name = "Winnie" };

    BindingList<MyData> data = new BindingList<MyData>() { m1, m2 };

    var dataBindingSource = new BindingSource();
    dataBindingSource.DataSource = data;

    this.dataGridView1.DataSource = dataBindingSource;
}

private void button1_Click(object sender, EventArgs e)
{
    BindingSource? src = this.dataGridView1.DataSource as BindingSource;
    if (src == null)
    {
        return;
    }

    BindingList<MyData>? data = src.DataSource as BindingList<MyData>;
    data?.Add(new MyData { Age = 32, Name = "Cooper" });
}




마이크로소프트는 범용 데이터 컨테이너인 DataTable/DataSet에 대한 연동도 빼놓지 않고 있습니다.

DataTable _dt = new DataTable();

private void Form1_Load(object sender, EventArgs e)
{
    _dt.Columns.Add("idx");
    _dt.Columns.Add("value");
    _dt.Rows.Add(1, "test1");
    _dt.Rows.Add(2, "test2");

    this.dataGridView1.DataSource = _dt;
}

private void button1_Click(object sender, EventArgs e)
{
    _dt.Rows.Add(3, "test3"); // 추가된 값이 DataGridView에 즉각 반영
}

여기서 재미있는 것은, DataTable의 경우 BindingList와는 달리 IBindingList 인터페이스를 구현한 개체가 아니라는 점입니다. 그래도 저렇게 (button1_Click에서) 데이터를 추가해도 DataGridView에 즉각 반영되는 것은, IListSource를 구현하면서 그것의 GetList 메서드에서 IBindingList를 구현한 DataView 타입으로 감싼 타입을 활용하기 때문입니다.

마지막으로, 다중 테이블을 담고 있는 DataSet의 경우에는 위의 코드에서 DataTable을 구분 지을 수 있는 값을 DataMember에 지정하면 됩니다.

DataSet _ds = new DataSet();

private void Form1_Load(object sender, EventArgs e)
{
    DataTable dt1 = new DataTable("List");
    dt1.Columns.Add("idx");
    dt1.Columns.Add("value");
    dt1.Rows.Add(1, "test1");
    dt1.Rows.Add(2, "test2");

    DataTable dt2 = new DataTable("Person");
    dt2.Columns.Add("age");
    dt2.Columns.Add("name");
    dt2.Rows.Add(30, "Kevin");
    dt2.Rows.Add(31, "Winnie");

    _ds = new DataSet();
    _ds.Tables.Add(dt1);
    _ds.Tables.Add(dt2);

    var dataBindingSource = new BindingSource();
    dataBindingSource.DataSource = _ds;
    dataBindingSource.DataMember = "Person";

    this.dataGridView1.DataSource = dataBindingSource;
}

private void button1_Click(object sender, EventArgs e)
{
    _ds.Tables[1].Rows.Add(32, "Cooper");
}

이 정도면, 대충 설명이 되었겠죠? ^^

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 10/3/2022]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2023-01-17 11시39분
[성태형 사랑해요] 성태형님 친절한 설명 감사합니다!
[guest]
2023-01-18 09시11분
[gwise] DataGridView에 List받아와서 DT로 변환해서 사용하고 있는데 BindingList이걸 테스트 해 봐야 겠습니다.
그래도 DataTable를 사용하는 이유는 UI에서 RowState를 구분해서 Add/Modi/Delete 별로 데이터를 처리 하기 때문에 어쩔수(?)없이 DataTable를
사용하는데 BindingList에도 row단위로 RowState같은게 있는지 찾아 보겠습니다.
만약 1000 row에서 사용자가 수정한게 10 row면 10개만 필터로 걸러서 서버로 보냅니다. 이게 안되면 전체 데이터를 다 서버로 보내야 해서...
[guest]

... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12123정성태1/23/202012242웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/20209178.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/20209768VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202011195.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202011230디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202011892개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202010894디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202011395디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202011112디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202013006디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202013603오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/202010139오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202013448디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
12110정성태1/11/202012031디버깅 기술: 154. Patch Guard로 인해 블루 스크린(BSOD)가 발생하는 사례 [5]파일 다운로드1
12109정성태1/10/20209930오류 유형: 588. Driver 프로젝트 빌드 오류 - Inf2Cat error -2: "Inf2Cat, signability test failed."
12108정성태1/10/20209996오류 유형: 587. Kernel Driver 시작 시 127(The specified procedure could not be found.) 오류 메시지 발생
12107정성태1/10/202010924.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
12106정성태1/8/202012434VC++: 136. C++ - OSR Driver Loader와 같은 Legacy 커널 드라이버 설치 프로그램 제작 [1]
12105정성태1/8/202011018디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
12104정성태1/7/202011690DDK: 9. 커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램 [4]
12103정성태1/7/202014458DDK: 8. Visual Studio 2019 + WDK Legacy Driver 제작- Hello World 예제 [1]파일 다운로드2
12102정성태1/6/202012012디버깅 기술: 152. User 권한(Ring 3)의 프로그램에서 _ETHREAD 주소(및 커널 메모리를 읽을 수 있다면 _EPROCESS 주소) 구하는 방법
12101정성태1/5/202011353.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
12100정성태1/3/20209380.NET Framework: 875. .NET 3.5 이하에서 IntPtr.Add 사용
12099정성태1/3/202011690디버깅 기술: 151. Windows 10 - Process Explorer로 확인한 Handle 정보를 windbg에서 조회 [1]
12098정성태1/2/202011276.NET Framework: 874. C# - 커널 구조체의 Offset 값을 하드 코딩하지 않고 사용하는 방법 [3]
... [61]  62  63  64  65  66  67  68  69  70  71  72  73  74  75  ...