Как начать работу с расширением Internet Explorer?

У кого-нибудь есть опыт работы с/в разработке расширений IE, которые могут делиться своими знаниями? Это будет включать примеры кода или ссылки на хорошие, или документацию по процессу или что-то еще.

Я действительно хочу это сделать, но я нажимаю гигантскую стену с паршивой документацией, паршивым кодом/примером кода/отсутствием этого. Любая помощь/ресурсы, которые вы могли бы предложить, будет с благодарностью.

В частности, я хотел бы начать с того, как получить доступ к /DOM изнутри расширения IE.

ИЗМЕНИТЬ, еще более подробно:

В идеале, я хотел бы установить кнопку панели инструментов, которая при щелчке вытащила меню вверх, которое содержит ссылки на внешние сайты. Я также хотел бы получить доступ к DOM и разместить JavaScript на странице в зависимости от некоторых условий.

Каков наилучший способ сохранить информацию в расширении IE? В Firefox/Chrome/большинстве современных браузеров вы используете window.localStorage, но, очевидно, с IE8/IE7, это не вариант. Может быть, SQLite DB или такой? Можно ли предположить, что .NET 4.0 будет установлен на пользовательский компьютер?

Я не хочу использовать Spice IE, поскольку я хочу создать тот, который совместим с IE9. Я добавил тег С++ к этому вопросу, потому что, если лучше построить его на С++, я могу это сделать.

Ответ 1

Человек... это была большая работа! Мне было так любопытно, как это сделать, что я сделал это сам.

Прежде всего... кредит не все мое. Это компиляция того, что я нашел, на этих сайтах:

И, конечно, я хотел, чтобы мой ответ имел функции, которые вы задали:

  • Обход DOM, чтобы найти что-то;
  • кнопка, которая показывает окно (в моем случае для настройки)
  • сохраняйте конфигурацию (я буду использовать для этого regitry)
  • и, наконец, выполнить javascript.

Я опишу его шаг за шагом, как мне это удалось, работая с Internet Explorer 8, в Windows 7 x64... обратите внимание, что я не смог проверить в других конфигурациях. Надеюсь, ты понимаешь =)

Создание рабочего обозревателя Internet Explorer 8

Я использую Visual Studio 2010, С# 4, .Net Framework 4, поэтому некоторые из этих шагов могут немного отличаться для вас.

Создан класс библиотеки. Я назвал мой InternetExplorerExtension.

Добавьте эти ссылки в проект:

  • Interop.SHDocVw
  • Microsoft.mshtml

Примечание. Эти ссылки могут быть в разных местах на каждом компьютере.

это то, что содержит раздел ссылок в csproj:

<Reference Include="Interop.SHDocVw, Version=1.1.0.0, Culture=neutral, PublicKeyToken=90ba9c70f846762e, processorArchitecture=MSIL">
  <SpecificVersion>False</SpecificVersion>
  <EmbedInteropTypes>True</EmbedInteropTypes>
  <HintPath>C:\Program Files (x86)\Microsoft Visual Studio 9.0\Common7\IDE\PrivateAssemblies\Interop.SHDocVw.dll</HintPath>
</Reference>
<Reference Include="Microsoft.CSharp" />
<Reference Include="Microsoft.mshtml, Version=7.0.3300.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
  <EmbedInteropTypes>True</EmbedInteropTypes>
</Reference>
<Reference Include="System" />
<Reference Include="System.Data" />
<Reference Include="System.Drawing" />
<Reference Include="System.Windows.Forms" />
<Reference Include="System.Xml" />

Создайте следующие файлы:

IEAddon.cs

using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using Microsoft.Win32;
using mshtml;
using SHDocVw;

namespace InternetExplorerExtension
{
    [ComVisible(true)]
    [ClassInterface(ClassInterfaceType.None)]
    [Guid("D40C654D-7C51-4EB3-95B2-1E23905C2A2D")]
    [ProgId("MyBHO.WordHighlighter")]
    public class WordHighlighterBHO : IObjectWithSite, IOleCommandTarget
    {
        const string DefaultTextToHighlight = "browser";

