Как получить интерфейс IPIEHTMLDocument2 в IE Mobile - PullRequest
2 голосов
/ 05 сентября 2008

Я написал плагин Active X для IE7, который реализует IObjectWithSite помимо некоторых других необходимых интерфейсов (обратите внимание, нет IOleClient). Этот интерфейс запрашивается и вызывается IE7. Во время вызова SetSite () я получаю указатель на интерфейс сайта IE7, который я могу использовать для получения интерфейса IHTMLDocument2, используя следующий подход:

IUnknown *site = pUnkSite; /* retrieved from IE7 during SetSite() call */
IServiceProvider *sp = NULL;
IHTMLWindow2 *win = NULL;
IHTMLDocument2 *doc = NULL;

if(site) {
    site->QueryInterface(IID_IServiceProvider, (void **)&sp);
    if(sp) {
        sp->QueryService(IID_IHTMLWindow2, IID_IHTMLWindow2, (void **)&win);
        if(win) {
            win->get_document(&doc);
        }
    }
}
if(doc) {
    /* found */
}

Я попробовал подобный подход и к PIE, используя следующий код, однако даже интерфейс IPIEHTMLWindow2 получить невозможно, поэтому я застрял:

IUnknown *site = pUnkSite; /* retrieved from PIE during SetSite() call */
IPIEHTMLWindow2 *win = NULL;
IPIEHTMLDocument1 *tmp = NULL;
IPIEHTMLDocument2 *doc = NULL;

if(site) {
    site->QueryInterface(__uuidof(*win), (void **)&win);
    if(win) { /* never the case */
        win->get_document(&tmp);
        if(tmp) {
            tmp->QueryInterface(__uuidof(*doc), (void **)&doc);
        }
    }
}
if(doc) {
    /* found */
}

Использование интерфейса IServiceProvider также не работает, поэтому я уже проверял это.

Есть идеи?

Ответы [ 2 ]

3 голосов
/ 15 сентября 2008

Я нашел следующий код в коде Google Gears, здесь . Я скопировал функции, я думаю, что вам нужно здесь. Тот, который вам нужен, находится внизу (GetHtmlWindow2), но остальные два также необходимы. Надеюсь, я ничего не пропустил, но если я сделал то, что вам нужно, вероятно, по ссылке.

#ifdef WINCE
// We can't get IWebBrowser2 for WinCE.
#else
HRESULT ActiveXUtils::GetWebBrowser2(IUnknown *site, IWebBrowser2 **browser2) {
  CComQIPtr<IServiceProvider> service_provider = site;
  if (!service_provider) { return E_FAIL; }

  return service_provider->QueryService(SID_SWebBrowserApp,
                                        IID_IWebBrowser2,
                                        reinterpret_cast<void**>(browser2));
}
#endif


HRESULT ActiveXUtils::GetHtmlDocument2(IUnknown *site,
                                       IHTMLDocument2 **document2) {
  HRESULT hr;

#ifdef WINCE
  // Follow path Window2 -> Window -> Document -> Document2
  CComPtr<IPIEHTMLWindow2> window2;
  hr = GetHtmlWindow2(site, &window2);
  if (FAILED(hr) || !window2) { return false; }
  CComQIPtr<IPIEHTMLWindow> window = window2;
  CComPtr<IHTMLDocument> document;
  hr = window->get_document(&document);
  if (FAILED(hr) || !document) { return E_FAIL; }
  return document->QueryInterface(__uuidof(*document2),
                                  reinterpret_cast<void**>(document2));
#else
  CComPtr<IWebBrowser2> web_browser2;
  hr = GetWebBrowser2(site, &web_browser2);
  if (FAILED(hr) || !web_browser2) { return E_FAIL; }

  CComPtr<IDispatch> doc_dispatch;
  hr = web_browser2->get_Document(&doc_dispatch);
  if (FAILED(hr) || !doc_dispatch) { return E_FAIL; }

  return doc_dispatch->QueryInterface(document2);
#endif
}


HRESULT ActiveXUtils::GetHtmlWindow2(IUnknown *site,
#ifdef WINCE
                                     IPIEHTMLWindow2 **window2) {
  // site is javascript IDispatch pointer.
  return site->QueryInterface(__uuidof(*window2),
                              reinterpret_cast<void**>(window2));
#else
                                     IHTMLWindow2 **window2) {
  CComPtr<IHTMLDocument2> html_document2;
  // To hook an event on a page's window object, follow the path
  // IWebBrowser2->document->parentWindow->IHTMLWindow2

  HRESULT hr = GetHtmlDocument2(site, &html_document2);
  if (FAILED(hr) || !html_document2) { return E_FAIL; }

  return html_document2->get_parentWindow(window2);
#endif
}
2 голосов
/ 16 сентября 2008

Ну, я уже знал код механизма. Механизм, который использует gears, основан на обходном пути, заключающемся в явном вызове метода в плагине gears из загрузчика gears, чтобы установить объект окна и использовать его в качестве интерфейса сайта вместо IUnknown, предоставляемого IE Mobile в вызове SetSite. Что касается кода механизма, инженеры Google знают о той же проблеме, которую я задаю, и придумали этот обходной путь, который я описал.

Тем не менее, я считаю, что должен существовать другой, более "официальный" способ решения этой проблемы, поскольку точная настройка сайта на элемент управления / плагин Active X не очень хороша. Я собираюсь обратиться к команде MS IE Mobile прямо сейчас и сообщу вам, как только я получу решение. Это может быть ошибка в IE Mobile, которая является наиболее вероятной вещью, которую я могу себе представить, но кто знает ...

Но все равно спасибо за ваш ответ;))

...