Update contrib.
1 // Copyright (c) 1995-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 #include <graphics/wscursor.h>
22 #include "windowgroup.h"
26 #include "graphics/windowserverconstants.h"
28 static const TInt64 KFlashRate(500000); // duration cursor is ON or OFF
30 void RWsTextCursor::ConstructL(CWsWindowGroup *aGroupWin)
34 iCustomTextCursor = NULL;
37 void RWsTextCursor::Close()
43 void RWsTextCursor::SetL(const TWsWinCmdSetTextCursor &aSet, TBool aClipped)
45 if (aSet.cursor.iType < TTextCursor::ETypeFirst ||
46 (aSet.cursor.iType > TTextCursor::ETypeLast &&
47 aSet.cursor.iType <= TTextCursor::ETypeLastBasic) ||
48 (aSet.cursor.iFlags&static_cast<TUint>(ETextCursorPrivateFlags)))
51 iGroupWin->OwnerPanic(EWservPanicInvalidTextCursor);
55 CWsClientWindow* win = NULL;
56 iGroupWin->WsOwner()->HandleToClientWindow(aSet.window, &win);
58 // Check window is a child of the group window
59 CWsWindowBase* searchWin = NULL;
60 for(searchWin=win; searchWin->WinType()!=EWinTypeGroup; searchWin=searchWin->BaseParent())
62 if (iGroupWin != searchWin)
65 iGroupWin->OwnerPanic(EWservPanicWindow);
68 TPoint pos(aSet.pos.iX, aSet.pos.iY-aSet.cursor.iAscent);
69 TSize size(aSet.cursor.iWidth, aSet.cursor.iHeight);
70 TUint flags = aSet.cursor.iFlags;
71 TInt type = aSet.cursor.iType;
72 TRect clipRect = iClipRect;
73 TRgb color = aSet.cursor.iColor;
74 CWsCustomTextCursor* customTextCursor = iCustomTextCursor;
75 TBool changed = EFalse;
80 if (type > TTextCursor::ETypeLastBasic)
84 customTextCursor = CWsClient::FindCustomTextCursor(type);
85 if (!customTextCursor)
88 iGroupWin->OwnerPanic(EWservPanicNoCustomTextCursor);
92 if( !customTextCursor->HasSpriteMember() )
94 iGroupWin->OwnerPanic(EWservPanicNoSpriteMember);
99 switch (customTextCursor->Alignment())
101 case RWsSession::ECustomTextCursorAlignTop:
103 case RWsSession::ECustomTextCursorAlignBaseline:
104 yAdjust = aSet.cursor.iAscent-1;
106 case RWsSession::ECustomTextCursorAlignBottom:
107 yAdjust = aSet.cursor.iHeight-1;
111 iGroupWin->OwnerPanic(EWservPanicCustomTextCursorAlign);
115 // Start with a clipping rect to be the whole window
116 // relative cursor pos and shrink down to what we want
118 clipSize = win->Size();
119 if (flags & TTextCursor::EFlagClipHorizontal)
122 clipSize.iWidth = size.iWidth;
124 if (flags & TTextCursor::EFlagClipVertical)
126 clipOrigo.iY = -yAdjust;
127 clipSize.iHeight = aSet.cursor.iHeight;
132 customTextCursor = NULL;
137 flags|=ETextCursorFlagClipped;
141 TPoint absPos(pos.iX,pos.iY);
142 absPos=absPos+win->Origin();
143 if (pos != iPos || absPos != iAbsPos || size != iSize || iType != type ||
144 flags != iFlags || clipRect != iClipRect || color != iColor ||
145 customTextCursor != iCustomTextCursor || win != iWin)
147 // There is a change in the cursor.
151 if (iInternalFlags&EHasFocus && changed)
153 if ((win != iWin && !iCustomTextCursor) || (customTextCursor && !iCustomTextCursor))
155 TCursorSprite::Hide();
158 UpdateAttributes(pos, absPos, size, type, flags, clipRect, color, customTextCursor, win);
160 if (customTextCursor && iInternalFlags&EHasFocus)
162 customTextCursor->CompleteL(win, !(flags&TTextCursor::EFlagNoFlash), flags & (TTextCursor::EFlagClipHorizontal | TTextCursor::EFlagClipVertical), clipOrigo, clipSize);
163 customTextCursor->SetPositionNoRedraw(pos);
166 if (iInternalFlags&EHasFocus && changed)
168 TCursorSprite::SetCurrentCursor(this, win);
172 void RWsTextCursor::UpdateAttributes(TPoint aPos, TPoint aAbsPos, TSize aSize, TInt aType, TUint aFlags, TRect aClipRect, TRgb aColor, CWsCustomTextCursor* aCustomTextCursor, CWsClientWindow* aWin)
174 if (aPos != iPos || aSize != iSize || aAbsPos != iAbsPos)
179 WS_ASSERT_DEBUG(iGroupWin->Screen(),EWsPanicNoScreen);
180 MWsWindowTreeObserver* const windowTreeObserver = iGroupWin->Screen()->WindowTreeObserver();
181 if (windowTreeObserver && iInternalFlags&EHasFocus && iInternalFlags&EActiveNode)
182 windowTreeObserver->NodeExtentChanged(*this, RectRelativeToScreen());
188 NotifyObserver(MWsWindowTreeObserver::ECursorType);
191 if (aClipRect != iClipRect)
193 iClipRect = aClipRect; // must update clip rect before sending clip rect set/unset notification
194 if ((aFlags&ETextCursorFlagClipped) && (iFlags&ETextCursorFlagClipped))
195 NotifyObserver(MWsWindowTreeObserver::ECursorClipRect); // clip rect changed
198 if (aFlags != iFlags)
200 TBool sendFlagChanged = EFalse;
201 if ((aFlags&ETextCursorFlagClipped) != (iFlags&ETextCursorFlagClipped))
203 if (iInternalFlags&EHasFocus && iInternalFlags&EActiveNode)
205 // We can't send flag changed till iFlags has been updated, as otherwise plugins responding to
206 // the flag changed notification by calling ClipRect() may get the wrong rect
207 sendFlagChanged = ETrue;
210 const TBool userFlagsChanged((aFlags&ETextCursorUserFlags) != (iFlags&ETextCursorUserFlags));
212 if (userFlagsChanged)
213 NotifyObserver(MWsWindowTreeObserver::ECursorFlags);
216 WS_ASSERT_DEBUG(iGroupWin->Screen(),EWsPanicNoScreen);
217 MWsWindowTreeObserver* const windowTreeObserver = iGroupWin->Screen()->WindowTreeObserver();
218 if (windowTreeObserver)
219 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::ECursorClipRectSet, !!(iFlags&ETextCursorFlagClipped)); // clip rect set/unset
223 if (aColor != iColor)
226 NotifyObserver(MWsWindowTreeObserver::ECursorColor);
228 iCustomTextCursor = aCustomTextCursor;
232 void RWsTextCursor::NotifyObserver(MWsWindowTreeObserver::TAttributes aAttribute) const
234 if (iInternalFlags&EHasFocus && iInternalFlags&EActiveNode)
236 WS_ASSERT_DEBUG(iGroupWin->Screen(),EWsPanicNoScreen);
237 MWsWindowTreeObserver* const windowTreeObserver = iGroupWin->Screen()->WindowTreeObserver();
238 if (windowTreeObserver)
239 windowTreeObserver->AttributeChanged(*this, aAttribute);
243 void RWsTextCursor::CreateNode()
245 WS_ASSERT_DEBUG(iGroupWin->Screen(),EWsPanicNoScreen);
246 MWsWindowTreeObserver* const windowTreeObserver = iGroupWin->Screen()->WindowTreeObserver();
247 if (windowTreeObserver && !(iInternalFlags&EActiveNode))
249 iInternalFlags |= EActiveNode;
250 windowTreeObserver->NodeCreated(*this, iWin);
251 windowTreeObserver->NodeExtentChanged(*this, RectRelativeToScreen());
252 if (iFlags&ETextCursorFlagClipped)
253 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::ECursorClipRectSet, ETrue);
254 windowTreeObserver->NodeActivated(*this);
258 void RWsTextCursor::ReleaseNode()
260 if (iInternalFlags&EActiveNode)
262 WS_ASSERT_DEBUG(iGroupWin->Screen(),EWsPanicNoScreen);
263 MWsWindowTreeObserver* const windowTreeObserver = iGroupWin->Screen()->WindowTreeObserver();
264 if (windowTreeObserver)
266 windowTreeObserver->NodeReleased(*this);
267 iInternalFlags &= ~EActiveNode;
272 void RWsTextCursor::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
274 if (iInternalFlags & EActiveNode)
276 aWindowTreeObserver.NodeCreated(*this, iWin);
277 aWindowTreeObserver.NodeExtentChanged(*this, RectRelativeToScreen());
278 if (iFlags&ETextCursorFlagClipped)
279 aWindowTreeObserver.FlagChanged(*this, MWsWindowTreeObserver::ECursorClipRectSet, ETrue);
280 aWindowTreeObserver.NodeActivated(*this);
284 void RWsTextCursor::Cancel()
286 if (iType!=TTextCursor::ETypeNone)
288 if (iInternalFlags&EHasFocus)
289 TCursorSprite::SetFocus(NULL);
290 iType=TTextCursor::ETypeNone;
295 void RWsTextCursor::Disable()
299 TCursorSprite::Hide();
303 void RWsTextCursor::Enable()
307 TCursorSprite::Reveal();
311 void RWsTextCursor::LostFocus()
313 TCursorSprite::SetFocus(NULL);
314 iInternalFlags &= ~EHasFocus;
317 void RWsTextCursor::ReceivedFocus()
319 iInternalFlags |= EHasFocus;
320 if (iType!=TTextCursor::ETypeNone && iWin)
322 TCursorSprite::SetFocus(this,iWin);
323 if (iCustomTextCursor)
325 iCustomTextCursor->SetPositionNoRedraw(iPos);
330 TRect RWsTextCursor::RectRelativeToScreen() const
333 rect.iTl=iPos+iWin->Origin();
334 rect.iBr=rect.iTl+iSize;
338 TRect RWsTextCursor::RectRelativeToWindow() const
342 rect.iBr=rect.iTl+iSize;
346 void RWsTextCursor::doDraw(const TRegion& aRegion)
348 TRegionFix<1> fallbackClipRegion;
349 const TRegion *clipRegion= &aRegion;
350 if (aRegion.CheckError())
352 fallbackClipRegion.AddRect(iWin->AbsRect());
353 clipRegion= &fallbackClipRegion;
356 if(!clipRegion->IsEmpty())
358 MWsTextCursor::TTextCursorInfo renderStageCursorInfo(
359 RectRelativeToWindow(),
361 iType, static_cast<MWsWindow *>(Win()), iColor
364 MWsTextCursor* textCursor = iWin->Screen()->RenderStageTextCursor();
366 textCursor->DrawTextCursor(renderStageCursorInfo);
368 TWindowServerEvent::NotifyScreenDrawingEvent(clipRegion);
372 void RWsTextCursor::Draw(const TRegion& aRegion)
374 iDrawRegion.Copy(iWin->VisibleRegion());
375 if (iFlags&ETextCursorFlagClipped)
377 TRect rect(iClipRect);
378 rect.Move(iWin->Origin());
379 iDrawRegion.ClipRect(rect);
382 // Need to clip against a possible recent screen size change.
383 iDrawRegion.ClipRect(iWin->Screen()->SizeInPixels());
386 tmpRegion.Intersection(iDrawRegion, aRegion);
387 if (tmpRegion.CheckError())
391 if (!tmpRegion.IsEmpty())
399 void RWsTextCursor::WindowDisconnected(CWsWindow *aWindow)
405 TBool RWsTextCursor::IsStandardCursorActive()
407 return TCursorSprite::IsStandardCursorActive();
410 TBool RWsTextCursor::IsFlashing() const
412 return !(iFlags&TTextCursor::EFlagNoFlash);
415 void RWsTextCursor::ScheduleReDrawNow()
417 if (!iGroupWin->Screen()->ChangeTracking())
418 iGroupWin->Screen()->ScheduleAnimation(ETextCursor, RectRelativeToScreen(), 0, 0, 0, iWin);
421 /** @see MWsWindowTreeNode */
422 MWsWindowTreeNode::TType RWsTextCursor::NodeType() const
424 return MWsWindowTreeNode::EWinTreeNodeStandardTextCursor;
427 /** @see MWsWindowTreeNode */
428 const MWsWindow* RWsTextCursor::Window() const
433 /** @see MWsWindowTreeNode */
434 const MWsSprite* RWsTextCursor::Sprite() const
439 /** @see MWsWindowTreeNode */
440 const MWsStandardTextCursor* RWsTextCursor::StandardTextCursor() const
445 /** @see MWsWindowTreeNode */
446 const MWsWindowGroup* RWsTextCursor::WindowGroup() const
448 return static_cast<MWsWindowGroup*>(iGroupWin);
451 /** @see MWsWindowTreeNode */
452 const MWsWindowTreeNode* RWsTextCursor::ParentNode() const
457 /** @see MWsStandardTextCursor */
458 TInt RWsTextCursor::Type() const
463 /** @see MWsStandardTextCursor */
464 TRect RWsTextCursor::Rect() const
466 return RectRelativeToScreen();
469 /** @see MWsStandardTextCursor */
470 TRect RWsTextCursor::ClipRect() const
472 if (iFlags&ETextCursorFlagClipped)
474 TRect clipRectRelativeToScreen(iClipRect);
475 clipRectRelativeToScreen.Move(iWin->Origin());
476 return clipRectRelativeToScreen;
484 /** @see MWsStandardTextCursor */
485 TUint RWsTextCursor::Flags() const
487 return iFlags&ETextCursorUserFlags;
490 /** @see MWsStandardTextCursor */
491 TRgb RWsTextCursor::Color() const
496 /** @see MWsStandardTextCursor */
497 TTimeIntervalMicroSeconds32 RWsTextCursor::FlashInterval() const
499 return iFlags&TTextCursor::EFlagNoFlash ? 0 : KFlashRate;
502 TFlashState RWsTextCursor::CurrentCursorFlashState() const
506 return (CWsTop::CurrentFocusScreen()->Now().DateTime().MicroSecond()<KFlashRate)?EFlashOn:EFlashOff;
515 // Cursor sprite handling
517 TBool TCursorSprite::iHidden=ETrue;
518 RWsTextCursor *TCursorSprite::iCurrentCursor=NULL;
522 // Hide / Reveal text cursors.
523 void TCursorSprite::Hide()
525 if (!iHidden && iCurrentCursor)
528 if (iCurrentCursor->iCustomTextCursor)
530 iCurrentCursor->iCustomTextCursor->Deactivate();
534 iCurrentCursor->ScheduleReDrawNow();
539 void TCursorSprite::Reveal()
541 if(iHidden && iCurrentCursor)
544 if (iCurrentCursor->iCustomTextCursor)
546 iCurrentCursor->iCustomTextCursor->Activate();
550 iCurrentCursor->ScheduleReDrawNow();
555 void TCursorSprite::SetFocus(RWsTextCursor* aFocus,CWsClientWindow* aWin/*=NULL*/)
557 if (iCurrentCursor!=aFocus)
560 iCurrentCursor->ReleaseNode();
562 SetCurrentCursor(aFocus, aWin);
566 void TCursorSprite::SetCurrentCursor(RWsTextCursor* aFocus, CWsClientWindow* aWin)
568 if (aFocus && !aFocus->iCustomTextCursor)
569 aFocus->CreateNode();
570 iCurrentCursor = aFocus;
571 if (aWin && iCurrentCursor && iCurrentCursor->iCustomTextCursor)
573 iCurrentCursor->iCustomTextCursor->SetWindow(aWin);
578 TBool TCursorSprite::IsStandardCursorActive()
580 return iCurrentCursor && !iCurrentCursor->iCustomTextCursor && !iHidden;