4

我需要在新的浏览器进程中打开一个 URL。当该浏览器进程退出时,我需要得到通知。我目前使用的代码如下:

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

        browser.Exited += new EventHandler(browser_Exited);

        browser.Start();

显然,这不会到期,因为“文件名”被固定为 iexplore,而不是用户的默认 Web 浏览器。如何确定用户的默认 Web 浏览器是什么?

我在 Vista->forward 上运行。虽然如果可能的话 XP 会很高兴支持。

更多背景信息:我创建了一个非常小的独立 Web 服务器,它为本地磁盘上的一些文件提供服务。在启动服务器结束时,我想启动浏览器。用户完成并关闭浏览器后,我想退出 Web 服务器。上面的代码完美运行,除了只使用 IE。

提前致谢!

4

7 回答 7

6

行。我现在有工作的 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);
        }
    }
}

哇!感谢大家帮助推动我找到正确的答案!

于 2009-05-03T06:32:08.530 回答
3

如果您将已知文件类型的路径传递给(文件)资源管理器应用程序,它将“做正确的事情”,例如

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

并在 PDF 阅读器中打开文件。

但是,如果您尝试使用 URL 进行相同的操作,例如

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

它会启动 IE(这不是我机器上的默认浏览器)。

我知道没有回答这个问题,但我认为这是一个有趣的旁注。

于 2009-04-22T08:40:17.203 回答
1

这篇博文解释了确定默认浏览器的方法:

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;
}
于 2009-04-12T23:12:26.307 回答
1

好的,我想我可能已经找到了 - IApplicationAssociationRegistration::QueryCurrentDefault [1]。根据文档,这是 ShellExecute 使用的。当我让它工作时,我会发布代码,但如果其他人认为这是正确的使用方法,我会很感兴趣(顺便说一句,我是 Vista 或更高版本的操作系统级别)。

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

于 2009-04-22T08:01:27.470 回答
1

行。在会议巡回赛上待了一个星期,现在回到这个问题上。我现在可以用 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# 代码!

于 2009-05-03T06:11:32.310 回答
1

我曾经为一个项目编写过这段代码……它记住了为默认浏览器设置的任何附加参数。它最初是为了在浏览器中打开 HTML 文档而创建的,原因很简单,我总是将我的默认 HTML 程序设置为编辑器而不是浏览器,而且看到某些程序在我的文本中打开它的 HTML 自述文件让我很恼火编辑。显然,它也适用于 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);
        }
    }
于 2011-01-05T04:02:59.527 回答
0

简短的回答,你不能。

如果默认浏览器是 Firefox,并且用户已经运行了一个 Firefox 实例,它只会在同一个 firefox.exe 进程的另一个窗口或选项卡中打开,即使他们关闭了您的页面,该进程也不会' t 退出,直到他们关闭每个窗口和选项卡。在这种情况下,由于临时 firefox.exe proc 会将 URL 编组到当前进程,您将在启动它时立即收到进程退出的通知。(假设这就是 Firefox 的单实例管理的工作方式)。

于 2009-05-03T06:18:00.500 回答