2015-05-03 22:32:37 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* 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/. */
|
2004-02-11 09:09:51 +03:00
|
|
|
#include "nsJSUtils.h"
|
1998-07-17 08:29:16 +04:00
|
|
|
#include "nsString.h"
|
1999-08-20 00:36:31 +04:00
|
|
|
#include "nsIServiceManager.h"
|
1999-09-18 00:13:52 +04:00
|
|
|
#include "nsIScriptSecurityManager.h"
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
#include "nsIScriptContext.h"
|
2006-06-13 07:07:47 +04:00
|
|
|
#include "nsIScriptGlobalObject.h"
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
#include "nsIXPConnect.h"
|
2006-06-13 07:07:47 +04:00
|
|
|
#include "nsIMutableArray.h"
|
|
|
|
#include "nsVariant.h"
|
2009-06-24 12:42:00 +04:00
|
|
|
#include "nsIDOMBeforeUnloadEvent.h"
|
2009-10-24 19:39:28 +04:00
|
|
|
#include "nsGkAtoms.h"
|
2012-01-26 19:03:21 +04:00
|
|
|
#include "xpcpublic.h"
|
|
|
|
#include "nsJSEnvironment.h"
|
2007-08-05 17:23:03 +04:00
|
|
|
#include "nsDOMJSUtils.h"
|
2014-03-21 21:46:15 +04:00
|
|
|
#include "WorkerPrivate.h"
|
2013-09-25 15:21:20 +04:00
|
|
|
#include "mozilla/ContentEvents.h"
|
2016-09-14 16:47:32 +03:00
|
|
|
#include "mozilla/CycleCollectedJSContext.h"
|
2014-05-22 14:58:29 +04:00
|
|
|
#include "mozilla/HoldDropJSObjects.h"
|
2014-04-02 06:00:45 +04:00
|
|
|
#include "mozilla/JSEventHandler.h"
|
2012-10-26 17:32:10 +04:00
|
|
|
#include "mozilla/Likely.h"
|
2014-02-23 09:01:12 +04:00
|
|
|
#include "mozilla/dom/ErrorEvent.h"
|
2012-11-09 20:00:25 +04:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
namespace mozilla {
|
2006-06-13 07:07:47 +04:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
using namespace dom;
|
2006-06-13 07:07:47 +04:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
JSEventHandler::JSEventHandler(nsISupports* aTarget,
|
|
|
|
nsIAtom* aType,
|
|
|
|
const TypedEventHandler& aTypedHandler)
|
2014-04-02 06:00:45 +04:00
|
|
|
: mEventName(aType)
|
2014-04-02 06:00:45 +04:00
|
|
|
, mTypedHandler(aTypedHandler)
|
1998-07-17 08:29:16 +04:00
|
|
|
{
|
2014-04-02 06:00:45 +04:00
|
|
|
nsCOMPtr<nsISupports> base = do_QueryInterface(aTarget);
|
|
|
|
mTarget = base.get();
|
2014-05-22 14:58:29 +04:00
|
|
|
// Note, we call HoldJSObjects to get CanSkip called before CC.
|
|
|
|
HoldJSObjects(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
JSEventHandler::~JSEventHandler()
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!mTarget, "Should have called Disconnect()!");
|
|
|
|
DropJSObjects(this);
|
2012-11-09 20:00:25 +04:00
|
|
|
}
|
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(JSEventHandler)
|
2013-08-02 05:29:05 +04:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(JSEventHandler)
|
2014-04-02 06:00:45 +04:00
|
|
|
tmp->mTypedHandler.ForgetHandler();
|
2007-03-08 14:17:16 +03:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INTERNAL(JSEventHandler)
|
2012-10-26 17:32:10 +04:00
|
|
|
if (MOZ_UNLIKELY(cb.WantDebugInfo()) && tmp->mEventName) {
|
2012-09-24 15:05:30 +04:00
|
|
|
nsAutoCString name;
|
2014-04-02 06:00:46 +04:00
|
|
|
name.AppendLiteral("JSEventHandler handlerName=");
|
2012-09-24 15:05:30 +04:00
|
|
|
name.Append(
|
|
|
|
NS_ConvertUTF16toUTF8(nsDependentAtomString(tmp->mEventName)).get());
|
|
|
|
cb.DescribeRefCountedNode(tmp->mRefCnt.get(), name.get());
|
|
|
|
} else {
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_DESCRIBE(JSEventHandler, tmp->mRefCnt.get())
|
2012-09-24 15:05:30 +04:00
|
|
|
}
|
2014-04-02 06:00:45 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_RAWPTR(mTypedHandler.Ptr())
|
2007-10-29 16:45:07 +03:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS
|
2007-03-08 14:17:16 +03:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
2007-01-05 01:31:26 +03:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_BEGIN(JSEventHandler)
|
2012-09-25 00:25:35 +04:00
|
|
|
if (tmp->IsBlackForCC()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// If we have a target, it is the one which has tmp as onfoo handler.
|
|
|
|
if (tmp->mTarget) {
|
|
|
|
nsXPCOMCycleCollectionParticipant* cp = nullptr;
|
|
|
|
CallQueryInterface(tmp->mTarget, &cp);
|
|
|
|
nsISupports* canonical = nullptr;
|
|
|
|
tmp->mTarget->QueryInterface(NS_GET_IID(nsCycleCollectionISupports),
|
|
|
|
reinterpret_cast<void**>(&canonical));
|
|
|
|
// Usually CanSkip ends up unmarking the event listeners of mTarget,
|
|
|
|
// so tmp may become black.
|
|
|
|
if (cp && canonical && cp->CanSkip(canonical, true)) {
|
|
|
|
return tmp->IsBlackForCC();
|
|
|
|
}
|
|
|
|
}
|
2012-01-26 19:03:21 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_END
|
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_BEGIN(JSEventHandler)
|
2012-01-26 19:03:21 +04:00
|
|
|
return tmp->IsBlackForCC();
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_END
|
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_BEGIN(JSEventHandler)
|
2012-01-26 19:03:21 +04:00
|
|
|
return tmp->IsBlackForCC();
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_END
|
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(JSEventHandler)
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
NS_INTERFACE_MAP_ENTRY(nsIDOMEventListener)
|
2011-08-24 23:49:25 +04:00
|
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_INTERFACE_MAP_ENTRY(JSEventHandler)
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
NS_INTERFACE_MAP_END
|
1998-07-17 08:29:16 +04:00
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(JSEventHandler)
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(JSEventHandler)
|
1998-07-17 08:29:16 +04:00
|
|
|
|
2012-01-26 19:03:21 +04:00
|
|
|
bool
|
2014-04-02 06:00:46 +04:00
|
|
|
JSEventHandler::IsBlackForCC()
|
2012-01-26 19:03:21 +04:00
|
|
|
{
|
2012-11-09 20:00:25 +04:00
|
|
|
// We can claim to be black if all the things we reference are
|
|
|
|
// effectively black already.
|
2014-04-02 06:00:45 +04:00
|
|
|
return !mTypedHandler.HasEventHandler() ||
|
|
|
|
!mTypedHandler.Ptr()->HasGrayCallable();
|
2012-01-26 19:03:21 +04:00
|
|
|
}
|
|
|
|
|
2009-10-17 18:40:44 +04:00
|
|
|
nsresult
|
2014-04-02 06:00:46 +04:00
|
|
|
JSEventHandler::HandleEvent(nsIDOMEvent* aEvent)
|
2009-10-16 12:57:32 +04:00
|
|
|
{
|
2013-04-20 02:18:32 +04:00
|
|
|
nsCOMPtr<EventTarget> target = do_QueryInterface(mTarget);
|
2014-04-02 06:00:45 +04:00
|
|
|
if (!target || !mTypedHandler.HasEventHandler() ||
|
|
|
|
!GetTypedEventHandler().Ptr()->CallbackPreserveColor()) {
|
2011-08-24 23:49:25 +04:00
|
|
|
return NS_ERROR_FAILURE;
|
2014-03-21 21:46:15 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
Event* event = aEvent->InternalDOMEvent();
|
|
|
|
bool isMainThread = event->IsMainThreadEvent();
|
|
|
|
bool isChromeHandler =
|
|
|
|
isMainThread ?
|
2014-05-13 13:58:00 +04:00
|
|
|
nsContentUtils::ObjectPrincipal(
|
2014-04-02 06:00:45 +04:00
|
|
|
GetTypedEventHandler().Ptr()->CallbackPreserveColor()) ==
|
2014-03-21 21:46:15 +04:00
|
|
|
nsContentUtils::GetSystemPrincipal() :
|
|
|
|
mozilla::dom::workers::IsCurrentThreadRunningChromeWorker();
|
2009-10-16 12:57:32 +04:00
|
|
|
|
2014-04-02 06:00:45 +04:00
|
|
|
if (mTypedHandler.Type() == TypedEventHandler::eOnError) {
|
2013-11-05 18:16:26 +04:00
|
|
|
MOZ_ASSERT_IF(mEventName, mEventName == nsGkAtoms::onerror);
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
|
2012-11-09 20:00:25 +04:00
|
|
|
nsString errorMsg, file;
|
|
|
|
EventOrString msgOrEvent;
|
|
|
|
Optional<nsAString> fileName;
|
|
|
|
Optional<uint32_t> lineNumber;
|
|
|
|
Optional<uint32_t> columnNumber;
|
2014-01-03 05:04:15 +04:00
|
|
|
Optional<JS::Handle<JS::Value>> error;
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
|
2012-11-09 20:00:25 +04:00
|
|
|
NS_ENSURE_TRUE(aEvent, NS_ERROR_UNEXPECTED);
|
2014-02-23 09:01:12 +04:00
|
|
|
ErrorEvent* scriptEvent = aEvent->InternalDOMEvent()->AsErrorEvent();
|
|
|
|
if (scriptEvent) {
|
|
|
|
scriptEvent->GetMessage(errorMsg);
|
2016-06-01 23:26:59 +03:00
|
|
|
msgOrEvent.SetAsString().ShareOrDependUpon(errorMsg);
|
2012-11-09 20:00:25 +04:00
|
|
|
|
2014-02-23 09:01:12 +04:00
|
|
|
scriptEvent->GetFilename(file);
|
2012-11-09 20:00:25 +04:00
|
|
|
fileName = &file;
|
|
|
|
|
|
|
|
lineNumber.Construct();
|
2014-02-23 09:01:12 +04:00
|
|
|
lineNumber.Value() = scriptEvent->Lineno();
|
2014-01-03 05:04:15 +04:00
|
|
|
|
2014-03-20 19:46:28 +04:00
|
|
|
columnNumber.Construct();
|
2014-04-23 15:55:58 +04:00
|
|
|
columnNumber.Value() = scriptEvent->Colno();
|
2014-03-20 19:46:28 +04:00
|
|
|
|
2016-08-11 15:39:22 +03:00
|
|
|
error.Construct(RootingCx());
|
2016-03-23 18:02:57 +03:00
|
|
|
scriptEvent->GetError(&error.Value());
|
2012-11-09 20:00:25 +04:00
|
|
|
} else {
|
2013-03-09 15:34:29 +04:00
|
|
|
msgOrEvent.SetAsEvent() = aEvent->InternalDOMEvent();
|
2001-11-14 02:35:12 +03:00
|
|
|
}
|
2001-11-02 07:49:07 +03:00
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<OnErrorEventHandlerNonNull> handler =
|
2014-04-02 06:00:45 +04:00
|
|
|
mTypedHandler.OnErrorEventHandler();
|
2012-11-09 20:00:25 +04:00
|
|
|
ErrorResult rv;
|
|
|
|
bool handled = handler->Call(mTarget, msgOrEvent, fileName, lineNumber,
|
2014-01-03 05:04:15 +04:00
|
|
|
columnNumber, error, rv);
|
2012-11-09 20:00:25 +04:00
|
|
|
if (rv.Failed()) {
|
2015-04-27 16:18:51 +03:00
|
|
|
return rv.StealNSResult();
|
2012-11-09 20:00:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (handled) {
|
2014-03-21 21:46:15 +04:00
|
|
|
event->PreventDefaultInternal(isChromeHandler);
|
2012-11-09 20:00:25 +04:00
|
|
|
}
|
|
|
|
return NS_OK;
|
2001-11-02 07:49:07 +03:00
|
|
|
}
|
1998-07-17 08:29:16 +04:00
|
|
|
|
2014-04-02 06:00:45 +04:00
|
|
|
if (mTypedHandler.Type() == TypedEventHandler::eOnBeforeUnload) {
|
2012-11-09 20:00:25 +04:00
|
|
|
MOZ_ASSERT(mEventName == nsGkAtoms::onbeforeunload);
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<OnBeforeUnloadEventHandlerNonNull> handler =
|
2014-04-02 06:00:45 +04:00
|
|
|
mTypedHandler.OnBeforeUnloadEventHandler();
|
2012-11-09 20:00:25 +04:00
|
|
|
ErrorResult rv;
|
|
|
|
nsString retval;
|
2013-03-09 15:34:29 +04:00
|
|
|
handler->Call(mTarget, *(aEvent->InternalDOMEvent()), retval, rv);
|
2012-11-09 20:00:25 +04:00
|
|
|
if (rv.Failed()) {
|
2015-04-27 16:18:51 +03:00
|
|
|
return rv.StealNSResult();
|
2012-11-09 20:00:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMBeforeUnloadEvent> beforeUnload = do_QueryInterface(aEvent);
|
|
|
|
NS_ENSURE_STATE(beforeUnload);
|
|
|
|
|
|
|
|
if (!DOMStringIsNull(retval)) {
|
2014-03-21 21:46:15 +04:00
|
|
|
event->PreventDefaultInternal(isChromeHandler);
|
2012-11-09 20:00:25 +04:00
|
|
|
|
|
|
|
nsAutoString text;
|
|
|
|
beforeUnload->GetReturnValue(text);
|
|
|
|
|
|
|
|
// Set the text in the beforeUnload event as long as it wasn't
|
|
|
|
// already set (through event.returnValue, which takes
|
|
|
|
// precedence over a value returned from a JS function in IE)
|
|
|
|
if (text.IsEmpty()) {
|
|
|
|
beforeUnload->SetReturnValue(retval);
|
2006-06-13 07:07:47 +04:00
|
|
|
}
|
2004-02-11 09:09:51 +03:00
|
|
|
}
|
2012-11-09 20:00:25 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2014-04-02 06:00:45 +04:00
|
|
|
MOZ_ASSERT(mTypedHandler.Type() == TypedEventHandler::eNormal);
|
2012-11-09 20:00:25 +04:00
|
|
|
ErrorResult rv;
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<EventHandlerNonNull> handler = mTypedHandler.NormalEventHandler();
|
2016-08-11 15:39:22 +03:00
|
|
|
JS::Rooted<JS::Value> retval(RootingCx());
|
2014-06-12 00:26:52 +04:00
|
|
|
handler->Call(mTarget, *(aEvent->InternalDOMEvent()), &retval, rv);
|
2012-11-09 20:00:25 +04:00
|
|
|
if (rv.Failed()) {
|
2015-04-27 16:18:51 +03:00
|
|
|
return rv.StealNSResult();
|
2003-12-21 21:01:19 +03:00
|
|
|
}
|
1998-07-17 08:29:16 +04:00
|
|
|
|
2012-11-09 20:00:25 +04:00
|
|
|
// If the handler returned false and its sense is not reversed,
|
|
|
|
// or the handler returned true and its sense is reversed from
|
|
|
|
// the usual (false means cancel), then prevent default.
|
|
|
|
if (retval.isBoolean() &&
|
|
|
|
retval.toBoolean() == (mEventName == nsGkAtoms::onerror ||
|
|
|
|
mEventName == nsGkAtoms::onmouseover)) {
|
2014-03-21 21:46:15 +04:00
|
|
|
event->PreventDefaultInternal(isChromeHandler);
|
2012-11-09 20:00:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
1999-11-25 03:01:30 +03:00
|
|
|
}
|
|
|
|
|
2014-04-02 06:00:46 +04:00
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
using namespace mozilla;
|
|
|
|
|
1998-07-17 08:29:16 +04:00
|
|
|
/*
|
|
|
|
* Factory functions
|
|
|
|
*/
|
|
|
|
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
nsresult
|
2014-04-02 06:00:45 +04:00
|
|
|
NS_NewJSEventHandler(nsISupports* aTarget,
|
|
|
|
nsIAtom* aEventType,
|
2014-04-02 06:00:45 +04:00
|
|
|
const TypedEventHandler& aTypedHandler,
|
2014-04-02 06:00:46 +04:00
|
|
|
JSEventHandler** aReturn)
|
1998-07-17 08:29:16 +04:00
|
|
|
{
|
2013-09-14 04:22:39 +04:00
|
|
|
NS_ENSURE_ARG(aEventType || !NS_IsMainThread());
|
2014-04-02 06:00:46 +04:00
|
|
|
JSEventHandler* it =
|
|
|
|
new JSEventHandler(aTarget, aEventType, aTypedHandler);
|
2005-07-31 00:57:07 +04:00
|
|
|
NS_ADDREF(*aReturn = it);
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 21:42:36 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
1998-07-17 08:29:16 +04:00
|
|
|
}
|