2010-08-20 23:24:40 +00:00
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* vim : sw = 2 ts = 8 et :
*/
2012-05-21 11:12:37 +00: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/. */
2010-08-20 23:24:40 +00:00
2012-07-17 23:59:45 +00:00
# include "base/basictypes.h"
2013-05-01 05:03:25 +00:00
# include "ClientLayerManager.h"
2012-07-17 23:59:45 +00:00
# include "gfxPlatform.h"
2011-08-09 19:38:27 +00:00
# if defined(MOZ_ENABLE_D3D10_LAYER)
# include "LayerManagerD3D10.h"
# endif
2012-07-17 23:59:45 +00:00
# include "mozilla/dom/TabChild.h"
2012-05-08 21:36:07 +00:00
# include "mozilla/Hal.h"
2014-03-08 01:20:07 +00:00
# include "mozilla/IMEStateManager.h"
2012-07-17 23:59:45 +00:00
# include "mozilla/layers/CompositorChild.h"
2013-04-24 18:42:40 +00:00
# include "mozilla/layers/PLayerTransactionChild.h"
2014-04-03 04:18:37 +00:00
# include "mozilla/TextComposition.h"
2013-09-25 11:21:19 +00:00
# include "mozilla/TextEvents.h"
2010-08-20 23:24:40 +00:00
# include "PuppetWidget.h"
2012-08-15 18:52:42 +00:00
# include "nsIWidgetListener.h"
2010-08-20 23:24:40 +00:00
2014-06-10 06:02:21 +00:00
using namespace mozilla ;
2012-05-08 21:36:07 +00:00
using namespace mozilla : : dom ;
using namespace mozilla : : hal ;
2014-02-09 08:04:38 +00:00
using namespace mozilla : : gfx ;
2010-08-20 23:24:40 +00:00
using namespace mozilla : : layers ;
using namespace mozilla : : widget ;
static void
InvalidateRegion ( nsIWidget * aWidget , const nsIntRegion & aRegion )
{
nsIntRegionRectIterator it ( aRegion ) ;
while ( const nsIntRect * r = it . Next ( ) ) {
2011-12-24 03:52:21 +00:00
aWidget - > Invalidate ( * r ) ;
2010-08-20 23:24:40 +00:00
}
}
/*static*/ already_AddRefed < nsIWidget >
2012-07-17 23:59:45 +00:00
nsIWidget : : CreatePuppetWidget ( TabChild * aTabChild )
2010-08-20 23:24:40 +00:00
{
2013-04-04 13:24:32 +00:00
NS_ABORT_IF_FALSE ( ! aTabChild | | nsIWidget : : UsePuppetWidgets ( ) ,
2010-08-20 23:24:40 +00:00
" PuppetWidgets not allowed in this configuration " ) ;
2010-09-24 03:28:15 +00:00
nsCOMPtr < nsIWidget > widget = new PuppetWidget ( aTabChild ) ;
2010-08-20 23:24:40 +00:00
return widget . forget ( ) ;
}
namespace mozilla {
namespace widget {
2011-02-23 17:45:09 +00:00
static bool
IsPopup ( const nsWidgetInitData * aInitData )
{
return aInitData & & aInitData - > mWindowType = = eWindowType_popup ;
}
static bool
MightNeedIMEFocus ( const nsWidgetInitData * aInitData )
{
// In the puppet-widget world, popup widgets are just dummies and
// shouldn't try to mess with IME state.
2013-01-28 23:56:28 +00:00
# ifdef MOZ_CROSS_PROCESS_IME
2011-02-23 17:45:09 +00:00
return ! IsPopup ( aInitData ) ;
2013-01-28 23:56:28 +00:00
# else
return false ;
# endif
2011-02-23 17:45:09 +00:00
}
2010-08-20 23:24:40 +00:00
// Arbitrary, fungible.
const size_t PuppetWidget : : kMaxDimension = 4000 ;
2014-04-27 07:06:00 +00:00
NS_IMPL_ISUPPORTS_INHERITED ( PuppetWidget , nsBaseWidget ,
nsISupportsWeakReference )
2010-08-20 23:24:40 +00:00
2012-07-17 23:59:45 +00:00
PuppetWidget : : PuppetWidget ( TabChild * aTabChild )
2010-09-24 03:28:15 +00:00
: mTabChild ( aTabChild )
2010-12-03 01:24:04 +00:00
, mDPI ( - 1 )
2013-05-01 23:06:19 +00:00
, mDefaultScale ( - 1 )
2014-04-21 20:40:09 +00:00
, mNativeKeyCommandsValid ( false )
2010-08-20 23:24:40 +00:00
{
MOZ_COUNT_CTOR ( PuppetWidget ) ;
2014-03-20 15:46:29 +00:00
mSingleLineCommands . SetCapacity ( 4 ) ;
mMultiLineCommands . SetCapacity ( 4 ) ;
mRichTextCommands . SetCapacity ( 4 ) ;
2010-08-20 23:24:40 +00:00
}
PuppetWidget : : ~ PuppetWidget ( )
{
MOZ_COUNT_DTOR ( PuppetWidget ) ;
}
NS_IMETHODIMP
PuppetWidget : : Create ( nsIWidget * aParent ,
nsNativeWidget aNativeParent ,
const nsIntRect & aRect ,
2011-04-17 01:22:44 +00:00
nsDeviceContext * aContext ,
2010-08-20 23:24:40 +00:00
nsWidgetInitData * aInitData )
{
NS_ABORT_IF_FALSE ( ! aNativeParent , " got a non-Puppet native parent " ) ;
2012-08-15 18:53:09 +00:00
BaseCreate ( nullptr , aRect , aContext , aInitData ) ;
2010-08-20 23:24:40 +00:00
mBounds = aRect ;
2011-10-17 14:59:28 +00:00
mEnabled = true ;
mVisible = true ;
2010-08-20 23:24:40 +00:00
2014-06-10 06:02:21 +00:00
mDrawTarget = gfxPlatform : : GetPlatform ( ) - >
CreateOffscreenContentDrawTarget ( IntSize ( 1 , 1 ) , SurfaceFormat : : B8G8R8A8 ) ;
2010-08-20 23:24:40 +00:00
2011-10-17 14:59:28 +00:00
mIMEComposing = false ;
2012-08-29 15:26:18 +00:00
mNeedIMEStateInit = MightNeedIMEFocus ( aInitData ) ;
2010-09-24 03:28:15 +00:00
2010-08-20 23:24:40 +00:00
PuppetWidget * parent = static_cast < PuppetWidget * > ( aParent ) ;
if ( parent ) {
parent - > SetChild ( this ) ;
2010-08-20 23:24:41 +00:00
mLayerManager = parent - > GetLayerManager ( ) ;
2010-08-20 23:24:40 +00:00
}
else {
2011-10-17 14:59:28 +00:00
Resize ( mBounds . x , mBounds . y , mBounds . width , mBounds . height , false ) ;
2010-08-20 23:24:40 +00:00
}
return NS_OK ;
}
2012-08-29 15:26:18 +00:00
void
PuppetWidget : : InitIMEState ( )
{
2013-06-17 05:42:20 +00:00
MOZ_ASSERT ( mTabChild ) ;
2012-08-29 15:26:18 +00:00
if ( mNeedIMEStateInit ) {
uint32_t chromeSeqno ;
2014-01-29 09:32:39 +00:00
mTabChild - > SendNotifyIMEFocus ( false , & mIMEPreferenceOfParent , & chromeSeqno ) ;
2012-08-29 15:26:18 +00:00
mIMELastBlurSeqno = mIMELastReceivedSeqno = chromeSeqno ;
mNeedIMEStateInit = false ;
}
}
2010-08-20 23:24:40 +00:00
already_AddRefed < nsIWidget >
PuppetWidget : : CreateChild ( const nsIntRect & aRect ,
2011-04-17 01:22:44 +00:00
nsDeviceContext * aContext ,
2010-08-20 23:24:40 +00:00
nsWidgetInitData * aInitData ,
2011-09-29 06:19:26 +00:00
bool aForceUseIWidgetParent )
2010-08-20 23:24:40 +00:00
{
2011-02-23 17:45:09 +00:00
bool isPopup = IsPopup ( aInitData ) ;
2010-09-24 03:28:15 +00:00
nsCOMPtr < nsIWidget > widget = nsIWidget : : CreatePuppetWidget ( mTabChild ) ;
2010-08-20 23:24:40 +00:00
return ( ( widget & &
2012-07-30 14:20:58 +00:00
NS_SUCCEEDED ( widget - > Create ( isPopup ? nullptr : this , nullptr , aRect ,
2011-10-25 15:05:32 +00:00
aContext , aInitData ) ) ) ?
2012-07-30 14:20:58 +00:00
widget . forget ( ) : nullptr ) ;
2010-08-20 23:24:40 +00:00
}
2010-08-20 23:24:40 +00:00
NS_IMETHODIMP
PuppetWidget : : Destroy ( )
{
2011-03-29 20:14:44 +00:00
Base : : OnDestroy ( ) ;
2010-08-20 23:24:40 +00:00
Base : : Destroy ( ) ;
mPaintTask . Revoke ( ) ;
2012-07-30 14:20:58 +00:00
mChild = nullptr ;
2010-12-07 02:05:25 +00:00
if ( mLayerManager ) {
mLayerManager - > Destroy ( ) ;
}
2012-07-30 14:20:58 +00:00
mLayerManager = nullptr ;
mTabChild = nullptr ;
2010-08-20 23:24:40 +00:00
return NS_OK ;
}
2010-08-20 23:24:40 +00:00
NS_IMETHODIMP
2011-09-29 06:19:26 +00:00
PuppetWidget : : Show ( bool aState )
2010-08-20 23:24:40 +00:00
{
NS_ASSERTION ( mEnabled ,
" does it make sense to Show()/Hide() a disabled widget? " ) ;
2011-09-29 06:19:26 +00:00
bool wasVisible = mVisible ;
2010-08-20 23:24:40 +00:00
mVisible = aState ;
2012-11-08 03:51:55 +00:00
if ( mChild ) {
mChild - > mVisible = aState ;
}
if ( ! mVisible & & mLayerManager ) {
mLayerManager - > ClearCachedResources ( ) ;
}
2010-08-20 23:24:40 +00:00
if ( ! wasVisible & & mVisible ) {
2011-10-17 14:59:28 +00:00
Resize ( mBounds . width , mBounds . height , false ) ;
2012-11-08 03:51:55 +00:00
Invalidate ( mBounds ) ;
2010-08-20 23:24:40 +00:00
}
return NS_OK ;
}
NS_IMETHODIMP
2012-12-12 09:57:38 +00:00
PuppetWidget : : Resize ( double aWidth ,
double aHeight ,
bool aRepaint )
2010-08-20 23:24:40 +00:00
{
nsIntRect oldBounds = mBounds ;
2012-12-12 09:57:38 +00:00
mBounds . SizeTo ( nsIntSize ( NSToIntRound ( aWidth ) , NSToIntRound ( aHeight ) ) ) ;
2010-08-20 23:24:40 +00:00
if ( mChild ) {
return mChild - > Resize ( aWidth , aHeight , aRepaint ) ;
}
// XXX: roc says that |aRepaint| dictates whether or not to
// invalidate the expanded area
if ( oldBounds . Size ( ) < mBounds . Size ( ) & & aRepaint ) {
nsIntRegion dirty ( mBounds ) ;
dirty . Sub ( dirty , oldBounds ) ;
InvalidateRegion ( this , dirty ) ;
}
2012-08-15 18:53:14 +00:00
if ( ! oldBounds . IsEqualEdges ( mBounds ) & & mAttachedWidgetListener ) {
mAttachedWidgetListener - > WindowResized ( this , mBounds . width , mBounds . height ) ;
2010-08-20 23:24:40 +00:00
}
return NS_OK ;
}
NS_IMETHODIMP
2011-09-29 06:19:26 +00:00
PuppetWidget : : SetFocus ( bool aRaise )
2010-08-20 23:24:40 +00:00
{
// XXX/cjones: someone who knows about event handling needs to
// decide how this should work.
return NS_OK ;
}
NS_IMETHODIMP
2011-12-24 03:52:21 +00:00
PuppetWidget : : Invalidate ( const nsIntRect & aRect )
2010-08-20 23:24:40 +00:00
{
# ifdef DEBUG
2011-12-24 03:52:21 +00:00
debug_DumpInvalidate ( stderr , this , & aRect ,
2012-09-02 02:35:17 +00:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 23:24:40 +00:00
# endif
if ( mChild ) {
2011-12-24 03:52:21 +00:00
return mChild - > Invalidate ( aRect ) ;
2010-08-20 23:24:40 +00:00
}
mDirtyRegion . Or ( mDirtyRegion , aRect ) ;
2011-12-24 03:52:21 +00:00
if ( ! mDirtyRegion . IsEmpty ( ) & & ! mPaintTask . IsPending ( ) ) {
2010-08-20 23:24:40 +00:00
mPaintTask = new PaintTask ( this ) ;
return NS_DispatchToCurrentThread ( mPaintTask . get ( ) ) ;
}
return NS_OK ;
}
2010-09-24 03:28:15 +00:00
void
2013-10-02 03:46:03 +00:00
PuppetWidget : : InitEvent ( WidgetGUIEvent & event , nsIntPoint * aPoint )
2010-09-24 03:28:15 +00:00
{
2012-07-30 14:20:58 +00:00
if ( nullptr = = aPoint ) {
2010-09-24 03:28:15 +00:00
event . refPoint . x = 0 ;
event . refPoint . y = 0 ;
}
else {
// use the point override if provided
event . refPoint . x = aPoint - > x ;
event . refPoint . y = aPoint - > y ;
}
event . time = PR_Now ( ) / 1000 ;
}
2010-08-20 23:24:40 +00:00
NS_IMETHODIMP
2013-10-02 03:46:03 +00:00
PuppetWidget : : DispatchEvent ( WidgetGUIEvent * event , nsEventStatus & aStatus )
2010-08-20 23:24:40 +00:00
{
# ifdef DEBUG
debug_DumpEvent ( stdout , event - > widget , event ,
2012-09-02 02:35:17 +00:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 23:24:40 +00:00
# endif
2011-02-09 20:13:18 +00:00
NS_ABORT_IF_FALSE ( ! mChild | | mChild - > mWindowType = = eWindowType_popup ,
" Unexpected event dispatch! " ) ;
2014-04-21 20:40:09 +00:00
AutoCacheNativeKeyCommands autoCache ( this ) ;
if ( event - > mFlags . mIsSynthesizedForTests & & ! mNativeKeyCommandsValid ) {
WidgetKeyboardEvent * keyEvent = event - > AsKeyboardEvent ( ) ;
if ( keyEvent ) {
mTabChild - > RequestNativeKeyBindings ( & autoCache , keyEvent ) ;
}
}
2010-08-20 23:24:40 +00:00
aStatus = nsEventStatus_eIgnore ;
2011-02-09 20:13:18 +00:00
if ( event - > message = = NS_COMPOSITION_START ) {
2011-10-17 14:59:28 +00:00
mIMEComposing = true ;
2011-02-09 20:13:18 +00:00
}
2014-01-15 14:41:39 +00:00
uint32_t seqno = kLatestSeqno ;
2011-02-09 20:13:18 +00:00
switch ( event - > eventStructType ) {
case NS_COMPOSITION_EVENT :
2014-01-15 14:41:39 +00:00
seqno = event - > AsCompositionEvent ( ) - > mSeqno ;
2011-02-09 20:13:18 +00:00
break ;
case NS_TEXT_EVENT :
2014-01-15 14:41:39 +00:00
seqno = event - > AsTextEvent ( ) - > mSeqno ;
2011-02-09 20:13:18 +00:00
break ;
case NS_SELECTION_EVENT :
2014-01-15 14:41:39 +00:00
seqno = event - > AsSelectionEvent ( ) - > mSeqno ;
2011-02-09 20:13:18 +00:00
break ;
2012-11-20 06:05:56 +00:00
default :
break ;
2011-02-09 20:13:18 +00:00
}
2014-01-15 14:41:39 +00:00
if ( seqno ! = kLatestSeqno ) {
mIMELastReceivedSeqno = seqno ;
if ( mIMELastReceivedSeqno < mIMELastBlurSeqno ) {
return NS_OK ;
}
}
2012-08-15 18:53:09 +00:00
2013-01-25 19:51:16 +00:00
if ( mAttachedWidgetListener ) {
aStatus = mAttachedWidgetListener - > HandleEvent ( event , mUseAttachedEvents ) ;
}
2011-02-09 20:13:18 +00:00
if ( event - > message = = NS_COMPOSITION_END ) {
2011-10-17 14:59:28 +00:00
mIMEComposing = false ;
2010-08-20 23:24:40 +00:00
}
return NS_OK ;
}
2014-03-20 15:46:29 +00:00
NS_IMETHODIMP_ ( bool )
PuppetWidget : : ExecuteNativeKeyBinding ( NativeKeyBindingsType aType ,
const mozilla : : WidgetKeyboardEvent & aEvent ,
DoCommandCallback aCallback ,
void * aCallbackData )
{
2014-04-21 20:40:09 +00:00
// B2G doesn't have native key bindings.
# ifdef MOZ_B2G
return false ;
# else // #ifdef MOZ_B2G
MOZ_ASSERT ( mNativeKeyCommandsValid ) ;
2014-03-20 15:46:29 +00:00
nsTArray < mozilla : : CommandInt > & commands = mSingleLineCommands ;
switch ( aType ) {
case nsIWidget : : NativeKeyBindingsForSingleLineEditor :
commands = mSingleLineCommands ;
break ;
case nsIWidget : : NativeKeyBindingsForMultiLineEditor :
commands = mMultiLineCommands ;
break ;
case nsIWidget : : NativeKeyBindingsForRichTextEditor :
commands = mRichTextCommands ;
break ;
}
if ( commands . IsEmpty ( ) ) {
return false ;
}
for ( uint32_t i = 0 ; i < commands . Length ( ) ; i + + ) {
aCallback ( static_cast < mozilla : : Command > ( commands [ i ] ) , aCallbackData ) ;
}
return true ;
2014-04-21 20:40:09 +00:00
# endif
2014-03-20 15:46:29 +00:00
}
2010-08-20 23:24:40 +00:00
LayerManager *
2013-04-24 18:42:40 +00:00
PuppetWidget : : GetLayerManager ( PLayerTransactionChild * aShadowManager ,
2011-08-09 19:38:26 +00:00
LayersBackend aBackendHint ,
LayerManagerPersistence aPersistence ,
bool * aAllowRetaining )
2010-08-20 23:24:40 +00:00
{
if ( ! mLayerManager ) {
2011-08-09 19:38:27 +00:00
// The backend hint is a temporary placeholder until Azure, when
// all content-process layer managers will be BasicLayerManagers.
# if defined(MOZ_ENABLE_D3D10_LAYER)
2014-01-23 18:26:41 +00:00
if ( mozilla : : layers : : LayersBackend : : LAYERS_D3D10 = = aBackendHint ) {
2011-08-09 19:38:27 +00:00
nsRefPtr < LayerManagerD3D10 > m = new LayerManagerD3D10 ( this ) ;
m - > AsShadowForwarder ( ) - > SetShadowManager ( aShadowManager ) ;
if ( m - > Initialize ( ) ) {
mLayerManager = m ;
}
}
# endif
if ( ! mLayerManager ) {
2013-05-01 05:03:25 +00:00
mLayerManager = new ClientLayerManager ( this ) ;
2011-08-09 19:38:27 +00:00
}
2010-08-20 23:24:40 +00:00
}
2013-11-01 06:36:02 +00:00
ShadowLayerForwarder * lf = mLayerManager - > AsShadowForwarder ( ) ;
if ( ! lf - > HasShadowManager ( ) & & aShadowManager ) {
lf - > SetShadowManager ( aShadowManager ) ;
}
2010-10-15 10:34:29 +00:00
if ( aAllowRetaining ) {
* aAllowRetaining = true ;
}
2010-08-20 23:24:40 +00:00
return mLayerManager ;
}
2010-09-24 03:28:15 +00:00
nsresult
2011-09-29 06:19:26 +00:00
PuppetWidget : : IMEEndComposition ( bool aCancel )
2010-09-24 03:28:15 +00:00
{
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-24 03:28:15 +00:00
nsEventStatus status ;
2013-10-01 07:22:59 +00:00
WidgetTextEvent textEvent ( true , NS_TEXT_TEXT , this ) ;
2012-07-30 14:20:58 +00:00
InitEvent ( textEvent , nullptr ) ;
2014-01-15 14:41:39 +00:00
textEvent . mSeqno = mIMELastReceivedSeqno ;
2010-12-15 19:22:15 +00:00
// SendEndIMEComposition is always called since ResetInputState
// should always be called even if we aren't composing something.
2010-09-24 03:28:15 +00:00
if ( ! mTabChild | |
! mTabChild - > SendEndIMEComposition ( aCancel , & textEvent . theText ) ) {
return NS_ERROR_FAILURE ;
}
2010-12-15 19:22:15 +00:00
if ( ! mIMEComposing )
return NS_OK ;
2010-09-24 03:28:15 +00:00
DispatchEvent ( & textEvent , status ) ;
2013-10-01 07:23:00 +00:00
WidgetCompositionEvent compEvent ( true , NS_COMPOSITION_END , this ) ;
2012-07-30 14:20:58 +00:00
InitEvent ( compEvent , nullptr ) ;
2014-01-15 14:41:39 +00:00
compEvent . mSeqno = mIMELastReceivedSeqno ;
2010-09-24 03:28:15 +00:00
DispatchEvent ( & compEvent , status ) ;
return NS_OK ;
}
NS_IMETHODIMP
2014-02-18 00:00:15 +00:00
PuppetWidget : : NotifyIME ( const IMENotification & aIMENotification )
2010-09-24 03:28:15 +00:00
{
2014-02-18 00:00:15 +00:00
switch ( aIMENotification . mMessage ) {
2013-03-06 06:14:31 +00:00
case NOTIFY_IME_OF_CURSOR_POS_CHANGED :
case REQUEST_TO_COMMIT_COMPOSITION :
return IMEEndComposition ( false ) ;
case REQUEST_TO_CANCEL_COMPOSITION :
return IMEEndComposition ( true ) ;
case NOTIFY_IME_OF_FOCUS :
return NotifyIMEOfFocusChange ( true ) ;
case NOTIFY_IME_OF_BLUR :
return NotifyIMEOfFocusChange ( false ) ;
case NOTIFY_IME_OF_SELECTION_CHANGE :
2014-02-26 00:48:02 +00:00
return NotifyIMEOfSelectionChange ( aIMENotification ) ;
2014-02-18 00:00:15 +00:00
case NOTIFY_IME_OF_TEXT_CHANGE :
return NotifyIMEOfTextChange ( aIMENotification ) ;
2013-11-07 00:11:11 +00:00
case NOTIFY_IME_OF_COMPOSITION_UPDATE :
return NotifyIMEOfUpdateComposition ( ) ;
2013-03-06 06:14:31 +00:00
default :
return NS_ERROR_NOT_IMPLEMENTED ;
}
2010-09-24 03:28:15 +00:00
}
2011-11-27 11:51:52 +00:00
NS_IMETHODIMP_ ( void )
PuppetWidget : : SetInputContext ( const InputContext & aContext ,
const InputContextAction & aAction )
2010-09-24 03:28:15 +00:00
{
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return ;
# endif
2011-11-27 11:51:52 +00:00
if ( ! mTabChild ) {
return ;
}
2011-11-27 11:51:53 +00:00
mTabChild - > SendSetInputContext (
2012-08-22 15:56:38 +00:00
static_cast < int32_t > ( aContext . mIMEState . mEnabled ) ,
static_cast < int32_t > ( aContext . mIMEState . mOpen ) ,
2011-11-27 11:51:53 +00:00
aContext . mHTMLInputType ,
2012-08-27 02:16:22 +00:00
aContext . mHTMLInputInputmode ,
2011-11-27 11:51:53 +00:00
aContext . mActionHint ,
2012-08-22 15:56:38 +00:00
static_cast < int32_t > ( aAction . mCause ) ,
static_cast < int32_t > ( aAction . mFocusChange ) ) ;
2010-09-24 03:28:15 +00:00
}
2011-11-27 11:51:52 +00:00
NS_IMETHODIMP_ ( InputContext )
PuppetWidget : : GetInputContext ( )
2010-09-24 03:28:15 +00:00
{
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return InputContext ( ) ;
# endif
2011-11-27 11:51:52 +00:00
InputContext context ;
if ( mTabChild ) {
2012-08-22 15:56:38 +00:00
int32_t enabled , open ;
2012-10-30 01:58:29 +00:00
intptr_t nativeIMEContext ;
2012-10-26 23:35:20 +00:00
mTabChild - > SendGetInputContext ( & enabled , & open , & nativeIMEContext ) ;
2011-11-27 11:51:53 +00:00
context . mIMEState . mEnabled = static_cast < IMEState : : Enabled > ( enabled ) ;
context . mIMEState . mOpen = static_cast < IMEState : : Open > ( open ) ;
2012-10-26 23:35:20 +00:00
context . mNativeIMEContext = reinterpret_cast < void * > ( nativeIMEContext ) ;
2011-11-27 11:51:52 +00:00
}
return context ;
2010-09-24 03:28:15 +00:00
}
2013-03-06 06:14:31 +00:00
nsresult
PuppetWidget : : NotifyIMEOfFocusChange ( bool aFocus )
2010-09-24 03:28:15 +00:00
{
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-24 03:28:15 +00:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
if ( aFocus ) {
nsEventStatus status ;
2013-10-01 07:23:00 +00:00
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_TEXT_CONTENT , this ) ;
2012-07-30 14:20:58 +00:00
InitEvent ( queryEvent , nullptr ) ;
2010-09-24 03:28:15 +00:00
// Query entire content
2012-09-28 06:57:33 +00:00
queryEvent . InitForQueryTextContent ( 0 , UINT32_MAX ) ;
2010-09-24 03:28:15 +00:00
DispatchEvent ( & queryEvent , status ) ;
if ( queryEvent . mSucceeded ) {
mTabChild - > SendNotifyIMETextHint ( queryEvent . mReply . mString ) ;
}
2010-09-24 03:28:15 +00:00
} else {
2013-03-06 06:14:31 +00:00
// Might not have been committed composition yet
IMEEndComposition ( false ) ;
2010-09-24 03:28:15 +00:00
}
2012-08-22 15:56:38 +00:00
uint32_t chromeSeqno ;
2014-02-26 00:48:02 +00:00
mIMEPreferenceOfParent = nsIMEUpdatePreference ( ) ;
2014-01-29 09:32:39 +00:00
if ( ! mTabChild - > SendNotifyIMEFocus ( aFocus , & mIMEPreferenceOfParent ,
& chromeSeqno ) ) {
2010-09-24 03:28:15 +00:00
return NS_ERROR_FAILURE ;
2014-01-29 09:32:39 +00:00
}
2010-09-24 03:28:15 +00:00
if ( aFocus ) {
2014-02-26 00:48:02 +00:00
IMENotification notification ( NOTIFY_IME_OF_SELECTION_CHANGE ) ;
notification . mSelectionChangeData . mCausedByComposition = false ;
NotifyIMEOfSelectionChange ( notification ) ; // Update selection
2010-10-01 14:17:37 +00:00
} else {
mIMELastBlurSeqno = chromeSeqno ;
2010-09-24 03:28:15 +00:00
}
return NS_OK ;
}
2013-11-07 00:11:11 +00:00
nsresult
PuppetWidget : : NotifyIMEOfUpdateComposition ( )
{
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
NS_ENSURE_TRUE ( mTabChild , NS_ERROR_FAILURE ) ;
2014-01-28 08:19:29 +00:00
nsRefPtr < TextComposition > textComposition =
2014-03-08 01:20:07 +00:00
IMEStateManager : : GetTextCompositionFor ( this ) ;
2013-11-07 00:11:11 +00:00
NS_ENSURE_TRUE ( textComposition , NS_ERROR_FAILURE ) ;
nsEventStatus status ;
uint32_t offset = textComposition - > OffsetOfTargetClause ( ) ;
WidgetQueryContentEvent textRect ( true , NS_QUERY_TEXT_RECT , this ) ;
InitEvent ( textRect , nullptr ) ;
textRect . InitForQueryTextRect ( offset , 1 ) ;
DispatchEvent ( & textRect , status ) ;
NS_ENSURE_TRUE ( textRect . mSucceeded , NS_ERROR_FAILURE ) ;
2014-01-16 10:04:39 +00:00
WidgetQueryContentEvent caretRect ( true , NS_QUERY_CARET_RECT , this ) ;
InitEvent ( caretRect , nullptr ) ;
caretRect . InitForQueryCaretRect ( offset ) ;
DispatchEvent ( & caretRect , status ) ;
NS_ENSURE_TRUE ( caretRect . mSucceeded , NS_ERROR_FAILURE ) ;
2013-11-07 00:11:11 +00:00
mTabChild - > SendNotifyIMESelectedCompositionRect ( offset ,
2014-01-16 10:04:39 +00:00
textRect . mReply . mRect ,
caretRect . mReply . mRect ) ;
2013-11-07 00:11:11 +00:00
return NS_OK ;
}
2012-11-13 13:04:44 +00:00
nsIMEUpdatePreference
PuppetWidget : : GetIMEUpdatePreference ( )
{
2014-02-25 05:51:01 +00:00
# ifdef MOZ_CROSS_PROCESS_IME
2014-01-29 09:32:39 +00:00
// e10s requires IME information cache into TabParent
return nsIMEUpdatePreference ( mIMEPreferenceOfParent . mWantUpdates |
nsIMEUpdatePreference : : NOTIFY_SELECTION_CHANGE |
nsIMEUpdatePreference : : NOTIFY_TEXT_CHANGE ) ;
# else
// B2G doesn't handle IME as widget-level.
return nsIMEUpdatePreference ( ) ;
# endif
2012-11-13 13:04:44 +00:00
}
2014-02-18 00:00:15 +00:00
nsresult
PuppetWidget : : NotifyIMEOfTextChange ( const IMENotification & aIMENotification )
2010-09-24 03:28:15 +00:00
{
2014-02-26 00:48:02 +00:00
MOZ_ASSERT ( aIMENotification . mMessage = = NOTIFY_IME_OF_TEXT_CHANGE ,
" Passed wrong notification " ) ;
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-24 03:28:15 +00:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
2014-01-29 09:32:35 +00:00
nsEventStatus status ;
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_TEXT_CONTENT , this ) ;
InitEvent ( queryEvent , nullptr ) ;
queryEvent . InitForQueryTextContent ( 0 , UINT32_MAX ) ;
DispatchEvent ( & queryEvent , status ) ;
2010-09-24 03:28:15 +00:00
2014-01-29 09:32:35 +00:00
if ( queryEvent . mSucceeded ) {
mTabChild - > SendNotifyIMETextHint ( queryEvent . mReply . mString ) ;
2010-09-24 03:28:15 +00:00
}
2014-01-29 09:32:39 +00:00
// TabParent doesn't this this to cache. we don't send the notification
// if parent process doesn't request NOTIFY_TEXT_CHANGE.
2014-02-26 00:48:02 +00:00
if ( mIMEPreferenceOfParent . WantTextChange ( ) & &
( mIMEPreferenceOfParent . WantChangesCausedByComposition ( ) | |
! aIMENotification . mTextChangeData . mCausedByComposition ) ) {
2014-02-18 00:00:15 +00:00
mTabChild - > SendNotifyIMETextChange (
aIMENotification . mTextChangeData . mStartOffset ,
aIMENotification . mTextChangeData . mOldEndOffset ,
2014-02-26 00:48:02 +00:00
aIMENotification . mTextChangeData . mNewEndOffset ,
aIMENotification . mTextChangeData . mCausedByComposition ) ;
2010-09-24 03:28:15 +00:00
}
return NS_OK ;
}
2013-03-06 06:14:31 +00:00
nsresult
2014-02-26 00:48:02 +00:00
PuppetWidget : : NotifyIMEOfSelectionChange (
const IMENotification & aIMENotification )
2010-09-24 03:28:15 +00:00
{
2014-02-26 00:48:02 +00:00
MOZ_ASSERT ( aIMENotification . mMessage = = NOTIFY_IME_OF_SELECTION_CHANGE ,
" Passed wrong notification " ) ;
2013-01-28 23:56:28 +00:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-24 03:28:15 +00:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
2014-01-29 09:32:39 +00:00
nsEventStatus status ;
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_SELECTED_TEXT , this ) ;
InitEvent ( queryEvent , nullptr ) ;
DispatchEvent ( & queryEvent , status ) ;
2010-09-24 03:28:15 +00:00
2014-01-29 09:32:39 +00:00
if ( queryEvent . mSucceeded ) {
2014-02-26 00:48:02 +00:00
mTabChild - > SendNotifyIMESelection (
mIMELastReceivedSeqno ,
queryEvent . GetSelectionStart ( ) ,
queryEvent . GetSelectionEnd ( ) ,
aIMENotification . mSelectionChangeData . mCausedByComposition ) ;
2010-09-24 03:28:15 +00:00
}
return NS_OK ;
}
2011-06-22 00:32:43 +00:00
NS_IMETHODIMP
PuppetWidget : : SetCursor ( nsCursor aCursor )
{
2014-05-28 01:12:29 +00:00
if ( mCursor = = aCursor & & ! mUpdateCursor ) {
2012-09-12 04:48:13 +00:00
return NS_OK ;
}
2014-05-28 01:12:29 +00:00
if ( mTabChild & &
! mTabChild - > SendSetCursor ( aCursor , mUpdateCursor ) ) {
2011-06-22 00:32:43 +00:00
return NS_ERROR_FAILURE ;
}
2012-09-12 04:48:13 +00:00
mCursor = aCursor ;
2014-05-28 01:12:29 +00:00
mUpdateCursor = false ;
2012-09-12 04:48:13 +00:00
2011-06-22 00:32:43 +00:00
return NS_OK ;
}
2010-08-20 23:24:40 +00:00
nsresult
2012-08-15 18:52:42 +00:00
PuppetWidget : : Paint ( )
2010-08-20 23:24:40 +00:00
{
NS_ABORT_IF_FALSE ( ! mDirtyRegion . IsEmpty ( ) , " paint event logic messed up " ) ;
2012-08-15 18:53:14 +00:00
if ( ! mAttachedWidgetListener )
2012-08-15 18:52:42 +00:00
return NS_OK ;
nsIntRegion region = mDirtyRegion ;
2010-08-20 23:24:40 +00:00
// reset repaint tracking
mDirtyRegion . SetEmpty ( ) ;
mPaintTask . Revoke ( ) ;
2013-01-28 19:34:08 +00:00
mAttachedWidgetListener - > WillPaintWindow ( this ) ;
2013-01-28 19:34:03 +00:00
if ( mAttachedWidgetListener ) {
2010-08-20 23:24:40 +00:00
# ifdef DEBUG
2012-08-15 18:52:42 +00:00
debug_DumpPaintEvent ( stderr , this , region ,
2012-09-02 02:35:17 +00:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 23:24:40 +00:00
# endif
2014-01-23 18:26:41 +00:00
if ( mozilla : : layers : : LayersBackend : : LAYERS_D3D10 = = mLayerManager - > GetBackendType ( ) ) {
2013-05-23 14:49:18 +00:00
mAttachedWidgetListener - > PaintWindow ( this , region ) ;
2014-01-23 18:26:41 +00:00
} else if ( mozilla : : layers : : LayersBackend : : LAYERS_CLIENT = = mLayerManager - > GetBackendType ( ) ) {
2013-05-01 05:03:25 +00:00
// Do nothing, the compositor will handle drawing
if ( mTabChild ) {
mTabChild - > NotifyPainted ( ) ;
}
2011-08-09 19:38:27 +00:00
} else {
2014-06-10 06:02:21 +00:00
nsRefPtr < gfxContext > ctx = new gfxContext ( mDrawTarget ) ;
2012-02-10 19:22:21 +00:00
ctx - > Rectangle ( gfxRect ( 0 , 0 , 0 , 0 ) ) ;
ctx - > Clip ( ) ;
2011-08-09 19:38:27 +00:00
AutoLayerManagerSetup setupLayerManager ( this , ctx ,
2014-01-23 18:26:41 +00:00
BufferMode : : BUFFER_NONE ) ;
2013-05-23 14:49:18 +00:00
mAttachedWidgetListener - > PaintWindow ( this , region ) ;
2013-04-02 18:32:59 +00:00
if ( mTabChild ) {
mTabChild - > NotifyPainted ( ) ;
}
2011-08-09 19:38:27 +00:00
}
2010-08-20 23:24:40 +00:00
}
2012-12-12 21:57:08 +00:00
if ( mAttachedWidgetListener ) {
mAttachedWidgetListener - > DidPaintWindow ( ) ;
}
2010-08-20 23:24:40 +00:00
return NS_OK ;
}
void
PuppetWidget : : SetChild ( PuppetWidget * aChild )
{
NS_ABORT_IF_FALSE ( this ! = aChild , " can't parent a widget to itself " ) ;
NS_ABORT_IF_FALSE ( ! aChild - > mChild ,
" fake widget 'hierarchy' only expected to have one level " ) ;
mChild = aChild ;
}
NS_IMETHODIMP
PuppetWidget : : PaintTask : : Run ( )
{
if ( mWidget ) {
2012-08-15 18:52:42 +00:00
mWidget - > Paint ( ) ;
2010-08-20 23:24:40 +00:00
}
return NS_OK ;
}
2012-11-08 03:51:55 +00:00
bool
PuppetWidget : : NeedsPaint ( )
{
return mVisible ;
}
2010-12-03 01:24:04 +00:00
float
PuppetWidget : : GetDPI ( )
{
if ( mDPI < 0 ) {
2013-04-02 18:32:59 +00:00
if ( mTabChild ) {
mTabChild - > GetDPI ( & mDPI ) ;
} else {
mDPI = 96.0 ;
}
2010-12-03 01:24:04 +00:00
}
return mDPI ;
}
2013-05-01 23:06:19 +00:00
double
PuppetWidget : : GetDefaultScaleInternal ( )
{
if ( mDefaultScale < 0 ) {
if ( mTabChild ) {
mTabChild - > GetDefaultScale ( & mDefaultScale ) ;
} else {
mDefaultScale = 1 ;
}
}
return mDefaultScale ;
}
2011-08-31 19:01:38 +00:00
void *
2012-08-22 15:56:38 +00:00
PuppetWidget : : GetNativeData ( uint32_t aDataType )
2011-08-31 19:01:38 +00:00
{
2012-05-08 21:36:07 +00:00
switch ( aDataType ) {
case NS_NATIVE_SHAREABLE_WINDOW : {
NS_ABORT_IF_FALSE ( mTabChild , " Need TabChild to get the nativeWindow from! " ) ;
2012-07-20 11:16:17 +00:00
mozilla : : WindowsHandle nativeData = 0 ;
2013-04-02 18:32:59 +00:00
if ( mTabChild ) {
mTabChild - > SendGetWidgetNativeData ( & nativeData ) ;
}
2012-05-08 21:36:07 +00:00
return ( void * ) nativeData ;
}
case NS_NATIVE_WINDOW :
case NS_NATIVE_DISPLAY :
case NS_NATIVE_PLUGIN_PORT :
case NS_NATIVE_GRAPHIC :
case NS_NATIVE_SHELLWIDGET :
case NS_NATIVE_WIDGET :
NS_WARNING ( " nsWindow::GetNativeData not implemented for this type " ) ;
break ;
default :
NS_WARNING ( " nsWindow::GetNativeData called with bad value " ) ;
break ;
}
2012-07-30 14:20:58 +00:00
return nullptr ;
2012-05-08 21:36:07 +00:00
}
PuppetScreen : : PuppetScreen ( void * nativeScreen )
{
}
PuppetScreen : : ~ PuppetScreen ( )
{
}
static ScreenConfiguration
ScreenConfig ( )
{
ScreenConfiguration config ;
hal : : GetCurrentScreenConfiguration ( & config ) ;
return config ;
}
2014-07-14 17:22:26 +00:00
NS_IMETHODIMP
PuppetScreen : : GetId ( uint32_t * outId )
{
* outId = 1 ;
return NS_OK ;
}
2012-05-08 21:36:07 +00:00
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : GetRect ( int32_t * outLeft , int32_t * outTop ,
int32_t * outWidth , int32_t * outHeight )
2012-05-08 21:36:07 +00:00
{
nsIntRect r = ScreenConfig ( ) . rect ( ) ;
* outLeft = r . x ;
* outTop = r . y ;
* outWidth = r . width ;
* outHeight = r . height ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : GetAvailRect ( int32_t * outLeft , int32_t * outTop ,
int32_t * outWidth , int32_t * outHeight )
2012-05-08 21:36:07 +00:00
{
return GetRect ( outLeft , outTop , outWidth , outHeight ) ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : GetPixelDepth ( int32_t * aPixelDepth )
2012-05-08 21:36:07 +00:00
{
* aPixelDepth = ScreenConfig ( ) . pixelDepth ( ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : GetColorDepth ( int32_t * aColorDepth )
2012-05-08 21:36:07 +00:00
{
* aColorDepth = ScreenConfig ( ) . colorDepth ( ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : GetRotation ( uint32_t * aRotation )
2012-05-08 21:36:07 +00:00
{
NS_WARNING ( " Attempt to get screen rotation through nsIScreen::GetRotation(). Nothing should know or care this in sandboxed contexts. If you want *orientation*, use hal. " ) ;
return NS_ERROR_NOT_AVAILABLE ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreen : : SetRotation ( uint32_t aRotation )
2012-05-08 21:36:07 +00:00
{
NS_WARNING ( " Attempt to set screen rotation through nsIScreen::GetRotation(). Nothing should know or care this in sandboxed contexts. If you want *orientation*, use hal. " ) ;
return NS_ERROR_NOT_AVAILABLE ;
}
2014-04-27 07:06:00 +00:00
NS_IMPL_ISUPPORTS ( PuppetScreenManager , nsIScreenManager )
2012-05-08 21:36:07 +00:00
PuppetScreenManager : : PuppetScreenManager ( )
{
2012-07-30 14:20:58 +00:00
mOneScreen = new PuppetScreen ( nullptr ) ;
2012-05-08 21:36:07 +00:00
}
PuppetScreenManager : : ~ PuppetScreenManager ( )
{
}
2014-07-14 17:22:26 +00:00
NS_IMETHODIMP
PuppetScreenManager : : ScreenForId ( uint32_t aId ,
nsIScreen * * outScreen )
{
NS_IF_ADDREF ( * outScreen = mOneScreen . get ( ) ) ;
return NS_OK ;
}
2012-05-08 21:36:07 +00:00
NS_IMETHODIMP
PuppetScreenManager : : GetPrimaryScreen ( nsIScreen * * outScreen )
{
NS_IF_ADDREF ( * outScreen = mOneScreen . get ( ) ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreenManager : : ScreenForRect ( int32_t inLeft ,
int32_t inTop ,
int32_t inWidth ,
int32_t inHeight ,
2012-05-08 21:36:07 +00:00
nsIScreen * * outScreen )
{
return GetPrimaryScreen ( outScreen ) ;
}
NS_IMETHODIMP
PuppetScreenManager : : ScreenForNativeWidget ( void * aWidget ,
nsIScreen * * outScreen )
{
return GetPrimaryScreen ( outScreen ) ;
}
NS_IMETHODIMP
2012-08-22 15:56:38 +00:00
PuppetScreenManager : : GetNumberOfScreens ( uint32_t * aNumberOfScreens )
2012-05-08 21:36:07 +00:00
{
* aNumberOfScreens = 1 ;
return NS_OK ;
2011-08-31 19:01:38 +00:00
}
2013-04-09 21:07:02 +00:00
NS_IMETHODIMP
PuppetScreenManager : : GetSystemDefaultScale ( float * aDefaultScale )
{
* aDefaultScale = 1.0f ;
return NS_OK ;
}
2010-08-20 23:24:40 +00:00
} // namespace widget
} // namespace mozilla