os/graphics/windowing/windowserver/nonnga/SERVER/TCURSOR.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// The text cursor
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <e32std.h>
sl@0
    19
#include "server.h"
sl@0
    20
#include "tcursor.h"
sl@0
    21
#include "windowgroup.h"
sl@0
    22
#include "wstop.h"
sl@0
    23
#include "panics.h"
sl@0
    24
#include "offscreenbitmap.h"
sl@0
    25
#include "EVENT.H"
sl@0
    26
#include "graphics/windowserverconstants.h"
sl@0
    27
sl@0
    28
void RWsTextCursor::ConstructL(CWsWindowGroup *aGroupWin)
sl@0
    29
	{
sl@0
    30
	iInternalFlags = 0;
sl@0
    31
	iGroupWin=aGroupWin;
sl@0
    32
	iCustomTextCursor = NULL;
sl@0
    33
	}
sl@0
    34
sl@0
    35
void RWsTextCursor::Close()
sl@0
    36
	{
sl@0
    37
	iDrawRegion.Close();
sl@0
    38
	Cancel();
sl@0
    39
	}
sl@0
    40
sl@0
    41
void RWsTextCursor::SetL(const TWsWinCmdSetTextCursor &aSet, TBool aClipped)
sl@0
    42
	{
sl@0
    43
	if (aSet.cursor.iType < TTextCursor::ETypeFirst ||
sl@0
    44
        (aSet.cursor.iType > TTextCursor::ETypeLast &&        
sl@0
    45
         aSet.cursor.iType <= TTextCursor::ETypeLastBasic) ||
sl@0
    46
		(aSet.cursor.iFlags&static_cast<TUint>(ETextCursorPrivateFlags)))
sl@0
    47
		{
sl@0
    48
		Cancel();
sl@0
    49
		iGroupWin->OwnerPanic(EWservPanicInvalidTextCursor);
sl@0
    50
		}
sl@0
    51
	else
sl@0
    52
		{
sl@0
    53
		CWsClientWindow* win = NULL;
sl@0
    54
		iGroupWin->WsOwner()->HandleToClientWindow(aSet.window, &win);
sl@0
    55
sl@0
    56
		// Check window is a child of the group window
sl@0
    57
		CWsWindowBase* searchWin = NULL;
sl@0
    58
		for(searchWin=win; searchWin->WinType()!=EWinTypeGroup; searchWin=searchWin->BaseParent())
sl@0
    59
			{}
sl@0
    60
		if (iGroupWin != searchWin)
sl@0
    61
			{
sl@0
    62
			Cancel();
sl@0
    63
			iGroupWin->OwnerPanic(EWservPanicWindow);
sl@0
    64
			}
sl@0
    65
sl@0
    66
		TPoint pos(aSet.pos.iX, aSet.pos.iY-aSet.cursor.iAscent);
sl@0
    67
		TSize size(aSet.cursor.iWidth, aSet.cursor.iHeight);
sl@0
    68
		TUint flags = aSet.cursor.iFlags;
sl@0
    69
		TInt type = aSet.cursor.iType;
sl@0
    70
		TRect clipRect = iClipRect;
sl@0
    71
		TRgb color = aSet.cursor.iColor;
sl@0
    72
		CWsCustomTextCursor* customTextCursor = iCustomTextCursor;
sl@0
    73
		TBool changed = EFalse;
sl@0
    74
sl@0
    75
		TPoint clipOrigo;
sl@0
    76
		TSize clipSize;
sl@0
    77
sl@0
    78
		if (type > TTextCursor::ETypeLastBasic)
sl@0
    79
			{
sl@0
    80
			changed = ETrue;
sl@0
    81
sl@0
    82
			customTextCursor = CWsClient::FindCustomTextCursor(type);
sl@0
    83
			if (!customTextCursor)
sl@0
    84
				{
sl@0
    85
				Cancel();
sl@0
    86
				iGroupWin->OwnerPanic(EWservPanicNoCustomTextCursor);
sl@0
    87
				return;
sl@0
    88
				}
sl@0
    89
			
sl@0
    90
			if( !customTextCursor->HasSpriteMember() )
sl@0
    91
				{
sl@0
    92
				iGroupWin->OwnerPanic(EWservPanicNoSpriteMember);
sl@0
    93
				return;
sl@0
    94
				}
sl@0
    95
			
sl@0
    96
			TInt yAdjust=0;
sl@0
    97
			switch (customTextCursor->Alignment())
sl@0
    98
				{
sl@0
    99
				case RWsSession::ECustomTextCursorAlignTop:
sl@0
   100
					break;
sl@0
   101
				case RWsSession::ECustomTextCursorAlignBaseline:
sl@0
   102
					yAdjust = aSet.cursor.iAscent-1;
sl@0
   103
					break;
sl@0
   104
				case RWsSession::ECustomTextCursorAlignBottom:
sl@0
   105
					yAdjust = aSet.cursor.iHeight-1;
sl@0
   106
					break;
sl@0
   107
				default:
sl@0
   108
					Cancel();
sl@0
   109
					iGroupWin->OwnerPanic(EWservPanicCustomTextCursorAlign);
sl@0
   110
					return;
sl@0
   111
				}
sl@0
   112
			pos.iY += yAdjust;
sl@0
   113
			// Start with a clipping rect to be the whole window
sl@0
   114
			// relative cursor pos and shrink down to what we want
sl@0
   115
			clipOrigo = -pos;
sl@0
   116
			clipSize = win->Size();
sl@0
   117
			if (flags & TTextCursor::EFlagClipHorizontal)
sl@0
   118
				{
sl@0
   119
				clipOrigo.iX = 0;
sl@0
   120
				clipSize.iWidth = size.iWidth;
sl@0
   121
				}
sl@0
   122
			if (flags & TTextCursor::EFlagClipVertical)
sl@0
   123
				{
sl@0
   124
				clipOrigo.iY = -yAdjust;
sl@0
   125
				clipSize.iHeight = aSet.cursor.iHeight;
sl@0
   126
				}
sl@0
   127
			}
sl@0
   128
		else
sl@0
   129
			{
sl@0
   130
			customTextCursor = NULL;
sl@0
   131
			}
sl@0
   132
sl@0
   133
		if (aClipped)
sl@0
   134
			{
sl@0
   135
			flags|=ETextCursorFlagClipped;
sl@0
   136
			clipRect=aSet.rect;
sl@0
   137
			}
sl@0
   138
sl@0
   139
		if (pos != iPos || size != iSize || iType != type ||
sl@0
   140
			flags != iFlags || clipRect != iClipRect || color != iColor ||
sl@0
   141
			customTextCursor != iCustomTextCursor || win != iWin)
sl@0
   142
			{
sl@0
   143
			// There is a change in the cursor.
sl@0
   144
			changed = ETrue;
sl@0
   145
			}
sl@0
   146
sl@0
   147
		if (iInternalFlags&EHasFocus && changed)
sl@0
   148
			{
sl@0
   149
			TCursorSprite::Hide();
sl@0
   150
			}
sl@0
   151
sl@0
   152
		iPos = pos;
sl@0
   153
		iSize = size;
sl@0
   154
		iType = type;
sl@0
   155
		iFlags= flags;
sl@0
   156
		iClipRect = clipRect;
sl@0
   157
		iColor = color;
sl@0
   158
		iCustomTextCursor = customTextCursor;
sl@0
   159
		iWin = win;
sl@0
   160
		if (customTextCursor && iInternalFlags&EHasFocus)
sl@0
   161
			{
sl@0
   162
			customTextCursor->CompleteL(win, !(flags&TTextCursor::EFlagNoFlash), flags & (TTextCursor::EFlagClipHorizontal | TTextCursor::EFlagClipVertical), clipOrigo, clipSize);
sl@0
   163
			customTextCursor->SetPositionNoRedraw(pos);
sl@0
   164
			}
sl@0
   165
sl@0
   166
		if (iInternalFlags&EHasFocus && changed)
sl@0
   167
			{
sl@0
   168
			TCursorSprite::SetCurrentCursor(this, win);
sl@0
   169
			}
sl@0
   170
		}
sl@0
   171
	}
