it-swarm-ru.tech

Как связать расширение файла с текущим исполняемым файлом в C #

Я хотел бы связать расширение файла с текущим исполняемым файлом в C # . Таким образом, когда пользователь щелкает файл впоследствии в Проводнике, он запускает мой исполняемый файл с указанным файлом в качестве первого аргумента .Идеально он также установил бы иконку для указанных расширений файла на иконку для моего исполняемого файла . Спасибо всем.

51
Chris

Похоже, что нет .Net API для прямого управления ассоциациями файлов, но вы можете использовать классы Registry для чтения и записи необходимых вам ключей.

Вам нужно создать ключ в HKEY_CLASSES_ROOT с именем, указанным для вашего расширения файла (например: ".txt"). Задайте в качестве значения по умолчанию для этого ключа уникальное имя для вашего типа файла, например «Acme.TextFile». Затем создайте другой ключ в HKEY_CLASSES_ROOT с именем, установленным в «Acme.TextFile». Добавьте подраздел «DefaultIcon» и установите значение ключа по умолчанию для файла, содержащего значок, который вы хотите использовать для этого типа файлов. Добавьте еще одного родного брата под названием «Shell». Под ключом «Shell» добавьте ключ для каждого действия, которое вы хотите сделать доступным через контекстное меню проводника, установив значение по умолчанию для каждого ключа, путь к вашему исполняемому файлу, затем пробел и «% 1» для представления пути. в выбранный файл.

Например, вот пример файла реестра для создания связи между файлами .txt и EmEditor:

 Редактор реестра Windows, версия 5.00 

 [HKEY_CLASSES_ROOT\.txt] 
 @ = "Emeditor.txt" 

 [HKEY_CLASSES_ROOT\emeditor.txt] 
 @ = "Text Документ»

 [HKEY_CLASSES_ROOT\emeditor.txt\DefaultIcon] 
 @ = "% SystemRoot% \\ \\ SysWow64 imageres.dll, -102" 

 [HKEY_CLASSES_ROOT\emeditor.txt\Shell] 
.__ [HKEY_CLASSES_ROOT\emeditor.txt\Shell\Open] 
.__ [HKEY_CLASSES_ROOT\emeditor.txt\Shell\Open\Command] .__ @ = "\" C:...\\ Program Files \\ EmEditor \\ EMEDITOR.EXE\"\"% 1\"" 

 [HKEY_CLASSES_ROOT\emeditor.txt\Shell\print] 

 [HKEY_CLASSES_ROOT\emeditor.txt\Оболочка\print\command] 
 @ = "\" C: \\ Program Files \\ EmEditor \\ EMEDITOR.EXE\"/ p \"% 1\"" 
40
X-Cubed

Кроме того, если вы решили пойти по пути регистрации, имейте в виду, что текущие ассоциации пользователей находятся в HKEY_CURRENT_USER\Software\Classes . Может быть, лучше добавить туда свое приложение, а не классы локальной машины.

Если ваша программа будет запускаться пользователями с ограниченными правами, вы все равно не сможете изменить CLASSES_ROOT.

25
Ishmaeel

Если вы используете развертывание ClickOnce, это все обрабатывается за вас (по крайней мере, в VS2008 SP1); просто:

  • Свойства проекта
  • Публиковать
  • Опции
  • Файловые ассоциации
  • (добавьте все, что вам нужно)

(обратите внимание, что он должен быть с полным доверием, целевым .NET 3.5 и быть установлен для автономного использования)

См. Также MSDN: Как создать файловые ассоциации для приложения ClickOnce

11
Marc Gravell

Вот полный пример:

public class FileAssociation
{
    public string Extension { get; set; }
    public string ProgId { get; set; }
    public string FileTypeDescription { get; set; }
    public string ExecutableFilePath { get; set; }
}

public class FileAssociations
{
    // needed so that Explorer windows get refreshed after the registry is updated
    [System.Runtime.InteropServices.DllImport("Shell32.dll")]
    private static extern int SHChangeNotify(int eventId, int flags, IntPtr item1, IntPtr item2);

