Откройте URL в новом процессе браузера - PullRequest
3 голосов
/ 13 апреля 2009

Мне нужно открыть URL в новом процессе браузера. Я должен быть уведомлен, когда тот процесс браузера выходит. Код, который я сейчас использую, следующий:

        Process browser = new Process();
        browser.EnableRaisingEvents = true;
        browser.StartInfo.Arguments = url;
        browser.StartInfo.FileName = "iexplore";

        browser.Exited += new EventHandler(browser_Exited);

        browser.Start();

Очевидно, что это не будет связано с тем, что «FileName» установлено в iexplore, а не в веб-браузере пользователя по умолчанию. Как мне определить, какой веб-браузер у пользователя по умолчанию?

Я работаю на Vista-> вперед. Хотя XP было бы неплохо поддержать, если это возможно.

Немного контекста: я создал очень маленький автономный веб-сервер, который обслуживает некоторые файлы с локального диска. В конце запуска сервера я хочу запустить браузер. Когда пользователь завершит работу и закроет браузер, я бы хотел выйти из веб-сервера. Приведенный выше код работает отлично, кроме использования только IE.

Заранее спасибо!

Ответы [ 7 ]

5 голосов
/ 03 мая 2009

Хорошо. Теперь у меня есть рабочий код C #, чтобы делать то, что я хочу. Это вернет «командную строку», которую вы должны запустить для загрузки текущего браузера по умолчанию:

using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using System.Linq;
using System.Text;

namespace testDefaultBrowser
{
    public enum ASSOCIATIONLEVEL
    {
        AL_MACHINE,
        AL_EFFECTIVE,
        AL_USER,
    };

    public enum ASSOCIATIONTYPE
    {
        AT_FILEEXTENSION,
        AT_URLPROTOCOL,
        AT_STARTMENUCLIENT,
        AT_MIMETYPE,
    };

