Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
VS.NET IDE: 158. C# - 디자인 타임(design-time)과 런타임(runtime)의 코드 실행 구분
; https://www.sysnet.pe.kr/2/0/12543

.NET Framework: 1065. Windows Forms - 속성 창의 디자인 설정 지원: 문자열 목록 내에서 항목을 선택하는 TypeConverter 제작
; https://www.sysnet.pe.kr/2/0/12663

오류 유형: 730. Windows Forms 디자이너 - The class Form1 can be designed, but is not the first class in the file.
; https://www.sysnet.pe.kr/2/0/12689




Windows Forms - 속성 창의 디자인 설정 지원: 문자열 목록 내에서 항목을 선택하는 TypeConverter 제작

아래와 같은 질문이 있군요.

User Control에 string array 속성 추가하는 방법
; https://www.sysnet.pe.kr/3/0/5511

Windows Forms 응용 프로그램은 Visual Studio의 Properties 창과 연동할 수 있는 다양한 디자인-타임 설정들이 존재합니다. 가령, 다음과 같이 컨트롤에 공용 속성을 정의하면,

using System.ComponentModel;
using System.Windows.Forms;

namespace WindowsFormsControlLibrary1
{
    public partial class UserControl1: UserControl
    {
        public UserControl1()
        {
            InitializeComponent();
        }

        AccountType _userType;
        [Category("MyUserControlType")]
        public AccountType UserType
        {
            get { return _userType; }
            set { _userType = value; }
        }
    }

    public enum AccountType
    {
        A,
        B,
        C
    }
}

다음과 같은 식으로 디자인 창에서 해당 컨트롤의 속성을 Properties 창으로 편집하는 것이 가능합니다.

winform_design_time_attr_1.png

(참고로, Category를 빼면 "Misc" 범주로 포함이 됩니다.)




질문자는, enum 대신 일련의 문자열 배열을 선택 사항으로 주고 그중에 하나를 취할 수 있는 것을 원하는 듯한데요. 바로 이럴 때 사용할 수 있는 방법이 TypeConverter입니다.

우선 기본 구현을 바탕으로,

Building Windows Forms Controls and Components with Rich Design-Time Features, Part 2
; https://docs.microsoft.com/en-us/archive/msdn-magazine/2003/may/design-time-features-for-windows-forms-controls-and-components

User Control Property that shows a list of all Forms at design time
; https://stackoverflow.com/questions/58514948/user-control-property-that-shows-a-list-of-all-forms-at-design-time

특정 문자열을 반환하는 TypeConverter를 다음과 같이 작성할 수 있고,

public class UserTypeConverter : TypeConverter
{
    public override bool GetStandardValuesExclusive(ITypeDescriptorContext context)
    {
        return true;
    }

    public override bool CanConvertTo(ITypeDescriptorContext pContext, Type pDestinationType)
    {
        return base.CanConvertTo(pContext, pDestinationType);
    }

    public override object ConvertTo(ITypeDescriptorContext pContext, CultureInfo pCulture, object pValue, Type pDestinationType)
    {
        return base.ConvertTo(pContext, pCulture, pValue, pDestinationType);
    }

    public override bool CanConvertFrom(ITypeDescriptorContext pContext, Type pSourceType)
    {
        if (pSourceType == typeof(string))
        {
            return true;
        }

        return base.CanConvertFrom(pContext, pSourceType);
    }

    public override object ConvertFrom(ITypeDescriptorContext pContext, CultureInfo pCulture, object pValue)
    {
        if (pValue is string)
        {
            return pValue.ToString();
        }

        return base.ConvertFrom(pContext, pCulture, pValue);
    }

    public override bool GetStandardValuesSupported(ITypeDescriptorContext pContext)
    {
        return true;
    }

    public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext pContext)
    {
        List<string> values = new List<string>();

        values.AddRange(new string[] { "UserA", "UserB", "UserC" });

        return new StandardValuesCollection(values);
    }
}

이것을 UserControl의 속성 하나에 지정하면,

public partial class UserControl1 : UserControl
{
    public UserControl1()
    {
        InitializeComponent();
    }

    AccountType _userType;
    [Category("MyUserControlType")]
    public AccountType UserType
    {
        get { return _userType; }
        set { _userType = value; }
    }

    string _guestType;
    [Category("MyUserControlType")]
    [TypeConverter(typeof(UserTypeConverter))]
    public string GuestType
    {
        get { return _guestType; }
        set { _guestType = value; }
    }
}

다음과 같이 디자인 타임에 문자열을 선택할 수 있습니다.

winform_design_time_attr_2.png




그런데, 채워줄 문자열 목록을 TypeConverter에 전달해 줄 방법이 없습니다. "[TypeConverter(typeof(UserTypeConverter))]" 특성은 오직 Type 정보만을 인자로 받기 때문인데요, 채워줄 문자열들의 종류 별로 TypeConverter를 만드는 것이 영 못마땅합니다.

이에 대한 우회 방법을 찾아 보면,

How to pass parameter to TypeConverter derived class
; https://stackoverflow.com/questions/14929681/how-to-pass-parameter-to-typeconverter-derived-class

그나마 현실성 있는 방법으로 별도의 특성(Attribute)을 지정해 해결하는 것입니다. 예를 들어, TypeConverter가 선택 목록으로 보여줄 항목을 다음과 같이 StringListAttribute 특성 타입으로 전달하고,