        IWebBrowser2 browser;
        private object site;

        #region Highlight Text
        void OnDocumentComplete(object pDisp, ref object URL)
        {
            try
            {
                // @Eric Stob: Thanks for this hint!
                // This will prevent this method being executed more than once.
                if (pDisp != this.site)
                    return;

                var document2 = browser.Document as IHTMLDocument2;
                var document3 = browser.Document as IHTMLDocument3;

                var window = document2.parentWindow;
                window.execScript(@"function FncAddedByAddon() { alert('Message added by addon.'); }");

                Queue<IHTMLDOMNode> queue = new Queue<IHTMLDOMNode>();
                foreach (IHTMLDOMNode eachChild in document3.childNodes)
                    queue.Enqueue(eachChild);

                while (queue.Count > 0)
                {
                    // replacing desired text with a highlighted version of it
                    var domNode = queue.Dequeue();

                    var textNode = domNode as IHTMLDOMTextNode;
                    if (textNode != null)
                    {
                        if (textNode.data.Contains(TextToHighlight))
                        {
                            var newText = textNode.data.Replace(TextToHighlight, "<span style='background-color: yellow; cursor: hand;' onclick='javascript:FncAddedByAddon()' title='Click to open script based alert window.'>" + TextToHighlight + "</span>");
                            var newNode = document2.createElement("span");
                            newNode.innerHTML = newText;
                            domNode.replaceNode((IHTMLDOMNode)newNode);
                        }
                    }
                    else
                    {
                        // adding children to collection
                        var x = (IHTMLDOMChildrenCollection)(domNode.childNodes);
                        foreach (IHTMLDOMNode eachChild in x)
                        {
                            if (eachChild is mshtml.IHTMLScriptElement)
                                continue;
                            if (eachChild is mshtml.IHTMLStyleElement)
                                continue;

                            queue.Enqueue(eachChild);
                        }
                    }
                }
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
        }
        #endregion
        #region Load and Save Data
        static string TextToHighlight = DefaultTextToHighlight;
        public static string RegData = "Software\\MyIEExtension";

        [DllImport("ieframe.dll")]
        public static extern int IEGetWriteableHKCU(ref IntPtr phKey);

        private static void SaveOptions()
        {
            // In IE 7,8,9,(desktop)10 tabs run in Protected Mode
            // which prohibits writes to HKLM, HKCU.
            // Must ask IE for "Writable" registry section pointer
            // which will be something like HKU/S-1-7***/Software/AppDataLow/
            // In "metro" IE 10 mode, tabs run in "Enhanced Protected Mode"
            // where BHOs are not allowed to run, except in edge cases.
            // see http://blogs.msdn.com/b/ieinternals/archive/2012/03/23/understanding-ie10-enhanced-protected-mode-network-security-addons-cookies-metro-desktop.aspx
            IntPtr phKey = new IntPtr();
            var answer = IEGetWriteableHKCU(ref phKey);
            RegistryKey writeable_registry = RegistryKey.FromHandle(
                new Microsoft.Win32.SafeHandles.SafeRegistryHandle(phKey, true)
            );
            RegistryKey registryKey = writeable_registry.OpenSubKey(RegData, true);

            if (registryKey == null)
                registryKey = writeable_registry.CreateSubKey(RegData);
            registryKey.SetValue("Data", TextToHighlight);

            writeable_registry.Close();
        }
        private static void LoadOptions()
        {
            // In IE 7,8,9,(desktop)10 tabs run in Protected Mode
            // which prohibits writes to HKLM, HKCU.
            // Must ask IE for "Writable" registry section pointer
            // which will be something like HKU/S-1-7***/Software/AppDataLow/
            // In "metro" IE 10 mode, tabs run in "Enhanced Protected Mode"
            // where BHOs are not allowed to run, except in edge cases.
            // see http://blogs.msdn.com/b/ieinternals/archive/2012/03/23/understanding-ie10-enhanced-protected-mode-network-security-addons-cookies-metro-desktop.aspx
            IntPtr phKey = new IntPtr();
            var answer = IEGetWriteableHKCU(ref phKey);
            RegistryKey writeable_registry = RegistryKey.FromHandle(
                new Microsoft.Win32.SafeHandles.SafeRegistryHandle(phKey, true)
            );
            RegistryKey registryKey = writeable_registry.OpenSubKey(RegData, true);

            if (registryKey == null)
                registryKey = writeable_registry.CreateSubKey(RegData);
            registryKey.SetValue("Data", TextToHighlight);

            if (registryKey == null)
            {
                TextToHighlight = DefaultTextToHighlight;
            }
            else
            {
                TextToHighlight = (string)registryKey.GetValue("Data");
            }
            writeable_registry.Close();
        }
        #endregion

        [Guid("6D5140C1-7436-11CE-8034-00AA006009FA")]
        [InterfaceType(1)]
        public interface IServiceProvider
        {
            int QueryService(ref Guid guidService, ref Guid riid, out IntPtr ppvObject);
        }

        #region Implementation of IObjectWithSite
        int IObjectWithSite.SetSite(object site)
        {
            this.site = site;

            if (site != null)
            {
                LoadOptions();

                var serviceProv = (IServiceProvider)this.site;
                var guidIWebBrowserApp = Marshal.GenerateGuidForType(typeof(IWebBrowserApp)); // new Guid("0002DF05-0000-0000-C000-000000000046");
                var guidIWebBrowser2 = Marshal.GenerateGuidForType(typeof(IWebBrowser2)); // new Guid("D30C1661-CDAF-11D0-8A3E-00C04FC9E26E");
                IntPtr intPtr;
                serviceProv.QueryService(ref guidIWebBrowserApp, ref guidIWebBrowser2, out intPtr);

                browser = (IWebBrowser2)Marshal.GetObjectForIUnknown(intPtr);

                ((DWebBrowserEvents2_Event)browser).DocumentComplete +=
                    new DWebBrowserEvents2_DocumentCompleteEventHandler(this.OnDocumentComplete);
            }
            else
            {
                ((DWebBrowserEvents2_Event)browser).DocumentComplete -=
                    new DWebBrowserEvents2_DocumentCompleteEventHandler(this.OnDocumentComplete);
                browser = null;
            }
            return 0;
        }
        int IObjectWithSite.GetSite(ref Guid guid, out IntPtr ppvSite)
        {
            IntPtr punk = Marshal.GetIUnknownForObject(browser);
            int hr = Marshal.QueryInterface(punk, ref guid, out ppvSite);
            Marshal.Release(punk);
            return hr;
        }
        #endregion
        #region Implementation of IOleCommandTarget
        int IOleCommandTarget.QueryStatus(IntPtr pguidCmdGroup, uint cCmds, ref OLECMD prgCmds, IntPtr pCmdText)
        {
            return 0;
        }
        int IOleCommandTarget.Exec(IntPtr pguidCmdGroup, uint nCmdID, uint nCmdexecopt, IntPtr pvaIn, IntPtr pvaOut)
        {
            try
            {
                // Accessing the document from the command-bar.
                var document = browser.Document as IHTMLDocument2;
                var window = document.parentWindow;
                var result = window.execScript(@"alert('You will now be allowed to configure the text to highlight...');");

                var form = new HighlighterOptionsForm();
                form.InputText = TextToHighlight;
                if (form.ShowDialog() != DialogResult.Cancel)
                {
                    TextToHighlight = form.InputText;
                    SaveOptions();
                }
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }

            return 0;
        }
        #endregion

        #region Registering with regasm
        public static string RegBHO = "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Browser Helper Objects";
        public static string RegCmd = "Software\\Microsoft\\Internet Explorer\\Extensions";

        [ComRegisterFunction]
        public static void RegisterBHO(Type type)
        {
            string guid = type.GUID.ToString("B");

            // BHO
            {
                RegistryKey registryKey = Registry.LocalMachine.OpenSubKey(RegBHO, true);
                if (registryKey == null)
                    registryKey = Registry.LocalMachine.CreateSubKey(RegBHO);
                RegistryKey key = registryKey.OpenSubKey(guid);
                if (key == null)
                    key = registryKey.CreateSubKey(guid);
                key.SetValue("Alright", 1);
                registryKey.Close();
                key.Close();
            }

            // Command
            {
                RegistryKey registryKey = Registry.LocalMachine.OpenSubKey(RegCmd, true);
                if (registryKey == null)
                    registryKey = Registry.LocalMachine.CreateSubKey(RegCmd);
                RegistryKey key = registryKey.OpenSubKey(guid);
                if (key == null)
                    key = registryKey.CreateSubKey(guid);
                key.SetValue("ButtonText", "Highlighter options");
                key.SetValue("CLSID", "{1FBA04EE-3024-11d2-8F1F-0000F87ABD16}");
                key.SetValue("ClsidExtension", guid);
                key.SetValue("Icon", "");
                key.SetValue("HotIcon", "");
                key.SetValue("Default Visible", "Yes");
                key.SetValue("MenuText", "&Highlighter options");
                key.SetValue("ToolTip", "Highlighter options");
                //key.SetValue("KeyPath", "no");
                registryKey.Close();
                key.Close();
            }
        }

        [ComUnregisterFunction]
        public static void UnregisterBHO(Type type)
        {
            string guid = type.GUID.ToString("B");
            // BHO
            {
                RegistryKey registryKey = Registry.LocalMachine.OpenSubKey(RegBHO, true);
                if (registryKey != null)
                    registryKey.DeleteSubKey(guid, false);
            }
            // Command
            {
                RegistryKey registryKey = Registry.LocalMachine.OpenSubKey(RegCmd, true);
                if (registryKey != null)
                    registryKey.DeleteSubKey(guid, false);
            }
        }
        #endregion
    }
}

Interop.cs

using System;
using System.Runtime.InteropServices;
namespace InternetExplorerExtension
{
    [ComVisible(true)]
    [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
    [Guid("FC4801A3-2BA9-11CF-A229-00AA003D7352")]
    public interface IObjectWithSite
    {
        [PreserveSig]
        int SetSite([MarshalAs(UnmanagedType.IUnknown)]object site);
        [PreserveSig]
        int GetSite(ref Guid guid, [MarshalAs(UnmanagedType.IUnknown)]out IntPtr ppvSite);
    }


    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    public struct OLECMDTEXT
    {
        public uint cmdtextf;
        public uint cwActual;
        public uint cwBuf;
        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 100)]
        public char rgwz;
    }

