Проблема двойной буферизации производного отрисованного владельцем TabControl? - PullRequest
2 голосов
/ 22 октября 2008

Я получил TabControl с целью включить двойную буферизацию, за исключением того, что ничего не работает, как ожидалось. Вот код TabControl:

class DoubleBufferedTabControl : TabControl
{
    public DoubleBufferedTabControl() : base()
    {
        this.DoubleBuffered = true;
        this.SetStyle
            (
                ControlStyles.UserPaint |
                ControlStyles.AllPaintingInWmPaint |
                ControlStyles.ResizeRedraw |
                ControlStyles.OptimizedDoubleBuffer |
                ControlStyles.SupportsTransparentBackColor,
                false
            );
    }
}

Этот Tabcontrol затем устанавливается с его режимом рисования как «OwnerDrawnFixed», так что я могу изменить цвета. Вот пользовательский метод рисования:

    private void Navigation_PageContent_DrawItem(object sender, DrawItemEventArgs e)
    {
        //Structure.
        Graphics g = e.Graphics;
        TabControl t = (TabControl)sender;
        TabPage CurrentPage = t.TabPages[e.Index];

        //Get the current tab
        Rectangle CurrentTabRect = t.GetTabRect(e.Index);

        //Get the last tab.
        Rectangle LastTab = t.GetTabRect(t.TabPages.Count - 1);

        //Main background rectangle.
        Rectangle BackgroundRect = new Rectangle(LastTab.Width, t.Bounds.Y - 4, t.Width - (LastTab.Width), t.Height);

        //Tab background rectangle.
        Rectangle TabBackgroundRect = new Rectangle(0, LastTab.Y + LastTab.Height, LastTab.Width, t.Bounds.Height - (LastTab.Y + LastTab.Height));

        //Set anitialiasing for the text.
        e.Graphics.TextRenderingHint = System.Drawing.Text.TextRenderingHint.ClearTypeGridFit;

        //String format for the text.
        StringFormat StringFormat = new StringFormat();
        StringFormat.Alignment = StringAlignment.Center;
        StringFormat.LineAlignment = StringAlignment.Center;

        //Fill the background.
        g.FillRectangle(Brushes.LightGray, BackgroundRect);
        g.FillRectangle(Brushes.Bisque, TabBackgroundRect);

        //Draw the selected tab.
        if(e.State == DrawItemState.Selected)
        {
            g.FillRectangle(Brushes.White, e.Bounds);
            Rectangle SelectedTabOutline = new Rectangle(e.Bounds.X + 2, e.Bounds.Y + 2, e.Bounds.Width, e.Bounds.Height - 4);
            g.DrawRectangle(new Pen(Brushes.LightGray, 4f), SelectedTabOutline);
            g.DrawString(CurrentPage.Text, new Font("Arial", 12f, FontStyle.Bold, GraphicsUnit.Point), new SolidBrush(Color.FromArgb(70, 70, 70)), CurrentTabRect, StringFormat);
        }
        else
        {
            g.FillRectangle(new SolidBrush(Color.FromArgb(230, 230, 230)), e.Bounds);
            g.DrawString(CurrentPage.Text, new Font("Arial", 12f, FontStyle.Regular, GraphicsUnit.Point), Brushes.Gray, CurrentTabRect, StringFormat);
        }

    }

Однако все безрезультатно, так как этот элемент управления не имеет двойной буферизации и продолжает мерцать при изменении размера.

Есть идеи?

Ответы [ 6 ]

2 голосов
/ 10 ноября 2008

Прежде всего, вы можете избавиться от своего TabControl кода - вы включаете буферизацию, а затем немедленно выключаете ее, так что на самом деле она не делает ничего полезного.

Часть вашей проблемы в том, что вы пытаетесь нарисовать только часть из TabControl.

Самое простое решение, которое дает примерно 90% -ое решение (все еще возможно получить мерцание), состоит в том, чтобы добавить это к вашему классу формы:

protected override CreateParams CreateParams
{
    get
    {
        CreateParams cp = base.CreateParams;
        cp.ExStyle |= 0x02000000;
        return cp;
    }
}

Если вы хотите, чтобы очень был уверен, что не мерцал, вам нужно нарисовать весь TabControl самостоятельно и не обращать внимания на запросы на рисование фона.

Редактировать: обратите внимание, что это будет работать только в XP и более поздних версиях.

2 голосов
/ 23 октября 2008

Если вы читаете документацию, там написано: «Этот элемент не имеет смысла для этого элемента управления». Если вы хотите, чтобы элемент управления отображался с использованием двойной буферизации, вам придется реализовать его самостоятельно. Помимо того факта, что если вы используете владелец-элемент управления, вам все равно придется реализовать двойную буферизацию.

1 голос
/ 19 ноября 2008

В прошлом у меня были проблемы с двойной буферизацией элементов управления, и единственный способ остановить мерцание состоял в том, чтобы гарантировать, что унаследованный метод OnPaintBackground не вызывался. (См. Код ниже). Вам также необходимо убедиться, что весь фон закрашен во время вызова.

protected override void OnPaintBackground( PaintEventArgs pevent )
{
    //do not call base - I don't want the background re-painted!
}
0 голосов
/ 10 ноября 2008

Я думаю, что это не работает, потому что вы отключаете двойную буферизацию!

Все this.DoubleBuffered = true делает для ControlStyles.OptimizedDoubleBuffer значение true. Поскольку вы отключаете этот флаг в следующей строке вашей программы, вы действительно ничего не делаете. Удалите ControlStyles.OptimizedDoubleBuffer (и, возможно, ControlStyles.AllPaintingInWmPaint), и он должен работать для вас.

0 голосов
/ 22 октября 2008

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

Для чего стоит отключить «Показывать содержимое окна при перетаскивании», это исправит это, но я понимаю, что это может быть не полезно.

0 голосов
/ 22 октября 2008

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

...