| /* |
| * 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(!doc->client) |
| return TRUE; |
| |
| 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(container->doc && container->doc->mime) { |
| DWORD len; |
| |
| heap_free(This->content); |
| |
| len = WideCharToMultiByte(CP_ACP, 0, container->doc->mime, -1, NULL, 0, NULL, NULL); |
| This->content = heap_alloc(len); |
| WideCharToMultiByte(CP_ACP, 0, container->doc->mime, -1, This->content, -1, NULL, NULL); |
| } |
| |
| 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_nsISupports %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); |
| } |