    [Guid("4e530b0a-e611-4c77-a3ac-9031d022281b"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
    public interface IApplicationAssociationRegistration
    {
        void QueryCurrentDefault([In, MarshalAs(UnmanagedType.LPWStr)] string pszQuery,
        [In, MarshalAs(UnmanagedType.I4)] ASSOCIATIONTYPE atQueryType,
        [In, MarshalAs(UnmanagedType.I4)] ASSOCIATIONLEVEL alQueryLevel,
        [Out, MarshalAs(UnmanagedType.LPWStr)] out string ppszAssociation);

        void QueryAppIsDefault(
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszQuery,
            [In] ASSOCIATIONTYPE atQueryType,
            [In] ASSOCIATIONLEVEL alQueryLevel,
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszAppRegistryName,
            [Out] out bool pfDefault);

        void QueryAppIsDefaultAll(
            [In] ASSOCIATIONLEVEL alQueryLevel,
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszAppRegistryName,
            [Out] out bool pfDefault);

        void SetAppAsDefault(
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszAppRegistryName,
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszSet,
            [In] ASSOCIATIONTYPE atSetType);

        void SetAppAsDefaultAll(
            [In, MarshalAs(UnmanagedType.LPWStr)] string pszAppRegistryName);

        void ClearUserAssociations();
    }

    [ComImport, Guid("591209c7-767b-42b2-9fba-44ee4615f2c7")]//
    class ApplicationAssociationRegistration
    {
    }

    class Program
    {
        static void Main(string[] args)
        {
            IApplicationAssociationRegistration reg = 
                (IApplicationAssociationRegistration) new ApplicationAssociationRegistration();

            string progID;
            reg.QueryCurrentDefault(".txt",
                ASSOCIATIONTYPE.AT_FILEEXTENSION,
                ASSOCIATIONLEVEL.AL_EFFECTIVE,
                out progID);
            Console.WriteLine(progID);

            reg.QueryCurrentDefault("http",
                ASSOCIATIONTYPE.AT_URLPROTOCOL,
                ASSOCIATIONLEVEL.AL_EFFECTIVE,
                out progID);
            Console.WriteLine(progID);
        }
    }
}

Уф! Спасибо всем за помощь в подталкивании меня к правильному ответу!

2 голосов
/ 22 апреля 2009

Если вы передадите путь файла известного типа в (проводник) приложения, оно будет ' делать правильные вещи ', например

 Process.Start("explorer.exe", @"\\path.to\filename.pdf");

и откройте файл в программе чтения PDF.

Но если вы попробуете то же самое с URL, например,

Process.Start("explorer.exe", @"http://www.stackoverflow.com/");

запускается IE (который не является браузером по умолчанию на моей машине).

Я знаю, что не отвечает на вопрос, но я подумал, что это была интересная информация.

1 голос
/ 05 января 2011

Я написал этот код для проекта один раз ... он учитывает любые дополнительные параметры, установленные для браузера по умолчанию. Первоначально он был создан для открытия документации HTML в браузере, по той простой причине, что я всегда устанавливал свою программу по умолчанию для HTML на редактор , а не на браузер, и это меня бесит, когда я вижу, что какая-то программа открыта его HTML readme в моем текстовом редакторе. Очевидно, что он отлично работает и для URL.

    /// <summary>
    ///     Opens a local file or url in the default web browser.
    /// </summary>
    /// <param name="path">Path of the local file or url</param>
    public static void openInDefaultBrowser(String pathOrUrl)
    {
        pathOrUrl = "\"" + pathOrUrl.Trim('"') + "\"";
        RegistryKey defBrowserKey = Registry.ClassesRoot.OpenSubKey(@"http\shell\open\command");
        if (defBrowserKey != null && defBrowserKey.ValueCount > 0 && defBrowserKey.GetValue("") != null)
        {
            String defBrowser = (String)defBrowserKey.GetValue("");
            if (defBrowser.Contains("%1"))
            {
                defBrowser = defBrowser.Replace("%1", pathOrUrl);
            }
            else
            {
                defBrowser += " " + pathOrUrl;
            }
            String defBrowserProcess;
            String defBrowserArgs;
            if (defBrowser[0] == '"')
            {
                defBrowserProcess = defBrowser.Substring(0, defBrowser.Substring(1).IndexOf('"') + 2).Trim();
                defBrowserArgs = defBrowser.Substring(defBrowser.Substring(1).IndexOf('"') + 2).TrimStart();
            }
            else
            {
                defBrowserProcess = defBrowser.Substring(0, defBrowser.IndexOf(" ")).Trim();
                defBrowserArgs = defBrowser.Substring(defBrowser.IndexOf(" ")).Trim();
            }
            if (new FileInfo(defBrowserProcess.Trim('"')).Exists)
                Process.Start(defBrowserProcess, defBrowserArgs);
        }
    }
1 голос
/ 03 мая 2009

Хорошо. Неделю пробыл на конференции, а теперь возвращаюсь к этому. Теперь я могу сделать это с C ++ - и, похоже, он ведет себя правильно! Однако мои попытки перевести это на C # (или .NET) не увенчались успехом ( Пост на вопрос ).

Вот код C ++ для других, кто наткнулся на этот вопрос:

#include "stdafx.h"
#include <iostream>
#include <shobjidl.h>
#define _ATL_CSTRING_EXPLICIT_CONSTRUCTORS      // some CString constructors will be explicit

#include <atlbase.h>
#include <atlstr.h>
#include <AtlDef.h>
#include <AtlConv.h>

using namespace std;
using namespace ATL;

int _tmain(int argc, _TCHAR* argv[])
{
    HRESULT hr = CoInitialize(NULL);
    if (!SUCCEEDED(hr)) {
        cout << "Failed to init COM instance" << endl;
        cout << hr << endl;
    }

    IApplicationAssociationRegistration *pAAR;
    hr = CoCreateInstance(CLSID_ApplicationAssociationRegistration,
        NULL, CLSCTX_INPROC, __uuidof(IApplicationAssociationRegistration),
        (void**) &pAAR);
    if (!SUCCEEDED(hr))
    {
        cout << "Failed to create COM object" << endl;
        cout << hr << endl;
        return 0;
    }

    LPWSTR progID;
    //wchar_t *ttype = ".txt";
    hr = pAAR->QueryCurrentDefault (L".txt", AT_FILEEXTENSION, AL_EFFECTIVE, &progID);
    if (!SUCCEEDED(hr)) {
        cout << "Failed to query default for .txt" << endl;
        cout << hr << endl;
    }
    CW2A myprogID (progID);
    cout << "Result is: " << static_cast<const char*>(myprogID) << endl;

    /// Now for http

    hr = pAAR->QueryCurrentDefault (L"http", AT_URLPROTOCOL, AL_EFFECTIVE, &progID);
    if (!SUCCEEDED(hr)) {
        cout << "Failed to query default for http" << endl;
        cout << hr << endl;
    }
    CW2A myprogID1 (progID);
    cout << "Result is: " << static_cast<const char*>(myprogID1) << endl;

    return 0;
}

Я опубликую код C #, когда я наконец заработаю!

1 голос
/ 22 апреля 2009

Хорошо, думаю, я мог бы найти это - IApplicationAssociationRegistration :: QueryCurrentDefault [1]. Согласно документам это именно то, что используется ShellExecute. Я опубликую код, когда заставлю его работать, но мне было бы интересно, если бы другие думали, что это правильная вещь (кстати, я Vista или выше для уровня ОС).

[1]: http://msdn.microsoft.com/en-us/library/bb776336(VS.85).aspx QueryCurrentDefault

1 голос
/ 13 апреля 2009

Способ определения браузера по умолчанию описан в этом сообщении в блоге:

http://ryanfarley.com/blog/archive/2004/05/16/649.aspx

Из поста блога выше:

private string getDefaultBrowser()
{
    string browser = string.Empty;
    RegistryKey key = null;
    try
    {
        key = Registry.ClassesRoot.OpenSubKey(@"HTTP\shell\open\command", false);

        //trim off quotes
        browser = key.GetValue(null).ToString().ToLower().Replace("\"", "");
        if (!browser.EndsWith("exe"))
        {
            //get rid of everything after the ".exe"
            browser = browser.Substring(0, browser.LastIndexOf(".exe")+4);
        }
    }
    finally
    {
        if (key != null) key.Close();
    }
    return browser;
}
0 голосов
/ 03 мая 2009

Короткий ответ, вы не можете.

Если браузером по умолчанию является, скажем, Firefox, и у пользователя уже запущен экземпляр Firefox, он будет просто открыт в другом окне или вкладке того же процесса firefox.exe, и даже после закрытия вашей страницы, процесс не завершится, пока они не закроют все окна и вкладки. В этом случае вы получите уведомление о выходе из процесса, как только вы его запустите, из-за временного процесса firefox.exe, который будет маршалировать URL-адрес текущего процесса. (Предполагая, что именно так работает управление отдельными экземплярами Firefox).

...