sl@0
   172
sl@0
   173
void RWsTextCursor::Cancel()
sl@0
   174
	{
sl@0
   175
	if (iType!=TTextCursor::ETypeNone)
sl@0
   176
		{
sl@0
   177
		if (iInternalFlags&EHasFocus)
sl@0
   178
			TCursorSprite::SetFocus(NULL);
sl@0
   179
		iType=TTextCursor::ETypeNone;
sl@0
   180
		iWin=NULL;
sl@0
   181
		}
sl@0
   182
	}
sl@0
   183
sl@0
   184
void RWsTextCursor::Disable()
sl@0
   185
	{
sl@0
   186
	if (iWin)
sl@0
   187
		{
sl@0
   188
		TCursorSprite::Hide();
sl@0
   189
		}
sl@0
   190
	}
sl@0
   191
sl@0
   192
void RWsTextCursor::Enable()
sl@0
   193
	{
sl@0
   194
	if (iWin)
sl@0
   195
		{
sl@0
   196
		TCursorSprite::Reveal();
sl@0
   197
		}
sl@0
   198
	}
sl@0
   199
sl@0
   200
void RWsTextCursor::LostFocus()
sl@0
   201
	{
sl@0
   202
	TCursorSprite::SetFocus(NULL);
sl@0
   203
	iInternalFlags &= ~EHasFocus;
sl@0
   204
	}
