using Remotely.Desktop.Win.Controls; using Remotely.Desktop.Win.Services; using Remotely.Shared.Models; using Remotely.ScreenCast.Core; using Remotely.ScreenCast.Core.Models; using Remotely.ScreenCast.Core.Services; using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Diagnostics; using System.Linq; using System.Threading.Tasks; using System.Windows; using System.Security.Principal; using System.Windows.Input; using Remotely.ScreenCast.Win.Services; using Remotely.ScreenCast.Core.Interfaces; using Remotely.ScreenCast.Core.Communication; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Remotely.Shared.Win32; namespace Remotely.Desktop.Win.ViewModels { public class MainWindowViewModel : ViewModelBase { private string host; private string sessionID; public MainWindowViewModel() { Application.Current.Exit += Application_Exit; Current = this; BuildServices(); CursorIconWatcher = Services.GetRequiredService(); CursorIconWatcher.OnChange += CursorIconWatcher_OnChange; Services.GetRequiredService().BeginWatching(); Conductor = Services.GetRequiredService(); CasterSocket = Services.GetRequiredService(); Conductor.SessionIDChanged += SessionIDChanged; Conductor.ViewerRemoved += ViewerRemoved; Conductor.ViewerAdded += ViewerAdded; Conductor.ScreenCastRequested += ScreenCastRequested; } public static MainWindowViewModel Current { get; private set; } public static IServiceProvider Services => ServiceContainer.Instance; public ICommand ChangeServerCommand { get { return new Executor(async (param) => { PromptForHostName(); await Init(); }); } } private Conductor Conductor { get; } private CasterSocket CasterSocket { get; } private ICursorIconWatcher CursorIconWatcher { get; set; } public ICommand ElevateToAdminCommand { get { return new Executor((param) => { try { //var filePath = Process.GetCurrentProcess().MainModule.FileName; var commandLine = Win32Interop.GetCommandLine().Replace(" -elevate", ""); var sections = commandLine.Split('"', StringSplitOptions.RemoveEmptyEntries); var filePath = sections.First(); var arguments = string.Join('"', sections.Skip(1)); var psi = new ProcessStartInfo(filePath, arguments) { Verb = "RunAs", UseShellExecute = true, WindowStyle = ProcessWindowStyle.Hidden }; Process.Start(psi); Environment.Exit(0); } // Exception can be thrown if UAC is dialog is cancelled. catch { } }, (param) => { return !IsAdministrator; }); } } public ICommand ElevateToServiceCommand { get { return new Executor((param) => { try { var psi = new ProcessStartInfo("cmd.exe") { WindowStyle = ProcessWindowStyle.Hidden, CreateNoWindow = true }; //var filePath = Process.GetCurrentProcess().MainModule.FileName; var commandLine = Win32Interop.GetCommandLine().Replace(" -elevate", ""); var sections = commandLine.Split('"', StringSplitOptions.RemoveEmptyEntries); var filePath = sections.First(); var arguments = string.Join('"', sections.Skip(1)); Logger.Write($"Creating temporary service with file path {filePath} and arguments {arguments}."); psi.Arguments = $"/c sc create Remotely_Temp binPath=\"{filePath} {arguments} -elevate\""; Process.Start(psi).WaitForExit(); psi.Arguments = "/c sc start Remotely_Temp"; Process.Start(psi).WaitForExit(); psi.Arguments = "/c sc delete Remotely_Temp"; Process.Start(psi).WaitForExit(); App.Current.Shutdown(); } catch { } }, (param) => { return IsAdministrator && !WindowsIdentity.GetCurrent().IsSystem; }); } } public string Host { get => host; set { host = value; FirePropertyChanged("Host"); } } public bool IsAdministrator => new WindowsPrincipal(WindowsIdentity.GetCurrent()).IsInRole(WindowsBuiltInRole.Administrator); public ICommand RemoveViewersCommand { get { return new Executor(async (param) => { foreach (Viewer viewer in (param as IList).ToArray()) { viewer.DisconnectRequested = true; ViewerRemoved(this, viewer.ViewerConnectionID); await CasterSocket.SendViewerRemoved(viewer.ViewerConnectionID); } }, (param) => { return (param as IList)?.Count > 0; }); } } public string SessionID { get => sessionID; set { sessionID = value; FirePropertyChanged("SessionID"); } } public ObservableCollection Viewers { get; } = new ObservableCollection(); public void CopyLink() { Clipboard.SetText($"{Host}/RemoteControl?sessionID={SessionID?.Replace(" ", "")}"); } public async Task GetSessionID() { await CasterSocket.SendDeviceInfo(Conductor.ServiceID, Environment.MachineName, Conductor.DeviceID); await CasterSocket.GetSessionID(); } public async Task Init() { SessionID = "Retrieving..."; Host = Config.GetConfig().Host; while (string.IsNullOrWhiteSpace(Host)) { Host = "https://"; PromptForHostName(); } Conductor.ProcessArgs(new string[] { "-mode", "Normal", "-host", Host }); try { await CasterSocket.Connect(Conductor.Host); CasterSocket.Connection.Closed += async (ex) => { await App.Current.Dispatcher.InvokeAsync(() => { Viewers.Clear(); SessionID = "Disconnected"; }); }; CasterSocket.Connection.Reconnecting += async (ex) => { await App.Current.Dispatcher.InvokeAsync(() => { Viewers.Clear(); SessionID = "Reconnecting"; }); }; CasterSocket.Connection.Reconnected += async (arg) => { await GetSessionID(); }; await GetSessionID(); } catch (Exception ex) { Logger.Write(ex); MessageBox.Show(Application.Current.MainWindow, "Failed to connect to server.", "Connection Failed", MessageBoxButton.OK, MessageBoxImage.Warning); return; } } public void PromptForHostName() { var prompt = new HostNamePrompt(); if (!string.IsNullOrWhiteSpace(Host)) { HostNamePromptViewModel.Current.Host = Host; } prompt.Owner = App.Current?.MainWindow; prompt.ShowDialog(); var result = HostNamePromptViewModel.Current.Host; if (!result.StartsWith("https://") && !result.StartsWith("http://")) { result = $"https://{result}"; } if (result != Host) { Host = result.TrimEnd('/'); var config = Config.GetConfig(); config.Host = Host; config.Save(); } } private void Application_Exit(object sender, ExitEventArgs e) { App.Current.Dispatcher.Invoke(() => { foreach (var viewer in Viewers) { viewer.DisconnectRequested = true; } Viewers.Clear(); }); } private void BuildServices() { var serviceCollection = new ServiceCollection(); serviceCollection.AddLogging(builder => { builder.AddConsole().AddDebug().AddEventLog(); }); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddSingleton(); serviceCollection.AddTransient(); serviceCollection.AddTransient(); serviceCollection.AddScoped(); serviceCollection.AddScoped(); ServiceContainer.Instance = serviceCollection.BuildServiceProvider(); } private async void CursorIconWatcher_OnChange(object sender, CursorInfo cursor) { if (Conductor?.Viewers?.Count > 0) { foreach (var viewer in Conductor.Viewers.Values) { await viewer.SendCursorChange(cursor); } } } private void ScreenCastRequested(object sender, ScreenCastRequest screenCastRequest) { App.Current.Dispatcher.Invoke(() => { App.Current.MainWindow.Activate(); var result = MessageBox.Show(Application.Current.MainWindow, $"You've received a connection request from {screenCastRequest.RequesterName}. Accept?", "Connection Request", MessageBoxButton.YesNo, MessageBoxImage.Question); if (result == MessageBoxResult.Yes) { Task.Run(() => { Services.GetRequiredService().BeginScreenCasting(screenCastRequest); }); } else { // Run on another thread so it doesn't tie up the UI thread. Task.Run(async () => { await CasterSocket.SendConnectionRequestDenied(screenCastRequest.ViewerID); }); } }); } private void SessionIDChanged(object sender, string sessionID) { var formattedSessionID = ""; for (var i = 0; i < sessionID.Length; i += 3) { formattedSessionID += sessionID.Substring(i, 3) + " "; } SessionID = formattedSessionID.Trim(); } private void ViewerAdded(object sender, Viewer viewer) { App.Current.Dispatcher.Invoke(() => { Viewers.Add(viewer); }); } private void ViewerRemoved(object sender, string viewerID) { App.Current.Dispatcher.Invoke(() => { var viewer = Viewers.FirstOrDefault(x => x.ViewerConnectionID == viewerID); if (viewer != null) { Viewers.Remove(viewer); viewer.Dispose(); } }); } } }