    [StructLayout(LayoutKind.Sequential)]
    public struct OLECMD
    {
        public uint cmdID;
        public uint cmdf;
    }

    [ComImport(), ComVisible(true),
    Guid("B722BCCB-4E68-101B-A2BC-00AA00404770"),
    InterfaceTypeAttribute(ComInterfaceType.InterfaceIsIUnknown)]
    public interface IOleCommandTarget
    {

        [return: MarshalAs(UnmanagedType.I4)]
        [PreserveSig]
        int QueryStatus(
            [In] IntPtr pguidCmdGroup,
            [In, MarshalAs(UnmanagedType.U4)] uint cCmds,
            [In, Out, MarshalAs(UnmanagedType.Struct)] ref OLECMD prgCmds,
            //This parameter must be IntPtr, as it can be null
            [In, Out] IntPtr pCmdText);

        [return: MarshalAs(UnmanagedType.I4)]
        [PreserveSig]
        int Exec(
            //[In] ref Guid pguidCmdGroup,
            //have to be IntPtr, since null values are unacceptable
            //and null is used as default group!
            [In] IntPtr pguidCmdGroup,
            [In, MarshalAs(UnmanagedType.U4)] uint nCmdID,
            [In, MarshalAs(UnmanagedType.U4)] uint nCmdexecopt,
            [In] IntPtr pvaIn,
            [In, Out] IntPtr pvaOut);
    }
}