sl@0
   205
sl@0
   206
void RWsTextCursor::ReceivedFocus()
sl@0
   207
	{
sl@0
   208
	iInternalFlags |= EHasFocus;
sl@0
   209
	if (iType!=TTextCursor::ETypeNone && iWin)
sl@0
   210
		{
sl@0
   211
		TCursorSprite::SetFocus(this,iWin);
sl@0
   212
		if (iCustomTextCursor)
sl@0
   213
			{
sl@0
   214
			iCustomTextCursor->SetPositionNoRedraw(iPos);
sl@0
   215
			}
sl@0
   216
		}
sl@0
   217
	}
sl@0
   218
sl@0
   219
TRect RWsTextCursor::RectRelativeToScreen() const
sl@0
   220
	{
sl@0
   221
	TRect rect;
sl@0
   222
	rect.iTl=iPos+iWin->Origin();
sl@0
   223
	rect.iBr=rect.iTl+iSize;
sl@0
   224
	return(rect);
sl@0
   225
	}
sl@0
   226
sl@0
   227
void RWsTextCursor::doDraw(CFbsBitGc* aGc, const TRegion& aRegion)
sl@0
   228
	{
sl@0
   229
	TRegionFix<1> justInCase;
sl@0
   230
	const TRegion *pr= &aRegion;
sl@0
   231
	if (aRegion.CheckError())
sl@0
   232
		{
sl@0
   233
		justInCase.AddRect(iWin->AbsRect());
sl@0
   234
		pr= &justInCase;
sl@0
   235
		}
sl@0
   236
	if (!pr->IsEmpty())
sl@0
   237
		{
sl@0
   238
		aGc->SetUserDisplayMode(iWin->DisplayMode());
sl@0
   239
		aGc->SetDitherOrigin(iWin->Origin());
sl@0
   240
		aGc->SetDrawMode(CGraphicsContext::EDrawModeXOR);
sl@0
   241
		switch (iType)
sl@0
   242
			{
sl@0
   243
			case TTextCursor::ETypeRectangle:
sl@0
   244
				{
sl@0
   245
				aGc->SetBrushStyle(CGraphicsContext::ESolidBrush);
sl@0
   246
				aGc->SetPenStyle(CGraphicsContext::ENullPen);
sl@0
   247
				aGc->SetBrushColor(iColor);
sl@0
   248
				}
sl@0
   249
				break;
sl@0
   250
			case TTextCursor::ETypeHollowRectangle:
sl@0
   251
				{
sl@0
   252
				aGc->SetBrushStyle(CGraphicsContext::ENullBrush);
sl@0
   253
				aGc->SetPenStyle(CGraphicsContext::ESolidPen);
sl@0
   254
				aGc->SetPenColor(iColor);
sl@0
   255
				}
sl@0
   256
				break;
sl@0
   257
			default:
sl@0
   258
				WS_PANIC_ALWAYS(EWsPanicInvalidCursorType);
sl@0
   259
			}
sl@0
   260
		aGc->SetClippingRegion(pr);
sl@0
   261
		aGc->DrawRect(RectRelativeToScreen());
sl@0
   262
		aGc->SetUserDisplayMode(ENone);
sl@0
   263
sl@0
   264
		TWindowServerEvent::NotifyScreenDrawingEvent(pr);
sl@0
   265
		}
sl@0
   266
	}
sl@0
   267
sl@0
   268
void RWsTextCursor::Draw(CFbsBitGc* aGc, const TRegion& aRegion)
sl@0
   269
	{
sl@0
   270
	iDrawRegion.Copy(iWin->VisibleRegion());
sl@0
   271
	if (iFlags&ETextCursorFlagClipped)
sl@0
   272
		{
sl@0
   273
		TRect rect(iClipRect);
sl@0
   274
		rect.Move(iWin->Origin());
sl@0
   275
		iDrawRegion.ClipRect(rect);
sl@0
   276
		}
sl@0
   277
sl@0
   278
	// Need to clip against a possible recent screen size change.
sl@0
   279
	iDrawRegion.ClipRect(iWin->Screen()->DrawDevice()->SizeInPixels());
sl@0
   280
sl@0
   281
sl@0
   282
	RWsRegion tmpRegion;
sl@0
   283
	tmpRegion.Intersection(iDrawRegion, aRegion);
sl@0
   284
	if (tmpRegion.CheckError())
sl@0
   285
		doDraw(aGc, iDrawRegion);
sl@0
   286
	else
sl@0
   287
		{
sl@0
   288
		if (!tmpRegion.IsEmpty())
sl@0
   289
			{
sl@0
   290
			doDraw(aGc, tmpRegion);
sl@0
   291
			}
sl@0
   292
		}
sl@0
   293
	tmpRegion.Close();
sl@0
   294
	}
