So I've been learning about MVVM and I had a problem. I often had to start a thread to download or do some longer task in background and when it is done change some properties so the UI updates. The problem is that the UI will update only if I changed it from the main thread. So I often had to create DispatchTimers to wait for the thread to finish and then execute some code in main. Well this is my solution to it.
With this class you can simply go to your ViewModel constructor and do something like this:
ViewModelEventHandler.RegisterEventListener("EventName", EventAction);
EventAction being an action that is executed when the event is raised.
You can simply create an async Task that does its work in the background and it raises the event "EventName" and then EventAction() is executed in main thread. You just have to put new ViewModelEventHandler(); inside your main function.
I also use it a lot for communication between view models. You can raise an event in view model A and a completely unrelated view model B can execute code.
Here is my code:
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Net;
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Input;
using System.Windows.Threading;
namespace MVVM {
  public class EventData < T1, T2 > {
    public T1 eventState {
      get;
      set;
    }
    public T2 subs {
      get;
      set;
    }
  }
  class ViewModelEventHandler {
    // holds all events and their subscribed actions, the string is the event name, bool keeps track of if the event is currently raised, and List<Action> holds all the functions that will be executed when the event is raised
    private static Dictionary < string, EventData < bool, List<Action> >> eventList = new Dictionary < string, EventData <bool,List<Action> >> ();
    public ViewModelEventHandler() {
      DispatcherTimer timer = new DispatcherTimer();
      timer.Interval = TimeSpan.FromMilliseconds(1);
      timer.Tick += eventListen_Tick;
      timer.Start();
    }
    public static void RaiseEvent(string eventName) {
      try {
        eventList[eventName].eventState = true;
      }
      catch {}
    }
    public static void RegisterEventListener(string eventName, Action eventMethod) {
      foreach(var regEvent in eventList) {
        if (regEvent.Key == eventName) {
          eventList[eventName].subs.Add(eventMethod);
          return;
        }
      }
      eventList[eventName] = new EventData < bool, List < Action >> ();
      eventList[eventName].eventState = false;
      eventList[eventName].subs = new List < Action > ();
      eventList[eventName].subs.Add(eventMethod);
    }
    public static void RemoveEventListener(string eventName, Action eventMethod) {
      try {
        eventList[eventName].subs.Remove(eventMethod);
      }
      catch {}
    }
    public static void RemoveEvent(string eventName) {
      try {
        eventList.Remove(eventName);
      }
      catch {}
    }
    private void eventListen_Tick(object sender, EventArgs e) {
      foreach(var ev in eventList) {
        if (ev.Value.eventState) {
          ev.Value.eventState = false;
          foreach(Action eventSub in ev.Value.subs) {
            eventSub();
          }
        }
      }
    }
  }
}
