430 lines
17 KiB
C#
430 lines
17 KiB
C#
|
|
using System;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Windows.Forms;
|
|
using System.Windows.Interop;
|
|
using System.Windows;
|
|
|
|
using static mpvnet.Global;
|
|
using System.Collections.Generic;
|
|
|
|
namespace mpvnet
|
|
{
|
|
public class Commands
|
|
{
|
|
public static void Execute(string id, string[] args)
|
|
{
|
|
switch (id)
|
|
{
|
|
case "add-files-to-playlist": OpenFiles("append"); break; // deprecated 2019
|
|
case "cycle-audio": CycleAudio(); break;
|
|
case "execute-mpv-command": Msg.ShowError("Command was removed, reset input.conf."); break;
|
|
case "load-audio": LoadAudio(); break;
|
|
case "load-sub": LoadSubtitle(); break;
|
|
case "open-conf-folder": ProcessHelp.ShellExecute(Core.ConfigFolder); break;
|
|
case "open-files": OpenFiles(args); break;
|
|
case "open-optical-media": Open_DVD_Or_BD_Folder(); break;
|
|
case "open-url": OpenURL(); break;
|
|
case "playlist-first": PlaylistFirst(); break;
|
|
case "playlist-last": PlaylistLast(); break;
|
|
case "scale-window": ScaleWindow(float.Parse(args[0], CultureInfo.InvariantCulture)); break;
|
|
case "shell-execute": ProcessHelp.ShellExecute(args[0]); break;
|
|
case "show-about": ShowDialog(typeof(AboutWindow)); break;
|
|
case "show-audio-devices": ShowTextWithEditor("audio-device-list", Core.GetPropertyOsdString("audio-device-list")); break;
|
|
case "show-command-palette": ShowCommandPalette(); break;
|
|
case "show-commands": ShowCommands(); break;
|
|
case "show-conf-editor": ShowDialog(typeof(ConfWindow)); break;
|
|
case "show-decoders": ShowTextWithEditor("decoder-list", mpvHelp.GetDecoders()); break;
|
|
case "show-demuxers": ShowTextWithEditor("demuxer-lavf-list", mpvHelp.GetDemuxers()); break;
|
|
case "show-history": ShowHistory(); break;
|
|
case "show-info": ShowInfo(); break;
|
|
case "show-input-editor": ShowDialog(typeof(InputWindow)); break;
|
|
case "show-keys": ShowTextWithEditor("input-key-list", Core.GetPropertyString("input-key-list").Replace(",", BR)); break;
|
|
case "show-media-info": ShowMediaInfo(args); break;
|
|
case "show-playlist": ShowPlaylist(); break;
|
|
case "show-profiles": ShowTextWithEditor("profile-list", mpvHelp.GetProfiles()); break;
|
|
case "show-properties": ShowProperties(); break;
|
|
case "show-protocols": ShowTextWithEditor("protocol-list", mpvHelp.GetProtocols()); break;
|
|
case "show-setup-dialog": ShowDialog(typeof(SetupWindow)); break;
|
|
case "show-text": ShowText(args[0], Convert.ToInt32(args[1]), Convert.ToInt32(args[2])); break;
|
|
case "update-check": UpdateCheck.CheckOnline(true); break;
|
|
case "window-scale": WindowScale(float.Parse(args[0], CultureInfo.InvariantCulture)); break;
|
|
|
|
default: Terminal.WriteError($"No command '{id}' found."); break;
|
|
}
|
|
}
|
|
|
|
public static void ShowDialog(Type winType)
|
|
{
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
Window win = Activator.CreateInstance(winType) as Window;
|
|
new WindowInteropHelper(win).Owner = MainForm.Instance.Handle;
|
|
win.ShowDialog();
|
|
}));
|
|
}
|
|
|
|
public static void OpenFiles(params string[] args)
|
|
{
|
|
bool append = Control.ModifierKeys.HasFlag(Keys.Control);
|
|
bool loadFolder = true;
|
|
|
|
foreach (string arg in args)
|
|
{
|
|
if (arg == "append") append = true;
|
|
if (arg == "no-folder") loadFolder = false;
|
|
}
|
|
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
using (var d = new OpenFileDialog() { Multiselect = true })
|
|
if (d.ShowDialog() == DialogResult.OK)
|
|
Core.LoadFiles(d.FileNames, loadFolder, append);
|
|
}));
|
|
}
|
|
|
|
public static void Open_DVD_Or_BD_Folder()
|
|
{
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
using (var dialog = new FolderBrowserDialog())
|
|
{
|
|
dialog.Description = "Select a DVD or Blu-ray folder.";
|
|
dialog.ShowNewFolderButton = false;
|
|
|
|
if (dialog.ShowDialog() == DialogResult.OK)
|
|
{
|
|
Core.Command("stop");
|
|
Thread.Sleep(500);
|
|
|
|
if (Directory.Exists(dialog.SelectedPath + "\\BDMV"))
|
|
{
|
|
Core.SetPropertyString("bluray-device", dialog.SelectedPath);
|
|
Core.LoadFiles(new[] { @"bd://" }, false, false);
|
|
}
|
|
else
|
|
{
|
|
Core.SetPropertyString("dvd-device", dialog.SelectedPath);
|
|
Core.LoadFiles(new[] { @"dvd://" }, false, false);
|
|
}
|
|
}
|
|
}
|
|
}));
|
|
}
|
|
|
|
public static void PlaylistFirst()
|
|
{
|
|
int pos = Core.GetPropertyInt("playlist-pos");
|
|
|
|
if (pos != 0)
|
|
Core.SetPropertyInt("playlist-pos", 0);
|
|
}
|
|
|
|
public static void PlaylistLast()
|
|
{
|
|
int pos = Core.GetPropertyInt("playlist-pos");
|
|
int count = Core.GetPropertyInt("playlist-count");
|
|
|
|
if (pos < count - 1)
|
|
Core.SetPropertyInt("playlist-pos", count - 1);
|
|
}
|
|
|
|
public static void ShowHistory()
|
|
{
|
|
if (File.Exists(Core.ConfigFolder + "history.txt"))
|
|
ProcessHelp.ShellExecute(Core.ConfigFolder + "history.txt");
|
|
else
|
|
{
|
|
if (Msg.ShowQuestion("Create history.txt file in config folder?",
|
|
"mpv.net will write the date, time and filename of opened files to it.") == DialogResult.OK)
|
|
|
|
File.WriteAllText(Core.ConfigFolder + "history.txt", "");
|
|
}
|
|
}
|
|
|
|
public static void ShowInfo()
|
|
{
|
|
try
|
|
{
|
|
string performer, title, album, genre, date, duration, text = "";
|
|
long fileSize = 0;
|
|
string path = Core.GetPropertyString("path");
|
|
|
|
if (path.Contains("://"))
|
|
path = Core.GetPropertyString("media-title");
|
|
|
|
int width = Core.GetPropertyInt("video-params/w");
|
|
int height = Core.GetPropertyInt("video-params/h");
|
|
|
|
if (File.Exists(path))
|
|
{
|
|
fileSize = new FileInfo(path).Length;
|
|
|
|
if (CorePlayer.AudioTypes.Contains(path.Ext()))
|
|
{
|
|
using (MediaInfo mediaInfo = new MediaInfo(path))
|
|
{
|
|
performer = mediaInfo.GetInfo(MediaInfoStreamKind.General, "Performer");
|
|
title = mediaInfo.GetInfo(MediaInfoStreamKind.General, "Title");
|
|
album = mediaInfo.GetInfo(MediaInfoStreamKind.General, "Album");
|
|
genre = mediaInfo.GetInfo(MediaInfoStreamKind.General, "Genre");
|
|
date = mediaInfo.GetInfo(MediaInfoStreamKind.General, "Recorded_Date");
|
|
duration = mediaInfo.GetInfo(MediaInfoStreamKind.Audio, "Duration/String");
|
|
|
|
if (performer != "") text += "Artist: " + performer + "\n";
|
|
if (title != "") text += "Title: " + title + "\n";
|
|
if (album != "") text += "Album: " + album + "\n";
|
|
if (genre != "") text += "Genre: " + genre + "\n";
|
|
if (date != "") text += "Year: " + date + "\n";
|
|
if (duration != "") text += "Length: " + duration + "\n";
|
|
text += "Size: " + mediaInfo.GetInfo(MediaInfoStreamKind.General, "FileSize/String") + "\n";
|
|
text += "Type: " + path.Ext().ToUpper();
|
|
|
|
Core.CommandV("show-text", text, "5000");
|
|
return;
|
|
}
|
|
}
|
|
else if (CorePlayer.ImageTypes.Contains(path.Ext()))
|
|
{
|
|
using (MediaInfo mediaInfo = new MediaInfo(path))
|
|
{
|
|
text =
|
|
"Width: " + mediaInfo.GetInfo(MediaInfoStreamKind.Image, "Width") + "\n" +
|
|
"Height: " + mediaInfo.GetInfo(MediaInfoStreamKind.Image, "Height") + "\n" +
|
|
"Size: " + mediaInfo.GetInfo(MediaInfoStreamKind.General, "FileSize/String") + "\n" +
|
|
"Type: " + path.Ext().ToUpper();
|
|
|
|
Core.CommandV("show-text", text, "5000");
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
TimeSpan position = TimeSpan.FromSeconds(Core.GetPropertyDouble("time-pos"));
|
|
TimeSpan duration2 = TimeSpan.FromSeconds(Core.GetPropertyDouble("duration"));
|
|
string videoFormat = Core.GetPropertyString("video-format").ToUpper();
|
|
string audioCodec = Core.GetPropertyString("audio-codec-name").ToUpper();
|
|
|
|
text = path.FileName() + "\n" +
|
|
FormatTime(position.TotalMinutes) + ":" +
|
|
FormatTime(position.Seconds) + " / " +
|
|
FormatTime(duration2.TotalMinutes) + ":" +
|
|
FormatTime(duration2.Seconds) + "\n" +
|
|
$"{width} x {height}\n";
|
|
|
|
if (fileSize > 0)
|
|
text += Convert.ToInt32(fileSize / 1024.0 / 1024.0) + " MB\n";
|
|
|
|
text += $"{videoFormat}\n{audioCodec}";
|
|
|
|
Core.CommandV("show-text", text, "5000");
|
|
string FormatTime(double value) => ((int)value).ToString("00");
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
App.ShowException(e);
|
|
}
|
|
}
|
|
|
|
public static void OpenURL()
|
|
{
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
string clipboard = System.Windows.Forms.Clipboard.GetText();
|
|
|
|
if (string.IsNullOrEmpty(clipboard) || (!clipboard.Contains("://") && !File.Exists(clipboard)) ||
|
|
clipboard.Contains("\n"))
|
|
{
|
|
App.ShowError("No URL found", "The clipboard does not contain a valid URL or file.");
|
|
return;
|
|
}
|
|
|
|
Core.LoadFiles(new [] { clipboard }, false, Control.ModifierKeys.HasFlag(Keys.Control));
|
|
}));
|
|
}
|
|
|
|
public static void LoadSubtitle()
|
|
{
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
using (var d = new OpenFileDialog())
|
|
{
|
|
string path = Core.GetPropertyString("path");
|
|
|
|
if (File.Exists(path))
|
|
d.InitialDirectory = Path.GetDirectoryName(path);
|
|
|
|
d.Multiselect = true;
|
|
|
|
if (d.ShowDialog() == DialogResult.OK)
|
|
foreach (string filename in d.FileNames)
|
|
Core.CommandV("sub-add", filename);
|
|
}
|
|
}));
|
|
}
|
|
|
|
public static void LoadAudio()
|
|
{
|
|
App.InvokeOnMainThread(new Action(() => {
|
|
using (var d = new OpenFileDialog())
|
|
{
|
|
string path = Core.GetPropertyString("path");
|
|
if (File.Exists(path))
|
|
d.InitialDirectory = Path.GetDirectoryName(path);
|
|
d.Multiselect = true;
|
|
|
|
if (d.ShowDialog() == DialogResult.OK)
|
|
foreach (string i in d.FileNames)
|
|
Core.CommandV("audio-add", i);
|
|
}
|
|
}));
|
|
}
|
|
|
|
public static void CycleAudio()
|
|
{
|
|
MediaTrack[] audioTracks = Core.MediaTracks.Where(track => track.Type == "a").ToArray();
|
|
int len = audioTracks.Length;
|
|
|
|
if (len < 1)
|
|
{
|
|
Core.CommandV("show-text", "No audio tracks");
|
|
return;
|
|
}
|
|
|
|
int aid = Core.GetPropertyInt("aid");
|
|
|
|
if (len > 1)
|
|
{
|
|
if (++aid > len)
|
|
aid = 1;
|
|
|
|
Core.CommandV("set", "aid", aid.ToString());
|
|
}
|
|
|
|
Core.CommandV("show-text", aid + "/" + len + ": " + audioTracks[aid - 1].Text.Substring(3), "5000");
|
|
}
|
|
|
|
public static void ShowCommands()
|
|
{
|
|
string code = @"
|
|
foreach ($item in ($json | ConvertFrom-Json | foreach { $_ } | sort name))
|
|
{
|
|
''
|
|
$item.name
|
|
|
|
foreach ($arg in $item.args)
|
|
{
|
|
$value = $arg.name + ' <' + $arg.type.ToLower() + '>'
|
|
|
|
if ($arg.optional -eq $true)
|
|
{
|
|
$value = '[' + $value + ']'
|
|
}
|
|
|
|
' ' + $value
|
|
}
|
|
}";
|
|
|
|
string json = Core.GetPropertyString("command-list");
|
|
ShowTextWithEditor("command-list", PowerShell.InvokeAndReturnString(code, "json", json));
|
|
}
|
|
|
|
public static void ShowProperties()
|
|
{
|
|
var props = Core.GetPropertyString("property-list").Split(',').OrderBy(prop => prop);
|
|
ShowTextWithEditor("property-list", string.Join(BR, props));
|
|
}
|
|
|
|
public static void ShowTextWithEditor(string name, string text)
|
|
{
|
|
string file = Path.Combine(Path.GetTempPath(), name + ".txt");
|
|
App.TempFiles.Add(file);
|
|
File.WriteAllText(file, BR + text.Trim() + BR);
|
|
ProcessHelp.ShellExecute(file);
|
|
}
|
|
|
|
public static void ScaleWindow(float factor) => Core.RaiseScaleWindow(factor);
|
|
|
|
public static void WindowScale(float value) => Core.RaiseWindowScale(value);
|
|
|
|
public static void ShowText(string text, int duration = 0, int fontSize = 0)
|
|
{
|
|
if (string.IsNullOrEmpty(text))
|
|
return;
|
|
|
|
if (duration == 0)
|
|
duration = Core.GetPropertyInt("osd-duration");
|
|
|
|
if (fontSize == 0)
|
|
fontSize = Core.GetPropertyInt("osd-font-size");
|
|
|
|
Core.Command("show-text \"${osd-ass-cc/0}{\\\\fs" + fontSize +
|
|
"}${osd-ass-cc/1}" + text + "\" " + duration);
|
|
}
|
|
|
|
public static void ShowMediaInfo(string[] args)
|
|
{
|
|
string path = Core.GetPropertyString("path");
|
|
|
|
if (File.Exists(path) && !path.Contains(@"\\.\pipe\"))
|
|
{
|
|
using (MediaInfo mediaInfo = new MediaInfo(path))
|
|
{
|
|
bool full = args.Contains("full");
|
|
bool raw = args.Contains("raw");
|
|
string text = mediaInfo.GetSummary(full, raw);
|
|
ShowTextWithEditor(Path.GetFileName(path), text);
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void ShowCommandPalette() => App.InvokeOnMainThread(ShowCommandPaletteInternal);
|
|
|
|
static void ShowCommandPaletteInternal()
|
|
{
|
|
CommandPalette.Instance.SetItems(CommandPalette.GetItems());
|
|
MainForm.Instance.ShowCommandPalette();
|
|
CommandPalette.Instance.SelectFirst();
|
|
}
|
|
|
|
public static void ShowPlaylist() => App.InvokeOnMainThread(ShowPlaylistInternal);
|
|
|
|
static void ShowPlaylistInternal()
|
|
{
|
|
int count = Core.GetPropertyInt("playlist-count");
|
|
string currentPath = Core.GetPropertyString("path");
|
|
CommandPaletteItem currentItem = null;
|
|
|
|
if (count <= 0)
|
|
return;
|
|
|
|
List<CommandPaletteItem> items = new List<CommandPaletteItem>();
|
|
|
|
for (int i = 0; i < count; i++)
|
|
{
|
|
int index = i;
|
|
string file = Core.GetPropertyString($"playlist/{i}/filename");
|
|
|
|
CommandPaletteItem item = new CommandPaletteItem() {
|
|
Text = PathHelp.GetFileName(file),
|
|
Action = () => Core.SetPropertyInt("playlist-pos", index)
|
|
};
|
|
|
|
items.Add(item);
|
|
|
|
if (currentPath.ToLowerEx() == file.ToLowerEx())
|
|
currentItem = item;
|
|
}
|
|
|
|
CommandPalette.Instance.SetItems(items);
|
|
|
|
if (currentItem != null)
|
|
{
|
|
CommandPalette.Instance.MainListView.SelectedItem = currentItem;
|
|
CommandPalette.Instance.MainListView.ScrollIntoView(
|
|
CommandPalette.Instance.MainListView.SelectedItem);
|
|
}
|
|
|
|
MainForm.Instance.ShowCommandPalette();
|
|
}
|
|
}
|
|
}
|