2010-03-31 4 views
7

я пытаюсь применить шаблон Decorator Design к следующей ситуации:Применяя декоратор к формам

У меня есть 3 различных вида форм: зеленый, желтый, красный.

Теперь каждая из этих форм может иметь различный набор атрибутов. У них может быть отключено окно минимума, отключено окно с максимальным полем, и они всегда могут быть на высоте.

Я попытался смоделировать это следующим образом:

   Form <---------------------------------------FormDecorator 
       /\             /\ 
    |---------|-----------|    |----------------------|-----------------| 
GreenForm YellowForm RedForm MinimizeButtonDisabled MaximizedButtonDisabled AlwaysOnTop 

Вот мой GreenForm код:

public class GreenForm : Form { 
    public GreenForm() { 
     this.BackColor = Color.GreenYellow; 
    } 

    public override sealed Color BackColor { 
     get { return base.BackColor; } 
     set { base.BackColor = value; } 
    } 
} 

FormDecorator:

public abstract class FormDecorator : Form { 
    private Form _decoratorForm; 

    protected FormDecorator(Form decoratorForm) { 
     this._decoratorForm = decoratorForm; 
    } 
} 

и, наконец, NoMaximizeDecorator:

public class NoMaximizeDecorator : FormDecorator 
{ 
    public NoMaximizeDecorator(Form decoratorForm) : base(decoratorForm) { 
     this.MaximizeBox = false; 
    } 
} 

Так вот работает код:

static void Main() 
{ 
    Application.EnableVisualStyles(); 
    Application.SetCompatibleTextRenderingDefault(false); 
    Application.Run(CreateForm()); 
} 

static Form CreateForm() { 
    Form form = new GreenForm(); 
    form = new NoMaximizeDecorator(form); 
    form = new NoMinimizeDecorator(form); 

    return form; 
} 

Проблема заключается в том, что я получаю форму, которая не зеленый, а еще позволяет мне развернуть его. Он учитывает только форму NoMinimizeDecorator. Я понимаю, почему это происходит, но мне трудно понять, как сделать эту работу с этим шаблоном.

Я знаю, что, возможно, есть лучшие способы достижения того, чего я хочу. Я сделал этот пример попыткой применить шаблон Decorator к чему-то. Может быть, это был не лучший образец, который я мог бы использовать (если вообще) для такого сценария. Есть ли другой образец, более подходящий, чем Decorator для достижения этого? Я что-то делаю неправильно при попытке реализовать шаблон Decorator?

Благодаря

ответ

6

Проблема здесь в том, что вы» на самом деле не реализует шаблон декоратора. Для правильной реализации шаблона вам необходимо подклассифицировать Form, чтобы создать ваш декоратор, а затем перехватить все операции, выполненные на вашем декораторе, и переслать их на ваш приватный экземпляр Form. Вы делаете это, за исключением того, что помимо назначения ссылки в конструкторе FormDecorator, вы никогда больше не будете использовать этот частный Form экземпляр. В результате вы создаете GreenForm, а затем оберните его в NoMaximizeDecorator, а затем оберните его в NoMinimizeDecorator. Но поскольку вы никогда не выполняете операции, предпринятые против NoMinimizeDecorator, к обернутому экземпляру Form, только экземпляр NoMinimizeDecorator фактически применяет любое поведение к используемому экземпляру. Это соответствует тому, что вы наблюдаете при запуске кода: стандартное окно с отключенной кнопкой «Минимизировать».

Form - очень плохой пример для создания декораторов на C#, поскольку большинство его свойств и методов не являются виртуальными, то есть если вы обращаетесь к оформленной форме с помощью ссылки Form, у вас нет возможности перехватить базу свойства класса - вы не можете эффективно «обернуть» Form.

EDIT

Это происходит со мной, что утверждение «формой является очень плохим примером для создания декораторов в C#» действительно возникает вопрос о том, что является хорошего примера. Как правило, вы используете шаблон декоратора, чтобы обеспечить реализацию пользовательского интерфейса без реализации всего всей реализации с нуля. A очень общий пример - общие коллекции. Большинство функций, которые требуют функции списка, не зависят, например, от List<String>, а скорее от IList<String>. Таким образом, если вы, например, хотите создать пользовательскую коллекцию, которая не примет строки длиной менее 5 символов, вы должны использовать примерно следующее:

2

Это растрата шаблона декоратора. Модель декоратора связана с поведением объектов. Вы - , строите объекты, которые подпадают под созданный зонт. В то время как вы могли бы обернуть свою голову вокруг «не имея кнопки максимизации», это поведение немного звучит.

Я не думаю, что есть реальный способ исправить ваш дизайн. Шаблон декоратора просто не подходит. Любая попытка исправить это просто невероятно круто, когда вы можете просто использовать Builder.

Что я мог видеть делать украшает Builder в форме для выполнения этих действий при строительстве. Это будет выглядеть примерно так ...

public interface IFormBuilder { 
    public Form BuildForm(); 
} 

public class FormBuilder : IFormBuilder { 
    public Form BuildForm(){ 
     return new Form(); 
    } 
} 

public class NoMaximizeFormBuilder : IFormBuilder { 
    private IFormBuilder _builder; 
    public NoMaximizeFormBuilder (IFormBuilder builder){ 
     _builder = builder;    
    } 
    public Form BuildForm(){ 
     f = _builder.BuildForm(); 
     f.MaximizeBox = false; 
     return f; 
    } 
} 

И вы могли бы использовать его как это ...

static void Main() 
{ 
    Application.EnableVisualStyles(); 
    Application.SetCompatibleTextRenderingDefault(false); 
    Application.Run(CreateForm()); 
} 

static Form CreateForm() { 

    var b = new FormBuilder(); 
    var b = new NoMaximizeFormBuilder(b); 
    return b.Build(); 
} 

Но даже это немного некрасиво. Возможно, вы сможете превратить это в свободный интерфейс для создания форм.

+0

+1. Это «немного уродливо», но я считаю, что у него есть потенциал. –

0

пытаются сделать свой шаблон, чтобы применить свойства декоратор к тому же объекту, не создавая новые формы:

public abstract class FormDecorator { 
    protected Form _decoratorForm; 

    protected FormDecorator(Form decoratorForm) { 
     this._decoratorForm = decoratorForm; 
    } 
    public abstract void Decorate(); 
} 

public class NoMaximizeDecorator : FormDecorator 
{ 
    public NoMaximizeDecorator(Form decoratorForm) : base(decoratorForm) { 
     Decorate(); 
    } 
    public override void Decorate() { 
     _decoratorForm.MaximizeBox = false; 
    } 
} 

И в главном:

static Form CreateForm() { 
    Form form = new GreenForm(); 
    new NoMaximizeDecorator(form); 
    new NoMinimizeDecorator(form); 

    return form; 
} 
+1

Рисунок декоратора не похож на этот. Иерархия классов декоратора должна наследовать базовый класс иерархии классов украшений. Абстрактный FormDecorator должен определять поведение по умолчанию для всех методов базового класса. – Seb

+1

Да, это не похоже на это. Однако он делает то, что предполагается. –

Смежные вопросы