    private const int SHCNE_ASSOCCHANGED = 0x8000000;
    private const int SHCNF_FLUSH = 0x1000;

    public static void EnsureAssociationsSet()
    {
        var filePath = Process.GetCurrentProcess().MainModule.FileName;
        EnsureAssociationsSet(
            new FileAssociation
            {
                Extension = ".binlog",
                ProgId = "MSBuildBinaryLog",
                FileTypeDescription = "MSBuild Binary Log",
                ExecutableFilePath = filePath
            },
            new FileAssociation
            {
                Extension = ".buildlog",
                ProgId = "MSBuildStructuredLog",
                FileTypeDescription = "MSBuild Structured Log",
                ExecutableFilePath = filePath
            });
    }

    public static void EnsureAssociationsSet(params FileAssociation[] associations)
    {
        bool madeChanges = false;
        foreach (var association in associations)
        {
            madeChanges |= SetAssociation(
                association.Extension,
                association.ProgId,
                association.FileTypeDescription,
                association.ExecutableFilePath);
        }

        if (madeChanges)
        {
            SHChangeNotify(SHCNE_ASSOCCHANGED, SHCNF_FLUSH, IntPtr.Zero, IntPtr.Zero);
        }
    }

    public static bool SetAssociation(string extension, string progId, string fileTypeDescription, string applicationFilePath)
    {
        bool madeChanges = false;
        madeChanges |= SetKeyDefaultValue(@"Software\Classes\" + extension, progId);
        madeChanges |= SetKeyDefaultValue(@"Software\Classes\" + progId, fileTypeDescription);
        madeChanges |= SetKeyDefaultValue([email protected]"Software\Classes\{progId}\Shell\open\command", "\"" + applicationFilePath + "\" \"%1\"");
        return madeChanges;
    }

    private static bool SetKeyDefaultValue(string keyPath, string value)
    {
        using (var key = Registry.CurrentUser.CreateSubKey(keyPath))
        {
            if (key.GetValue(null) as string != value)
            {
                key.SetValue(null, value);
                return true;
            }
        }

        return false;
    }
9
Kirill Osenkov

Могут быть конкретные причины, по которым вы решили не использовать установочный пакет для своего проекта, но установочный пакет является отличным местом для простого выполнения задач по настройке приложения, таких как регистрация расширений файлов, добавление ярлыков на рабочем столе и т.д.

Вот как создать ассоциацию расширения файлов с помощью встроенных инструментов установки Visual Studio:

  1. В существующем решении C # добавьте новый проект и выберите тип проекта как Other Project Types -> Setup and Deployment -> Setup Project (или попробуйте мастер установки)

  2. Сконфигурируйте свой установщик (достаточно документации для этого, если вам нужна помощь)

  3. Щелкните правой кнопкой мыши проект установки в обозревателе решений, выберите View -> File Types, а затем добавьте расширение, которое вы хотите зарегистрировать, вместе с программой для его запуска.

Этот метод имеет дополнительное преимущество очистки после себя, если пользователь запускает деинсталляцию для вашего приложения.

7
Paul J

Чтобы быть точным в отношении «реестра Windows»:

Я создаю ключи под HKEY_CURRENT_USER\Software\Classes (как сказал Измаил)

и следуйте инструкциям, полученным от X-Cubed.

Пример кода выглядит так:

private void Create_abc_FileAssociation()
{
    /***********************************/
    /**** Key1: Create ".abc" entry ****/
    /***********************************/
    Microsoft.Win32.RegistryKey key1 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key1.CreateSubKey("Classes");
    key1 = key1.OpenSubKey("Classes", true);

    key1.CreateSubKey(".abc");
    key1 = key1.OpenSubKey(".abc", true);
    key1.SetValue("", "DemoKeyValue"); // Set default key value

    key1.Close();

    /*******************************************************/
    /**** Key2: Create "DemoKeyValue\DefaultIcon" entry ****/
    /*******************************************************/
    Microsoft.Win32.RegistryKey key2 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key2.CreateSubKey("Classes");
    key2 = key2.OpenSubKey("Classes", true);

    key2.CreateSubKey("DemoKeyValue");
    key2 = key2.OpenSubKey("DemoKeyValue", true);

    key2.CreateSubKey("DefaultIcon");
    key2 = key2.OpenSubKey("DefaultIcon", true);
    key2.SetValue("", "\"" + "(The icon path you desire)" + "\""); // Set default key value

    key2.Close();

    /**************************************************************/
    /**** Key3: Create "DemoKeyValue\Shell\open\command" entry ****/
    /**************************************************************/
    Microsoft.Win32.RegistryKey key3 = Microsoft.Win32.Registry.CurrentUser.OpenSubKey("Software", true);

    key3.CreateSubKey("Classes");
    key3 = key3.OpenSubKey("Classes", true);

    key3.CreateSubKey("DemoKeyValue");
    key3 = key3.OpenSubKey("DemoKeyValue", true);

    key3.CreateSubKey("Shell");
    key3 = key3.OpenSubKey("Shell", true);

    key3.CreateSubKey("open");
    key3 = key3.OpenSubKey("open", true);

    key3.CreateSubKey("command");
    key3 = key3.OpenSubKey("command", true);
    key3.SetValue("", "\"" + "(The application path you desire)" + "\"" + " \"%1\""); // Set default key value

    key3.Close();
}

Просто покажите вам, ребята, быстрое демо, очень простое для понимания. Вы можете изменить эти ключевые значения, и все хорошо.

5
Strong

Ассоциации файлов определены в реестре в разделе HKEY_CLASSES_ROOT.

Вот пример VB.NET здесь который я могу легко перенести на C #.

2
Steve Morgan

Приведенный ниже код является функцией, которая должна работать, он добавляет необходимые значения в реестр Windows. Обычно я запускаю SelfCreateAssociation (". Abc") в моем исполняемом файле. (конструктор формы или onload или onshown) Он будет обновлять запись в реестре для текущего пользователя каждый раз, когда исполняемый файл выполняется. (хорошо для отладки, если у вас есть некоторые изменения) . Если вам нужна подробная информация о ключах реестра, проверьте эту ссылку MSDN.

https://msdn.Microsoft.com/en-us/library/windows/desktop/dd758090(v=vs.85).aspx

Чтобы получить больше информации об общем разделе реестра ClassesRoot. Смотрите эту статью MSDN.

https://msdn.Microsoft.com/en-us/library/windows/desktop/ms724475(v=vs.85).aspx

public enum KeyHiveSmall
{
    ClassesRoot,
    CurrentUser,
    LocalMachine,
}

/// <summary>
/// Create an associaten for a file extension in the windows registry
/// CreateAssociation(@"vendor.application",".tmf","Tool file",@"C:\Windows\SYSWOW64\notepad.exe",@"%SystemRoot%\SYSWOW64\notepad.exe,0");
/// </summary>
/// <param name="ProgID">e.g. vendor.application</param>
/// <param name="extension">e.g. .tmf</param>
/// <param name="description">e.g. Tool file</param>
/// <param name="application">e.g.  @"C:\Windows\SYSWOW64\notepad.exe"</param>
/// <param name="icon">@"%SystemRoot%\SYSWOW64\notepad.exe,0"</param>
/// <param name="Hive">e.g. The user-specific settings have priority over the computer settings. KeyHive.LocalMachine  need admin rights</param>
public static void CreateAssociation(string ProgID, string extension, string description, string application, string icon, KeyHiveSmall Hive = KeyHiveSmall.CurrentUser)
{
    RegistryKey selectedKey = null;

    switch (Hive)
    {
        case KeyHiveSmall.ClassesRoot:
            Microsoft.Win32.Registry.ClassesRoot.CreateSubKey(extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.ClassesRoot.CreateSubKey(ProgID);
            break;

        case KeyHiveSmall.CurrentUser:
            Microsoft.Win32.Registry.CurrentUser.CreateSubKey(@"Software\Classes\" + extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.CurrentUser.CreateSubKey(@"Software\Classes\" + ProgID);
            break;

        case KeyHiveSmall.LocalMachine:
            Microsoft.Win32.Registry.LocalMachine.CreateSubKey(@"Software\Classes\" + extension).SetValue("", ProgID);
            selectedKey = Microsoft.Win32.Registry.LocalMachine.CreateSubKey(@"Software\Classes\" + ProgID);
            break;
    }

    if (selectedKey != null)
    {
        if (description != null)
        {
            selectedKey.SetValue("", description);
        }
        if (icon != null)
        {
            selectedKey.CreateSubKey("DefaultIcon").SetValue("", icon, RegistryValueKind.ExpandString);
            selectedKey.CreateSubKey(@"Shell\Open").SetValue("icon", icon, RegistryValueKind.ExpandString);
        }
        if (application != null)
        {
            selectedKey.CreateSubKey(@"Shell\Open\command").SetValue("", "\"" + application + "\"" + " \"%1\"", RegistryValueKind.ExpandString);
        }
    }
    selectedKey.Flush();
    selectedKey.Close();
}

 /// <summary>
    /// Creates a association for current running executable
    /// </summary>
    /// <param name="extension">e.g. .tmf</param>
    /// <param name="Hive">e.g. KeyHive.LocalMachine need admin rights</param>
    /// <param name="description">e.g. Tool file. Displayed in Explorer</param>
    public static void SelfCreateAssociation(string extension, KeyHiveSmall Hive = KeyHiveSmall.CurrentUser, string description = "")
    {
        string ProgID = System.Reflection.Assembly.GetExecutingAssembly().EntryPoint.DeclaringType.FullName;
        string FileLocation = System.Reflection.Assembly.GetExecutingAssembly().Location;
        CreateAssociation(ProgID, extension, description, FileLocation, FileLocation + ",0", Hive);
    }
2
Carsten R.

Начиная с Windows 7 существует два инструмента cmd, которые позволяют легко создавать простые ассоциации файлов. Это assoc и ftype . Вот основное объяснение каждой команды.

  • Assoc - связывает расширение файла (например, «.txt») с «типом файла».
  • FType - определяет исполняемый файл, запускаемый, когда пользователь открывает данный «тип файла».

Обратите внимание, что это инструменты cmd, а не исполняемые файлы (exe). Это означает, что их можно запускать только в окне cmd или с помощью ShellExecute с «cmd/c assoc.» Вы можете узнать о них больше по ссылкам или набрав "assoc /?" и "ftype /?" в командной строке. 

Таким образом, чтобы связать приложение с расширением .bob, вы можете открыть окно cmd (WindowKey + R, введите cmd, нажмите enter) и выполнить следующее:

assoc .bob=BobFile
ftype BobFile=c:\temp\BobView.exe "%1"

Это намного проще, чем возиться с реестром, и с большей вероятностью будет работать в будущих версиях Windows. 

В завершение, вот функция C # для создания ассоциации файлов:

public static int setFileAssociation(string[] extensions, string fileType, string openCommandString) {
    int v = execute("cmd", "/c ftype " + fileType + "=" + openCommandString);
    foreach (string ext in extensions) {
        v = execute("cmd", "/c assoc " + ext + "=" + fileType);
        if (v != 0) return v;
    }
    return v;
}
public static int execute(string exeFilename, string arguments) {
    ProcessStartInfo startInfo = new ProcessStartInfo();
    startInfo.CreateNoWindow = false;
    startInfo.UseShellExecute = true;
    startInfo.FileName = exeFilename;
    startInfo.WindowStyle = ProcessWindowStyle.Hidden;
    startInfo.Arguments = arguments;
    try {
        using (Process exeProcess = Process.Start(startInfo)) {
            exeProcess.WaitForExit();
            return exeProcess.ExitCode;
        }
    } catch {
        return 1;
    }
}
0
Mike