и, наконец, форму, которую мы будем использовать для настройки параметров. В этой форме поместите a TextBox и Ok Button. Установите DialogResult на кнопку ОК. Поместите этот код в код формы:

using System.Windows.Forms;
namespace InternetExplorerExtension
{
    public partial class HighlighterOptionsForm : Form
    {
        public HighlighterOptionsForm()
        {
            InitializeComponent();
        }

        public string InputText
        {
            get { return this.textBox1.Text; }
            set { this.textBox1.Text = value; }
        }
    }
}

В свойствах проекта сделайте следующее:

  • Подпишите сборку с помощью сильной клавиши;
  • На вкладке "Отладка" установите Запустить внешнюю программу на C:\Program Files (x86)\Internet Explorer\iexplore.exe
  • На вкладке "Отладка" установите Аргументы командной строки на http://msdn.microsoft.com/en-us/library/ms976373.aspx#bho_getintouch
  • На вкладке "События" установите Командная строка событий после сборки:

    "C:\Program Files (x86)\Microsoft SDKs\Windows\v7.0A\Bin\NETFX 4.0 Tools\x64\gacutil.exe" /f /i "$(TargetDir)$(TargetFileName)"
    
    "C:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm.exe" /unregister "$(TargetDir)$(TargetFileName)"
    
    "C:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm.exe" "$(TargetDir)$(TargetFileName)"

