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)
12083정성태12/17/201922232Linux: 27. linux - lldb를 이용한 .NET Core 응용 프로그램의 메모리 덤프 분석 방법 [2]
12082정성태12/17/201920519오류 유형: 585. lsof: WARNING: can't stat() fuse.gvfsd-fuse file system
12081정성태12/16/201922375개발 환경 구성: 465. 로컬 PC에서 개발 중인 ASP.NET Core 웹 응용 프로그램을 다른 PC에서도 접근하는 방법 [5]
12080정성태12/16/201919518.NET Framework: 870. C# - 프로세스의 모든 핸들을 열람
12079정성태12/13/201921392오류 유형: 584. 원격 데스크톱(rdp) 환경에서 다중 또는 고용량 파일 복사 시 "Unspecified error" 오류 발생
12078정성태12/13/201921206Linux: 26. .NET Core 응용 프로그램을 위한 메모리 덤프 방법 [3]
12077정성태12/13/201920323Linux: 25. 자주 실행할 명령어 또는 초기 환경을 "~/.bashrc" 파일에 등록
12076정성태12/12/201918805디버깅 기술: 142. Linux - lldb 환경에서 sos 확장 명령어를 이용한 닷넷 프로세스 디버깅 - 배포 방법에 따른 차이
12075정성태12/11/201919594디버깅 기술: 141. Linux - lldb 환경에서 sos 확장 명령어를 이용한 닷넷 프로세스 디버깅
12074정성태12/10/201919244디버깅 기술: 140. windbg/Visual Studio - 값이 변경된 경우를 위한 정지점(BP) 설정(Data Breakpoint)
12073정성태12/10/201920837Linux: 24. Linux/C# - 실행 파일이 아닌 스크립트 형식의 명령어를 Process.Start로 실행하는 방법
12072정성태12/9/201917639오류 유형: 583. iisreset 수행 시 "No such interface supported" 오류
12071정성태12/9/201921155오류 유형: 582. 리눅스 디스크 공간 부족 및 safemode 부팅 방법
12070정성태12/9/201923066오류 유형: 581. resize2fs: Bad magic number in super-block while trying to open /dev/.../root
12069정성태12/2/201919464디버깅 기술: 139. windbg - x64 덤프 분석 시 메서드의 인자 또는 로컬 변수의 값을 확인하는 방법
12068정성태11/28/201928090디버깅 기술: 138. windbg와 Win32 API로 알아보는 Windows Heap 정보 분석 [3]파일 다운로드2
12067정성태11/27/201919524디버깅 기술: 137. 실제 사례를 통해 Debug Diagnostics 도구가 생성한 닷넷 웹 응용 프로그램의 성능 장애 보고서 설명 [1]파일 다운로드1
12066정성태11/27/201919154디버깅 기술: 136. windbg - C# PInvoke 호출 시 마샬링을 담당하는 함수 분석 - OracleCommand.ExecuteReader에서 OpsSql.Prepare2 PInvoke 호출 분석
12065정성태11/25/201917497디버깅 기술: 135. windbg - C# PInvoke 호출 시 마샬링을 담당하는 함수 분석파일 다운로드1
12064정성태11/25/201920383오류 유형: 580. HTTP Error 500.0/500.33 - ANCM In-Process Handler Load Failure
12063정성태11/21/201919321디버깅 기술: 134. windbg - RtlReportCriticalFailure로부터 parameters 정보 찾는 방법
12062정성태11/21/201918844디버깅 기술: 133. windbg - CoTaskMemFree/FreeCoTaskMem에서 발생한 덤프 분석 사례 - 두 번째 이야기
12061정성태11/20/201919278Windows: 167. CoTaskMemAlloc/CoTaskMemFree과 윈도우 Heap의 관계
12060정성태11/20/201920855디버깅 기술: 132. windbg/Visual Studio - HeapFree x64의 동작 분석
12059정성태11/20/201920018디버깅 기술: 131. windbg/Visual Studio - HeapFree x86의 동작 분석
12058정성태11/19/201920664디버깅 기술: 130. windbg - CoTaskMemFree/FreeCoTaskMem에서 발생한 덤프 분석 사례
... 61  62  63  64  65  66  67  68  69  70  71  72  73  [74]  75  ...