WPF делает ServiceController [] Наблюдаемым - PullRequest
1 голос
/ 14 октября 2011

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

Есть предложения?

Ответы [ 3 ]

3 голосов
/ 27 октября 2011

Это то, что я в конечном итоге навязал. По большей части он работает довольно хорошо, однако я готов предложить любые варианты кода.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ComponentModel;
using System.ServiceProcess;

namespace v7quickbar
{
    class NotifiableServiceController : INotifyPropertyChanged
    {

        private ServiceController m_oServiceController = null;
        private System.Timers.Timer m_oServiceCheckTimer = new System.Timers.Timer();

        public ServiceControllerStatus Status { get { return this.m_oServiceController.Status; } }

        public string DisplayName { get { return this.m_oServiceController.DisplayName; } }

        public string ServiceName { get { return this.m_oServiceController.ServiceName; } }

        public bool CanStop { get { return this.m_oServiceController.CanStop; } }

        public NotifiableServiceController(ServiceController oService)
        {
            CreateObject(oService, TimeSpan.FromSeconds(.5));
        }

        public NotifiableServiceController(ServiceController oService, TimeSpan oInterval)
        {
            CreateObject(oService, oInterval);
        }

        private void CreateObject(ServiceController oService, TimeSpan oInterval)
        {
            m_oServiceController = oService;
            m_oServiceCheckTimer.Interval = oInterval.TotalMilliseconds;

            m_oServiceCheckTimer.Elapsed += new System.Timers.ElapsedEventHandler(m_oServiceCheckTimer_Elapsed);
            m_oServiceCheckTimer.Start();
        }

        public void Start()
        {
            try
            {
                this.m_oServiceController.Start();
                this.m_oServiceController.WaitForStatus(ServiceControllerStatus.Running);
            }
            catch (Exception)
            {
            }
        }

        public void Stop()
        {
            try
            {
                this.m_oServiceController.Stop();
                this.m_oServiceController.WaitForStatus(ServiceControllerStatus.Stopped);
            }
            catch (Exception)
            {
            }
        }

        public void Restart()
        {
            try
            {
                if (m_oServiceController.CanStop && (m_oServiceController.Status == ServiceControllerStatus.Running || m_oServiceController.Status == ServiceControllerStatus.Paused))
                {
                    this.Stop();
                    this.m_oServiceController.WaitForStatus(ServiceControllerStatus.Stopped);
                }

                if (m_oServiceController.Status == ServiceControllerStatus.Stopped)
                {
                    this.Start();
                    this.m_oServiceController.WaitForStatus(ServiceControllerStatus.Running);
                }
            }
            catch (Exception)
            {
            }
        }

        void m_oServiceCheckTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
        {
            ServiceControllerStatus oCurrentStatus = m_oServiceController.Status;
            m_oServiceController.Refresh();

            if (oCurrentStatus != m_oServiceController.Status)
            {
                PropertyChanged.Invoke(this, new PropertyChangedEventArgs("Status"));
            }

        }

        public static IEnumerable<NotifiableServiceController> GetServices()
        {
            List<NotifiableServiceController> oaServices = new List<NotifiableServiceController>();
            foreach (ServiceController sc in ServiceController.GetServices())
            {
                oaServices.Add(new NotifiableServiceController(sc));
            }

            return oaServices;
        }

        public event PropertyChangedEventHandler PropertyChanged;
    }
}
3 голосов
/ 14 октября 2011

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

1 голос
/ 14 октября 2011

К сожалению, из-за того, что вызов ServiceController.GetServices() всегда будет возвращать массив, мы должны иметь DispatcherTimer и в его тике выполнить вызов ServiceController.GetServices() и изменить свойство notify, измененное для этого свойства, которое содержит массив служб.

Делать это наблюдаемым ради наблюдаемости не практично, верно?В любом случае, мы не получим от этого никаких преимуществ.

...