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

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12888정성태12/21/202115146.NET Framework: 1122. C# - ImageCodecInfo 사용 시 System.Drawing.Image와 System.Drawing.Bitmap에 따른 Save 성능 차이파일 다운로드1
12887정성태12/21/202118662오류 유형: 777. OpenCVSharp4를 사용한 프로그램 실행 시 "The type initializer for 'OpenCvSharp.Internal.NativeMethods' threw an exception." 예외 발생
12886정성태12/20/202114698스크립트: 37. 파이썬 - uwsgi의 --enable-threads 옵션 [2]
12885정성태12/20/202115738오류 유형: 776. uwsgi-plugin-python3 환경에서 MySQLdb 사용 환경
12884정성태12/20/202114669개발 환경 구성: 620. Windows 10+에서 WMI root/Microsoft/Windows/WindowsUpdate 네임스페이스 제거
12883정성태12/19/202115093오류 유형: 775. uwsgi-plugin-python3 환경에서 "ModuleNotFoundError: No module named 'django'" 오류 발생
12882정성태12/18/202114585개발 환경 구성: 619. Windows Server에서 WSL을 위한 리눅스 배포본을 설치하는 방법
12881정성태12/17/202114176개발 환경 구성: 618. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법 (2)
12880정성태12/16/202115176VS.NET IDE: 170. Visual Studio에서 .NET Core/5+ 역어셈블 소스코드 확인하는 방법
12879정성태12/16/202121714오류 유형: 774. Windows Server 2022 + docker desktop 설치 시 WSL 2로 선택한 경우 "Failed to deploy distro docker-desktop to ..." 오류 발생
12878정성태12/15/202115976개발 환경 구성: 617. 윈도우 WSL 환경에서 같은 종류의 리눅스를 다중으로 설치하는 방법
12877정성태12/15/202115301스크립트: 36. 파이썬 - pymysql 기본 예제 코드
12876정성태12/14/202115159개발 환경 구성: 616. Custom Sources를 이용한 Azure Monitor Metric 만들기
12875정성태12/13/202114020스크립트: 35. python - time.sleep(...) 호출 시 hang이 걸리는 듯한 문제
12874정성태12/13/202113862오류 유형: 773. shell script 실행 시 "$'\r': command not found" 오류
12873정성태12/12/202115241오류 유형: 772. 리눅스 - PATH에 등록했는데도 "command not found"가 나온다면?
12872정성태12/12/202115650개발 환경 구성: 615. GoLang과 Python 빌드가 모두 가능한 docker 이미지 만들기
12871정성태12/12/202114688오류 유형: 771. docker: Error response from daemon: OCI runtime create failed
12870정성태12/9/202113784개발 환경 구성: 614. 파이썬 - PyPI 패키지 만들기 (4) package_data 옵션
12869정성태12/8/202116464개발 환경 구성: 613. git clone 실행 시 fingerprint 묻는 단계를 생략하는 방법
12868정성태12/7/202114847오류 유형: 770. twine 업로드 시 "HTTPError: 400 Bad Request ..." 오류 [1]
12867정성태12/7/202114600개발 환경 구성: 612. 파이썬 - PyPI 패키지 만들기 (3) entry_points 옵션
12866정성태12/7/202121541오류 유형: 769. "docker build ..." 시 "failed to solve with frontend dockerfile.v0: failed to read dockerfile ..." 오류
12865정성태12/6/202114864개발 환경 구성: 611. 파이썬 - PyPI 패키지 만들기 (2) long_description, cmdclass 옵션
12864정성태12/6/202112532Linux: 46. WSL 환경에서 find 명령을 사용해 파일을 찾는 방법
12863정성태12/4/202114714개발 환경 구성: 610. 파이썬 - PyPI 패키지 만들기
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...