sl@0
   295
sl@0
   296
void RWsTextCursor::WindowDisconnected(CWsWindow *aWindow)
sl@0
   297
	{
sl@0
   298
	if (iWin==aWindow)
sl@0
   299
		Cancel();
sl@0
   300
	}
sl@0
   301
sl@0
   302
TBool RWsTextCursor::IsStandardCursorActive()
sl@0
   303
	{
sl@0
   304
	return TCursorSprite::IsStandardCursorActive();
sl@0
   305
	}
sl@0
   306
sl@0
   307
TBool RWsTextCursor::IsFlashing() const
sl@0
   308
	{
sl@0
   309
	return !(iFlags&TTextCursor::EFlagNoFlash);
sl@0
   310
	}
sl@0
   311
sl@0
   312
void RWsTextCursor::ScheduleReDrawNow()
sl@0
   313
	{
sl@0
   314
	iGroupWin->Screen()->ScheduleAnimation(RectRelativeToScreen(), 0, 0, 0);
sl@0
   315
	}
sl@0
   316
sl@0
   317
sl@0
   318
// Cursor sprite handling
sl@0
   319
sl@0
   320
TBool TCursorSprite::iHidden=ETrue;
sl@0
   321
RWsTextCursor *TCursorSprite::iCurrentCursor=NULL;
sl@0
   322
sl@0
   323
//
sl@0
   324
sl@0
   325
// Hide / Reveal text cursors.
sl@0
   326
void TCursorSprite::Hide()
sl@0
   327
	{
sl@0
   328
	if (!iHidden && iCurrentCursor)
sl@0
   329
		{
sl@0
   330
		iHidden=ETrue;
sl@0
   331
		if (iCurrentCursor->iCustomTextCursor)
sl@0
   332
			{
sl@0
   333
			iCurrentCursor->iCustomTextCursor->Deactivate();
sl@0
   334
			}
sl@0
   335
		else
sl@0
   336
			{
sl@0
   337
			iCurrentCursor->ScheduleReDrawNow();
sl@0
   338
			}
sl@0
   339
		}
sl@0
   340
	}
sl@0
   341
	
sl@0
   342
void TCursorSprite::Reveal()
sl@0
   343
	{
sl@0
   344
	if(iHidden && iCurrentCursor)
sl@0
   345
		{
sl@0
   346
		iHidden=EFalse;
sl@0
   347
		if (iCurrentCursor->iCustomTextCursor)
sl@0
   348
			{
sl@0
   349
			iCurrentCursor->iCustomTextCursor->Activate();
sl@0
   350
			}
sl@0
   351
		else
sl@0
   352
			{
sl@0
   353
			iCurrentCursor->ScheduleReDrawNow();
sl@0
   354
			}
sl@0
   355
		}
sl@0
   356
	}
sl@0
   357
sl@0
   358
void TCursorSprite::SetFocus(RWsTextCursor* aFocus,CWsClientWindow* aWin/*=NULL*/)
sl@0
   359
	{
sl@0
   360
	if (iCurrentCursor!=aFocus)
sl@0
   361
		{
sl@0
   362
		Hide();
sl@0
   363
		SetCurrentCursor(aFocus, aWin);
sl@0
   364
		}
sl@0
   365
	}
sl@0
   366
sl@0
   367
void TCursorSprite::SetCurrentCursor(RWsTextCursor* aFocus, CWsClientWindow* aWin)
sl@0
   368
	{
sl@0
   369
	iCurrentCursor = aFocus;
sl@0
   370
	if (aWin && iCurrentCursor && iCurrentCursor->iCustomTextCursor)
sl@0
   371
		{
sl@0
   372
		iCurrentCursor->iCustomTextCursor->SetWindow(aWin);
sl@0
   373
		}
sl@0
   374
	Reveal();
sl@0
   375
	}
sl@0
   376
sl@0
   377
TBool TCursorSprite::IsStandardCursorActive()
sl@0
   378
	{
sl@0
   379
	return iCurrentCursor && !iCurrentCursor->iCustomTextCursor && !iHidden;
sl@0
   380
	}
sl@0
   381