2012-02-24 13:19:49 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#include "DOMRequest.h"
|
|
|
|
|
|
|
|
#include "DOMError.h"
|
2012-09-04 20:19:22 +00:00
|
|
|
#include "nsThreadUtils.h"
|
2013-02-08 20:25:37 +00:00
|
|
|
#include "DOMCursor.h"
|
2013-09-25 11:21:19 +00:00
|
|
|
#include "nsIDOMEvent.h"
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2014-05-19 20:37:58 +00:00
|
|
|
using mozilla::dom::DOMError;
|
2012-02-24 13:19:49 +00:00
|
|
|
using mozilla::dom::DOMRequest;
|
|
|
|
using mozilla::dom::DOMRequestService;
|
2013-02-08 20:25:37 +00:00
|
|
|
using mozilla::dom::DOMCursor;
|
2013-12-04 16:53:50 +00:00
|
|
|
using mozilla::AutoSafeJSContext;
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2014-01-07 02:53:23 +00:00
|
|
|
DOMRequest::DOMRequest(nsPIDOMWindow* aWindow)
|
2014-04-01 06:13:50 +00:00
|
|
|
: DOMEventTargetHelper(aWindow->IsInnerWindow() ?
|
|
|
|
aWindow : aWindow->GetCurrentInnerWindow())
|
2014-01-07 02:53:23 +00:00
|
|
|
, mResult(JSVAL_VOID)
|
2012-06-03 16:33:52 +00:00
|
|
|
, mDone(false)
|
2012-07-20 15:41:30 +00:00
|
|
|
{
|
2012-02-24 13:19:49 +00:00
|
|
|
}
|
|
|
|
|
2013-08-02 01:29:05 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(DOMRequest)
|
|
|
|
|
2012-02-24 13:19:49 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(DOMRequest,
|
2014-04-01 06:13:50 +00:00
|
|
|
DOMEventTargetHelper)
|
2012-11-15 07:32:40 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mError)
|
2012-02-24 13:19:49 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(DOMRequest,
|
2014-04-01 06:13:50 +00:00
|
|
|
DOMEventTargetHelper)
|
2012-11-15 07:32:40 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mError)
|
2013-08-14 18:29:20 +00:00
|
|
|
tmp->mResult = JSVAL_VOID;
|
2012-02-24 13:19:49 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(DOMRequest,
|
2014-04-01 06:13:50 +00:00
|
|
|
DOMEventTargetHelper)
|
2012-02-24 13:19:49 +00:00
|
|
|
// Don't need NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER because
|
2014-04-01 06:13:50 +00:00
|
|
|
// DOMEventTargetHelper does it for us.
|
2012-05-15 16:56:39 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_JSVAL_MEMBER_CALLBACK(mResult)
|
2012-02-24 13:19:49 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
|
|
|
|
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(DOMRequest)
|
|
|
|
NS_INTERFACE_MAP_ENTRY(nsIDOMDOMRequest)
|
2014-04-01 06:13:50 +00:00
|
|
|
NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper)
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2014-04-01 06:13:50 +00:00
|
|
|
NS_IMPL_ADDREF_INHERITED(DOMRequest, DOMEventTargetHelper)
|
|
|
|
NS_IMPL_RELEASE_INHERITED(DOMRequest, DOMEventTargetHelper)
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2013-03-17 08:51:36 +00:00
|
|
|
/* virtual */ JSObject*
|
2014-04-08 22:27:18 +00:00
|
|
|
DOMRequest::WrapObject(JSContext* aCx)
|
2013-03-17 08:51:36 +00:00
|
|
|
{
|
Bug 991742 part 6. Remove the "aScope" argument of binding Wrap() methods. r=bholley
This patch was mostly generated with this command:
find . -name "*.h" -o -name "*.cpp" | xargs sed -e 's/Binding::Wrap(aCx, aScope, this/Binding::Wrap(aCx, this/' -e 's/Binding_workers::Wrap(aCx, aScope, this/Binding_workers::Wrap(aCx, this/' -e 's/Binding::Wrap(cx, scope, this/Binding::Wrap(cx, this/' -i ""
plus a few manual fixes to dom/bindings/Codegen.py, js/xpconnect/src/event_impl_gen.py, and a few C++ files that were not caught in the search-and-replace above.
2014-04-08 22:27:17 +00:00
|
|
|
return DOMRequestBinding::Wrap(aCx, this);
|
2013-03-17 08:51:36 +00:00
|
|
|
}
|
|
|
|
|
2012-08-31 03:45:16 +00:00
|
|
|
NS_IMPL_EVENT_HANDLER(DOMRequest, success)
|
|
|
|
NS_IMPL_EVENT_HANDLER(DOMRequest, error)
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequest::GetReadyState(nsAString& aReadyState)
|
|
|
|
{
|
2013-03-17 08:51:36 +00:00
|
|
|
DOMRequestReadyState readyState = ReadyState();
|
|
|
|
switch (readyState) {
|
2013-05-06 19:28:13 +00:00
|
|
|
case DOMRequestReadyState::Pending:
|
2013-03-17 08:51:36 +00:00
|
|
|
aReadyState.AssignLiteral("pending");
|
|
|
|
break;
|
2013-05-06 19:28:13 +00:00
|
|
|
case DOMRequestReadyState::Done:
|
2013-03-17 08:51:36 +00:00
|
|
|
aReadyState.AssignLiteral("done");
|
|
|
|
break;
|
|
|
|
default:
|
2013-06-29 01:38:30 +00:00
|
|
|
MOZ_CRASH("Unrecognized readyState.");
|
2013-03-17 08:51:36 +00:00
|
|
|
}
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2014-01-09 17:39:36 +00:00
|
|
|
DOMRequest::GetResult(JS::MutableHandle<JS::Value> aResult)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2014-06-11 20:26:52 +00:00
|
|
|
GetResult(nullptr, aResult);
|
2012-02-24 13:19:49 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2013-05-18 17:52:06 +00:00
|
|
|
DOMRequest::GetError(nsISupports** aError)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2013-03-17 08:51:36 +00:00
|
|
|
NS_IF_ADDREF(*aError = GetError());
|
2012-02-24 13:19:49 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-05-18 01:48:25 +00:00
|
|
|
DOMRequest::FireSuccess(JS::Handle<JS::Value> aResult)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2012-06-03 16:33:52 +00:00
|
|
|
NS_ASSERTION(!mDone, "mDone shouldn't have been set to true already!");
|
|
|
|
NS_ASSERTION(!mError, "mError shouldn't have been set!");
|
|
|
|
NS_ASSERTION(mResult == JSVAL_VOID, "mResult shouldn't have been set!");
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
mDone = true;
|
2014-04-28 23:01:30 +00:00
|
|
|
if (aResult.isGCThing()) {
|
2012-06-03 16:33:52 +00:00
|
|
|
RootResultVal();
|
|
|
|
}
|
2012-02-24 13:19:49 +00:00
|
|
|
mResult = aResult;
|
|
|
|
|
2012-08-08 21:07:39 +00:00
|
|
|
FireEvent(NS_LITERAL_STRING("success"), false, false);
|
2012-02-24 13:19:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-08-08 21:07:39 +00:00
|
|
|
DOMRequest::FireError(const nsAString& aError)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2012-06-03 16:33:52 +00:00
|
|
|
NS_ASSERTION(!mDone, "mDone shouldn't have been set to true already!");
|
|
|
|
NS_ASSERTION(!mError, "mError shouldn't have been set!");
|
|
|
|
NS_ASSERTION(mResult == JSVAL_VOID, "mResult shouldn't have been set!");
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
mDone = true;
|
2013-05-18 17:52:06 +00:00
|
|
|
mError = new DOMError(GetOwner(), aError);
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2012-08-08 21:07:39 +00:00
|
|
|
FireEvent(NS_LITERAL_STRING("error"), true, true);
|
2012-06-03 16:33:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-08-08 21:07:39 +00:00
|
|
|
DOMRequest::FireError(nsresult aError)
|
2012-06-03 16:33:52 +00:00
|
|
|
{
|
|
|
|
NS_ASSERTION(!mDone, "mDone shouldn't have been set to true already!");
|
|
|
|
NS_ASSERTION(!mError, "mError shouldn't have been set!");
|
|
|
|
NS_ASSERTION(mResult == JSVAL_VOID, "mResult shouldn't have been set!");
|
|
|
|
|
|
|
|
mDone = true;
|
2013-05-18 17:52:06 +00:00
|
|
|
mError = new DOMError(GetOwner(), aError);
|
2012-06-03 16:33:52 +00:00
|
|
|
|
2012-08-08 21:07:39 +00:00
|
|
|
FireEvent(NS_LITERAL_STRING("error"), true, true);
|
2012-02-24 13:19:49 +00:00
|
|
|
}
|
|
|
|
|
2013-06-28 02:53:44 +00:00
|
|
|
void
|
2014-05-19 20:37:58 +00:00
|
|
|
DOMRequest::FireDetailedError(DOMError* aError)
|
2013-06-28 02:53:44 +00:00
|
|
|
{
|
|
|
|
NS_ASSERTION(!mDone, "mDone shouldn't have been set to true already!");
|
|
|
|
NS_ASSERTION(!mError, "mError shouldn't have been set!");
|
|
|
|
NS_ASSERTION(mResult == JSVAL_VOID, "mResult shouldn't have been set!");
|
|
|
|
NS_ASSERTION(aError, "No detailed error provided");
|
|
|
|
|
|
|
|
mDone = true;
|
|
|
|
mError = aError;
|
|
|
|
|
|
|
|
FireEvent(NS_LITERAL_STRING("error"), true, true);
|
|
|
|
}
|
|
|
|
|
2012-02-24 13:19:49 +00:00
|
|
|
void
|
2012-08-08 21:07:39 +00:00
|
|
|
DOMRequest::FireEvent(const nsAString& aType, bool aBubble, bool aCancelable)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
|
|
|
if (NS_FAILED(CheckInnerWindowCorrectness())) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-03-09 11:34:29 +00:00
|
|
|
nsCOMPtr<nsIDOMEvent> event;
|
|
|
|
NS_NewDOMEvent(getter_AddRefs(event), this, nullptr, nullptr);
|
2012-06-03 16:33:52 +00:00
|
|
|
nsresult rv = event->InitEvent(aType, aBubble, aCancelable);
|
2012-02-24 13:19:49 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-12-22 08:18:08 +00:00
|
|
|
event->SetTrusted(true);
|
2012-02-24 13:19:49 +00:00
|
|
|
|
2012-08-08 21:07:39 +00:00
|
|
|
bool dummy;
|
|
|
|
DispatchEvent(event, &dummy);
|
2012-02-24 13:19:49 +00:00
|
|
|
}
|
|
|
|
|
2012-06-03 16:33:52 +00:00
|
|
|
void
|
|
|
|
DOMRequest::RootResultVal()
|
|
|
|
{
|
2013-08-16 20:10:17 +00:00
|
|
|
mozilla::HoldJSObjects(this);
|
2012-06-03 16:33:52 +00:00
|
|
|
}
|
|
|
|
|
2014-04-27 07:06:00 +00:00
|
|
|
NS_IMPL_ISUPPORTS(DOMRequestService, nsIDOMRequestService)
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::CreateRequest(nsIDOMWindow* aWindow,
|
|
|
|
nsIDOMDOMRequest** aRequest)
|
|
|
|
{
|
2014-01-07 02:53:23 +00:00
|
|
|
nsCOMPtr<nsPIDOMWindow> win(do_QueryInterface(aWindow));
|
|
|
|
NS_ENSURE_STATE(win);
|
|
|
|
NS_ADDREF(*aRequest = new DOMRequest(win));
|
2013-02-08 20:25:37 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::CreateCursor(nsIDOMWindow* aWindow,
|
|
|
|
nsICursorContinueCallback* aCallback,
|
2014-01-07 02:53:23 +00:00
|
|
|
nsIDOMDOMCursor** aCursor)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsPIDOMWindow> win(do_QueryInterface(aWindow));
|
|
|
|
NS_ENSURE_STATE(win);
|
|
|
|
NS_ADDREF(*aCursor = new DOMCursor(win, aCallback));
|
2013-02-08 20:25:37 +00:00
|
|
|
|
2012-02-24 13:19:49 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireSuccess(nsIDOMDOMRequest* aRequest,
|
2014-01-09 17:39:36 +00:00
|
|
|
JS::Handle<JS::Value> aResult)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2012-06-13 21:26:10 +00:00
|
|
|
NS_ENSURE_STATE(aRequest);
|
2014-01-09 17:39:36 +00:00
|
|
|
static_cast<DOMRequest*>(aRequest)->FireSuccess(aResult);
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireError(nsIDOMDOMRequest* aRequest,
|
2012-08-08 21:07:39 +00:00
|
|
|
const nsAString& aError)
|
2012-02-24 13:19:49 +00:00
|
|
|
{
|
2012-06-13 21:26:10 +00:00
|
|
|
NS_ENSURE_STATE(aRequest);
|
2012-08-08 21:07:39 +00:00
|
|
|
static_cast<DOMRequest*>(aRequest)->FireError(aError);
|
2013-06-28 02:53:44 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireDetailedError(nsIDOMDOMRequest* aRequest,
|
|
|
|
nsISupports* aError)
|
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(aRequest);
|
2014-05-19 20:37:58 +00:00
|
|
|
nsCOMPtr<DOMError> err = do_QueryInterface(aError);
|
|
|
|
NS_ENSURE_STATE(err);
|
|
|
|
static_cast<DOMRequest*>(aRequest)->FireDetailedError(err);
|
2012-02-24 13:19:49 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2012-09-04 20:19:22 +00:00
|
|
|
|
|
|
|
class FireSuccessAsyncTask : public nsRunnable
|
|
|
|
{
|
2013-04-16 21:54:00 +00:00
|
|
|
|
2014-04-16 08:47:53 +00:00
|
|
|
FireSuccessAsyncTask(JSContext* aCx,
|
|
|
|
DOMRequest* aRequest,
|
2013-04-11 22:52:10 +00:00
|
|
|
const JS::Value& aResult) :
|
2012-09-04 20:19:22 +00:00
|
|
|
mReq(aRequest),
|
2014-04-16 08:47:53 +00:00
|
|
|
mResult(aCx, aResult)
|
2013-04-16 21:54:00 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
// Due to the fact that initialization can fail during shutdown (since we
|
|
|
|
// can't fetch a js context), set up an initiatization function to make sure
|
|
|
|
// we can return the failure appropriately
|
|
|
|
static nsresult
|
|
|
|
Dispatch(DOMRequest* aRequest,
|
|
|
|
const JS::Value& aResult)
|
|
|
|
{
|
|
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
2014-04-16 08:47:53 +00:00
|
|
|
AutoSafeJSContext cx;
|
|
|
|
nsRefPtr<FireSuccessAsyncTask> asyncTask = new FireSuccessAsyncTask(cx, aRequest, aResult);
|
2013-04-16 21:54:00 +00:00
|
|
|
if (NS_FAILED(NS_DispatchToMainThread(asyncTask))) {
|
|
|
|
NS_WARNING("Failed to dispatch to main thread!");
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
return NS_OK;
|
2012-09-04 20:19:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
Run()
|
|
|
|
{
|
2014-04-16 08:47:53 +00:00
|
|
|
mReq->FireSuccess(JS::Handle<JS::Value>::fromMarkedLocation(mResult.address()));
|
2012-09-04 20:19:22 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
~FireSuccessAsyncTask()
|
|
|
|
{
|
|
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
}
|
2014-04-16 08:47:53 +00:00
|
|
|
|
2012-09-04 20:19:22 +00:00
|
|
|
private:
|
|
|
|
nsRefPtr<DOMRequest> mReq;
|
2014-04-16 08:47:53 +00:00
|
|
|
JS::PersistentRooted<JS::Value> mResult;
|
2012-09-04 20:19:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class FireErrorAsyncTask : public nsRunnable
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FireErrorAsyncTask(DOMRequest* aRequest,
|
|
|
|
const nsAString& aError) :
|
|
|
|
mReq(aRequest),
|
|
|
|
mError(aError)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
Run()
|
|
|
|
{
|
|
|
|
mReq->FireError(mError);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
nsRefPtr<DOMRequest> mReq;
|
|
|
|
nsString mError;
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireSuccessAsync(nsIDOMDOMRequest* aRequest,
|
2014-01-09 17:39:36 +00:00
|
|
|
JS::Handle<JS::Value> aResult)
|
2012-09-04 20:19:22 +00:00
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(aRequest);
|
2013-04-16 21:54:00 +00:00
|
|
|
return FireSuccessAsyncTask::Dispatch(static_cast<DOMRequest*>(aRequest), aResult);
|
2012-09-04 20:19:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireErrorAsync(nsIDOMDOMRequest* aRequest,
|
|
|
|
const nsAString& aError)
|
|
|
|
{
|
|
|
|
NS_ENSURE_STATE(aRequest);
|
|
|
|
nsCOMPtr<nsIRunnable> asyncTask =
|
|
|
|
new FireErrorAsyncTask(static_cast<DOMRequest*>(aRequest), aError);
|
|
|
|
if (NS_FAILED(NS_DispatchToMainThread(asyncTask))) {
|
|
|
|
NS_WARNING("Failed to dispatch to main thread!");
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2013-02-08 20:25:37 +00:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
DOMRequestService::FireDone(nsIDOMDOMCursor* aCursor) {
|
|
|
|
NS_ENSURE_STATE(aCursor);
|
|
|
|
static_cast<DOMCursor*>(aCursor)->FireDone();
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|