Внимание:, так как мой компьютер равен x64, на моем компьютере есть определенный x64 внутри исполняемого файла gacutil, который может отличаться от вашего.

64-битный IE. Требуется 64-битный и 64-битный BHO. Используйте 64-битный RegAsm.exe(обычно он живет в C:\Windows\Microsoft.NET\Framework64\v4.0.30319\RegAsm.exe)

Как работает этот аддон

Он перемещает все дерево DOM, заменяя текст, настроенный с помощью кнопки, сам по себе с желтым фоном. Если вы нажмете на пожелтевшие тексты, он вызывает функцию javascript, которая была вставлена ​​на страницу динамически. Слово по умолчанию - "браузер", так что оно соответствует многим из них! ИЗМЕНИТЬ: после изменения выделенной строки, вы должны нажать на поле URL и нажать Enter... F5 не будет работать, я думаю, что это потому, что F5 считается "навигацией", и это для прослушивания события навигации (возможно). Я попытаюсь это исправить позже.

Теперь пора идти. Я очень устал. Не стесняйтесь задавать вопросы... может быть, я не буду отвечать, потому что я еду в путешествие... через 3 дня я вернулся, но я постараюсь приехать сюда тем временем.

Ответ 2

Еще один классный подход - проверить:

http://www.crossrider.org

Это фреймворк, основанный на JS с jquery, который позволяет вам разрабатывать расширения браузеров для IE, FF и Chrome с использованием единого общего JS-кода. В основном структура делает всю неприятную работу, и вам остается писать код вашего приложения.

Ответ 3

Состояние расширений IE на самом деле довольно грустно. У вас есть старая модель IE5 Browser Helper Object (да, эти печально известные BHOs, которым все любят блокировать в тот же день), панели инструментов и новые ускорители для IE. Даже тогда совместимость иногда ломается. Я использовал для поддержки расширения для IE6, который сломался с IE7, поэтому есть некоторые вещи, которые изменились. По большей части, насколько я знаю (я не прикасался к BHOs ​​годами), вам все равно нужно закодировать их, используя Active Template Libraries (вроде STL для Microsoft COM), а также, как таковой, только для С++. Вы можете сделать COM Interop с С# и уйти с этим в С#, но, вероятно, это будет слишком сложно для того, что стоит. В любом случае, если вы заинтересованы в кодировании собственного расширения для IE (что является правдоподобным, если вы хотите, чтобы ваши расширения были доступны во всех основных браузерах), вот официальные ресурсы Microsoft.

http://msdn.microsoft.com/en-us/library/aa753587(v=vs.85).aspx

А для ускорителей, которые являются новыми в IE8, вы можете проверить это.

http://msdn.microsoft.com/en-us/library/cc289775(v=vs.85).aspx

Я согласен, что документация ужасная, и API-интерфейсы довольно устарели. Тем не менее я надеюсь, что это поможет.

