2010-08-20 16:24:40 -07:00
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* vim : sw = 2 ts = 8 et :
*/
2012-05-21 04:12:37 -07: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 16:24:40 -07:00
2012-07-17 16:59:45 -07:00
# include "base/basictypes.h"
2013-04-30 22:03:25 -07:00
# include "ClientLayerManager.h"
2012-07-17 16:59:45 -07:00
# include "gfxPlatform.h"
# include "mozilla/dom/TabChild.h"
2012-05-08 14:36:07 -07:00
# include "mozilla/Hal.h"
2014-03-07 17:20:07 -08:00
# include "mozilla/IMEStateManager.h"
2012-07-17 16:59:45 -07:00
# include "mozilla/layers/CompositorChild.h"
2013-04-24 11:42:40 -07:00
# include "mozilla/layers/PLayerTransactionChild.h"
2014-11-20 09:28:58 -08:00
# include "mozilla/Preferences.h"
2014-04-02 21:18:37 -07:00
# include "mozilla/TextComposition.h"
2013-09-25 04:21:19 -07:00
# include "mozilla/TextEvents.h"
2010-08-20 16:24:40 -07:00
# include "PuppetWidget.h"
2012-08-15 11:52:42 -07:00
# include "nsIWidgetListener.h"
2010-08-20 16:24:40 -07:00
2014-06-09 23:02:21 -07:00
using namespace mozilla ;
2012-05-08 14:36:07 -07:00
using namespace mozilla : : dom ;
using namespace mozilla : : hal ;
2014-02-09 00:04:38 -08:00
using namespace mozilla : : gfx ;
2010-08-20 16:24:40 -07: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-23 19:52:21 -08:00
aWidget - > Invalidate ( * r ) ;
2010-08-20 16:24:40 -07:00
}
}
/*static*/ already_AddRefed < nsIWidget >
2012-07-17 16:59:45 -07:00
nsIWidget : : CreatePuppetWidget ( TabChild * aTabChild )
2010-08-20 16:24:40 -07:00
{
2013-04-04 06:24:32 -07:00
NS_ABORT_IF_FALSE ( ! aTabChild | | nsIWidget : : UsePuppetWidgets ( ) ,
2010-08-20 16:24:40 -07:00
" PuppetWidgets not allowed in this configuration " ) ;
2010-09-23 20:28:15 -07:00
nsCOMPtr < nsIWidget > widget = new PuppetWidget ( aTabChild ) ;
2010-08-20 16:24:40 -07:00
return widget . forget ( ) ;
}
namespace mozilla {
namespace widget {
2011-02-23 09:45:09 -08: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 15:56:28 -08:00
# ifdef MOZ_CROSS_PROCESS_IME
2011-02-23 09:45:09 -08:00
return ! IsPopup ( aInitData ) ;
2013-01-28 15:56:28 -08:00
# else
return false ;
# endif
2011-02-23 09:45:09 -08:00
}
2010-08-20 16:24:40 -07:00
// Arbitrary, fungible.
const size_t PuppetWidget : : kMaxDimension = 4000 ;
2014-04-27 00:06:00 -07:00
NS_IMPL_ISUPPORTS_INHERITED ( PuppetWidget , nsBaseWidget ,
nsISupportsWeakReference )
2010-08-20 16:24:40 -07:00
2012-07-17 16:59:45 -07:00
PuppetWidget : : PuppetWidget ( TabChild * aTabChild )
2010-09-23 20:28:15 -07:00
: mTabChild ( aTabChild )
2010-12-02 17:24:04 -08:00
, mDPI ( - 1 )
2013-05-01 16:06:19 -07:00
, mDefaultScale ( - 1 )
2014-04-21 13:40:09 -07:00
, mNativeKeyCommandsValid ( false )
2010-08-20 16:24:40 -07:00
{
MOZ_COUNT_CTOR ( PuppetWidget ) ;
2014-03-20 08:46:29 -07:00
mSingleLineCommands . SetCapacity ( 4 ) ;
mMultiLineCommands . SetCapacity ( 4 ) ;
mRichTextCommands . SetCapacity ( 4 ) ;
2010-08-20 16:24:40 -07:00
}
PuppetWidget : : ~ PuppetWidget ( )
{
MOZ_COUNT_DTOR ( PuppetWidget ) ;
}
NS_IMETHODIMP
PuppetWidget : : Create ( nsIWidget * aParent ,
nsNativeWidget aNativeParent ,
const nsIntRect & aRect ,
2011-04-16 18:22:44 -07:00
nsDeviceContext * aContext ,
2010-08-20 16:24:40 -07:00
nsWidgetInitData * aInitData )
{
NS_ABORT_IF_FALSE ( ! aNativeParent , " got a non-Puppet native parent " ) ;
2012-08-15 11:53:09 -07:00
BaseCreate ( nullptr , aRect , aContext , aInitData ) ;
2010-08-20 16:24:40 -07:00
mBounds = aRect ;
2011-10-17 07:59:28 -07:00
mEnabled = true ;
mVisible = true ;
2010-08-20 16:24:40 -07:00
2014-06-09 23:02:21 -07:00
mDrawTarget = gfxPlatform : : GetPlatform ( ) - >
CreateOffscreenContentDrawTarget ( IntSize ( 1 , 1 ) , SurfaceFormat : : B8G8R8A8 ) ;
2010-08-20 16:24:40 -07:00
2011-10-17 07:59:28 -07:00
mIMEComposing = false ;
2012-08-29 08:26:18 -07:00
mNeedIMEStateInit = MightNeedIMEFocus ( aInitData ) ;
2010-09-23 20:28:15 -07:00
2010-08-20 16:24:40 -07:00
PuppetWidget * parent = static_cast < PuppetWidget * > ( aParent ) ;
if ( parent ) {
parent - > SetChild ( this ) ;
2010-08-20 16:24:41 -07:00
mLayerManager = parent - > GetLayerManager ( ) ;
2010-08-20 16:24:40 -07:00
}
else {
2011-10-17 07:59:28 -07:00
Resize ( mBounds . x , mBounds . y , mBounds . width , mBounds . height , false ) ;
2010-08-20 16:24:40 -07:00
}
return NS_OK ;
}
2012-08-29 08:26:18 -07:00
void
PuppetWidget : : InitIMEState ( )
{
2013-06-16 22:42:20 -07:00
MOZ_ASSERT ( mTabChild ) ;
2012-08-29 08:26:18 -07:00
if ( mNeedIMEStateInit ) {
uint32_t chromeSeqno ;
2014-01-29 01:32:39 -08:00
mTabChild - > SendNotifyIMEFocus ( false , & mIMEPreferenceOfParent , & chromeSeqno ) ;
2012-08-29 08:26:18 -07:00
mIMELastBlurSeqno = mIMELastReceivedSeqno = chromeSeqno ;
mNeedIMEStateInit = false ;
}
}
2010-08-20 16:24:40 -07:00
already_AddRefed < nsIWidget >
PuppetWidget : : CreateChild ( const nsIntRect & aRect ,
2011-04-16 18:22:44 -07:00
nsDeviceContext * aContext ,
2010-08-20 16:24:40 -07:00
nsWidgetInitData * aInitData ,
2011-09-28 23:19:26 -07:00
bool aForceUseIWidgetParent )
2010-08-20 16:24:40 -07:00
{
2011-02-23 09:45:09 -08:00
bool isPopup = IsPopup ( aInitData ) ;
2010-09-23 20:28:15 -07:00
nsCOMPtr < nsIWidget > widget = nsIWidget : : CreatePuppetWidget ( mTabChild ) ;
2010-08-20 16:24:40 -07:00
return ( ( widget & &
2012-07-30 07:20:58 -07:00
NS_SUCCEEDED ( widget - > Create ( isPopup ? nullptr : this , nullptr , aRect ,
2011-10-25 08:05:32 -07:00
aContext , aInitData ) ) ) ?
2012-07-30 07:20:58 -07:00
widget . forget ( ) : nullptr ) ;
2010-08-20 16:24:40 -07:00
}
2010-08-20 16:24:40 -07:00
NS_IMETHODIMP
PuppetWidget : : Destroy ( )
{
2011-03-29 13:14:44 -07:00
Base : : OnDestroy ( ) ;
2010-08-20 16:24:40 -07:00
Base : : Destroy ( ) ;
mPaintTask . Revoke ( ) ;
2012-07-30 07:20:58 -07:00
mChild = nullptr ;
2010-12-06 18:05:25 -08:00
if ( mLayerManager ) {
mLayerManager - > Destroy ( ) ;
}
2012-07-30 07:20:58 -07:00
mLayerManager = nullptr ;
mTabChild = nullptr ;
2010-08-20 16:24:40 -07:00
return NS_OK ;
}
2010-08-20 16:24:40 -07:00
NS_IMETHODIMP
2011-09-28 23:19:26 -07:00
PuppetWidget : : Show ( bool aState )
2010-08-20 16:24:40 -07:00
{
NS_ASSERTION ( mEnabled ,
" does it make sense to Show()/Hide() a disabled widget? " ) ;
2011-09-28 23:19:26 -07:00
bool wasVisible = mVisible ;
2010-08-20 16:24:40 -07:00
mVisible = aState ;
2012-11-07 19:51:55 -08:00
if ( mChild ) {
mChild - > mVisible = aState ;
}
if ( ! mVisible & & mLayerManager ) {
mLayerManager - > ClearCachedResources ( ) ;
}
2010-08-20 16:24:40 -07:00
if ( ! wasVisible & & mVisible ) {
2011-10-17 07:59:28 -07:00
Resize ( mBounds . width , mBounds . height , false ) ;
2012-11-07 19:51:55 -08:00
Invalidate ( mBounds ) ;
2010-08-20 16:24:40 -07:00
}
return NS_OK ;
}
NS_IMETHODIMP
2012-12-12 01:57:38 -08:00
PuppetWidget : : Resize ( double aWidth ,
double aHeight ,
bool aRepaint )
2010-08-20 16:24:40 -07:00
{
nsIntRect oldBounds = mBounds ;
2012-12-12 01:57:38 -08:00
mBounds . SizeTo ( nsIntSize ( NSToIntRound ( aWidth ) , NSToIntRound ( aHeight ) ) ) ;
2010-08-20 16:24:40 -07: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 11:53:14 -07:00
if ( ! oldBounds . IsEqualEdges ( mBounds ) & & mAttachedWidgetListener ) {
mAttachedWidgetListener - > WindowResized ( this , mBounds . width , mBounds . height ) ;
2010-08-20 16:24:40 -07:00
}
return NS_OK ;
}
2014-11-12 12:59:19 -08:00
nsresult
PuppetWidget : : ConfigureChildren ( const nsTArray < Configuration > & aConfigurations )
{
for ( uint32_t i = 0 ; i < aConfigurations . Length ( ) ; + + i ) {
const Configuration & configuration = aConfigurations [ i ] ;
PuppetWidget * w = static_cast < PuppetWidget * > ( configuration . mChild ) ;
NS_ASSERTION ( w - > GetParent ( ) = = this ,
" Configured widget is not a child " ) ;
w - > SetWindowClipRegion ( configuration . mClipRegion , true ) ;
nsIntRect bounds ;
w - > GetBounds ( bounds ) ;
if ( bounds . Size ( ) ! = configuration . mBounds . Size ( ) ) {
w - > Resize ( configuration . mBounds . x , configuration . mBounds . y ,
configuration . mBounds . width , configuration . mBounds . height ,
true ) ;
} else if ( bounds . TopLeft ( ) ! = configuration . mBounds . TopLeft ( ) ) {
w - > Move ( configuration . mBounds . x , configuration . mBounds . y ) ;
}
w - > SetWindowClipRegion ( configuration . mClipRegion , false ) ;
}
return NS_OK ;
}
2010-08-20 16:24:40 -07:00
NS_IMETHODIMP
2011-09-28 23:19:26 -07:00
PuppetWidget : : SetFocus ( bool aRaise )
2010-08-20 16:24:40 -07:00
{
// XXX/cjones: someone who knows about event handling needs to
// decide how this should work.
return NS_OK ;
}
NS_IMETHODIMP
2011-12-23 19:52:21 -08:00
PuppetWidget : : Invalidate ( const nsIntRect & aRect )
2010-08-20 16:24:40 -07:00
{
# ifdef DEBUG
2011-12-23 19:52:21 -08:00
debug_DumpInvalidate ( stderr , this , & aRect ,
2012-09-01 19:35:17 -07:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 16:24:40 -07:00
# endif
if ( mChild ) {
2011-12-23 19:52:21 -08:00
return mChild - > Invalidate ( aRect ) ;
2010-08-20 16:24:40 -07:00
}
mDirtyRegion . Or ( mDirtyRegion , aRect ) ;
2011-12-23 19:52:21 -08:00
if ( ! mDirtyRegion . IsEmpty ( ) & & ! mPaintTask . IsPending ( ) ) {
2010-08-20 16:24:40 -07:00
mPaintTask = new PaintTask ( this ) ;
return NS_DispatchToCurrentThread ( mPaintTask . get ( ) ) ;
}
return NS_OK ;
}
2010-09-23 20:28:15 -07:00
void
2013-10-01 20:46:03 -07:00
PuppetWidget : : InitEvent ( WidgetGUIEvent & event , nsIntPoint * aPoint )
2010-09-23 20:28:15 -07:00
{
2012-07-30 07:20:58 -07:00
if ( nullptr = = aPoint ) {
2010-09-23 20:28:15 -07: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 16:24:40 -07:00
NS_IMETHODIMP
2013-10-01 20:46:03 -07:00
PuppetWidget : : DispatchEvent ( WidgetGUIEvent * event , nsEventStatus & aStatus )
2010-08-20 16:24:40 -07:00
{
# ifdef DEBUG
debug_DumpEvent ( stdout , event - > widget , event ,
2012-09-01 19:35:17 -07:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 16:24:40 -07:00
# endif
2011-02-09 12:13:18 -08:00
NS_ABORT_IF_FALSE ( ! mChild | | mChild - > mWindowType = = eWindowType_popup ,
" Unexpected event dispatch! " ) ;
2014-04-21 13:40:09 -07:00
AutoCacheNativeKeyCommands autoCache ( this ) ;
if ( event - > mFlags . mIsSynthesizedForTests & & ! mNativeKeyCommandsValid ) {
WidgetKeyboardEvent * keyEvent = event - > AsKeyboardEvent ( ) ;
if ( keyEvent ) {
mTabChild - > RequestNativeKeyBindings ( & autoCache , keyEvent ) ;
}
}
2010-08-20 16:24:40 -07:00
aStatus = nsEventStatus_eIgnore ;
2011-02-09 12:13:18 -08:00
if ( event - > message = = NS_COMPOSITION_START ) {
2011-10-17 07:59:28 -07:00
mIMEComposing = true ;
2011-02-09 12:13:18 -08:00
}
2014-01-15 06:41:39 -08:00
uint32_t seqno = kLatestSeqno ;
2014-08-03 22:28:46 -07:00
switch ( event - > mClass ) {
2014-08-03 22:28:48 -07:00
case eCompositionEventClass :
2014-01-15 06:41:39 -08:00
seqno = event - > AsCompositionEvent ( ) - > mSeqno ;
2011-02-09 12:13:18 -08:00
break ;
2014-08-03 22:28:49 -07:00
case eSelectionEventClass :
2014-01-15 06:41:39 -08:00
seqno = event - > AsSelectionEvent ( ) - > mSeqno ;
2011-02-09 12:13:18 -08:00
break ;
2012-11-19 22:05:56 -08:00
default :
break ;
2011-02-09 12:13:18 -08:00
}
2014-01-15 06:41:39 -08:00
if ( seqno ! = kLatestSeqno ) {
mIMELastReceivedSeqno = seqno ;
if ( mIMELastReceivedSeqno < mIMELastBlurSeqno ) {
return NS_OK ;
}
}
2012-08-15 11:53:09 -07:00
2013-01-25 11:51:16 -08:00
if ( mAttachedWidgetListener ) {
aStatus = mAttachedWidgetListener - > HandleEvent ( event , mUseAttachedEvents ) ;
}
2011-02-09 12:13:18 -08:00
2014-11-24 21:02:30 -08:00
if ( event - > mClass = = eCompositionEventClass & &
event - > AsCompositionEvent ( ) - > CausesDOMCompositionEndEvent ( ) ) {
2011-10-17 07:59:28 -07:00
mIMEComposing = false ;
2010-08-20 16:24:40 -07:00
}
return NS_OK ;
}
2014-03-20 08:46:29 -07:00
NS_IMETHODIMP_ ( bool )
PuppetWidget : : ExecuteNativeKeyBinding ( NativeKeyBindingsType aType ,
const mozilla : : WidgetKeyboardEvent & aEvent ,
DoCommandCallback aCallback ,
void * aCallbackData )
{
2014-04-21 13:40:09 -07:00
// B2G doesn't have native key bindings.
# ifdef MOZ_B2G
return false ;
# else // #ifdef MOZ_B2G
MOZ_ASSERT ( mNativeKeyCommandsValid ) ;
2014-03-20 08:46:29 -07: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 13:40:09 -07:00
# endif
2014-03-20 08:46:29 -07:00
}
2010-08-20 16:24:40 -07:00
LayerManager *
2013-04-24 11:42:40 -07:00
PuppetWidget : : GetLayerManager ( PLayerTransactionChild * aShadowManager ,
2011-08-09 12:38:26 -07:00
LayersBackend aBackendHint ,
LayerManagerPersistence aPersistence ,
bool * aAllowRetaining )
2010-08-20 16:24:40 -07:00
{
if ( ! mLayerManager ) {
2014-08-21 17:16:44 -07:00
mLayerManager = new ClientLayerManager ( this ) ;
2010-08-20 16:24:40 -07:00
}
2013-10-31 23:36:02 -07:00
ShadowLayerForwarder * lf = mLayerManager - > AsShadowForwarder ( ) ;
if ( ! lf - > HasShadowManager ( ) & & aShadowManager ) {
lf - > SetShadowManager ( aShadowManager ) ;
}
2010-10-15 03:34:29 -07:00
if ( aAllowRetaining ) {
* aAllowRetaining = true ;
}
2010-08-20 16:24:40 -07:00
return mLayerManager ;
}
2010-09-23 20:28:15 -07:00
nsresult
2011-09-28 23:19:26 -07:00
PuppetWidget : : IMEEndComposition ( bool aCancel )
2010-09-23 20:28:15 -07:00
{
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-23 20:28:15 -07:00
nsEventStatus status ;
2014-10-07 03:01:49 -07:00
WidgetCompositionEvent compositionChangeEvent ( true , NS_COMPOSITION_CHANGE ,
this ) ;
InitEvent ( compositionChangeEvent , nullptr ) ;
compositionChangeEvent . mSeqno = mIMELastReceivedSeqno ;
2010-12-15 11:22:15 -08:00
// SendEndIMEComposition is always called since ResetInputState
// should always be called even if we aren't composing something.
2010-09-23 20:28:15 -07:00
if ( ! mTabChild | |
2014-10-07 03:01:49 -07:00
! mTabChild - > SendEndIMEComposition ( aCancel ,
& compositionChangeEvent . mData ) ) {
2010-09-23 20:28:15 -07:00
return NS_ERROR_FAILURE ;
}
2010-12-15 11:22:15 -08:00
if ( ! mIMEComposing )
return NS_OK ;
2014-10-07 03:01:49 -07:00
DispatchEvent ( & compositionChangeEvent , status ) ;
2010-09-23 20:28:15 -07:00
2014-10-07 03:01:49 -07:00
WidgetCompositionEvent compositionEndEvent ( true , NS_COMPOSITION_END , this ) ;
InitEvent ( compositionEndEvent , nullptr ) ;
compositionEndEvent . mSeqno = mIMELastReceivedSeqno ;
DispatchEvent ( & compositionEndEvent , status ) ;
2010-09-23 20:28:15 -07:00
return NS_OK ;
}
NS_IMETHODIMP
2014-02-17 16:00:15 -08:00
PuppetWidget : : NotifyIME ( const IMENotification & aIMENotification )
2010-09-23 20:28:15 -07:00
{
2014-02-17 16:00:15 -08:00
switch ( aIMENotification . mMessage ) {
2013-03-05 22:14:31 -08:00
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-25 16:48:02 -08:00
return NotifyIMEOfSelectionChange ( aIMENotification ) ;
2014-02-17 16:00:15 -08:00
case NOTIFY_IME_OF_TEXT_CHANGE :
return NotifyIMEOfTextChange ( aIMENotification ) ;
2013-11-06 16:11:11 -08:00
case NOTIFY_IME_OF_COMPOSITION_UPDATE :
return NotifyIMEOfUpdateComposition ( ) ;
2014-09-11 06:46:17 -07:00
case NOTIFY_IME_OF_MOUSE_BUTTON_EVENT :
return NotifyIMEOfMouseButtonEvent ( aIMENotification ) ;
2013-03-05 22:14:31 -08:00
default :
return NS_ERROR_NOT_IMPLEMENTED ;
}
2010-09-23 20:28:15 -07:00
}
2011-11-27 03:51:52 -08:00
NS_IMETHODIMP_ ( void )
PuppetWidget : : SetInputContext ( const InputContext & aContext ,
const InputContextAction & aAction )
2010-09-23 20:28:15 -07:00
{
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return ;
# endif
2011-11-27 03:51:52 -08:00
if ( ! mTabChild ) {
return ;
}
2011-11-27 03:51:53 -08:00
mTabChild - > SendSetInputContext (
2012-08-22 08:56:38 -07:00
static_cast < int32_t > ( aContext . mIMEState . mEnabled ) ,
static_cast < int32_t > ( aContext . mIMEState . mOpen ) ,
2011-11-27 03:51:53 -08:00
aContext . mHTMLInputType ,
2012-08-26 19:16:22 -07:00
aContext . mHTMLInputInputmode ,
2011-11-27 03:51:53 -08:00
aContext . mActionHint ,
2012-08-22 08:56:38 -07:00
static_cast < int32_t > ( aAction . mCause ) ,
static_cast < int32_t > ( aAction . mFocusChange ) ) ;
2010-09-23 20:28:15 -07:00
}
2011-11-27 03:51:52 -08:00
NS_IMETHODIMP_ ( InputContext )
PuppetWidget : : GetInputContext ( )
2010-09-23 20:28:15 -07:00
{
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return InputContext ( ) ;
# endif
2011-11-27 03:51:52 -08:00
InputContext context ;
if ( mTabChild ) {
2012-08-22 08:56:38 -07:00
int32_t enabled , open ;
2012-10-29 18:58:29 -07:00
intptr_t nativeIMEContext ;
2012-10-26 16:35:20 -07:00
mTabChild - > SendGetInputContext ( & enabled , & open , & nativeIMEContext ) ;
2011-11-27 03:51:53 -08:00
context . mIMEState . mEnabled = static_cast < IMEState : : Enabled > ( enabled ) ;
context . mIMEState . mOpen = static_cast < IMEState : : Open > ( open ) ;
2012-10-26 16:35:20 -07:00
context . mNativeIMEContext = reinterpret_cast < void * > ( nativeIMEContext ) ;
2011-11-27 03:51:52 -08:00
}
return context ;
2010-09-23 20:28:15 -07:00
}
2013-03-05 22:14:31 -08:00
nsresult
PuppetWidget : : NotifyIMEOfFocusChange ( bool aFocus )
2010-09-23 20:28:15 -07:00
{
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-23 20:28:15 -07:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
if ( aFocus ) {
nsEventStatus status ;
2013-10-01 00:23:00 -07:00
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_TEXT_CONTENT , this ) ;
2012-07-30 07:20:58 -07:00
InitEvent ( queryEvent , nullptr ) ;
2010-09-23 20:28:15 -07:00
// Query entire content
2012-09-27 23:57:33 -07:00
queryEvent . InitForQueryTextContent ( 0 , UINT32_MAX ) ;
2010-09-23 20:28:15 -07:00
DispatchEvent ( & queryEvent , status ) ;
if ( queryEvent . mSucceeded ) {
mTabChild - > SendNotifyIMETextHint ( queryEvent . mReply . mString ) ;
}
2010-09-23 20:28:15 -07:00
} else {
2013-03-05 22:14:31 -08:00
// Might not have been committed composition yet
IMEEndComposition ( false ) ;
2010-09-23 20:28:15 -07:00
}
2012-08-22 08:56:38 -07:00
uint32_t chromeSeqno ;
2014-02-25 16:48:02 -08:00
mIMEPreferenceOfParent = nsIMEUpdatePreference ( ) ;
2014-01-29 01:32:39 -08:00
if ( ! mTabChild - > SendNotifyIMEFocus ( aFocus , & mIMEPreferenceOfParent ,
& chromeSeqno ) ) {
2010-09-23 20:28:15 -07:00
return NS_ERROR_FAILURE ;
2014-01-29 01:32:39 -08:00
}
2010-09-23 20:28:15 -07:00
if ( aFocus ) {
2014-02-25 16:48:02 -08:00
IMENotification notification ( NOTIFY_IME_OF_SELECTION_CHANGE ) ;
notification . mSelectionChangeData . mCausedByComposition = false ;
NotifyIMEOfSelectionChange ( notification ) ; // Update selection
2010-10-01 07:17:37 -07:00
} else {
mIMELastBlurSeqno = chromeSeqno ;
2010-09-23 20:28:15 -07:00
}
return NS_OK ;
}
2013-11-06 16:11:11 -08:00
nsresult
PuppetWidget : : NotifyIMEOfUpdateComposition ( )
{
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
NS_ENSURE_TRUE ( mTabChild , NS_ERROR_FAILURE ) ;
2014-01-28 00:19:29 -08:00
nsRefPtr < TextComposition > textComposition =
2014-03-07 17:20:07 -08:00
IMEStateManager : : GetTextCompositionFor ( this ) ;
2013-11-06 16:11:11 -08:00
NS_ENSURE_TRUE ( textComposition , NS_ERROR_FAILURE ) ;
nsEventStatus status ;
2014-09-15 00:11:08 -07:00
nsTArray < nsIntRect > textRectArray ( textComposition - > String ( ) . Length ( ) ) ;
uint32_t startOffset = textComposition - > NativeOffsetOfStartComposition ( ) ;
uint32_t endOffset = textComposition - > String ( ) . Length ( ) + startOffset ;
for ( uint32_t i = startOffset ; i < endOffset ; i + + ) {
WidgetQueryContentEvent textRect ( true , NS_QUERY_TEXT_RECT , this ) ;
InitEvent ( textRect , nullptr ) ;
textRect . InitForQueryTextRect ( i , 1 ) ;
DispatchEvent ( & textRect , status ) ;
NS_ENSURE_TRUE ( textRect . mSucceeded , NS_ERROR_FAILURE ) ;
2013-11-06 16:11:11 -08:00
2014-09-15 00:11:08 -07:00
textRectArray . AppendElement ( textRect . mReply . mRect ) ;
}
uint32_t targetCauseOffset = textComposition - > OffsetOfTargetClause ( ) ;
2014-01-16 02:04:39 -08:00
WidgetQueryContentEvent caretRect ( true , NS_QUERY_CARET_RECT , this ) ;
InitEvent ( caretRect , nullptr ) ;
2014-09-15 00:11:08 -07:00
caretRect . InitForQueryCaretRect ( targetCauseOffset ) ;
2014-01-16 02:04:39 -08:00
DispatchEvent ( & caretRect , status ) ;
NS_ENSURE_TRUE ( caretRect . mSucceeded , NS_ERROR_FAILURE ) ;
2014-09-15 00:11:08 -07:00
mTabChild - > SendNotifyIMESelectedCompositionRect ( startOffset ,
textRectArray ,
targetCauseOffset ,
2014-01-16 02:04:39 -08:00
caretRect . mReply . mRect ) ;
2013-11-06 16:11:11 -08:00
return NS_OK ;
}
2012-11-13 05:04:44 -08:00
nsIMEUpdatePreference
PuppetWidget : : GetIMEUpdatePreference ( )
{
2014-02-24 21:51:01 -08:00
# ifdef MOZ_CROSS_PROCESS_IME
2014-01-29 01:32:39 -08: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 05:04:44 -08:00
}
2014-02-17 16:00:15 -08:00
nsresult
PuppetWidget : : NotifyIMEOfTextChange ( const IMENotification & aIMENotification )
2010-09-23 20:28:15 -07:00
{
2014-02-25 16:48:02 -08:00
MOZ_ASSERT ( aIMENotification . mMessage = = NOTIFY_IME_OF_TEXT_CHANGE ,
" Passed wrong notification " ) ;
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-23 20:28:15 -07:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
2014-01-29 01:32:35 -08:00
nsEventStatus status ;
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_TEXT_CONTENT , this ) ;
InitEvent ( queryEvent , nullptr ) ;
queryEvent . InitForQueryTextContent ( 0 , UINT32_MAX ) ;
DispatchEvent ( & queryEvent , status ) ;
2010-09-23 20:28:15 -07:00
2014-01-29 01:32:35 -08:00
if ( queryEvent . mSucceeded ) {
mTabChild - > SendNotifyIMETextHint ( queryEvent . mReply . mString ) ;
2010-09-23 20:28:15 -07:00
}
2014-01-29 01:32:39 -08: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-25 16:48:02 -08:00
if ( mIMEPreferenceOfParent . WantTextChange ( ) & &
( mIMEPreferenceOfParent . WantChangesCausedByComposition ( ) | |
! aIMENotification . mTextChangeData . mCausedByComposition ) ) {
2014-02-17 16:00:15 -08:00
mTabChild - > SendNotifyIMETextChange (
aIMENotification . mTextChangeData . mStartOffset ,
aIMENotification . mTextChangeData . mOldEndOffset ,
2014-02-25 16:48:02 -08:00
aIMENotification . mTextChangeData . mNewEndOffset ,
aIMENotification . mTextChangeData . mCausedByComposition ) ;
2010-09-23 20:28:15 -07:00
}
return NS_OK ;
}
2013-03-05 22:14:31 -08:00
nsresult
2014-02-25 16:48:02 -08:00
PuppetWidget : : NotifyIMEOfSelectionChange (
const IMENotification & aIMENotification )
2010-09-23 20:28:15 -07:00
{
2014-02-25 16:48:02 -08:00
MOZ_ASSERT ( aIMENotification . mMessage = = NOTIFY_IME_OF_SELECTION_CHANGE ,
" Passed wrong notification " ) ;
2013-01-28 15:56:28 -08:00
# ifndef MOZ_CROSS_PROCESS_IME
return NS_OK ;
# endif
2010-09-23 20:28:15 -07:00
if ( ! mTabChild )
return NS_ERROR_FAILURE ;
2014-01-29 01:32:39 -08:00
nsEventStatus status ;
WidgetQueryContentEvent queryEvent ( true , NS_QUERY_SELECTED_TEXT , this ) ;
InitEvent ( queryEvent , nullptr ) ;
DispatchEvent ( & queryEvent , status ) ;
2010-09-23 20:28:15 -07:00
2014-01-29 01:32:39 -08:00
if ( queryEvent . mSucceeded ) {
2014-02-25 16:48:02 -08:00
mTabChild - > SendNotifyIMESelection (
mIMELastReceivedSeqno ,
queryEvent . GetSelectionStart ( ) ,
queryEvent . GetSelectionEnd ( ) ,
2014-12-04 18:15:43 -08:00
queryEvent . GetWritingMode ( ) ,
2014-02-25 16:48:02 -08:00
aIMENotification . mSelectionChangeData . mCausedByComposition ) ;
2010-09-23 20:28:15 -07:00
}
return NS_OK ;
2014-09-11 06:46:17 -07:00
}
nsresult
PuppetWidget : : NotifyIMEOfMouseButtonEvent (
const IMENotification & aIMENotification )
{
if ( ! mTabChild ) {
return NS_ERROR_FAILURE ;
}
bool consumedByIME = false ;
if ( ! mTabChild - > SendNotifyIMEMouseButtonEvent ( aIMENotification ,
& consumedByIME ) ) {
return NS_ERROR_FAILURE ;
}
return consumedByIME ? NS_SUCCESS_EVENT_CONSUMED : NS_OK ;
2010-09-23 20:28:15 -07:00
}
2011-06-21 17:32:43 -07:00
NS_IMETHODIMP
PuppetWidget : : SetCursor ( nsCursor aCursor )
{
2014-05-27 18:12:29 -07:00
if ( mCursor = = aCursor & & ! mUpdateCursor ) {
2012-09-11 21:48:13 -07:00
return NS_OK ;
}
2014-05-27 18:12:29 -07:00
if ( mTabChild & &
! mTabChild - > SendSetCursor ( aCursor , mUpdateCursor ) ) {
2011-06-21 17:32:43 -07:00
return NS_ERROR_FAILURE ;
}
2012-09-11 21:48:13 -07:00
mCursor = aCursor ;
2014-05-27 18:12:29 -07:00
mUpdateCursor = false ;
2012-09-11 21:48:13 -07:00
2011-06-21 17:32:43 -07:00
return NS_OK ;
}
2010-08-20 16:24:40 -07:00
nsresult
2012-08-15 11:52:42 -07:00
PuppetWidget : : Paint ( )
2010-08-20 16:24:40 -07:00
{
NS_ABORT_IF_FALSE ( ! mDirtyRegion . IsEmpty ( ) , " paint event logic messed up " ) ;
2012-08-15 11:53:14 -07:00
if ( ! mAttachedWidgetListener )
2012-08-15 11:52:42 -07:00
return NS_OK ;
nsIntRegion region = mDirtyRegion ;
2010-08-20 16:24:40 -07:00
// reset repaint tracking
mDirtyRegion . SetEmpty ( ) ;
mPaintTask . Revoke ( ) ;
2013-01-28 11:34:08 -08:00
mAttachedWidgetListener - > WillPaintWindow ( this ) ;
2013-01-28 11:34:03 -08:00
if ( mAttachedWidgetListener ) {
2010-08-20 16:24:40 -07:00
# ifdef DEBUG
2012-08-15 11:52:42 -07:00
debug_DumpPaintEvent ( stderr , this , region ,
2012-09-01 19:35:17 -07:00
nsAutoCString ( " PuppetWidget " ) , 0 ) ;
2010-08-20 16:24:40 -07:00
# endif
2014-01-23 10:26:41 -08:00
if ( mozilla : : layers : : LayersBackend : : LAYERS_D3D10 = = mLayerManager - > GetBackendType ( ) ) {
2013-05-23 07:49:18 -07:00
mAttachedWidgetListener - > PaintWindow ( this , region ) ;
2014-01-23 10:26:41 -08:00
} else if ( mozilla : : layers : : LayersBackend : : LAYERS_CLIENT = = mLayerManager - > GetBackendType ( ) ) {
2013-04-30 22:03:25 -07:00
// Do nothing, the compositor will handle drawing
if ( mTabChild ) {
mTabChild - > NotifyPainted ( ) ;
}
2011-08-09 12:38:27 -07:00
} else {
2014-06-09 23:02:21 -07:00
nsRefPtr < gfxContext > ctx = new gfxContext ( mDrawTarget ) ;
2012-02-10 11:22:21 -08:00
ctx - > Rectangle ( gfxRect ( 0 , 0 , 0 , 0 ) ) ;
ctx - > Clip ( ) ;
2011-08-09 12:38:27 -07:00
AutoLayerManagerSetup setupLayerManager ( this , ctx ,
2014-01-23 10:26:41 -08:00
BufferMode : : BUFFER_NONE ) ;
2013-05-23 07:49:18 -07:00
mAttachedWidgetListener - > PaintWindow ( this , region ) ;
2013-04-02 11:32:59 -07:00
if ( mTabChild ) {
mTabChild - > NotifyPainted ( ) ;
}
2011-08-09 12:38:27 -07:00
}
2010-08-20 16:24:40 -07:00
}
2012-12-12 13:57:08 -08:00
if ( mAttachedWidgetListener ) {
mAttachedWidgetListener - > DidPaintWindow ( ) ;
}
2010-08-20 16:24:40 -07: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 11:52:42 -07:00
mWidget - > Paint ( ) ;
2010-08-20 16:24:40 -07:00
}
return NS_OK ;
}
2012-11-07 19:51:55 -08:00
bool
PuppetWidget : : NeedsPaint ( )
{
2014-11-20 09:28:58 -08:00
// e10s popups are handled by the parent process, so never should be painted here
if ( XRE_GetProcessType ( ) = = GeckoProcessType_Content & &
Preferences : : GetBool ( " browser.tabs.remote.desktopbehavior " , false ) & &
mWindowType = = eWindowType_popup ) {
NS_WARNING ( " Trying to paint an e10s popup in the child process! " ) ;
return false ;
}
2012-11-07 19:51:55 -08:00
return mVisible ;
}
2010-12-02 17:24:04 -08:00
float
PuppetWidget : : GetDPI ( )
{
if ( mDPI < 0 ) {
2013-04-02 11:32:59 -07:00
if ( mTabChild ) {
mTabChild - > GetDPI ( & mDPI ) ;
} else {
mDPI = 96.0 ;
}
2010-12-02 17:24:04 -08:00
}
return mDPI ;
}
2013-05-01 16:06:19 -07:00
double
PuppetWidget : : GetDefaultScaleInternal ( )
{
if ( mDefaultScale < 0 ) {
if ( mTabChild ) {
mTabChild - > GetDefaultScale ( & mDefaultScale ) ;
} else {
mDefaultScale = 1 ;
}
}
return mDefaultScale ;
}
2011-08-31 12:01:38 -07:00
void *
2012-08-22 08:56:38 -07:00
PuppetWidget : : GetNativeData ( uint32_t aDataType )
2011-08-31 12:01:38 -07:00
{
2012-05-08 14:36:07 -07:00
switch ( aDataType ) {
case NS_NATIVE_SHAREABLE_WINDOW : {
NS_ABORT_IF_FALSE ( mTabChild , " Need TabChild to get the nativeWindow from! " ) ;
2012-07-20 04:16:17 -07:00
mozilla : : WindowsHandle nativeData = 0 ;
2013-04-02 11:32:59 -07:00
if ( mTabChild ) {
mTabChild - > SendGetWidgetNativeData ( & nativeData ) ;
}
2012-05-08 14:36:07 -07: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 07:20:58 -07:00
return nullptr ;
2012-05-08 14:36:07 -07:00
}
PuppetScreen : : PuppetScreen ( void * nativeScreen )
{
}
PuppetScreen : : ~ PuppetScreen ( )
{
}
static ScreenConfiguration
ScreenConfig ( )
{
ScreenConfiguration config ;
hal : : GetCurrentScreenConfiguration ( & config ) ;
return config ;
}
2014-07-14 10:22:26 -07:00
NS_IMETHODIMP
PuppetScreen : : GetId ( uint32_t * outId )
{
* outId = 1 ;
return NS_OK ;
}
2012-05-08 14:36:07 -07:00
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreen : : GetRect ( int32_t * outLeft , int32_t * outTop ,
int32_t * outWidth , int32_t * outHeight )
2012-05-08 14:36:07 -07: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 08:56:38 -07:00
PuppetScreen : : GetAvailRect ( int32_t * outLeft , int32_t * outTop ,
int32_t * outWidth , int32_t * outHeight )
2012-05-08 14:36:07 -07:00
{
return GetRect ( outLeft , outTop , outWidth , outHeight ) ;
}
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreen : : GetPixelDepth ( int32_t * aPixelDepth )
2012-05-08 14:36:07 -07:00
{
* aPixelDepth = ScreenConfig ( ) . pixelDepth ( ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreen : : GetColorDepth ( int32_t * aColorDepth )
2012-05-08 14:36:07 -07:00
{
* aColorDepth = ScreenConfig ( ) . colorDepth ( ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreen : : GetRotation ( uint32_t * aRotation )
2012-05-08 14:36:07 -07: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 08:56:38 -07:00
PuppetScreen : : SetRotation ( uint32_t aRotation )
2012-05-08 14:36:07 -07: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 00:06:00 -07:00
NS_IMPL_ISUPPORTS ( PuppetScreenManager , nsIScreenManager )
2012-05-08 14:36:07 -07:00
PuppetScreenManager : : PuppetScreenManager ( )
{
2012-07-30 07:20:58 -07:00
mOneScreen = new PuppetScreen ( nullptr ) ;
2012-05-08 14:36:07 -07:00
}
PuppetScreenManager : : ~ PuppetScreenManager ( )
{
}
2014-07-14 10:22:26 -07:00
NS_IMETHODIMP
PuppetScreenManager : : ScreenForId ( uint32_t aId ,
nsIScreen * * outScreen )
{
NS_IF_ADDREF ( * outScreen = mOneScreen . get ( ) ) ;
return NS_OK ;
}
2012-05-08 14:36:07 -07:00
NS_IMETHODIMP
PuppetScreenManager : : GetPrimaryScreen ( nsIScreen * * outScreen )
{
NS_IF_ADDREF ( * outScreen = mOneScreen . get ( ) ) ;
return NS_OK ;
}
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreenManager : : ScreenForRect ( int32_t inLeft ,
int32_t inTop ,
int32_t inWidth ,
int32_t inHeight ,
2012-05-08 14:36:07 -07:00
nsIScreen * * outScreen )
{
return GetPrimaryScreen ( outScreen ) ;
}
NS_IMETHODIMP
PuppetScreenManager : : ScreenForNativeWidget ( void * aWidget ,
nsIScreen * * outScreen )
{
return GetPrimaryScreen ( outScreen ) ;
}
NS_IMETHODIMP
2012-08-22 08:56:38 -07:00
PuppetScreenManager : : GetNumberOfScreens ( uint32_t * aNumberOfScreens )
2012-05-08 14:36:07 -07:00
{
* aNumberOfScreens = 1 ;
return NS_OK ;
2011-08-31 12:01:38 -07:00
}
2013-04-09 14:07:02 -07:00
NS_IMETHODIMP
PuppetScreenManager : : GetSystemDefaultScale ( float * aDefaultScale )
{
* aDefaultScale = 1.0f ;
return NS_OK ;
}
2010-08-20 16:24:40 -07:00
} // namespace widget
} // namespace mozilla