string _guestType;
[Category("MyUserControlType")]
[TypeConverter(typeof(UserTypeConverter))]
[StringList(new string[] { "UserA", "UserB", "UserC" })]
public string GuestType
{
    get { return _guestType; }
    set { _guestType = value; }
}

public class StringListAttribute : Attribute
{
    string[] _list;
    public string [] List
    {
        get { return _list; }
    }

    public StringListAttribute(string [] list)
    {
        _list = list;
    }
}

TypeConverter의 GetStandardValues 메서드에서 다음과 같이 해당 속성에 지정된 특성을 열거해 StringList를 찾아 그것에 설정된 목록을 반환하는 것입니다.

public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext pContext)
{
    List<string> values = new List<string>();

    AttributeCollection ua = pContext.PropertyDescriptor.Attributes;

    foreach (Attribute attr in ua)
    {
        if (attr is StringListAttribute listAttr)
        {
            values.AddRange(listAttr.List);
        }
    }

    return new StandardValuesCollection(values);
}

그런대로 이 정도면 해결된 것 같군요. ^^

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




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/9/2021]

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

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  69  70  71  72  73  74  [75]  ...
NoWriterDateCnt.TitleFile(s)
12061정성태11/20/201919332Windows: 167. CoTaskMemAlloc/CoTaskMemFree과 윈도우 Heap의 관계
12060정성태11/20/201920938디버깅 기술: 132. windbg/Visual Studio - HeapFree x64의 동작 분석
12059정성태11/20/201920130디버깅 기술: 131. windbg/Visual Studio - HeapFree x86의 동작 분석
12058정성태11/19/201920760디버깅 기술: 130. windbg - CoTaskMemFree/FreeCoTaskMem에서 발생한 덤프 분석 사례
12057정성태11/18/201916632오류 유형: 579. Visual Studio - Memory 창에서 유효한 주소 영역임에도 "Unable to evaluate the expression." 오류 출력
12056정성태11/18/201922303개발 환경 구성: 464. "Microsoft Visual Studio Installer Projects" 프로젝트로 EXE 서명 및 MSI 파일 서명 방법파일 다운로드1
12055정성태11/17/201916454개발 환경 구성: 463. Visual Studio의 Ctrl + Alt + M, 1 (Memory 1) 등의 단축키가 동작하지 않는 경우
12054정성태11/15/201918028.NET Framework: 869. C# - 일부러 GC Heap을 깨뜨려 GC 수행 시 비정상 종료시키는 예제
12053정성태11/15/201919783Windows: 166. 윈도우 10 - 명령행 창(cmd.exe) 속성에 (DotumChe, GulimChe, GungsuhChe 등의) 한글 폰트가 없는 경우
12052정성태11/15/201918585오류 유형: 578. Azure - 일정(schedule)에 등록한 runbook이 1년 후 실행이 안 되는 문제(Reason - The key used is expired.)
12051정성태11/14/201922017개발 환경 구성: 462. 시작하자마자 비정상 종료하는 프로세스의 메모리 덤프 - procdump [1]
12050정성태11/14/201919626Windows: 165. AcLayers의 API 후킹과 FaultTolerantHeap
12049정성태11/13/201920091.NET Framework: 868. (닷넷 프로세스를 대상으로) 디버거 방식이 아닌 CLR Profiler를 이용해 procdump.exe 기능 구현
12048정성태11/12/201920245Windows: 164. GUID 이름의 볼륨에 해당하는 파티션을 찾는 방법
12047정성태11/12/201922519Windows: 163. 안전하게 eject시킨 USB 장치를 물리적인 재연결 없이 다시 인식시키는 방법
12046정성태10/29/201917073오류 유형: 577. windbg - The call to LoadLibrary(...\sos.dll) failed, Win32 error 0n193
12045정성태10/27/201917024오류 유형: 576. mstest.exe 실행 시 "Visual Studio Enterprise is required to execute the test." 오류 - 두 번째 이야기
12044정성태10/27/201916618오류 유형: 575. mstest.exe - System.Resources.MissingSatelliteAssemblyException: The satellite assembly named "Microsoft.VisualStudio.ProductKeyDialog.resources.dll, ..."
12043정성태10/27/201918174오류 유형: 574. Windows 10 설치 시 오류 - 0xC1900101 - 0x4001E
12042정성태10/26/201917894오류 유형: 573. OneDrive 하위에 위치한 Documents, Desktop 폴더에 대한 권한 변경 시 "Unable to display current owner"
12041정성태10/23/201918854오류 유형: 572. mstest.exe - The load test results database could not be opened.
12040정성태10/23/201919238오류 유형: 571. Unhandled Exception: System.Net.Mail.SmtpException: Transaction failed. The server response was: 5.2.0 STOREDRV.Submission.Exception:SendAsDeniedException.MapiExceptionSendAsDenied
12039정성태10/22/201916662스크립트: 16. cmd.exe의 for 문에서는 ERRORLEVEL이 설정되지 않는 문제
12038정성태10/17/201916749오류 유형: 570. SQL Server 2019 RC1 - SQL Client Connectivity SDK 설치 오류
12037정성태10/15/201924224.NET Framework: 867. C# - Encoding.Default 값을 바꿀 수 있을까요?파일 다운로드1
12036정성태10/14/201925297.NET Framework: 866. C# - 고성능이 필요한 환경에서 GC가 발생하지 않는 네이티브 힙 사용파일 다운로드1
... 61  62  63  64  65  66  67  68  69  70  71  72  73  74  [75]  ...