EDIT: Думаю, я могу бросить здесь последний источник информации. Я просматривал свои записи назад, когда я работал над BHOs. И это статья, которая заставила меня начать с них. Это своего рода старый, но имеет хорошее объяснение интерфейсов ATL, которые вы будете использовать при работе с IE BHOs ​​(например, IObjectWithSite). Я думаю, что это довольно хорошо объяснилось и мне тогда очень помогло. http://msdn.microsoft.com/en-us/library/bb250436.aspx Я также проверил пример, который опубликовал GregC. Он работает, по крайней мере, с IE8, и он совместим с VS 2010, поэтому, если вы хотите сделать С#, вы можете начать там и взглянуть на книгу Jon Skeet Book. (С# в Depth 2nd edition) В главе 13 содержится много информации о новых функциях на С# 4, которые вы можете использовать для улучшения взаимодействия с COM. (Я бы порекомендовал вам сделать свой админ на С++)

Ответ 4

Разработка С# BHOs ​​- это боль в заднице. Это связано с большим количеством нехорошего COM-кода и вызовов p/invoke.

У меня есть в основном готовый С# BHO здесь, который вы можете использовать источник для всего, что вы хотите. Я говорю "в основном", потому что никогда не выяснял, как сохранить appdata в защищенном режиме IE.

Ответ 5

Я работаю с IE webbrowser control уже много лет, и по ходу их одно имя снова и снова появляется с полезными сообщениями: Igor Tandetnik

Если бы я разрабатывал расширение, я бы нацелился на BHO и начинал искать по:

BHO Игорь Тандетник

ИЛИ

Объект помощника браузера Igor Tandetnik

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

Вы будете подходить к своим ушам в программировании COM и ATL. Для примера прохождение игры, проверьте: http://msdn.microsoft.com/en-us/library/ms976373.aspx

Ответ 7

Если вы не пытаетесь изобрести колесо, вы можете попробовать Добавить в Express для IE. Я использовал продукт для материал VSTO, и это довольно хорошо. Также у них есть полезный форум и быстрая поддержка.

Ответ 8

Он, очевидно, решен, но для других пользователей я бы рекомендовал инфраструктуру SpicIE. Я сделал свое собственное расширение на основе этого. Он поддерживает только Internet Explorer 7/8, но я проверял, что в Internet Explorer 6-10 (от Windows XP до Windows 8 Consumer Preview) и работает отлично. К сожалению, в последней версии были некоторые ошибки, поэтому мне пришлось исправить их и сделать свой собственный выпуск: http://archive.msdn.microsoft.com/SpicIE/Thread/View.aspx?ThreadId=5251

Ответ 9

Я тепло предлагаю вам этот пост Павла Зольникова, опубликованный в 2002 году!

http://www.codeproject.com/Articles/2219/Extending-Explorer-with-Band-Objects-using-NET-and

Он основан на использовании объектов Band и скомпилирован с использованием .Net 2.0. Исходный код предоставляется и открывается и хорошо компилируется с помощью Visual Studio 2013. Как вы прочтете на комментариях, это отлично работает для IE 11 и Windows 7 и Windows 10. Он отлично работал у меня в Windows 7 + SP1 и IE 11 Наслаждайтесь!

Ответ 10

enter image description here

На вкладке "События сборки" установите командную строку "События после сборки", чтобы: (x64) указана ниже

"C:\Program Files (x86)\Microsoft SDKs\Windows\v7.0A\Bin\NETFX 4.0 Tools\x64\gacutil.exe" /if "$(TargetDir)$(TargetFileName)"    
"C:\Windows\Microsoft.NET\Framework64\v4.0.30319\RegAsm.exe" /u "$(TargetDir)$(TargetFileName)"    
"C:\Windows\Microsoft.NET\Framework64\v4.0.30319\RegAsm.exe" "$(TargetDir)$(TargetFileName)"

Мне нужна вкладка "События сборки", установите командную строку "События после сборки" (32-разрядная операционная система)

"C:\Program Files\Microsoft SDKs\Windows\v7.1\Bin\gacutil.exe" /if "$(TargetDir)$(TargetFileName)"    
"C:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm.exe" /u "$(TargetDir)$(TargetFileName)"    
"C:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm.exe" "$(TargetDir)$(TargetFileName)"