mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-01 22:55:23 +00:00
236 lines
6.3 KiB
C++
236 lines
6.3 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/*
|
|
* The contents of this file are subject to the Netscape Public
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is Mozilla Communicator client code,
|
|
* released March 31, 1998.
|
|
*
|
|
* The Initial Developer of the Original Code is Netscape Communications
|
|
* Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998-1999 Netscape Communications Corporation. All
|
|
* Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Daniel Veditz <dveditz@netscape.com>
|
|
*/
|
|
|
|
#include "nscore.h"
|
|
#include "nsXPITriggerInfo.h"
|
|
#include "nsDebug.h"
|
|
#include "nsIServiceManager.h"
|
|
#include "nsIEventQueueService.h"
|
|
|
|
static NS_DEFINE_IID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID);
|
|
|
|
|
|
//
|
|
// nsXPITriggerItem
|
|
//
|
|
MOZ_DECL_CTOR_COUNTER(nsXPITriggerItem)
|
|
|
|
nsXPITriggerItem::nsXPITriggerItem( const PRUnichar* aName,
|
|
const PRUnichar* aURL,
|
|
PRInt32 aFlags )
|
|
: mName(aName), mURL(aURL), mFlags(aFlags)
|
|
{
|
|
MOZ_COUNT_CTOR(nsXPITriggerItem);
|
|
|
|
// check for arguments
|
|
PRInt32 qmark = mURL.FindChar('?');
|
|
if ( qmark != kNotFound )
|
|
{
|
|
mArguments = Substring( mURL, qmark+1, mURL.Length() );
|
|
}
|
|
|
|
// construct name if not passed in
|
|
if ( mName.IsEmpty() )
|
|
{
|
|
// Use the filename as the display name by starting after the last
|
|
// slash in the URL, looking backwards from the arguments delimiter if
|
|
// we found one. By good fortune using kNotFound as the offset for
|
|
// RFindChar() starts at the end, so we can use qmark in all cases.
|
|
|
|
PRInt32 namestart = mURL.RFindChar( '/', qmark );
|
|
|
|
// the real start is after the slash (or 0 if not found)
|
|
namestart = ( namestart==kNotFound ) ? 0 : namestart + 1;
|
|
|
|
PRInt32 length;
|
|
if (qmark == kNotFound)
|
|
length = mURL.Length(); // no '?', slurp up rest of URL
|
|
else
|
|
length = (qmark - namestart); // filename stops at the '?'
|
|
|
|
mName = Substring( mURL, namestart, length );
|
|
}
|
|
}
|
|
|
|
nsXPITriggerItem::~nsXPITriggerItem()
|
|
{
|
|
MOZ_COUNT_DTOR(nsXPITriggerItem);
|
|
}
|
|
|
|
PRBool nsXPITriggerItem::IsRelativeURL()
|
|
{
|
|
PRInt32 cpos = mURL.FindChar(':');
|
|
if (cpos == kNotFound)
|
|
return PR_TRUE;
|
|
|
|
PRInt32 spos = mURL.FindChar('/');
|
|
return (cpos > spos);
|
|
}
|
|
|
|
|
|
//
|
|
// nsXPITriggerInfo
|
|
//
|
|
|
|
MOZ_DECL_CTOR_COUNTER(nsXPITriggerInfo)
|
|
|
|
nsXPITriggerInfo::nsXPITriggerInfo()
|
|
: mCx(0), mCbval(JSVAL_NULL)
|
|
{
|
|
MOZ_COUNT_CTOR(nsXPITriggerInfo);
|
|
}
|
|
|
|
nsXPITriggerInfo::~nsXPITriggerInfo()
|
|
{
|
|
nsXPITriggerItem* item;
|
|
|
|
for(PRUint32 i=0; i < Size(); i++)
|
|
{
|
|
item = Get(i);
|
|
if (item)
|
|
delete item;
|
|
}
|
|
mItems.Clear();
|
|
|
|
if ( mCx && !JSVAL_IS_NULL(mCbval) )
|
|
JS_RemoveRoot( mCx, &mCbval );
|
|
|
|
MOZ_COUNT_DTOR(nsXPITriggerInfo);
|
|
}
|
|
|
|
void nsXPITriggerInfo::SaveCallback( JSContext *aCx, jsval aVal )
|
|
{
|
|
NS_ASSERTION( mCx == 0, "callback set twice, memory leak" );
|
|
mCx = aCx;
|
|
JSObject *obj = JS_GetGlobalObject( mCx );
|
|
|
|
JSClass* clazz;
|
|
|
|
#ifdef JS_THREADSAFE
|
|
clazz = ::JS_GetClass(aCx, obj);
|
|
#else
|
|
clazz = ::JS_GetClass(obj);
|
|
#endif
|
|
|
|
if (clazz &&
|
|
(clazz->flags & JSCLASS_HAS_PRIVATE) &&
|
|
(clazz->flags & JSCLASS_PRIVATE_IS_NSISUPPORTS)) {
|
|
mGlobalWrapper =
|
|
do_QueryInterface((nsISupports*)::JS_GetPrivate(aCx, obj));
|
|
}
|
|
|
|
mCbval = aVal;
|
|
mThread = PR_GetCurrentThread();
|
|
|
|
if ( !JSVAL_IS_NULL(mCbval) )
|
|
JS_AddRoot( mCx, &mCbval );
|
|
}
|
|
|
|
static void destroyTriggerEvent(XPITriggerEvent* event)
|
|
{
|
|
JS_RemoveRoot( event->cx, &event->cbval );
|
|
delete event;
|
|
}
|
|
|
|
static void* handleTriggerEvent(XPITriggerEvent* event)
|
|
{
|
|
jsval ret;
|
|
void* mark;
|
|
jsval* args;
|
|
|
|
args = JS_PushArguments( event->cx, &mark, "Wi",
|
|
event->URL.get(),
|
|
event->status );
|
|
if ( args )
|
|
{
|
|
JS_CallFunctionValue( event->cx,
|
|
JSVAL_TO_OBJECT(event->global),
|
|
event->cbval,
|
|
2,
|
|
args,
|
|
&ret );
|
|
|
|
JS_PopArguments( event->cx, mark );
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
void nsXPITriggerInfo::SendStatus(const PRUnichar* URL, PRInt32 status)
|
|
{
|
|
nsCOMPtr<nsIEventQueue> eq;
|
|
nsresult rv;
|
|
|
|
if ( mCx && mGlobalWrapper && mCbval )
|
|
{
|
|
nsCOMPtr<nsIEventQueueService> EQService =
|
|
do_GetService(kEventQueueServiceCID, &rv);
|
|
if ( NS_SUCCEEDED( rv ) )
|
|
{
|
|
rv = EQService->GetThreadEventQueue(mThread, getter_AddRefs(eq));
|
|
if ( NS_SUCCEEDED(rv) )
|
|
{
|
|
// create event and post it
|
|
XPITriggerEvent* event = new XPITriggerEvent();
|
|
if (event)
|
|
{
|
|
PL_InitEvent(&event->e, 0,
|
|
(PLHandleEventProc)handleTriggerEvent,
|
|
(PLDestroyEventProc)destroyTriggerEvent);
|
|
|
|
event->URL = URL;
|
|
event->status = status;
|
|
event->cx = mCx;
|
|
|
|
JSObject *obj = nsnull;
|
|
|
|
mGlobalWrapper->GetJSObject(&obj);
|
|
|
|
event->global = OBJECT_TO_JSVAL(obj);
|
|
|
|
event->cbval = mCbval;
|
|
JS_AddNamedRoot( event->cx, &event->cbval,
|
|
"XPITriggerEvent::cbval" );
|
|
|
|
// Hold a strong reference to keep the underlying
|
|
// JSContext from dying before we handle this event.
|
|
event->ref = mGlobalWrapper;
|
|
|
|
eq->PostEvent(&event->e);
|
|
}
|
|
else
|
|
rv = NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
|
|
if ( NS_FAILED( rv ) )
|
|
{
|
|
// couldn't get event queue -- maybe window is gone or
|
|
// some similarly catastrophic occurrance
|
|
}
|
|
}
|
|
}
|
|
|