2254 lines
62 KiB
C
2254 lines
62 KiB
C
/*
|
|
* Copyright 2006-2007 Jacek Caban for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winuser.h"
|
|
#include "winreg.h"
|
|
#include "ole2.h"
|
|
#include "shlguid.h"
|
|
#include "wininet.h"
|
|
#include "shlwapi.h"
|
|
|
|
#include "wine/debug.h"
|
|
#include "wine/unicode.h"
|
|
|
|
#include "mshtml_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
|
|
|
|
#define LOAD_INITIAL_DOCUMENT_URI 0x80000
|
|
|
|
#define NS_IOSERVICE_CLASSNAME "nsIOService"
|
|
#define NS_IOSERVICE_CONTRACTID "@mozilla.org/network/io-service;1"
|
|
|
|
static const IID NS_IOSERVICE_CID =
|
|
{0x9ac9e770, 0x18bc, 0x11d3, {0x93, 0x37, 0x00, 0x10, 0x4b, 0xa0, 0xfd, 0x40}};
|
|
|
|
static nsIIOService *nsio = NULL;
|
|
|
|
typedef struct {
|
|
const nsIWineURIVtbl *lpWineURIVtbl;
|
|
|
|
LONG ref;
|
|
|
|
nsIURI *uri;
|
|
NSContainer *container;
|
|
LPWSTR wine_url;
|
|
PRBool is_doc_uri;
|
|
BOOL use_wine_url;
|
|
} nsURI;
|
|
|
|
#define NSURI(x) ((nsIURI*) &(x)->lpWineURIVtbl)
|
|
#define NSWINEURI(x) ((nsIWineURI*) &(x)->lpWineURIVtbl)
|
|
|
|
static nsresult create_uri(nsIURI*,NSContainer*,nsIWineURI**);
|
|
|
|
static BOOL exec_shldocvw_67(HTMLDocument *doc, LPCWSTR url)
|
|
{
|
|
IOleCommandTarget *cmdtrg = NULL;
|
|
HRESULT hres;
|
|
|
|
hres = IOleClientSite_QueryInterface(doc->client, &IID_IOleCommandTarget,
|
|
(void**)&cmdtrg);
|
|
if(SUCCEEDED(hres)) {
|
|
VARIANT varUrl, varRes;
|
|
|
|
V_VT(&varUrl) = VT_BSTR;
|
|
V_BSTR(&varUrl) = SysAllocString(url);
|
|
V_VT(&varRes) = VT_BOOL;
|
|
|
|
hres = IOleCommandTarget_Exec(cmdtrg, &CGID_ShellDocView, 67, 0, &varUrl, &varRes);
|
|
|
|
IOleCommandTarget_Release(cmdtrg);
|
|
SysFreeString(V_BSTR(&varUrl));
|
|
|
|
if(SUCCEEDED(hres) && !V_BOOL(&varRes)) {
|
|
TRACE("got VARIANT_FALSE, do not load\n");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL before_async_open(nsChannel *channel, NSContainer *container)
|
|
{
|
|
IServiceProvider *service_provider;
|
|
HTMLDocument *doc = container->doc;
|
|
DWORD hlnf = 0;
|
|
LPCWSTR uri;
|
|
HRESULT hres;
|
|
|
|
nsIWineURI_GetWineURL(channel->uri, &uri);
|
|
if(!uri) {
|
|
ERR("GetWineURL returned NULL\n");
|
|
return TRUE;
|
|
}
|
|
|
|
if(!doc) {
|
|
NSContainer *container_iter = container;
|
|
|
|
hlnf = HLNF_OPENINNEWWINDOW;
|
|
while(!container_iter->doc)
|
|
container_iter = container_iter->parent;
|
|
doc = container_iter->doc;
|
|
}
|
|
|
|
if(!hlnf && !exec_shldocvw_67(doc, uri))
|
|
return FALSE;
|
|
|
|
hres = IOleClientSite_QueryInterface(doc->client, &IID_IServiceProvider,
|
|
(void**)&service_provider);
|
|
if(SUCCEEDED(hres)) {
|
|
IHlinkFrame *hlink_frame;
|
|
|
|
hres = IServiceProvider_QueryService(service_provider, &IID_IHlinkFrame,
|
|
&IID_IHlinkFrame, (void**)&hlink_frame);
|
|
IServiceProvider_Release(service_provider);
|
|
if(SUCCEEDED(hres)) {
|
|
hlink_frame_navigate(doc, hlink_frame, uri, channel->post_data_stream, hlnf);
|
|
IHlinkFrame_Release(hlink_frame);
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define NSCHANNEL_THIS(iface) DEFINE_THIS(nsChannel, HttpChannel, iface)
|
|
|
|
static nsresult NSAPI nsChannel_QueryInterface(nsIHttpChannel *iface, nsIIDRef riid, nsQIResult result)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
*result = NULL;
|
|
|
|
if(IsEqualGUID(&IID_nsISupports, riid)) {
|
|
TRACE("(%p)->(IID_nsISupports %p)\n", This, result);
|
|
*result = NSCHANNEL(This);
|
|
}else if(IsEqualGUID(&IID_nsIRequest, riid)) {
|
|
TRACE("(%p)->(IID_nsIRequest %p)\n", This, result);
|
|
*result = NSCHANNEL(This);
|
|
}else if(IsEqualGUID(&IID_nsIChannel, riid)) {
|
|
TRACE("(%p)->(IID_nsIChannel %p)\n", This, result);
|
|
*result = NSCHANNEL(This);
|
|
}else if(This->http_channel && IsEqualGUID(&IID_nsIHttpChannel, riid)) {
|
|
TRACE("(%p)->(IID_nsIHttpChannel %p)\n", This, result);
|
|
*result = NSHTTPCHANNEL(This);
|
|
}else if(IsEqualGUID(&IID_nsIUploadChannel, riid)) {
|
|
TRACE("(%p)->(IID_nsIUploadChannel %p)\n", This, result);
|
|
*result = NSUPCHANNEL(This);
|
|
}
|
|
|
|
if(*result) {
|
|
nsIChannel_AddRef(NSCHANNEL(This));
|
|
return NS_OK;
|
|
}
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), result);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_QueryInterface(This->channel, riid, result);
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsChannel_AddRef(nsIHttpChannel *iface)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
nsrefcnt ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsChannel_Release(nsIHttpChannel *iface)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
if(!ref) {
|
|
nsIWineURI_Release(This->uri);
|
|
if(This->channel)
|
|
nsIChannel_Release(This->channel);
|
|
if(This->http_channel)
|
|
nsIHttpChannel_Release(This->http_channel);
|
|
if(This->post_data_stream)
|
|
nsIInputStream_Release(This->post_data_stream);
|
|
if(This->load_group)
|
|
nsILoadGroup_Release(This->load_group);
|
|
if(This->notif_callback)
|
|
nsIInterfaceRequestor_Release(This->notif_callback);
|
|
if(This->original_uri)
|
|
nsIURI_Release(This->original_uri);
|
|
heap_free(This->content);
|
|
heap_free(This->charset);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetName(nsIHttpChannel *iface, nsACString *aName)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aName);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetName(This->channel, aName);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_IsPending(nsIHttpChannel *iface, PRBool *_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, _retval);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_IsPending(This->channel, _retval);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetStatus(nsIHttpChannel *iface, nsresult *aStatus)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aStatus);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetStatus(This->channel, aStatus);
|
|
|
|
TRACE("returning NS_OK\n");
|
|
return *aStatus = NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_Cancel(nsIHttpChannel *iface, nsresult aStatus)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%08x)\n", This, aStatus);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_Cancel(This->channel, aStatus);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_Suspend(nsIHttpChannel *iface)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_Suspend(This->channel);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_Resume(nsIHttpChannel *iface)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_Resume(This->channel);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetLoadGroup(nsIHttpChannel *iface, nsILoadGroup **aLoadGroup)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aLoadGroup);
|
|
|
|
if(This->load_group)
|
|
nsILoadGroup_AddRef(This->load_group);
|
|
|
|
*aLoadGroup = This->load_group;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetLoadGroup(nsIHttpChannel *iface, nsILoadGroup *aLoadGroup)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aLoadGroup);
|
|
|
|
if(This->load_group)
|
|
nsILoadGroup_Release(This->load_group);
|
|
if(aLoadGroup)
|
|
nsILoadGroup_AddRef(aLoadGroup);
|
|
|
|
This->load_group = aLoadGroup;
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetLoadGroup(This->channel, aLoadGroup);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetLoadFlags(nsIHttpChannel *iface, nsLoadFlags *aLoadFlags)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aLoadFlags);
|
|
|
|
*aLoadFlags = This->load_flags;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetLoadFlags(nsIHttpChannel *iface, nsLoadFlags aLoadFlags)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%08x)\n", This, aLoadFlags);
|
|
|
|
This->load_flags = aLoadFlags;
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetLoadFlags(This->channel, aLoadFlags);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetOriginalURI(nsIHttpChannel *iface, nsIURI **aOriginalURI)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aOriginalURI);
|
|
|
|
if(This->original_uri)
|
|
nsIURI_AddRef(This->original_uri);
|
|
|
|
*aOriginalURI = This->original_uri;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetOriginalURI(nsIHttpChannel *iface, nsIURI *aOriginalURI)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aOriginalURI);
|
|
|
|
if(This->original_uri)
|
|
nsIURI_Release(This->original_uri);
|
|
|
|
nsIURI_AddRef(aOriginalURI);
|
|
This->original_uri = aOriginalURI;
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetOriginalURI(This->channel, aOriginalURI);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetURI(nsIHttpChannel *iface, nsIURI **aURI)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aURI);
|
|
|
|
nsIWineURI_AddRef(This->uri);
|
|
*aURI = (nsIURI*)This->uri;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetOwner(nsIHttpChannel *iface, nsISupports **aOwner)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aOwner);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetOwner(This->channel, aOwner);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetOwner(nsIHttpChannel *iface, nsISupports *aOwner)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aOwner);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetOwner(This->channel, aOwner);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetNotificationCallbacks(nsIHttpChannel *iface,
|
|
nsIInterfaceRequestor **aNotificationCallbacks)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aNotificationCallbacks);
|
|
|
|
if(This->notif_callback)
|
|
nsIInterfaceRequestor_AddRef(This->notif_callback);
|
|
*aNotificationCallbacks = This->notif_callback;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetNotificationCallbacks(nsIHttpChannel *iface,
|
|
nsIInterfaceRequestor *aNotificationCallbacks)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aNotificationCallbacks);
|
|
|
|
if(This->notif_callback)
|
|
nsIInterfaceRequestor_Release(This->notif_callback);
|
|
if(aNotificationCallbacks)
|
|
nsIInterfaceRequestor_AddRef(aNotificationCallbacks);
|
|
|
|
This->notif_callback = aNotificationCallbacks;
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetNotificationCallbacks(This->channel, aNotificationCallbacks);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetSecurityInfo(nsIHttpChannel *iface, nsISupports **aSecurityInfo)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aSecurityInfo);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetSecurityInfo(This->channel, aSecurityInfo);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetContentType(nsIHttpChannel *iface, nsACString *aContentType)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContentType);
|
|
|
|
if(This->content) {
|
|
nsACString_SetData(aContentType, This->content);
|
|
return S_OK;
|
|
}
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetContentType(This->channel, aContentType);
|
|
|
|
TRACE("returning default text/html\n");
|
|
nsACString_SetData(aContentType, "text/html");
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetContentType(nsIHttpChannel *iface,
|
|
const nsACString *aContentType)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContentType);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetContentType(This->channel, aContentType);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetContentCharset(nsIHttpChannel *iface,
|
|
nsACString *aContentCharset)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContentCharset);
|
|
|
|
if(This->charset) {
|
|
nsACString_SetData(aContentCharset, This->charset);
|
|
return NS_OK;
|
|
}
|
|
|
|
if(This->channel) {
|
|
nsresult nsres = nsIChannel_GetContentCharset(This->channel, aContentCharset);
|
|
const char *ch;
|
|
nsACString_GetData(aContentCharset, &ch);
|
|
return nsres;
|
|
}
|
|
|
|
nsACString_SetData(aContentCharset, "");
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetContentCharset(nsIHttpChannel *iface,
|
|
const nsACString *aContentCharset)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContentCharset);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetContentCharset(This->channel, aContentCharset);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetContentLength(nsIHttpChannel *iface, PRInt32 *aContentLength)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContentLength);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_GetContentLength(This->channel, aContentLength);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetContentLength(nsIHttpChannel *iface, PRInt32 aContentLength)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%d)\n", This, aContentLength);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_SetContentLength(This->channel, aContentLength);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_Open(nsIHttpChannel *iface, nsIInputStream **_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, _retval);
|
|
|
|
if(This->channel)
|
|
return nsIChannel_Open(This->channel, _retval);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static BOOL do_load_from_moniker_hack(nsChannel *This)
|
|
{
|
|
nsACString scheme_str;
|
|
nsresult nsres;
|
|
BOOL ret = TRUE;
|
|
|
|
/*
|
|
* We should always load the page from IMoniker, but Wine is not yet
|
|
* ready for this. This function is a heuristic, that decides which
|
|
* way of loading is better (Gecko implementation or IMoniker). The
|
|
* aim is to always return TRUE.
|
|
*/
|
|
|
|
/* Load from moniker if there is no Gecko channel available */
|
|
if(!This->channel)
|
|
return TRUE;
|
|
|
|
nsACString_Init(&scheme_str, NULL);
|
|
nsres = nsIWineURI_GetScheme(This->uri, &scheme_str);
|
|
|
|
if(NS_SUCCEEDED(nsres)) {
|
|
const char *scheme;
|
|
|
|
nsACString_GetData(&scheme_str, &scheme);
|
|
ret = !strcmp(scheme, "wine");
|
|
}
|
|
|
|
nsACString_Finish(&scheme_str);
|
|
return ret;
|
|
}
|
|
|
|
static HRESULT create_mon_for_nschannel(nsChannel *channel, IMoniker **mon)
|
|
{
|
|
nsIWineURI *wine_uri;
|
|
LPCWSTR wine_url;
|
|
nsresult nsres;
|
|
HRESULT hres;
|
|
|
|
if(!channel->original_uri) {
|
|
ERR("original_uri == NULL\n");
|
|
return E_FAIL;
|
|
}
|
|
|
|
nsres = nsIURI_QueryInterface(channel->original_uri, &IID_nsIWineURI, (void**)&wine_uri);
|
|
if(NS_FAILED(nsres)) {
|
|
ERR("Could not get nsIWineURI: %08x\n", nsres);
|
|
return E_FAIL;
|
|
}
|
|
|
|
nsIWineURI_GetWineURL(wine_uri, &wine_url);
|
|
nsIWineURI_Release(wine_uri);
|
|
if(!wine_url) {
|
|
TRACE("wine_url == NULL\n");
|
|
return E_FAIL;
|
|
}
|
|
|
|
hres = CreateURLMoniker(NULL, wine_url, mon);
|
|
if(FAILED(hres))
|
|
WARN("CreateURLMonikrer failed: %08x\n", hres);
|
|
|
|
return hres;
|
|
}
|
|
|
|
static nsresult async_open_doc_uri(nsChannel *This, NSContainer *container,
|
|
nsIStreamListener *listener, nsISupports *context, BOOL *open)
|
|
{
|
|
IMoniker *mon;
|
|
HRESULT hres;
|
|
|
|
*open = FALSE;
|
|
|
|
if(container->bscallback) {
|
|
nsIChannel_AddRef(NSCHANNEL(This));
|
|
container->bscallback->nschannel = This;
|
|
|
|
nsIStreamListener_AddRef(listener);
|
|
container->bscallback->nslistener = listener;
|
|
|
|
if(context) {
|
|
nsISupports_AddRef(context);
|
|
container->bscallback->nscontext = context;
|
|
}
|
|
|
|
if(do_load_from_moniker_hack(This))
|
|
return WINE_NS_LOAD_FROM_MONIKER;
|
|
}else {
|
|
BOOL cont = before_async_open(This, container);
|
|
|
|
if(!cont) {
|
|
TRACE("canceled\n");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
if(!container->doc) {
|
|
return This->channel
|
|
? nsIChannel_AsyncOpen(This->channel, listener, context)
|
|
: NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
hres = create_mon_for_nschannel(This, &mon);
|
|
if(FAILED(hres)) {
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
set_current_mon(container->doc, mon);
|
|
}
|
|
|
|
*open = TRUE;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult async_open(nsChannel *This, NSContainer *container, nsIStreamListener *listener,
|
|
nsISupports *context)
|
|
{
|
|
BSCallback *bscallback;
|
|
IMoniker *mon = NULL;
|
|
nsresult nsres;
|
|
task_t *task;
|
|
HRESULT hres;
|
|
|
|
if(This->channel) {
|
|
if(This->post_data_stream) {
|
|
nsIUploadChannel *upload_channel;
|
|
|
|
nsres = nsIChannel_QueryInterface(This->channel, &IID_nsIUploadChannel,
|
|
(void**)&upload_channel);
|
|
if(NS_SUCCEEDED(nsres)) {
|
|
nsACString empty_string;
|
|
nsACString_Init(&empty_string, "");
|
|
|
|
nsres = nsIUploadChannel_SetUploadStream(upload_channel, This->post_data_stream,
|
|
&empty_string, -1);
|
|
nsIUploadChannel_Release(upload_channel);
|
|
if(NS_FAILED(nsres))
|
|
WARN("SetUploadStream failed: %08x\n", nsres);
|
|
|
|
nsACString_Finish(&empty_string);
|
|
}
|
|
}
|
|
|
|
nsres = nsIChannel_AsyncOpen(This->channel, listener, context);
|
|
|
|
if(mon)
|
|
IMoniker_Release(mon);
|
|
|
|
if(NS_FAILED(nsres) && (This->load_flags & LOAD_INITIAL_DOCUMENT_URI))
|
|
return WINE_NS_LOAD_FROM_MONIKER;
|
|
return nsres;
|
|
}
|
|
|
|
TRACE("channel == NULL\n");
|
|
|
|
hres = create_mon_for_nschannel(This, &mon);
|
|
if(FAILED(hres))
|
|
return NS_ERROR_UNEXPECTED;
|
|
|
|
bscallback = create_bscallback(mon);
|
|
IMoniker_Release(mon);
|
|
|
|
nsIChannel_AddRef(NSCHANNEL(This));
|
|
bscallback->nschannel = This;
|
|
|
|
nsIStreamListener_AddRef(listener);
|
|
bscallback->nslistener = listener;
|
|
|
|
if(context) {
|
|
nsISupports_AddRef(context);
|
|
bscallback->nscontext = context;
|
|
}
|
|
|
|
task = heap_alloc(sizeof(task_t));
|
|
|
|
task->doc = container->doc;
|
|
task->task_id = TASK_START_BINDING;
|
|
task->next = NULL;
|
|
task->bscallback = bscallback;
|
|
|
|
push_task(task);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_AsyncOpen(nsIHttpChannel *iface, nsIStreamListener *aListener,
|
|
nsISupports *aContext)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
NSContainer *container;
|
|
PRBool is_doc_uri;
|
|
BOOL open = TRUE;
|
|
nsresult nsres = NS_OK;
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, aListener, aContext);
|
|
|
|
nsIWineURI_GetNSContainer(This->uri, &container);
|
|
if(!container) {
|
|
TRACE("container = NULL\n");
|
|
return This->channel
|
|
? nsIChannel_AsyncOpen(This->channel, aListener, aContext)
|
|
: NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
nsIWineURI_GetIsDocumentURI(This->uri, &is_doc_uri);
|
|
|
|
if(is_doc_uri && (This->load_flags & LOAD_INITIAL_DOCUMENT_URI))
|
|
nsres = async_open_doc_uri(This, container, aListener, aContext, &open);
|
|
|
|
if(open)
|
|
nsres = async_open(This, container, aListener, aContext);
|
|
|
|
nsIWebBrowserChrome_Release(NSWBCHROME(container));
|
|
return nsres;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetRequestMethod(nsIHttpChannel *iface, nsACString *aRequestMethod)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aRequestMethod);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetRequestMethod(This->http_channel, aRequestMethod);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetRequestMethod(nsIHttpChannel *iface,
|
|
const nsACString *aRequestMethod)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aRequestMethod);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetRequestMethod(This->http_channel, aRequestMethod);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetReferrer(nsIHttpChannel *iface, nsIURI **aReferrer)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aReferrer);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetReferrer(This->http_channel, aReferrer);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetReferrer(nsIHttpChannel *iface, nsIURI *aReferrer)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aReferrer);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetReferrer(This->http_channel, aReferrer);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetRequestHeader(nsIHttpChannel *iface,
|
|
const nsACString *aHeader, nsACString *_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, aHeader, _retval);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetRequestHeader(This->http_channel, aHeader, _retval);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetRequestHeader(nsIHttpChannel *iface,
|
|
const nsACString *aHeader, const nsACString *aValue, PRBool aMerge)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p %x)\n", This, aHeader, aValue, aMerge);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetRequestHeader(This->http_channel, aHeader, aValue, aMerge);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_VisitRequestHeaders(nsIHttpChannel *iface,
|
|
nsIHttpHeaderVisitor *aVisitor)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aVisitor);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_VisitRequestHeaders(This->http_channel, aVisitor);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetAllowPipelining(nsIHttpChannel *iface, PRBool *aAllowPipelining)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aAllowPipelining);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetAllowPipelining(This->http_channel, aAllowPipelining);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetAllowPipelining(nsIHttpChannel *iface, PRBool aAllowPipelining)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%x)\n", This, aAllowPipelining);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetAllowPipelining(This->http_channel, aAllowPipelining);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetRedirectionLimit(nsIHttpChannel *iface, PRUint32 *aRedirectionLimit)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aRedirectionLimit);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetRedirectionLimit(This->http_channel, aRedirectionLimit);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetRedirectionLimit(nsIHttpChannel *iface, PRUint32 aRedirectionLimit)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%u)\n", This, aRedirectionLimit);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetRedirectionLimit(This->http_channel, aRedirectionLimit);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetResponseStatus(nsIHttpChannel *iface, PRUint32 *aResponseStatus)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aResponseStatus);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetResponseStatus(This->http_channel, aResponseStatus);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetResponseStatusText(nsIHttpChannel *iface,
|
|
nsACString *aResponseStatusText)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aResponseStatusText);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetResponseStatusText(This->http_channel, aResponseStatusText);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetRequestSucceeded(nsIHttpChannel *iface,
|
|
PRBool *aRequestSucceeded)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aRequestSucceeded);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetRequestSucceeded(This->http_channel, aRequestSucceeded);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_GetResponseHeader(nsIHttpChannel *iface,
|
|
const nsACString *header, nsACString *_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, header, _retval);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_GetResponseHeader(This->http_channel, header, _retval);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_SetResponseHeader(nsIHttpChannel *iface,
|
|
const nsACString *header, const nsACString *value, PRBool merge)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p %x)\n", This, header, value, merge);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_SetResponseHeader(This->http_channel, header, value, merge);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_VisitResponseHeaders(nsIHttpChannel *iface,
|
|
nsIHttpHeaderVisitor *aVisitor)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aVisitor);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_VisitResponseHeaders(This->http_channel, aVisitor);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_IsNoStoreResponse(nsIHttpChannel *iface, PRBool *_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, _retval);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_IsNoStoreResponse(This->http_channel, _retval);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsChannel_IsNoCacheResponse(nsIHttpChannel *iface, PRBool *_retval)
|
|
{
|
|
nsChannel *This = NSCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, _retval);
|
|
|
|
if(This->http_channel)
|
|
return nsIHttpChannel_IsNoCacheResponse(This->http_channel, _retval);
|
|
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
#undef NSCHANNEL_THIS
|
|
|
|
static const nsIHttpChannelVtbl nsChannelVtbl = {
|
|
nsChannel_QueryInterface,
|
|
nsChannel_AddRef,
|
|
nsChannel_Release,
|
|
nsChannel_GetName,
|
|
nsChannel_IsPending,
|
|
nsChannel_GetStatus,
|
|
nsChannel_Cancel,
|
|
nsChannel_Suspend,
|
|
nsChannel_Resume,
|
|
nsChannel_GetLoadGroup,
|
|
nsChannel_SetLoadGroup,
|
|
nsChannel_GetLoadFlags,
|
|
nsChannel_SetLoadFlags,
|
|
nsChannel_GetOriginalURI,
|
|
nsChannel_SetOriginalURI,
|
|
nsChannel_GetURI,
|
|
nsChannel_GetOwner,
|
|
nsChannel_SetOwner,
|
|
nsChannel_GetNotificationCallbacks,
|
|
nsChannel_SetNotificationCallbacks,
|
|
nsChannel_GetSecurityInfo,
|
|
nsChannel_GetContentType,
|
|
nsChannel_SetContentType,
|
|
nsChannel_GetContentCharset,
|
|
nsChannel_SetContentCharset,
|
|
nsChannel_GetContentLength,
|
|
nsChannel_SetContentLength,
|
|
nsChannel_Open,
|
|
nsChannel_AsyncOpen,
|
|
nsChannel_GetRequestMethod,
|
|
nsChannel_SetRequestMethod,
|
|
nsChannel_GetReferrer,
|
|
nsChannel_SetReferrer,
|
|
nsChannel_GetRequestHeader,
|
|
nsChannel_SetRequestHeader,
|
|
nsChannel_VisitRequestHeaders,
|
|
nsChannel_GetAllowPipelining,
|
|
nsChannel_SetAllowPipelining,
|
|
nsChannel_GetRedirectionLimit,
|
|
nsChannel_SetRedirectionLimit,
|
|
nsChannel_GetResponseStatus,
|
|
nsChannel_GetResponseStatusText,
|
|
nsChannel_GetRequestSucceeded,
|
|
nsChannel_GetResponseHeader,
|
|
nsChannel_SetResponseHeader,
|
|
nsChannel_VisitResponseHeaders,
|
|
nsChannel_IsNoStoreResponse,
|
|
nsChannel_IsNoCacheResponse
|
|
};
|
|
|
|
#define NSUPCHANNEL_THIS(iface) DEFINE_THIS(nsChannel, UploadChannel, iface)
|
|
|
|
static nsresult NSAPI nsUploadChannel_QueryInterface(nsIUploadChannel *iface, nsIIDRef riid,
|
|
nsQIResult result)
|
|
{
|
|
nsChannel *This = NSUPCHANNEL_THIS(iface);
|
|
return nsIChannel_QueryInterface(NSCHANNEL(This), riid, result);
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsUploadChannel_AddRef(nsIUploadChannel *iface)
|
|
{
|
|
nsChannel *This = NSUPCHANNEL_THIS(iface);
|
|
return nsIChannel_AddRef(NSCHANNEL(This));
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsUploadChannel_Release(nsIUploadChannel *iface)
|
|
{
|
|
nsChannel *This = NSUPCHANNEL_THIS(iface);
|
|
return nsIChannel_Release(NSCHANNEL(This));
|
|
}
|
|
|
|
static nsresult NSAPI nsUploadChannel_SetUploadStream(nsIUploadChannel *iface,
|
|
nsIInputStream *aStream, const nsACString *aContentType, PRInt32 aContentLength)
|
|
{
|
|
nsChannel *This = NSUPCHANNEL_THIS(iface);
|
|
const char *content_type;
|
|
|
|
TRACE("(%p)->(%p %p %d)\n", This, aStream, aContentType, aContentLength);
|
|
|
|
if(This->post_data_stream)
|
|
nsIInputStream_Release(This->post_data_stream);
|
|
|
|
if(aContentType) {
|
|
nsACString_GetData(aContentType, &content_type);
|
|
if(*content_type)
|
|
FIXME("Unsupported aContentType argument: %s\n", debugstr_a(content_type));
|
|
}
|
|
|
|
if(aContentLength != -1)
|
|
FIXME("Unsupported acontentLength = %d\n", aContentLength);
|
|
|
|
if(aStream)
|
|
nsIInputStream_AddRef(aStream);
|
|
This->post_data_stream = aStream;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsUploadChannel_GetUploadStream(nsIUploadChannel *iface,
|
|
nsIInputStream **aUploadStream)
|
|
{
|
|
nsChannel *This = NSUPCHANNEL_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aUploadStream);
|
|
|
|
if(This->post_data_stream)
|
|
nsIInputStream_AddRef(This->post_data_stream);
|
|
|
|
*aUploadStream = This->post_data_stream;
|
|
return NS_OK;
|
|
}
|
|
|
|
#undef NSUPCHANNEL_THIS
|
|
|
|
static const nsIUploadChannelVtbl nsUploadChannelVtbl = {
|
|
nsUploadChannel_QueryInterface,
|
|
nsUploadChannel_AddRef,
|
|
nsUploadChannel_Release,
|
|
nsUploadChannel_SetUploadStream,
|
|
nsUploadChannel_GetUploadStream
|
|
};
|
|
|
|
#define NSURI_THIS(iface) DEFINE_THIS(nsURI, WineURI, iface)
|
|
|
|
static nsresult NSAPI nsURI_QueryInterface(nsIWineURI *iface, nsIIDRef riid, nsQIResult result)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
*result = NULL;
|
|
|
|
if(IsEqualGUID(&IID_nsISupports, riid)) {
|
|
TRACE("(%p)->(IID_nsISupports %p)\n", This, result);
|
|
*result = NSURI(This);
|
|
}else if(IsEqualGUID(&IID_nsIURI, riid)) {
|
|
TRACE("(%p)->(IID_nsIURI %p)\n", This, result);
|
|
*result = NSURI(This);
|
|
}else if(IsEqualGUID(&IID_nsIWineURI, riid)) {
|
|
TRACE("(%p)->(IID_nsIWineURI %p)\n", This, result);
|
|
*result = NSURI(This);
|
|
}
|
|
|
|
if(*result) {
|
|
nsIURI_AddRef(NSURI(This));
|
|
return NS_OK;
|
|
}
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), result);
|
|
return This->uri ? nsIURI_QueryInterface(This->uri, riid, result) : NS_NOINTERFACE;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsURI_AddRef(nsIWineURI *iface)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsURI_Release(nsIWineURI *iface)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if(!ref) {
|
|
if(This->container)
|
|
nsIWebBrowserChrome_Release(NSWBCHROME(This->container));
|
|
if(This->uri)
|
|
nsIURI_Release(This->uri);
|
|
heap_free(This->wine_url);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetSpec(nsIWineURI *iface, nsACString *aSpec)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aSpec);
|
|
|
|
if(This->use_wine_url) {
|
|
char speca[INTERNET_MAX_URL_LENGTH] = "wine:";
|
|
WideCharToMultiByte(CP_ACP, 0, This->wine_url, -1, speca+5, sizeof(speca)-5, NULL, NULL);
|
|
nsACString_SetData(aSpec, speca);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetSpec(This->uri, aSpec);
|
|
|
|
TRACE("returning error\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetSpec(nsIWineURI *iface, const nsACString *aSpec)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aSpec);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetSpec(This->uri, aSpec);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetPrePath(nsIWineURI *iface, nsACString *aPrePath)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPrePath);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetPrePath(This->uri, aPrePath);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetScheme(nsIWineURI *iface, nsACString *aScheme)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aScheme);
|
|
|
|
if(This->use_wine_url) {
|
|
/*
|
|
* For Gecko we set scheme to unknown so it won't be handled
|
|
* as any special case.
|
|
*/
|
|
nsACString_SetData(aScheme, "wine");
|
|
return NS_OK;
|
|
}
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetScheme(This->uri, aScheme);
|
|
|
|
TRACE("returning error\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetScheme(nsIWineURI *iface, const nsACString *aScheme)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aScheme);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetScheme(This->uri, aScheme);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetUserPass(nsIWineURI *iface, nsACString *aUserPass)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aUserPass);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetUserPass(This->uri, aUserPass);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetUserPass(nsIWineURI *iface, const nsACString *aUserPass)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aUserPass);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetUserPass(This->uri, aUserPass);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetUsername(nsIWineURI *iface, nsACString *aUsername)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aUsername);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetUsername(This->uri, aUsername);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetUsername(nsIWineURI *iface, const nsACString *aUsername)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aUsername);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetUsername(This->uri, aUsername);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetPassword(nsIWineURI *iface, nsACString *aPassword)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPassword);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetPassword(This->uri, aPassword);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetPassword(nsIWineURI *iface, const nsACString *aPassword)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPassword);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetPassword(This->uri, aPassword);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetHostPort(nsIWineURI *iface, nsACString *aHostPort)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aHostPort);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetHostPort(This->uri, aHostPort);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetHostPort(nsIWineURI *iface, const nsACString *aHostPort)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aHostPort);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetHostPort(This->uri, aHostPort);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetHost(nsIWineURI *iface, nsACString *aHost)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aHost);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetHost(This->uri, aHost);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetHost(nsIWineURI *iface, const nsACString *aHost)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aHost);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetHost(This->uri, aHost);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetPort(nsIWineURI *iface, PRInt32 *aPort)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPort);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetPort(This->uri, aPort);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetPort(nsIWineURI *iface, PRInt32 aPort)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%d)\n", This, aPort);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetPort(This->uri, aPort);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetPath(nsIWineURI *iface, nsACString *aPath)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPath);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetPath(This->uri, aPath);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetPath(nsIWineURI *iface, const nsACString *aPath)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aPath);
|
|
|
|
if(This->uri)
|
|
return nsIURI_SetPath(This->uri, aPath);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_Equals(nsIWineURI *iface, nsIURI *other, PRBool *_retval)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, other, _retval);
|
|
|
|
if(This->uri)
|
|
return nsIURI_Equals(This->uri, other, _retval);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SchemeIs(nsIWineURI *iface, const char *scheme, PRBool *_retval)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, debugstr_a(scheme), _retval);
|
|
|
|
if(This->use_wine_url) {
|
|
WCHAR buf[INTERNET_MAX_SCHEME_LENGTH];
|
|
int len = MultiByteToWideChar(CP_ACP, 0, scheme, -1, buf, sizeof(buf)/sizeof(WCHAR))-1;
|
|
|
|
*_retval = strlenW(This->wine_url) > len
|
|
&& This->wine_url[len] == ':'
|
|
&& !memcmp(buf, This->wine_url, len*sizeof(WCHAR));
|
|
return NS_OK;
|
|
}
|
|
|
|
if(This->uri)
|
|
return nsIURI_SchemeIs(This->uri, scheme, _retval);
|
|
|
|
TRACE("returning error\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_Clone(nsIWineURI *iface, nsIURI **_retval)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, _retval);
|
|
|
|
if(This->uri) {
|
|
nsIURI *uri;
|
|
nsIWineURI *wine_uri;
|
|
nsresult nsres;
|
|
|
|
nsres = nsIURI_Clone(This->uri, &uri);
|
|
if(NS_FAILED(nsres)) {
|
|
WARN("Clone failed: %08x\n", nsres);
|
|
return nsres;
|
|
}
|
|
|
|
nsres = create_uri(uri, This->container, &wine_uri);
|
|
*_retval = (nsIURI*)wine_uri;
|
|
if(NS_SUCCEEDED(nsres))
|
|
return nsIWineURI_SetWineURL(wine_uri, This->wine_url);
|
|
return nsres;
|
|
}
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_Resolve(nsIWineURI *iface, const nsACString *arelativePath,
|
|
nsACString *_retval)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, arelativePath, _retval);
|
|
|
|
if(This->uri)
|
|
return nsIURI_Resolve(This->uri, arelativePath, _retval);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetAsciiSpec(nsIWineURI *iface, nsACString *aAsciiSpec)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aAsciiSpec);
|
|
|
|
if(This->use_wine_url)
|
|
return nsIURI_GetSpec(NSURI(This), aAsciiSpec);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetAsciiSpec(This->uri, aAsciiSpec);
|
|
|
|
TRACE("returning error\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetAsciiHost(nsIWineURI *iface, nsACString *aAsciiHost)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aAsciiHost);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetAsciiHost(This->uri, aAsciiHost);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetOriginCharset(nsIWineURI *iface, nsACString *aOriginCharset)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aOriginCharset);
|
|
|
|
if(This->uri)
|
|
return nsIURI_GetOriginCharset(This->uri, aOriginCharset);
|
|
|
|
FIXME("default action not implemented\n");
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetNSContainer(nsIWineURI *iface, NSContainer **aContainer)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContainer);
|
|
|
|
if(This->container)
|
|
nsIWebBrowserChrome_AddRef(NSWBCHROME(This->container));
|
|
*aContainer = This->container;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetNSContainer(nsIWineURI *iface, NSContainer *aContainer)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aContainer);
|
|
|
|
if(This->container) {
|
|
if(This->container == aContainer)
|
|
return NS_OK;
|
|
TRACE("Changing %p -> %p\n", This->container, aContainer);
|
|
nsIWebBrowserChrome_Release(NSWBCHROME(This->container));
|
|
}
|
|
|
|
if(aContainer)
|
|
nsIWebBrowserChrome_AddRef(NSWBCHROME(aContainer));
|
|
This->container = aContainer;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetIsDocumentURI(nsIWineURI *iface, PRBool *aIsDocumentURI)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aIsDocumentURI);
|
|
|
|
*aIsDocumentURI = This->is_doc_uri;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetIsDocumentURI(nsIWineURI *iface, PRBool aIsDocumentURI)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%x)\n", This, aIsDocumentURI);
|
|
|
|
This->is_doc_uri = aIsDocumentURI;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_GetWineURL(nsIWineURI *iface, LPCWSTR *aURL)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aURL);
|
|
|
|
*aURL = This->wine_url;
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsURI_SetWineURL(nsIWineURI *iface, LPCWSTR aURL)
|
|
{
|
|
nsURI *This = NSURI_THIS(iface);
|
|
|
|
static const WCHAR wszFtp[] = {'f','t','p',':'};
|
|
static const WCHAR wszHttp[] = {'h','t','t','p',':'};
|
|
static const WCHAR wszHttps[] = {'h','t','t','p','s',':'};
|
|
|
|
TRACE("(%p)->(%s)\n", This, debugstr_w(aURL));
|
|
|
|
heap_free(This->wine_url);
|
|
|
|
if(aURL) {
|
|
int len = strlenW(aURL)+1;
|
|
This->wine_url = heap_alloc(len*sizeof(WCHAR));
|
|
memcpy(This->wine_url, aURL, len*sizeof(WCHAR));
|
|
|
|
/* FIXME: Always use wine url */
|
|
This->use_wine_url =
|
|
strncmpW(aURL, wszFtp, sizeof(wszFtp)/sizeof(WCHAR))
|
|
&& strncmpW(aURL, wszHttp, sizeof(wszHttp)/sizeof(WCHAR))
|
|
&& strncmpW(aURL, wszHttps, sizeof(wszHttps)/sizeof(WCHAR));
|
|
}else {
|
|
This->wine_url = NULL;
|
|
This->use_wine_url = FALSE;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
#undef NSURI_THIS
|
|
|
|
static const nsIWineURIVtbl nsWineURIVtbl = {
|
|
nsURI_QueryInterface,
|
|
nsURI_AddRef,
|
|
nsURI_Release,
|
|
nsURI_GetSpec,
|
|
nsURI_SetSpec,
|
|
nsURI_GetPrePath,
|
|
nsURI_GetScheme,
|
|
nsURI_SetScheme,
|
|
nsURI_GetUserPass,
|
|
nsURI_SetUserPass,
|
|
nsURI_GetUsername,
|
|
nsURI_SetUsername,
|
|
nsURI_GetPassword,
|
|
nsURI_SetPassword,
|
|
nsURI_GetHostPort,
|
|
nsURI_SetHostPort,
|
|
nsURI_GetHost,
|
|
nsURI_SetHost,
|
|
nsURI_GetPort,
|
|
nsURI_SetPort,
|
|
nsURI_GetPath,
|
|
nsURI_SetPath,
|
|
nsURI_Equals,
|
|
nsURI_SchemeIs,
|
|
nsURI_Clone,
|
|
nsURI_Resolve,
|
|
nsURI_GetAsciiSpec,
|
|
nsURI_GetAsciiHost,
|
|
nsURI_GetOriginCharset,
|
|
nsURI_GetNSContainer,
|
|
nsURI_SetNSContainer,
|
|
nsURI_GetIsDocumentURI,
|
|
nsURI_SetIsDocumentURI,
|
|
nsURI_GetWineURL,
|
|
nsURI_SetWineURL
|
|
};
|
|
|
|
static nsresult create_uri(nsIURI *uri, NSContainer *container, nsIWineURI **_retval)
|
|
{
|
|
nsURI *ret = heap_alloc(sizeof(nsURI));
|
|
|
|
ret->lpWineURIVtbl = &nsWineURIVtbl;
|
|
ret->ref = 1;
|
|
ret->uri = uri;
|
|
ret->container = container;
|
|
ret->wine_url = NULL;
|
|
ret->is_doc_uri = FALSE;
|
|
ret->use_wine_url = FALSE;
|
|
|
|
if(container)
|
|
nsIWebBrowserChrome_AddRef(NSWBCHROME(container));
|
|
|
|
TRACE("retval=%p\n", ret);
|
|
*_retval = NSWINEURI(ret);
|
|
return NS_OK;
|
|
}
|
|
|
|
typedef struct {
|
|
const nsIProtocolHandlerVtbl *lpProtocolHandlerVtbl;
|
|
|
|
LONG ref;
|
|
|
|
nsIProtocolHandler *nshandler;
|
|
} nsProtocolHandler;
|
|
|
|
#define NSPROTHANDLER(x) ((nsIProtocolHandler*) &(x)->lpProtocolHandlerVtbl)
|
|
|
|
#define NSPROTHANDLER_THIS(iface) DEFINE_THIS(nsProtocolHandler, ProtocolHandler, iface)
|
|
|
|
static nsresult NSAPI nsProtocolHandler_QueryInterface(nsIProtocolHandler *iface, nsIIDRef riid,
|
|
nsQIResult result)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
*result = NULL;
|
|
|
|
if(IsEqualGUID(&IID_nsISupports, riid)) {
|
|
TRACE("(%p)->(IID_nsISupports %p)\n", This, result);
|
|
*result = NSPROTHANDLER(This);
|
|
}else if(IsEqualGUID(&IID_nsIProtocolHandler, riid)) {
|
|
TRACE("(%p)->(IID_nsIProtocolHandler %p)\n", This, result);
|
|
*result = NSPROTHANDLER(This);
|
|
}else if(IsEqualGUID(&IID_nsIExternalProtocolHandler, riid)) {
|
|
TRACE("(%p)->(IID_nsIExternalProtocolHandler %p), returning NULL\n", This, result);
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
if(*result) {
|
|
nsISupports_AddRef((nsISupports*)*result);
|
|
return NS_OK;
|
|
}
|
|
|
|
WARN("(%s %p)\n", debugstr_guid(riid), result);
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsProtocolHandler_AddRef(nsIProtocolHandler *iface)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsProtocolHandler_Release(nsIProtocolHandler *iface)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if(!ref) {
|
|
if(This->nshandler)
|
|
nsIProtocolHandler_Release(This->nshandler);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_GetScheme(nsIProtocolHandler *iface, nsACString *aScheme)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aScheme);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_GetScheme(This->nshandler, aScheme);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_GetDefaultPort(nsIProtocolHandler *iface,
|
|
PRInt32 *aDefaultPort)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aDefaultPort);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_GetDefaultPort(This->nshandler, aDefaultPort);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_GetProtocolFlags(nsIProtocolHandler *iface,
|
|
PRUint32 *aProtocolFlags)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, aProtocolFlags);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_GetProtocolFlags(This->nshandler, aProtocolFlags);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_NewURI(nsIProtocolHandler *iface,
|
|
const nsACString *aSpec, const char *aOriginCharset, nsIURI *aBaseURI, nsIURI **_retval)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("((%p)->%p %s %p %p)\n", This, aSpec, debugstr_a(aOriginCharset), aBaseURI, _retval);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_NewURI(This->nshandler, aSpec, aOriginCharset, aBaseURI, _retval);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_NewChannel(nsIProtocolHandler *iface,
|
|
nsIURI *aURI, nsIChannel **_retval)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("(%p)->(%p %p)\n", This, aURI, _retval);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_NewChannel(This->nshandler, aURI, _retval);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
static nsresult NSAPI nsProtocolHandler_AllowPort(nsIProtocolHandler *iface,
|
|
PRInt32 port, const char *scheme, PRBool *_retval)
|
|
{
|
|
nsProtocolHandler *This = NSPROTHANDLER_THIS(iface);
|
|
|
|
TRACE("(%p)->(%d %s %p)\n", This, port, debugstr_a(scheme), _retval);
|
|
|
|
if(This->nshandler)
|
|
return nsIProtocolHandler_AllowPort(This->nshandler, port, scheme, _retval);
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
#undef NSPROTHANDLER_THIS
|
|
|
|
static const nsIProtocolHandlerVtbl nsProtocolHandlerVtbl = {
|
|
nsProtocolHandler_QueryInterface,
|
|
nsProtocolHandler_AddRef,
|
|
nsProtocolHandler_Release,
|
|
nsProtocolHandler_GetScheme,
|
|
nsProtocolHandler_GetDefaultPort,
|
|
nsProtocolHandler_GetProtocolFlags,
|
|
nsProtocolHandler_NewURI,
|
|
nsProtocolHandler_NewChannel,
|
|
nsProtocolHandler_AllowPort
|
|
};
|
|
|
|
static nsIProtocolHandler *create_protocol_handler(nsIProtocolHandler *nshandler)
|
|
{
|
|
nsProtocolHandler *ret = heap_alloc(sizeof(nsProtocolHandler));
|
|
|
|
ret->lpProtocolHandlerVtbl = &nsProtocolHandlerVtbl;
|
|
ret->ref = 1;
|
|
ret->nshandler = nshandler;
|
|
|
|
return NSPROTHANDLER(ret);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_QueryInterface(nsIIOService *iface, nsIIDRef riid,
|
|
nsQIResult result)
|
|
{
|
|
*result = NULL;
|
|
|
|
if(IsEqualGUID(&IID_nsISupports, riid)
|
|
|| IsEqualGUID(&IID_nsIIOService, riid)) {
|
|
*result = iface;
|
|
nsIIOService_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("(%s %p)\n", debugstr_guid(riid), result);
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsIOService_AddRef(nsIIOService *iface)
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsIOService_Release(nsIIOService *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_GetProtocolHandler(nsIIOService *iface, const char *aScheme,
|
|
nsIProtocolHandler **_retval)
|
|
{
|
|
nsIExternalProtocolHandler *nsexthandler;
|
|
nsIProtocolHandler *nshandler;
|
|
nsresult nsres;
|
|
|
|
TRACE("(%s %p)\n", debugstr_a(aScheme), _retval);
|
|
|
|
nsres = nsIIOService_GetProtocolHandler(nsio, aScheme, &nshandler);
|
|
if(NS_FAILED(nsres)) {
|
|
WARN("GetProtocolHandler failed: %08x\n", nsres);
|
|
return nsres;
|
|
}
|
|
|
|
nsres = nsIProtocolHandler_QueryInterface(nshandler, &IID_nsIExternalProtocolHandler,
|
|
(void**)&nsexthandler);
|
|
if(NS_FAILED(nsres)) {
|
|
*_retval = nshandler;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIExternalProtocolHandler_Release(nsexthandler);
|
|
*_retval = create_protocol_handler(nshandler);
|
|
TRACE("return %p\n", *_retval);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_GetProtocolFlags(nsIIOService *iface, const char *aScheme,
|
|
PRUint32 *_retval)
|
|
{
|
|
TRACE("(%s %p)\n", debugstr_a(aScheme), _retval);
|
|
return nsIIOService_GetProtocolFlags(nsio, aScheme, _retval);
|
|
}
|
|
|
|
static BOOL is_gecko_special_uri(const char *spec)
|
|
{
|
|
static const char chromeW[] = "chrome:";
|
|
static const char jarW[] = "jar:";
|
|
static const char resourceW[] = "resource:";
|
|
static const char javascriptW[] = "javascript:";
|
|
|
|
return !strncasecmp(spec, chromeW, sizeof(chromeW)-1)
|
|
|| !strncasecmp(spec, resourceW, sizeof(resourceW)-1)
|
|
|| !strncasecmp(spec, jarW, sizeof(jarW)-1)
|
|
|| !strncasecmp(spec, javascriptW, sizeof(javascriptW)-1);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_NewURI(nsIIOService *iface, const nsACString *aSpec,
|
|
const char *aOriginCharset, nsIURI *aBaseURI, nsIURI **_retval)
|
|
{
|
|
const char *spec = NULL;
|
|
nsACString spec_str;
|
|
NSContainer *nscontainer = NULL;
|
|
nsIURI *uri = NULL;
|
|
LPCWSTR base_wine_url = NULL;
|
|
nsIWineURI *base_wine_uri = NULL, *wine_uri;
|
|
BOOL is_wine_uri = FALSE;
|
|
nsresult nsres;
|
|
|
|
nsACString_GetData(aSpec, &spec);
|
|
|
|
TRACE("(%p(%s) %s %p %p)\n", aSpec, debugstr_a(spec), debugstr_a(aOriginCharset),
|
|
aBaseURI, _retval);
|
|
|
|
if(is_gecko_special_uri(spec))
|
|
return nsIIOService_NewURI(nsio, aSpec, aOriginCharset, aBaseURI, _retval);
|
|
|
|
if(!strncmp(spec, "wine:", 5)) {
|
|
spec += 5;
|
|
is_wine_uri = TRUE;
|
|
}
|
|
|
|
if(aBaseURI) {
|
|
nsACString base_uri_str;
|
|
const char *base_uri = NULL;
|
|
|
|
nsACString_Init(&base_uri_str, NULL);
|
|
|
|
nsres = nsIURI_GetSpec(aBaseURI, &base_uri_str);
|
|
if(NS_SUCCEEDED(nsres)) {
|
|
nsACString_GetData(&base_uri_str, &base_uri);
|
|
TRACE("base_uri=%s\n", debugstr_a(base_uri));
|
|
}else {
|
|
ERR("GetSpec failed: %08x\n", nsres);
|
|
}
|
|
|
|
nsACString_Finish(&base_uri_str);
|
|
}
|
|
|
|
nsACString_Init(&spec_str, spec);
|
|
nsres = nsIIOService_NewURI(nsio, aSpec, aOriginCharset, aBaseURI, &uri);
|
|
nsACString_Finish(&spec_str);
|
|
if(NS_FAILED(nsres))
|
|
TRACE("NewURI failed: %08x\n", nsres);
|
|
|
|
if(aBaseURI) {
|
|
nsres = nsIURI_QueryInterface(aBaseURI, &IID_nsIWineURI, (void**)&base_wine_uri);
|
|
if(NS_SUCCEEDED(nsres)) {
|
|
nsIWineURI_GetNSContainer(base_wine_uri, &nscontainer);
|
|
nsIWineURI_GetWineURL(base_wine_uri, &base_wine_url);
|
|
}else {
|
|
TRACE("Could not get base nsIWineURI: %08x\n", nsres);
|
|
}
|
|
}
|
|
|
|
nsres = create_uri(uri, nscontainer, &wine_uri);
|
|
*_retval = (nsIURI*)wine_uri;
|
|
|
|
if(nscontainer)
|
|
nsIWebBrowserChrome_Release(NSWBCHROME(nscontainer));
|
|
|
|
if(base_wine_url) {
|
|
WCHAR url[INTERNET_MAX_URL_LENGTH], rel_url[INTERNET_MAX_URL_LENGTH];
|
|
DWORD len;
|
|
HRESULT hres;
|
|
|
|
MultiByteToWideChar(CP_ACP, 0, spec, -1, rel_url, sizeof(rel_url)/sizeof(WCHAR));
|
|
|
|
hres = CoInternetCombineUrl(base_wine_url, rel_url,
|
|
URL_ESCAPE_SPACES_ONLY|URL_DONT_ESCAPE_EXTRA_INFO,
|
|
url, sizeof(url)/sizeof(WCHAR), &len, 0);
|
|
if(SUCCEEDED(hres))
|
|
nsIWineURI_SetWineURL(wine_uri, url);
|
|
else
|
|
WARN("CoCombineUrl failed: %08x\n", hres);
|
|
}else if(is_wine_uri) {
|
|
WCHAR url[INTERNET_MAX_URL_LENGTH];
|
|
|
|
MultiByteToWideChar(CP_ACP, 0, spec, -1, url, sizeof(url)/sizeof(WCHAR));
|
|
nsIWineURI_SetWineURL(wine_uri, url);
|
|
}
|
|
|
|
if(base_wine_uri)
|
|
nsIWineURI_Release(base_wine_uri);
|
|
|
|
return nsres;
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_NewFileURI(nsIIOService *iface, nsIFile *aFile,
|
|
nsIURI **_retval)
|
|
{
|
|
TRACE("(%p %p)\n", aFile, _retval);
|
|
return nsIIOService_NewFileURI(nsio, aFile, _retval);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_NewChannelFromURI(nsIIOService *iface, nsIURI *aURI,
|
|
nsIChannel **_retval)
|
|
{
|
|
nsIChannel *channel = NULL;
|
|
nsChannel *ret;
|
|
nsIWineURI *wine_uri;
|
|
nsresult nsres;
|
|
|
|
TRACE("(%p %p)\n", aURI, _retval);
|
|
|
|
nsres = nsIIOService_NewChannelFromURI(nsio, aURI, &channel);
|
|
if(NS_FAILED(nsres) && nsres != NS_ERROR_UNKNOWN_PROTOCOL) {
|
|
WARN("NewChannelFromURI failed: %08x\n", nsres);
|
|
*_retval = channel;
|
|
return nsres;
|
|
}
|
|
|
|
nsres = nsIURI_QueryInterface(aURI, &IID_nsIWineURI, (void**)&wine_uri);
|
|
if(NS_FAILED(nsres)) {
|
|
WARN("Could not get nsIWineURI: %08x\n", nsres);
|
|
*_retval = channel;
|
|
return channel ? NS_OK : NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
ret = heap_alloc(sizeof(nsChannel));
|
|
|
|
ret->lpHttpChannelVtbl = &nsChannelVtbl;
|
|
ret->lpUploadChannelVtbl = &nsUploadChannelVtbl;
|
|
ret->ref = 1;
|
|
ret->channel = channel;
|
|
ret->http_channel = NULL;
|
|
ret->uri = wine_uri;
|
|
ret->post_data_stream = NULL;
|
|
ret->load_group = NULL;
|
|
ret->notif_callback = NULL;
|
|
ret->load_flags = 0;
|
|
ret->content = NULL;
|
|
ret->charset = NULL;
|
|
|
|
nsIURI_AddRef(aURI);
|
|
ret->original_uri = aURI;
|
|
|
|
if(channel)
|
|
nsIChannel_QueryInterface(channel, &IID_nsIHttpChannel, (void**)&ret->http_channel);
|
|
|
|
*_retval = NSCHANNEL(ret);
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_NewChannel(nsIIOService *iface, const nsACString *aSpec,
|
|
const char *aOriginCharset, nsIURI *aBaseURI, nsIChannel **_retval)
|
|
{
|
|
TRACE("(%p %s %p %p)\n", aSpec, debugstr_a(aOriginCharset), aBaseURI, _retval);
|
|
return nsIIOService_NewChannel(nsio, aSpec, aOriginCharset, aBaseURI, _retval);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_GetOffline(nsIIOService *iface, PRBool *aOffline)
|
|
{
|
|
TRACE("(%p)\n", aOffline);
|
|
return nsIIOService_GetOffline(nsio, aOffline);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_SetOffline(nsIIOService *iface, PRBool aOffline)
|
|
{
|
|
TRACE("(%x)\n", aOffline);
|
|
return nsIIOService_SetOffline(nsio, aOffline);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_AllowPort(nsIIOService *iface, PRInt32 aPort,
|
|
const char *aScheme, PRBool *_retval)
|
|
{
|
|
TRACE("(%d %s %p)\n", aPort, debugstr_a(aScheme), _retval);
|
|
return nsIIOService_AllowPort(nsio, aPort, debugstr_a(aScheme), _retval);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOService_ExtractScheme(nsIIOService *iface, const nsACString *urlString,
|
|
nsACString * _retval)
|
|
{
|
|
TRACE("(%p %p)\n", urlString, _retval);
|
|
return nsIIOService_ExtractScheme(nsio, urlString, _retval);
|
|
}
|
|
|
|
static const nsIIOServiceVtbl nsIOServiceVtbl = {
|
|
nsIOService_QueryInterface,
|
|
nsIOService_AddRef,
|
|
nsIOService_Release,
|
|
nsIOService_GetProtocolHandler,
|
|
nsIOService_GetProtocolFlags,
|
|
nsIOService_NewURI,
|
|
nsIOService_NewFileURI,
|
|
nsIOService_NewChannelFromURI,
|
|
nsIOService_NewChannel,
|
|
nsIOService_GetOffline,
|
|
nsIOService_SetOffline,
|
|
nsIOService_AllowPort,
|
|
nsIOService_ExtractScheme
|
|
};
|
|
|
|
static nsIIOService nsIOService = { &nsIOServiceVtbl };
|
|
|
|
static nsresult NSAPI nsIOServiceFactory_QueryInterface(nsIFactory *iface, nsIIDRef riid,
|
|
nsQIResult result)
|
|
{
|
|
*result = NULL;
|
|
|
|
if(IsEqualGUID(&IID_nsISupports, riid)) {
|
|
TRACE("(IID_nsISupoprts %p)\n", result);
|
|
*result = iface;
|
|
}else if(IsEqualGUID(&IID_nsIFactory, riid)) {
|
|
TRACE("(IID_nsIFactory %p)\n", result);
|
|
*result = iface;
|
|
}
|
|
|
|
if(*result) {
|
|
nsIFactory_AddRef(iface);
|
|
return NS_OK;
|
|
}
|
|
|
|
WARN("(%s %p)\n", debugstr_guid(riid), result);
|
|
return NS_NOINTERFACE;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsIOServiceFactory_AddRef(nsIFactory *iface)
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
static nsrefcnt NSAPI nsIOServiceFactory_Release(nsIFactory *iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static nsresult NSAPI nsIOServiceFactory_CreateInstance(nsIFactory *iface,
|
|
nsISupports *aOuter, const nsIID *iid, void **result)
|
|
{
|
|
return nsIIOService_QueryInterface(&nsIOService, iid, result);
|
|
}
|
|
|
|
static nsresult NSAPI nsIOServiceFactory_LockFactory(nsIFactory *iface, PRBool lock)
|
|
{
|
|
WARN("(%x)\n", lock);
|
|
return NS_OK;
|
|
}
|
|
|
|
static const nsIFactoryVtbl nsIOServiceFactoryVtbl = {
|
|
nsIOServiceFactory_QueryInterface,
|
|
nsIOServiceFactory_AddRef,
|
|
nsIOServiceFactory_Release,
|
|
nsIOServiceFactory_CreateInstance,
|
|
nsIOServiceFactory_LockFactory
|
|
};
|
|
|
|
static nsIFactory nsIOServiceFactory = { &nsIOServiceFactoryVtbl };
|
|
|
|
void init_nsio(nsIComponentManager *component_manager, nsIComponentRegistrar *registrar)
|
|
{
|
|
nsIFactory *old_factory = NULL;
|
|
nsresult nsres;
|
|
|
|
nsres = nsIComponentManager_GetClassObject(component_manager, &NS_IOSERVICE_CID,
|
|
&IID_nsIFactory, (void**)&old_factory);
|
|
if(NS_FAILED(nsres)) {
|
|
ERR("Could not get factory: %08x\n", nsres);
|
|
return;
|
|
}
|
|
|
|
nsres = nsIFactory_CreateInstance(old_factory, NULL, &IID_nsIIOService, (void**)&nsio);
|
|
if(NS_FAILED(nsres)) {
|
|
ERR("Couldn not create nsIOService instance %08x\n", nsres);
|
|
nsIFactory_Release(old_factory);
|
|
return;
|
|
}
|
|
|
|
nsres = nsIComponentRegistrar_UnregisterFactory(registrar, &NS_IOSERVICE_CID, old_factory);
|
|
nsIFactory_Release(old_factory);
|
|
if(NS_FAILED(nsres))
|
|
ERR("UnregisterFactory failed: %08x\n", nsres);
|
|
|
|
nsres = nsIComponentRegistrar_RegisterFactory(registrar, &NS_IOSERVICE_CID,
|
|
NS_IOSERVICE_CLASSNAME, NS_IOSERVICE_CONTRACTID, &nsIOServiceFactory);
|
|
if(NS_FAILED(nsres))
|
|
ERR("RegisterFactory failed: %08x\n", nsres);
|
|
}
|