using Microsoft.Xna.Framework; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; namespace Barotrauma { public static class FileSelection { private static bool open; public static bool Open { get { return open; } set { if (value && backgroundFrame == null) { Init(); } if (!value) { fileSystemWatcher?.Dispose(); fileSystemWatcher = null; } open = value; } } private static GUIFrame backgroundFrame; private static GUIFrame window; private static GUIListBox sidebar; private static GUIListBox fileList; private static GUITextBox directoryBox; private static GUITextBox filterBox; private static GUITextBox fileBox; private static GUIDropDown fileTypeDropdown; private static GUIButton openButton; private static FileSystemWatcher fileSystemWatcher; private static string currentFileTypePattern; private static readonly string[] ignoredDrivePrefixes = new string[] { "/sys/", "/snap/" }; private static string currentDirectory; public static string CurrentDirectory { get { return currentDirectory; } set { string[] dirSplit = value.Replace('\\', '/').Split('/'); List dirs = new List(); for (int i = 0; i < dirSplit.Length; i++) { if (dirSplit[i].Trim() == "..") { if (dirs.Count > 1) { dirs.RemoveAt(dirs.Count - 1); } } else if (dirSplit[i].Trim() != ".") { dirs.Add(dirSplit[i]); } } currentDirectory = string.Join("/", dirs); if (!currentDirectory.EndsWith("/")) { currentDirectory += "/"; } fileSystemWatcher?.Dispose(); fileSystemWatcher = new FileSystemWatcher(currentDirectory) { Filter = "*", NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName }; fileSystemWatcher.Created += OnFileSystemChanges; fileSystemWatcher.Deleted += OnFileSystemChanges; fileSystemWatcher.Renamed += OnFileSystemChanges; fileSystemWatcher.EnableRaisingEvents = true; RefreshFileList(); } } public static Action OnFileSelected { get; set; } private static void OnFileSystemChanges(object sender, FileSystemEventArgs e) { switch (e.ChangeType) { case WatcherChangeTypes.Created: { var itemFrame = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.05f), fileList.Content.RectTransform), e.Name) { UserData = (bool?)Directory.Exists(e.FullPath) }; if ((itemFrame.UserData as bool?) ?? false) { itemFrame.Text += "/"; } fileList.Content.RectTransform.SortChildren(SortFiles); } break; case WatcherChangeTypes.Deleted: { var itemFrame = fileList.Content.FindChild(c => (c is GUITextBlock tb) && (tb.Text == e.Name || tb.Text == e.Name + "/")); if (itemFrame != null) { fileList.RemoveChild(itemFrame); } } break; case WatcherChangeTypes.Renamed: { RenamedEventArgs renameArgs = e as RenamedEventArgs; var itemFrame = fileList.Content.FindChild(c => (c is GUITextBlock tb) && (tb.Text == renameArgs.OldName || tb.Text == renameArgs.OldName + "/")) as GUITextBlock; itemFrame.UserData = (bool?)Directory.Exists(e.FullPath); itemFrame.Text = renameArgs.Name; if ((itemFrame.UserData as bool?) ?? false) { itemFrame.Text += "/"; } fileList.Content.RectTransform.SortChildren(SortFiles); } break; } } private static int SortFiles(RectTransform r1, RectTransform r2) { string file1 = (r1.GUIComponent as GUITextBlock)?.Text ?? ""; string file2 = (r2.GUIComponent as GUITextBlock)?.Text ?? ""; bool dir1 = (r1.GUIComponent.UserData as bool?) ?? false; bool dir2 = (r2.GUIComponent.UserData as bool?) ?? false; if (dir1 && !dir2) { return -1; } else if (!dir1 && dir2) { return 1; } return string.Compare(file1, file2); } public static void Init() { backgroundFrame = new GUIFrame(new RectTransform(Vector2.One, GUI.Canvas), style: null) { Color = Color.Black * 0.5f, HoverColor = Color.Black * 0.5f, SelectedColor = Color.Black * 0.5f, PressedColor = Color.Black * 0.5f, }; window = new GUIFrame(new RectTransform(Vector2.One * 0.8f, backgroundFrame.RectTransform, Anchor.Center)); var horizontalLayout = new GUILayoutGroup(new RectTransform(Vector2.One * 0.9f, window.RectTransform, Anchor.Center), true); sidebar = new GUIListBox(new RectTransform(new Vector2(0.29f, 1.0f), horizontalLayout.RectTransform)); var drives = DriveInfo.GetDrives(); foreach (var drive in drives) { if (drive.DriveType == DriveType.Ram) { continue; } if (ignoredDrivePrefixes.Any(p => drive.Name.StartsWith(p))) { continue; } new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.05f), sidebar.Content.RectTransform), drive.Name.Replace('\\','/')); } sidebar.OnSelected = (child, userdata) => { CurrentDirectory = (child as GUITextBlock).Text; return false; }; //spacing between sidebar and fileListLayout new GUIFrame(new RectTransform(new Vector2(0.01f, 1.0f), horizontalLayout.RectTransform), style: null); var fileListLayout = new GUILayoutGroup(new RectTransform(new Vector2(0.7f, 1.0f), horizontalLayout.RectTransform)); var firstRow = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.04f), fileListLayout.RectTransform), isHorizontal: true, childAnchor: Anchor.CenterLeft); new GUIButton(new RectTransform(new Vector2(0.05f, 1.0f), firstRow.RectTransform), "^") { OnClicked = MoveToParentDirectory }; directoryBox = new GUITextBox(new RectTransform(new Vector2(0.7f, 1.0f), firstRow.RectTransform)) { OverflowClip = true, OnEnterPressed = (tb, txt) => { if (Directory.Exists(txt)) { CurrentDirectory = txt; return true; } else { tb.Text = CurrentDirectory; return false; } } }; filterBox = new GUITextBox(new RectTransform(new Vector2(0.25f, 1.0f), firstRow.RectTransform)) { OverflowClip = true }; firstRow.RectTransform.MinSize = new Point(0, firstRow.RectTransform.Children.Max(c => c.MinSize.Y)); filterBox.OnTextChanged += (txtbox, txt) => { RefreshFileList(); return true; }; //spacing between rows new GUIFrame(new RectTransform(new Vector2(1.0f, 0.01f), fileListLayout.RectTransform), style: null); fileList = new GUIListBox(new RectTransform(new Vector2(1.0f, 0.85f), fileListLayout.RectTransform)) { OnSelected = (child, userdata) => { if (userdata == null) { return false; } var fileName = (child as GUITextBlock).Text; fileBox.Text = fileName; if (PlayerInput.DoubleClicked()) { bool isDir = (userdata as bool?).Value; if (isDir) { CurrentDirectory += fileName; } else { OnFileSelected?.Invoke(CurrentDirectory + fileName); Open = false; } } return true; } }; //spacing between rows new GUIFrame(new RectTransform(new Vector2(1.0f, 0.01f), fileListLayout.RectTransform), style: null); var thirdRow = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.04f), fileListLayout.RectTransform), true); fileBox = new GUITextBox(new RectTransform(new Vector2(0.7f, 1.0f), thirdRow.RectTransform)) { OnEnterPressed = (tb, txt) => openButton?.OnClicked?.Invoke(openButton, null) ?? false }; fileTypeDropdown = new GUIDropDown(new RectTransform(new Vector2(0.3f, 1.0f), thirdRow.RectTransform), dropAbove: true) { OnSelected = (child, userdata) => { currentFileTypePattern = (child as GUITextBlock).UserData as string; RefreshFileList(); return true; } }; fileTypeDropdown.Select(4); //spacing between rows new GUIFrame(new RectTransform(new Vector2(1.0f, 0.01f), fileListLayout.RectTransform), style: null); var fourthRow = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.04f), fileListLayout.RectTransform), true); //padding for open/cancel buttons new GUIFrame(new RectTransform(new Vector2(0.7f, 1.0f), fourthRow.RectTransform), style: null); openButton = new GUIButton(new RectTransform(new Vector2(0.15f, 1.0f), fourthRow.RectTransform), TextManager.Get("opensubbutton")) { OnClicked = (btn, obj) => { if (Directory.Exists(Path.Combine(CurrentDirectory, fileBox.Text))) { CurrentDirectory += fileBox.Text; } if (!File.Exists(CurrentDirectory + fileBox.Text)) { return false; } OnFileSelected?.Invoke(CurrentDirectory + fileBox.Text); Open = false; return false; } }; new GUIButton(new RectTransform(new Vector2(0.15f, 1.0f), fourthRow.RectTransform), TextManager.Get("cancel")) { OnClicked = (btn, obj) => { Open = false; return false; } }; CurrentDirectory = Directory.GetCurrentDirectory(); } public static void ClearFileTypeFilters() { if (backgroundFrame == null) { Init(); } fileTypeDropdown.ClearChildren(); } public static void AddFileTypeFilter(string name, string pattern) { if (backgroundFrame == null) { Init(); } fileTypeDropdown.AddItem(name + " (" + pattern + ")", pattern); } public static void SelectFileTypeFilter(string pattern) { if (backgroundFrame == null) { Init(); } fileTypeDropdown.SelectItem(pattern); } public static void RefreshFileList() { fileList.Content.ClearChildren(); fileList.BarScroll = 0.0f; try { var directories = Directory.EnumerateDirectories(currentDirectory, "*" + filterBox.Text + "*"); foreach (var directory in directories) { string txt = directory; if (txt.StartsWith(currentDirectory)) { txt = txt.Substring(currentDirectory.Length); } if (!txt.EndsWith("/")) { txt += "/"; } var itemFrame = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.05f), fileList.Content.RectTransform), txt) { UserData = (bool?)true }; var folderIcon = new GUIImage(new RectTransform(new Point((int)(itemFrame.Rect.Height * 0.8f)), itemFrame.RectTransform, Anchor.CenterLeft) { AbsoluteOffset = new Point((int)(itemFrame.Rect.Height * 0.25f), 0) }, style: "OpenButton", scaleToFit: true); itemFrame.Padding = new Vector4(folderIcon.Rect.Width * 1.5f, itemFrame.Padding.Y, itemFrame.Padding.Z, itemFrame.Padding.W); } IEnumerable files = null; foreach (string pattern in currentFileTypePattern.Split(',')) { string patternTrimmed = pattern.Trim(); patternTrimmed = "*" + filterBox.Text + "*" + patternTrimmed; if (files == null) { files = Directory.EnumerateFiles(currentDirectory, patternTrimmed); } else { files = files.Concat(Directory.EnumerateFiles(currentDirectory, patternTrimmed)); } } foreach (var file in files) { string txt = file; if (txt.StartsWith(currentDirectory)) { txt = txt.Substring(currentDirectory.Length); } var itemFrame = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.05f), fileList.Content.RectTransform), txt) { UserData = (bool?)false }; } } catch (Exception e) { new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.05f), fileList.Content.RectTransform), "Could not list items in directory: " + e.Message) { CanBeFocused = false }; } directoryBox.Text = currentDirectory; fileBox.Text = ""; fileList.Deselect(); } public static bool MoveToParentDirectory(GUIButton button, object userdata) { string dir = CurrentDirectory; if (dir.EndsWith("/")) { dir = dir.Substring(0, dir.Length - 1); } int index = dir.LastIndexOf("/"); if (index < 0) { return false; } CurrentDirectory = CurrentDirectory.Substring(0, index+1); return false; } public static void AddToGUIUpdateList() { if (!Open) { return; } backgroundFrame?.AddToGUIUpdateList(); } } }