#region License Information (GPL v3) /* ShareX - A program that allows you to take screenshots and share any file type Copyright (c) 2007-2021 ShareX Team This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Optionally you can also view the license at . */ #endregion License Information (GPL v3) // http://www.lyquidity.com/devblog/?p=136 using ShareX.HelpersLib.Properties; using System; using System.Reflection; using System.Windows.Forms; namespace ShareX.HelpersLib { /// /// Wraps System.Windows.Forms.OpenFileDialog to make it present /// a vista-style dialog. /// public class FolderSelectDialog : IDisposable { // Wrapped dialog private OpenFileDialog ofd; /// /// Default constructor /// public FolderSelectDialog() { ofd = new OpenFileDialog(); ofd.Filter = "Folders|\n"; ofd.AddExtension = false; ofd.CheckFileExists = false; ofd.DereferenceLinks = true; ofd.Multiselect = false; } /// /// Gets/Sets the initial folder to be selected. A null value selects the current directory. /// public string InitialDirectory { get { return ofd.InitialDirectory; } set { ofd.InitialDirectory = string.IsNullOrEmpty(value) ? Environment.CurrentDirectory : value; } } /// /// Gets/Sets the title to show in the dialog /// public string Title { get { return ofd.Title; } set { ofd.Title = string.IsNullOrEmpty(value) ? Resources.FolderSelectDialog_Title_Select_a_folder : value; } } /// /// Gets the selected folder /// public string FileName { get { return ofd.FileName; } } /// /// Shows the dialog /// /// True if the user presses OK else false public bool ShowDialog() { return ShowDialog(NativeMethods.GetActiveWindow()); } /// /// Shows the dialog /// /// Handle of the control to be parent /// True if the user presses OK else false public bool ShowDialog(IntPtr hWndOwner) { bool flag; if (Helpers.IsWindowsVistaOrGreater()) { Reflector r = new Reflector("System.Windows.Forms"); uint num = 0; Type typeIFileDialog = r.GetType("FileDialogNative.IFileDialog"); object dialog = r.Call(ofd, "CreateVistaDialog"); r.Call(ofd, "OnBeforeVistaDialog", dialog); uint options = (uint)r.CallAs(typeof(FileDialog), ofd, "GetOptions"); options |= (uint)r.GetEnum("FileDialogNative.FOS", "FOS_PICKFOLDERS"); r.CallAs(typeIFileDialog, dialog, "SetOptions", options); object pfde = r.New("FileDialog.VistaDialogEvents", ofd); object[] parameters = new object[] { pfde, num }; r.CallAs2(typeIFileDialog, dialog, "Advise", parameters); num = (uint)parameters[1]; try { int num2 = (int)r.CallAs(typeIFileDialog, dialog, "Show", hWndOwner); flag = num2 == 0; } finally { r.CallAs(typeIFileDialog, dialog, "Unadvise", num); GC.KeepAlive(pfde); } } else { using (FolderBrowserDialog fbd = new FolderBrowserDialog()) { fbd.Description = Title; fbd.SelectedPath = InitialDirectory; if (fbd.ShowDialog(new WindowWrapper(hWndOwner)) != DialogResult.OK) return false; ofd.FileName = fbd.SelectedPath; flag = true; } } return flag; } public void Dispose() { if (ofd != null) { ofd.Dispose(); } } } /// /// Creates IWin32Window around an IntPtr /// internal class WindowWrapper : IWin32Window { /// /// Original ptr /// public IntPtr Handle { get { return hwnd; } } private IntPtr hwnd; /// /// Constructor /// /// Handle to wrap public WindowWrapper(IntPtr handle) { hwnd = handle; } } /// /// This class is from the Front-End for Dosbox and is used to present a 'vista' dialog box to select folders. /// Being able to use a vista style dialog box to select folders is much better then using the shell folder browser. /// http://code.google.com/p/fed/ /// /// Example: /// var r = new Reflector("System.Windows.Forms"); /// internal class Reflector { #region variables private string m_ns; private Assembly m_asmb; #endregion variables #region Constructors /// /// Constructor /// /// The namespace containing types to be used public Reflector(string ns) : this(ns, ns) { } /// /// Constructor /// /// A specific assembly name (used if the assembly name does not tie exactly with the namespace) /// The namespace containing types to be used public Reflector(string an, string ns) { m_ns = ns; m_asmb = null; foreach (AssemblyName aN in Assembly.GetExecutingAssembly().GetReferencedAssemblies()) { if (aN.FullName.StartsWith(an)) { m_asmb = Assembly.Load(aN); break; } } } #endregion Constructors #region Methods /// /// Return a Type instance for a type 'typeName' /// /// The name of the type /// A type instance public Type GetType(string typeName) { Type type = null; string[] names = typeName.Split('.'); if (names.Length > 0) type = m_asmb.GetType(m_ns + "." + names[0]); for (int i = 1; i < names.Length; ++i) { type = type.GetNestedType(names[i], BindingFlags.NonPublic); } return type; } /// /// Create a new object of a named type passing along any params /// /// The name of the type to create /// /// An instantiated type public object New(string name, params object[] parameters) { Type type = GetType(name); ConstructorInfo[] ctorInfos = type.GetConstructors(); foreach (ConstructorInfo ci in ctorInfos) { try { return ci.Invoke(parameters); } catch { } } return null; } /// /// Calls method 'func' on object 'obj' passing parameters 'parameters' /// /// The object on which to excute function 'func' /// The function to execute /// The parameters to pass to function 'func' /// The result of the function invocation public object Call(object obj, string func, params object[] parameters) { return Call2(obj, func, parameters); } /// /// Calls method 'func' on object 'obj' passing parameters 'parameters' /// /// The object on which to excute function 'func' /// The function to execute /// The parameters to pass to function 'func' /// The result of the function invocation public object Call2(object obj, string func, object[] parameters) { return CallAs2(obj.GetType(), obj, func, parameters); } /// /// Calls method 'func' on object 'obj' which is of type 'type' passing parameters 'parameters' /// /// The type of 'obj' /// The object on which to excute function 'func' /// The function to execute /// The parameters to pass to function 'func' /// The result of the function invocation public object CallAs(Type type, object obj, string func, params object[] parameters) { return CallAs2(type, obj, func, parameters); } /// /// Calls method 'func' on object 'obj' which is of type 'type' passing parameters 'parameters' /// /// The type of 'obj' /// The object on which to excute function 'func' /// The function to execute /// The parameters to pass to function 'func' /// The result of the function invocation public object CallAs2(Type type, object obj, string func, object[] parameters) { MethodInfo methInfo = type.GetMethod(func, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); return methInfo.Invoke(obj, parameters); } /// /// Returns the value of property 'prop' of object 'obj' /// /// The object containing 'prop' /// The property name /// The property value public object Get(object obj, string prop) { return GetAs(obj.GetType(), obj, prop); } /// /// Returns the value of property 'prop' of object 'obj' which has type 'type' /// /// The type of 'obj' /// The object containing 'prop' /// The property name /// The property value public object GetAs(Type type, object obj, string prop) { PropertyInfo propInfo = type.GetProperty(prop, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic); return propInfo.GetValue(obj, null); } /// /// Returns an enum value /// /// The name of enum type /// The name of the value /// The enum value public object GetEnum(string typeName, string name) { Type type = GetType(typeName); FieldInfo fieldInfo = type.GetField(name); return fieldInfo.GetValue(null); } #endregion Methods } }