Вопрос

Я использую ToolStripDropDown Чтобы показать SelectionPopup.

А ToolStripDropDown Содержит несколько пользовательских элементов управления с запущенными потоками. Нить заканчивается событием в Onhandledestroyed, но по какой -то причине ToolStripDropDown не утилизирует/разрушает его ручку после закрытия.

Утилизация ToolStripDropDown В закрытом событии мне исключение, потому что все еще доступно к ToolStripdropdown.

Как я узнаю, используется ли пользовательский элемент управления или не заканчивать поток?

Пользовательский контроль:

public class CControl : Control
{
    Thread StyleThread;
    Object lockOBJ = new Object();
    bool abortthread = false;

    public CControl()
    {
        this.SetStyle(ControlStyles.AllPaintingInWmPaint | ControlStyles.UserPaint | ControlStyles.OptimizedDoubleBuffer, true);
        this.SetStyle(ControlStyles.Selectable, false);

        StyleThread = new Thread(new ThreadStart(this.StyleDelegate));
    }


    protected override void OnHandleCreated(EventArgs e)
    {
        base.OnHandleCreated(e);
        if(!StyleThread.IsAlive)
            StyleThread.Start();
    }

    protected override void OnHandleDestroyed(EventArgs e)
    {
        base.OnHandleDestroyed(e);
        lock (lockOBJ)
        {
            abortthread = true;
        }
        if (StyleThread.IsAlive)
        {
            StyleThread.Join(100);
        }
    }

    ...
}

ToolStripDropDown-Control:

public class AddPopUp : UserControl
{
    /*
        ------------------------------------
        This Control contains some CControls
        ------------------------------------
    */

    public void Show(Control control)
    {
        TSDD tsdd = new TSDD(this);
        Point screenpoint = control.PointToScreen(new Point(0, 0));
        tsdd.Show(control,new Point(0, -tsdd.Height ));
    }

    class TSDD : ToolStripDropDown
    {
        private Control Control { get; set; }

        public TSDD(Control control)
        {
            this.Control = control;
            this.DropShadowEnabled = false;
            ToolStripControlHost ch = new ToolStripControlHost(control);
            this.Items.Add(ch);
        }

        protected override void SetBoundsCore(int x, int y, int width, int height, BoundsSpecified specified)
        {
            base.SetBoundsCore(x, y, Control.Size.Width + 16, Control.Size.Height + 18, specified);
        }

        protected override CreateParams CreateParams
        {
            get
            {
                CreateParams cp = base.CreateParams;
                unchecked
                {
                    cp.Style |= (int)0x80000000;
                    cp.Style |= 0x40000;
                }
                return cp;
            }
        }


        protected override void WndProc(ref Message m)
        {
            if (m.Msg == 0x84)
            {
                m.Result = (IntPtr)1;
                return;
            }
            base.WndProc(ref m);
        }
    }
}

РЕДАКТИРОВАТЬ:

После того, что еще большее тестирование я не смог избавиться от ToolStripdropdown. В качестве обходного пути я просто разрушаю ручку на закрытии ToolStripDropdown.

        protected override void OnClosed(ToolStripDropDownClosedEventArgs e)
        {
            base.OnClosed(e);
            this.DestroyHandle();
        }
Это было полезно?

Решение

ToolStrips имеет очень постоянную ошибку (утверждается, что это исправлена, но у меня все еще есть проблемы), когда они регистрируют обработчик событий на SystemEvents.UserPreferenceChanged, чтобы иметь возможность перекрасить, если пользователь изменит настройки глобального стиля в ОС.

Единственный обходной путь, который я обнаружил, - это удалить прикрепленные инструментальные структуры путем перечисления обработчиков в SystemEvents.

Утечка памяти ToolStrip

http://connect.microsoft.com/visualstudio/feedback/details/115600/toolstrip-leaks-after-calling-dispose#

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top