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.
14 // Client window functions
19 #include <graphics/wselement.h>
24 #include "windowgroup.h"
25 #include "walkwindowtree.h"
34 #include "backedupwindow.h"
35 #include "redrawmsgwindow.h"
38 #include "windowelementset.h"
43 TBool CWsClientWindow::iAbsoluteFading = EFalse;
45 const TPoint corner1[1]={TPoint(1,1)};
46 const TPoint corner2[2]={TPoint(2,1),TPoint(1,1)};
47 const TPoint corner3[2]={TPoint(3,1),TPoint(1,2)};
48 const TPoint corner5[4]={TPoint(5,1),TPoint(3,1),TPoint(2,1),TPoint(1,2)};
52 CWsClientWindow::CWsClientWindow(CWsClient* aOwner, CScreen* aScreen) : CWsWindow(aOwner,WS_HANDLE_WINDOW,aScreen)
54 iWinType=EWinTypeClient;
57 void CWsClientWindow::ConstructL(const TWsClCmdCreateWindow &aCmd, CWsWindowBase *aParent, TBool aScreenDeviceIsInvalid)
59 CWsWindow::Construct();
61 if (aCmd.clientHandle==NULL)
62 OwnerPanic(EWservPanicNullHandle);
64 if (IsClientHandleInUse(aCmd.clientHandle))
65 OwnerPanic(EWservPanicDuplicateHandle);
67 iPointerFilter=EPointerFilterEnterExit|EPointerFilterMove|EPointerFilterDrag;
68 iClientHandle=aCmd.clientHandle;
69 CWsWindow* inherit=static_cast<CWsWindow *>(aParent);
70 if (aParent->WinType()==EWinTypeGroup)
72 SetPointerCursor(aParent->PointerCursor());
74 iOrigin=aParent->Origin();
75 iRel.iBr.iX=inherit->Rel().iBr.iX-inherit->Rel().iTl.iX;
76 iRel.iBr.iY=inherit->Rel().iBr.iY-inherit->Rel().iTl.iY;
80 iRedraw=new(ELeave) CWsRedrawMsgWindow(this);
83 iRedraw=new(ELeave) CWsBackedUpWindow(this, aCmd.displayMode);
88 iRedraw=new(ELeave) CWsBlankWindow(this);
91 OwnerPanic(EWservPanicRedrawType);
94 SetCornerTypeL(EWindowCornerSquare,0,NULL,EFalse);
95 CWsWindowBase::ConstructL(aParent);
96 if (aScreenDeviceIsInvalid)
98 iFlags|=EFlagScreenDeviceInvalid;
101 iRedraw->ConstructL();
104 void CWsClientWindow::GetBaseAreaOfNode(RWsRegion &aRegion) const
108 aRegion.Copy(*iBaseArea);
110 aRegion.ClipRect(iAbs);
113 void CWsClientWindow::ClipRegionToBaseArea(RWsRegion &aRegion) const
117 aRegion.Intersect(*iBaseArea);
119 aRegion.ClipRect(iAbs);
122 void CWsClientWindow::GetClippedBaseArea(RWsRegion &aRegion) const
124 const CWsWindowBase* ancestor = BaseParent();
125 GetBaseAreaOfNode(aRegion);
126 while (ancestor && ancestor->WinType() == EWinTypeClient)
128 static_cast<const CWsClientWindow*>(ancestor)->ClipRegionToBaseArea(aRegion);
129 ancestor = ancestor->BaseParent();
133 void CWsClientWindow::GetOpaqueBaseAreaOfNode(RWsRegion &aRegion) const
137 if (iUserDefinedOpaqueRegion)
139 aRegion.Copy(*iUserDefinedOpaqueRegion);
140 aRegion.ClipRect(iAbs);
149 GetBaseAreaOfNode(aRegion);
153 void CWsClientWindow::ClipRegionToOpaqueBaseArea(RWsRegion& aRegion) const
157 if (iUserDefinedOpaqueRegion)
159 aRegion.Intersect(*iUserDefinedOpaqueRegion);
160 aRegion.ClipRect(iAbs);
169 ClipRegionToBaseArea(aRegion);
173 void CWsClientWindow::GetOpaqueClippedBaseArea(RWsRegion &aRegion) const
175 const CWsWindowBase* ancestor = BaseParent();
176 GetOpaqueBaseAreaOfNode(aRegion);
177 while (ancestor && ancestor->WinType() == EWinTypeClient)
179 static_cast<const CWsClientWindow*>(ancestor)->ClipRegionToOpaqueBaseArea(aRegion);
180 ancestor = ancestor->BaseParent();
184 TInt CWsClientWindow::GetNonOpaqueBaseAreaOfNode(RWsRegion &aRegion) const
189 if(iUserDefinedTransparentRegion)
191 aRegion.Copy(*iUserDefinedTransparentRegion);
192 aRegion.ClipRect(iAbs);
199 void CWsClientWindow::ResetHiddenFlag()
201 // Reset the status of the hidden flag based on the current states of the active and invisible flags
204 CWsClientWindow *parent=static_cast<CWsClientWindow*>(iParent);
206 TBool wasHidden = iFlags&EFlagHidden;
207 TBool nowHidden = (parent==NULL ||
208 (parent->WinType()==EWinTypeClient && !parent->IsVisible()) ||
209 !(iFlags&EFlagActive) ||
210 (iFlags&EFlagInvisible) ||
211 (iFlags&EFlagScreenDeviceInvalid));
216 iFlags&=~EFlagDrawnToScreen;
220 iFlags&=~EFlagHidden;
222 if ((!nowHidden) != (!wasHidden))
224 // intentionally call the screen directly
225 iScreen->ScheduleRegionUpdate(&iVisibleRegion);
227 WS_ASSERT_DEBUG(!iDirtyWindowRegion.CheckError(),EWsPanicErrorInRegion);
228 WS_ASSERT_DEBUG(!iDirtySpriteRegion.CheckError(),EWsPanicErrorInRegion);
229 // WS_ASSERT_DEBUG(!InvalidArea().CheckError(),EWsPanicErrorInRegion); //error flag in invalid area may be set in OOM cases
230 //thus the assert statement is caused to fail
231 if (wasHidden && iScreen->ChangeTracking() &&
232 (!iDirtyWindowRegion.IsEmpty() || !iDirtySpriteRegion.IsEmpty() || !InvalidArea().IsEmpty()))
234 // Window has just become visible, schedule it
235 iScreen->ScheduleWindow(this);
240 void CWsClientWindow::ResetHiddenFlags()
242 CWsClientWindow *win=this;
245 TUint oldHiddenFlag=win->iFlags&EFlagHidden;
246 win->ResetHiddenFlag();
247 if ((win->iFlags&EFlagHidden)!=oldHiddenFlag) // If hidden status hasn't changed nothing to do
249 win->SetElementOpacity(IsVisible() ? 0xFF : 0x00); // Update element visibility
258 while(!win->NextSibling())
260 win=(CWsClientWindow *)win->BaseParent();
264 win=win->NextSibling();
268 void CWsClientWindow::OffsetBaseArea(const TPoint &aOffset)
270 iBaseArea->Offset(aOffset);
271 //If the given window's position changes, then update.
272 if (aOffset.iX || aOffset.iY)
274 UpdateElementExtent(&aOffset);
278 void CWsClientWindow::CalcBaseArea()
280 // The windows basic area before any clipping is done
283 TInt cornerType=iCornerData&ECornerTypeMask;
284 if (cornerType==EWindowCornerRegion)
285 iBaseArea->ClipRect(FullRect());
290 const TPoint *corners=NULL;
295 count=sizeof(corner1)/sizeof(TPoint);
299 count=sizeof(corner2)/sizeof(TPoint);
303 count=sizeof(corner3)/sizeof(TPoint);
307 count=sizeof(corner5)/sizeof(TPoint);
314 TInt bot=size.iHeight;
315 for(TInt index=0;index<count;index++)
317 TInt xadjust=corners[index].iX;
318 TInt yadjust=corners[index].iY;
319 if ((iCornerData&(EWindowCornerNotTL|EWindowCornerNotTR))!=(EWindowCornerNotTL|EWindowCornerNotTR))
321 iBaseArea->AddRect(TRect(iCornerData&EWindowCornerNotTL?0:xadjust,top,
322 size.iWidth-(iCornerData&EWindowCornerNotTR?0:xadjust),top+yadjust));
325 if ((iCornerData&(EWindowCornerNotBL|EWindowCornerNotBR))!=(EWindowCornerNotBL|EWindowCornerNotBR))
327 iBaseArea->AddRect(TRect(iCornerData&EWindowCornerNotBL?0:xadjust,bot-yadjust,
328 size.iWidth-(iCornerData&EWindowCornerNotBR?0:xadjust),bot));
332 iBaseArea->AddRect(TRect(0,top,size.iWidth,bot));
333 iBaseArea->Offset(Origin());
334 iBaseArea->ClipRect(FullRect());
339 void CWsClientWindow::GenerateArea(RWsRegion &aArea, TBool aClipTranslucent) const
341 // Create the window area list.
347 aArea.Copy(*iBaseArea);
348 aArea.ClipRect(iAbs);
349 const CWsClientWindow *win=this;
352 if (win->IsTopClientWindow())
354 ClipWindows(aArea,(CWsClientWindow *)win->BaseParent()->BaseChild(),win,aClipTranslucent);
355 win=(CWsClientWindow *)win->iParent;
358 for(const CWsTopClientWindow *cwin=RootWindow()->FirstTopClientWindow();aArea.Count() && cwin!=win;cwin=cwin->NextSiblingMultiParent())
363 tidyCount=ETidyCountSetting; // Tidy every ETidyCountSetting times around
365 if (cwin->IsVisible())
367 if (cwin->IsTranslucent() && !aClipTranslucent)
369 if (cwin->iUserDefinedOpaqueRegion)
371 aArea.SubRegion(*cwin->iUserDefinedOpaqueRegion);
376 aArea.SubRegion(*cwin->iBaseArea);
384 void CWsClientWindow::ClipWindows(TRegion ®ion,const CWsClientWindow *start, const CWsClientWindow *end, TBool aClipTranslucent)
386 // Remove out of the region the opaque part of the abs rect of all the windows starting from 'start'
387 // along the sibling list to (and not including) the end window.
390 for(const CWsClientWindow *win=start;region.Count() && win!=end;win=win->NextSibling())
392 if (win->IsVisible())
394 if (win->IsTranslucent() && !aClipTranslucent)
396 if (win->iUserDefinedOpaqueRegion)
398 region.SubRegion(*win->iUserDefinedOpaqueRegion);
403 region.SubRegion(*win->iBaseArea);
409 void CWsClientWindow::GenerateTopRegion(RWsRegion& aRegion) const
411 GenerateArea(aRegion,ETrue);
413 ClipWindows(aRegion,Child(),NULL,ETrue);
416 void CWsClientWindow::GenerateWindowRegion(RWsRegion &aRegion) const
418 // Calculate the windows clipping region without using the usual stored iArea or iRegion fields
419 // this function is used by the screen backup code to calculate "what if" regions to work out
420 // whether something would be visible if the backed up window didn't exist, on this basis we
421 // don't want to modify the existing copies of iArea & iRegion.
424 GenerateArea(aRegion,EFalse);
426 ClipWindows(aRegion,Child(),NULL,EFalse);
429 const TRegion *CWsClientWindow::DrawingRegion()
431 return (&iRedraw->BaseDrawRegion());
434 void CWsClientWindow::RecalcChildAbs(const TPoint *aOffset)
436 CWsClientWindow *win=this;
441 win->SetAbsFromRel();
443 win->OffsetBaseArea(*aOffset);
444 if (win->Child()==NULL)
452 if (win->NextSibling()!=NULL)
454 win=win->NextSibling();
457 win=(CWsClientWindow *)win->iParent; // The cast is safe as the loop is aborted when win==this
462 void CWsClientWindow::SetAbsFromRel()
464 iOrigin=iRel.iTl+iParent->Origin();
466 iAbs.Move(iParent->Origin());
467 iAbs.Intersection(iParent->AbsRect());
470 void CWsClientWindow::SetExtentL(const TPoint *aPos,const TSize *aSize)
473 OwnerPanic(EWservPanicParentDeleted);
474 TPoint offset = TPoint(0,0);
477 TBool sizeChanged = EFalse;
478 TBool posChanged = EFalse;
482 offset = *aPos+iParent->Origin()-iOrigin;
483 if (offset.iX != 0 || offset.iY != 0)
491 TWalkWindowTreeScheduleRedraws wwt;
492 WalkWindowTree(wwt, EWalkChildren);
498 if (newSize.iWidth<0)
500 if (newSize.iHeight<0)
502 // This should be the only part of resizing that can fail
503 // and it can only fail for backedup windows.
504 iRedraw->PrepareForResizeL(newSize,oldSize);
505 sizeChanged = *aSize != iRel.Size();
511 RecalcChildAbs(&offset); // Also calls UpdateElementExtent(offset)
512 TWalkWindowTreeOffsetTransparentRegions offsetTransparent(offset);
513 WalkWindowTree(offsetTransparent, EWalkChildren);
518 iRel.SetSize(newSize);
519 RecalcChildAbs(NULL);
521 iRedraw->Resize(newSize,oldSize);
522 if (Redraw()->HasElement())
523 UpdateElementExtent();
526 if ((posChanged || sizeChanged) && Redraw()->HasElement())
528 TRect interSection(iParent->Origin(), iParent->Size());
529 interSection.Intersection(FullRect());
530 if (interSection == FullRect())
532 // There is no any clipping in this case
533 interSection = TRect();
535 // Get the corresponding source rectangle for the element
536 if (!iOriginalSrcElementRect.IsEmpty() && !iOriginalDestElementRect.IsEmpty() && !iAbs.IsEmpty())
538 MWsElement* element = Screen()->WindowElements().GetElementFromWindow(*this);
541 element->SetDestinationClippingRect(interSection);
546 if (posChanged || sizeChanged)
548 iRedraw->ClipInvalidRegion(TRect(iRel.Size()));
550 ScheduleRegionUpdate(NULL);
551 TWalkWindowTreeRecalcOpaque recalcOpaque;
552 WalkWindowTree(recalcOpaque, EWalkChildren);
554 MWsWindowTreeObserver* windowTreeObserver = Screen()->WindowTreeObserver();
555 if (windowTreeObserver)
557 TRect rect = FullRect();
558 windowTreeObserver->NodeExtentChanged(*this, rect);
560 for (CWsAnim* anim = iAnimList; anim; anim = anim->Next())
562 windowTreeObserver->NodeExtentChanged(*anim, rect);
569 void CWsClientWindow::GetElementFlipAndRotation(TBool& aElemetFlip, MWsElement::TElementRotation& aElemenetRotation)
571 MWsElement* element = Screen()->WindowElements().GetElementFromWindow(*this);
574 aElemetFlip = element->SourceFlipping();
575 aElemenetRotation = element->SourceRotation();
579 void CWsClientWindow::Scroll(const TRect &aClipRect, const TPoint &aOffset, const TRect &aRect)
582 OwnerPanic(EWservPanicParentDeleted);
585 iRedraw->Scroll(aClipRect, aOffset,aRect);
587 CWsTop::TriggerRedraws(RootWindow());
590 void CWsClientWindow::DeleteBaseArea()
592 WS_ASSERT_DEBUG(iBaseArea!=&nullRegion, EWsPanicRegionNull);
593 if ((iCornerData&ECornerTypeMask)==EWindowCornerRegion)
594 ((RWsRegion *)iBaseArea)->Destroy();
602 CWsClientWindow::~CWsClientWindow()
604 while(iVisibleRegionTrackingCounter>0)
606 SetupVisibleRegionTracking(EFalse);
608 if (iBaseWinFlags&EBaseWinNodeCreated)
610 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
611 if (windowTreeObserver)
613 windowTreeObserver->NodeReleased(*this);
614 iBaseWinFlags &= ~EBaseWinNodeCreated;
618 SetUserTransparentRegion(0);
619 CWsPassword::WindowDestroyed(this);
622 void CWsClientWindow::Shutdown()
624 // Destroy a window, disconnects from the window tree and destroys all it's child windows
627 iFlags|=EFlagShutDownInProgress;
628 if (CClick::IsHandler())
630 TWindowCloseData params;
631 params.iClientHandle=iClientHandle;
632 //if parent already shutdown (or disconnected) send 0
633 params.iWindowGroupId = (iParent) ? WinGroup()->Identifier() : 0;
634 CClick::OtherEvent(EEventWindowClose,¶ms);
639 iWinGcList->Deactivate();
641 iFlags|=EFlagInvisible; // First make it invisble
642 if (iParent) // In case window wasn't fully constructed
645 CWsWindow::Shutdown();
647 CWsPointerBuffer::Disconnect(this);
648 iFlags&=~EFlagShutDownInProgress;
651 void CWsClientWindow::Activate()
653 if (iFlags&EFlagActive)
654 OwnerPanic(EWservPanicWindowActive);
659 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
660 if (windowTreeObserver)
662 windowTreeObserver->NodeExtentChanged(*this, FullRect());
663 windowTreeObserver->NodeActivated(*this);
667 TBool CWsClientWindow::IsActivated() const
669 return (iFlags&EFlagActive)!=EFalse;
672 void CWsClientWindow::SetCornerTypeL(TCornerType aCornerType, TInt aCornerFlags, TRegion *aNewBaseArea, TBool aNotifyShapeChanged)
674 TRegion *baseArea=NULL;
675 if (aCornerFlags&ECornerTypeMask)
676 OwnerPanic(EWservPanicCornerParams);
680 case EWindowCornerSquare:
681 baseArea=new(ELeave) TRegionFix<1>();
684 baseArea=new(ELeave) TRegionFix<3>();
688 baseArea=new(ELeave) TRegionFix<5>();
691 baseArea=new(ELeave) TRegionFix<9>();
693 case EWindowCornerRegion:
694 User::LeaveIfNull(baseArea=aNewBaseArea);
695 baseArea->Offset(Origin());
698 OwnerPanic(EWservPanicCornerParams);
701 iCornerData=aCornerType;
702 iCornerData|=aCornerFlags;
705 ScheduleRegionUpdate(NULL);
707 if ( aNotifyShapeChanged )
709 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
710 if (windowTreeObserver)
712 windowTreeObserver->AttributeChanged(*this, MWsWindowTreeObserver::EWindowShape);
717 void CWsClientWindow::SetVisible(TBool aState)
722 OwnerPanic(EWservPanicParentDeleted);
723 if (!(iFlags&EFlagInvisible)) // Already visible
725 iFlags&=~EFlagInvisible;
730 if (iFlags&EFlagInvisible || !iParent) // Already invisible or parent has been deleted
732 TWalkWindowTreePurgeEvents wwt;
733 WalkWindowTree(wwt,EWalkChildren); // Destroy all events on this and all children
734 iFlags|=EFlagInvisible;
738 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
739 if (windowTreeObserver)
741 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EVisible, aState);
745 void CWsClientWindow::CommandL(TInt aOpcode, const TAny *aCmdData)
748 // Save root window for performing CheckTree at the end of this func.
749 // When aOpcode is EWsWinOpFree, this object would've been destroyed
750 // and a call to RootWindow() in that case would be impossible
751 CWsRootWindow* rootWindow=RootWindow();
753 TWsWinCmdUnion pData;
755 if (CWsWindowBase::CommandL(aOpcode,pData)==EFalse)
759 case EWsWinOpActivate:
763 SetExtentL(pData.pos,NULL);
765 case EWsWinOpSetExtent:
766 case EWsWinOpSetExtentErr:
767 SetExtentL(&pData.SetEx->pos,&pData.SetEx->size);
769 case EWsWinOpSetSize:
770 case EWsWinOpSetSizeErr:
771 SetExtentL(NULL,pData.size);
773 case EWsWinOpInquireOffset:
774 CWsClient::ReplyPoint(InquireOffset(*pData.UInt));
776 case EWsWinOpPosition:
777 CWsClient::ReplyPoint(iRel.iTl);
779 case EWsWinOpAbsPosition:
780 CWsClient::ReplyPoint(iOrigin);
783 CWsClient::ReplySize(iRel.Size());
785 case EWsWinOpTestInvariant:
788 case EWsWinOpPointerFilter:
790 TUint old=iPointerFilter;
791 iPointerFilter&=~pData.PointerFilter->mask;
792 iPointerFilter|=pData.PointerFilter->mask&pData.PointerFilter->flags;
793 if (old&EPointerFilterEnterExit)
794 TWsPointer::ReLogWindow(this);
797 case EWsWinOpSetPointerGrab:
798 if (*pData.Bool==EFalse)
799 iFlags&=~EFlagPointerGrab;
801 iFlags|=EFlagPointerGrab;
803 case EWsWinOpClaimPointerGrab:
806 OwnerPanic(EWservPanicParentDeleted);
808 TInt errNo = TWsPointer::ClaimGrab(this,*pData.GrabControl);
809 if(TWsWinCmdGrabControl::ESendReply & pData.GrabControl->flags)
811 // To avoid the reply-generated-flush, only do this for the new APIs, not the old ones.
816 case EWsWinOpSetPointerCapture:
817 iFlags&=~(EFlagPointerCaptured|EFlagPointerCaptureDragDrop|EFlagPointerCaptureAllGroups);
818 if ((*pData.UInt)&RWindowBase::TCaptureFlagEnabled)
820 iFlags|=EFlagPointerCaptured;
821 if ((*pData.UInt)&RWindowBase::TCaptureFlagDragDrop)
822 iFlags|=EFlagPointerCaptureDragDrop;
823 if ((*pData.UInt)&RWindowBase::TCaptureFlagAllGroups)
824 iFlags|=EFlagPointerCaptureAllGroups;
827 TWsPointer::ReLogPointersCurrentWindows();
829 case EWsWinOpSetPointerCapturePriority:
830 iPointerCapturePriority=*pData.Int;
832 case EWsWinOpGetPointerCapturePriority:
833 SetReply(iPointerCapturePriority);
835 case EWsWinOpSetVisible:
836 SetVisible(*pData.Bool);
841 TRect src(TRect(origin,iRel.Size()));
842 src.Move(-pData.ScrollRect->offset);
843 Scroll(TRect(origin,iRel.Size()),pData.ScrollRect->offset,src);
846 case EWsWinOpScrollClip:
849 TRect src(TRect(origin,iRel.Size()));
850 src.Move(-pData.ScrollRect->offset);
851 TRect clip(pData.ScrollRect->clip);
852 Scroll(clip,pData.ScrollRect->offset,src);
855 case EWsWinOpScrollRect:
857 TRect src(pData.ScrollRect->rect);
858 Scroll(TRect(TPoint(0,0),iRel.Size()),pData.ScrollRect->offset,src);
861 case EWsWinOpScrollClipRect:
863 TRect src(pData.ScrollRect->rect);
864 TRect clip(pData.ScrollRect->clip);
865 Scroll(clip, pData.ScrollRect->offset,src);
868 case EWsWinOpSetOrdinalPositionPri:
869 iOrdinalPriority=pData.OrdinalPos->ordinalPriority;
870 SetOrdinalPosition(pData.OrdinalPos->pos);
872 case EWsWinOpSetShadowHeight:
873 OwnerPanic(EWservPanicOpcode); //this op code is deprecated, should never be generated by the client
875 case EWsWinOpShadowDisabled:
876 OwnerPanic(EWservPanicOpcode); //this op code is deprecated, should never be generated by the client
878 case EWsWinOpRequiredDisplayMode:
880 OwnerPanic(EWservPanicBackupDisplayMode);
881 SetReply(SetRequiredDisplayModeL(*pData.DisplayMode));
883 case EWsWinOpGetDisplayMode:
884 SetReply(DisplayMode());
886 case EWsWinOpRequestPointerRepeatEvent:
889 OwnerPanic(EWservPanicParentDeleted);
890 TInt errNo = TWsPointer::RequestPointerRepeatEvent(this,*pData.RequestPointerRepeatEvent);
891 if(TWsWinCmdRequestPointerRepeatEvent::ERepeatFlagsSendReply & pData.RequestPointerRepeatEvent->repeatFlags)
897 case EWsWinOpCancelPointerRepeatEventRequest:
899 TInt errNo = TWsPointer::CancelPointerRepeatEventRequest(*pData.CancelPointerRepeatEventRequest);
900 if(TWsWinCmdCancelPointerRepeatEventRequest::ECancelRepeatFlagsSendReply & pData.RequestPointerRepeatEvent->repeatFlags)
906 case EWsWinOpAllocPointerMoveBuffer:
907 CWsPointerBuffer::ConnectL(this,pData.AllocPointerMoveBuffer->maxNumPoints,pData.AllocPointerMoveBuffer->flags);
908 iFlags|=EFlagUsingPointerBuffer|EFlagHasPointerBuffer;
910 case EWsWinOpFreePointerMoveBuffer:
911 CWsPointerBuffer::Disconnect(this);
912 iFlags&=~(EFlagUsingPointerBuffer|EFlagHasPointerBuffer);
914 case EWsWinOpRetrievePointerMoveBuffer:
915 CWsPointerBuffer::RetrievePointerMoveBuffer(this,*pData.Int);
917 case EWsWinOpEnablePointerMoveBuffer:
918 if (!(iFlags&EFlagHasPointerBuffer))
919 OwnerPanic(EWservPanicNoPointerBuffer);
920 iFlags|=EFlagUsingPointerBuffer;
922 case EWsWinOpDisablePointerMoveBuffer:
923 iFlags&=~EFlagUsingPointerBuffer;
925 case EWsWinOpDiscardPointerMoveBuffer:
926 CWsPointerBuffer::DiscardPointerMoveBuffer(this);
928 case EWsWinOpAddKeyRect:
929 AddKeyRectL(pData.AddKeyRect->rect, pData.AddKeyRect->scanCode, pData.AddKeyRect->activatedByPointerSwitchOn);
931 case EWsWinOpRemoveAllKeyRects:
934 case EWsWinOpPasswordWindow:
936 OwnerPanic(EWservPanicParentDeleted);
937 CWsPassword::SetPasswordWindowL(this, *pData.PasswordMode);
939 case EWsWinOpEnableBackup:
941 OwnerPanic(EWservPanicParentDeleted);
943 iBackupsRequested|=EWindowBackupAreaBehind; //For backwards compatibility
945 iBackupsRequested|=*pData.UInt;
947 case EWsWinOpFadeBehind:
950 OwnerPanic(EWservPanicParentDeleted);
954 iScreen->GetFadingParams(blackMap,whiteMap);
955 SetFadeBehind(*pData.Bool);
956 TWalkWindowTreeSetFaded wwt(*pData.Bool,this,blackMap,whiteMap);
957 WalkWindowTree(wwt,EWalkBehind);
960 case EWsWinOpGetIsFaded:
961 SetReply(iFadeCount);
963 case EWsWinOpGetIsNonFading:
964 SetReply(iFlags&EFlagNonFadingWindow);
966 case EWsWinOpMoveToGroup:
968 OwnerPanic(EWservPanicParentDeleted);
969 if (iParent->WinType()!=EWinTypeGroup)
970 OwnerPanic(EWservPanicNotTopClient);
971 ((CWsTopClientWindow*)this)->DoMoveWindowL(*pData.Int);
973 case EWsWinOpTestLowPriorityRedraw:
975 // This is purely for testing purposes
976 // Returns the redraw priority
978 TPckgBuf<TUint> priBuf;
979 priority=WsOwner()->RedrawQueue()->RedrawPriority((CWsWindowRedraw*)this->iRedraw);
981 CWsClient::ReplyBuf(priBuf);
984 case EWsWinOpEnableVisibilityChangeEvents:
985 iFlags |= EFlagGeneratesVisibilityEvents;
986 SetupVisibleRegionTracking(ETrue);
987 if (iFlags&EFlagActive)
989 iScreen->DoRedrawNow();
990 PossibleVisibilityChangedEvent(ETrue);
993 case EWsWinOpDisableVisibilityChangeEvents:
994 iFlags &= ~EFlagGeneratesVisibilityEvents;
995 SetupVisibleRegionTracking(EFalse);
997 case EWsWinOpSetTransparentRegion:
1001 TInt recs=*pData.Int;
1002 RWsRegion* reg=recs>0? GetRegionFromClientL(iWsOwner,recs) : new(ELeave) RWsRegion;
1003 SetUserTransparentRegion(reg);
1008 OwnerPanic(EWservPanicTransparencyObjNotCreated);
1012 case EWsWinOpSetTransparencyPolicy:
1014 if (IsTranslucent())
1017 OwnerPanic(EWservPanicTransparencyObjNotCreated);
1020 case EWsWinOpSetTransparencyAlphaChannel:
1022 iFlags |= static_cast<TUint>(EFlagHasAlpha);
1025 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
1026 if (windowTreeObserver)
1028 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EAlphaChannelTransparencyEnabled, ETrue);
1033 if (iRedraw->CommandL(aOpcode,pData)==EFalse)
1035 OwnerPanic(EWservPanicOpcode);
1040 rootWindow->CheckTree();
1044 void CWsClientWindow::GcActivated(CWsGc *aGc)
1046 aGc->SetNextWinGc(iWinGcList);
1050 void CWsClientWindow::GcDeactivated(CWsGc *aGc)
1052 if (aGc==iWinGcList)
1053 iWinGcList=aGc->NextWinGc();
1056 CWsGc *gc=iWinGcList;
1060 next=gc->NextWinGc();
1061 WS_ASSERT_DEBUG(next!=NULL, EWsPanicBadActiveGcList);
1064 gc->SetNextWinGc(next->NextWinGc());
1070 aGc->SetNextWinGc(NULL);
1073 void CWsClientWindow::ReactivateGcs()
1075 for (CWsGc * gc = iWinGcList; gc; gc = gc->NextWinGc())
1081 void CWsClientWindow::OffsetUserTransparentRegion(const TPoint& aOffset)
1083 if (iUserDefinedTransparentRegion)
1085 iUserDefinedTransparentRegion->Offset(aOffset);
1089 void CWsClientWindow::SetUserTransparentRegion(RWsRegion* aRegion)
1091 if (iUserDefinedTransparentRegion)
1093 iUserDefinedTransparentRegion->Close();
1094 delete iUserDefinedTransparentRegion;
1095 iUserDefinedTransparentRegion = 0;
1100 aRegion->Offset(iOrigin);
1101 iUserDefinedTransparentRegion=aRegion;
1104 SetUserOpaqueRegion();
1107 void CWsClientWindow::SetUserOpaqueRegion()
1109 if (iUserDefinedOpaqueRegion)
1111 iUserDefinedOpaqueRegion->Close();
1112 delete iUserDefinedOpaqueRegion;
1113 iUserDefinedOpaqueRegion = 0;
1115 if (iUserDefinedTransparentRegion)
1117 iUserDefinedOpaqueRegion=new RWsRegion;
1118 if (iUserDefinedOpaqueRegion)
1120 iUserDefinedOpaqueRegion->Copy(*iBaseArea);
1121 iUserDefinedOpaqueRegion->SubRegion(*iUserDefinedTransparentRegion);
1122 if (iUserDefinedOpaqueRegion->CheckError() || iUserDefinedOpaqueRegion->Count() == 0)
1124 iUserDefinedOpaqueRegion->Close();
1125 delete iUserDefinedOpaqueRegion;
1126 iUserDefinedOpaqueRegion = 0;
1129 // Intentionally not sending this notification during destruction (when SetUserTransparentRegion is called from d'tor)
1130 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
1131 if (windowTreeObserver)
1133 windowTreeObserver->TransparentRegionChanged(*this, *iUserDefinedTransparentRegion, iUserDefinedOpaqueRegion);
1138 /** Checks whether this window is in front of aWin.
1140 @param aWin A window.
1141 @return EFalse if aWin is the same or is in front of this, ETrue otherwise.
1145 TBool CWsClientWindow::IsInfrontOf(const CWsWindowBase* aWin) const
1147 TInt thisDepth=Depth();
1148 TInt otherDepth=aWin->Depth();
1149 const CWsWindowBase *thisWin=this;
1150 const CWsWindowBase *otherWin=aWin;
1151 if (thisDepth>otherDepth)
1153 for (TInt count=thisDepth-otherDepth;count>0;count--)
1154 thisWin=thisWin->BaseParent();
1158 for (TInt count=otherDepth-thisDepth;count>0;count--)
1159 otherWin=otherWin->BaseParent();
1161 if (thisWin==otherWin)
1162 return thisDepth>otherDepth;
1163 while(thisWin->BaseParent()!=otherWin->BaseParent())
1165 thisWin=thisWin->BaseParent();
1166 otherWin=otherWin->BaseParent();
1168 const CWsWindowBase *win=thisWin->BaseParent()->BaseChild();
1177 win=win->NextSibling();
1181 CWsTopClientWindow* CWsClientWindow::TopClientWindow()
1184 OwnerPanic(EWservPanicParentDeleted);
1185 CWsWindowBase* win=this;
1186 while(win->BaseParent()->WinType()!=EWinTypeGroup)
1187 win=win->BaseParent();
1188 return static_cast<CWsTopClientWindow*>(win);
1191 const TRegion &CWsClientWindow::InvalidArea() const
1193 return(iRedraw->InvalidArea());
1197 TUint CWsClientWindow::RedrawPriority(TInt *aShift) const
1199 TUint ordinalPos=OrdinalPosition(EFalse)+1;
1200 if (ordinalPos>15) // Algorithm only works upto 15 , make all windows after 15 equal in priority
1203 TUint parent=((CWsClientWindow *)iParent)->RedrawPriority(&shift);
1208 return(parent+(ordinalPos<<(shift*KWinRedrawPriBitsPerLevel)));
1211 TDblQue<TPointerKeyList> *CWsClientWindow::PointerKeyList() const
1213 return(iPointerKeyList);
1216 void CWsClientWindow::AddKeyRectL(const TRect &aRect, TInt aScanCode, TBool aActivatedByPointerSwitchOn)
1218 if (!iPointerKeyList)
1219 iPointerKeyList=new(ELeave) TDblQue<TPointerKeyList>(_FOFF(TPointerKeyList,iQue));
1220 TPointerKeyList *pkl=new(ELeave) TPointerKeyList();
1221 iPointerKeyList->AddLast(*pkl);
1223 pkl->iScanCode=aScanCode;
1224 pkl->iActivatedByPointerSwitchOn=aActivatedByPointerSwitchOn;
1227 void CWsClientWindow::RemoveAllKeyRects()
1229 if (iPointerKeyList)
1231 TPointerKeyList *pkl=NULL;
1232 for(TDblQueIter<TPointerKeyList> iter(*iPointerKeyList);(pkl=iter++)!=NULL;)
1237 delete iPointerKeyList;
1238 iPointerKeyList=NULL;
1242 TBool CWsClientWindow::IsHidden()
1244 return (!IsVisible()) || VisibleRegion().IsEmpty();
1247 void CWsClientWindow::SetFaded(TBool aFade, TUint8 aBlackMap, TUint8 aWhiteMap, TBool aNotifyObserver)
1250 SetFaded(aFade, aBlackMap, aWhiteMap, aNotifyObserver, stateChanged);
1253 void CWsClientWindow::SetFaded(TBool aFade, TUint8 aBlackMap, TUint8 aWhiteMap, TBool aNotifyObserver, TBool& aStateChanged)
1255 iBlackMap=aBlackMap;
1256 iWhiteMap=aWhiteMap;
1257 const TBool mapAltered = (iBlackMap != aBlackMap) || (iWhiteMap != aWhiteMap);
1258 const TInt oldFadeCount = iFadeCount;
1260 if (iAbsoluteFading)
1277 else if (iFadeCount > 0)
1283 //Fade state only changes when iFadeCount transitions from 0 to 1 or from 1 to 0.
1284 aStateChanged = (iFadeCount==0 || oldFadeCount==0) && (oldFadeCount != iFadeCount);
1285 if (!Screen()->ChangeTracking() && CWsTop::IsFadeEnabled() && (aStateChanged || mapAltered) )
1287 Screen()->AcceptFadeRequest(this, (iFadeCount > 0));
1290 const TBool doNotify = iAbsoluteFading ? aStateChanged : (oldFadeCount != iFadeCount);
1291 MWsWindowTreeObserver* const windowTreeObserver = Screen()->WindowTreeObserver();
1292 if (windowTreeObserver && aNotifyObserver && doNotify)
1294 windowTreeObserver->FadeCountChanged(*this, iFadeCount);
1298 void CWsClientWindow::ResetHiddenFlagsInParentAndChildren()
1301 SetElementOpacity(IsVisible() ? 0xFF : 0x00); // Update element visibility
1302 for(CWsClientWindow* child=Child();child;child=child->NextSibling())
1304 child->ResetHiddenFlagsInParentAndChildren();
1308 const TRegion& CWsClientWindow::WindowArea() const
1310 WS_ASSERT_DEBUG(iBaseArea, EWsPanicRegionNull);
1314 void CWsClientWindow::Invalidate(const TRect * aRect)
1316 iRedraw->Invalidate(aRect);
1319 void CWsClientWindow::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
1323 iScreen->ScheduleRegionUpdate(aDefinitelyDirty);
1327 void CWsClientWindow::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
1331 iScreen->AddRedrawRegion(aRegion, aSchedule, aDepth);
1335 void CWsClientWindow::SendState(MWsWindowTreeObserver& aWindowTreeObserver) const
1337 CWsWindow::SendState(aWindowTreeObserver);
1341 aWindowTreeObserver.FadeCountChanged(*this, iFadeCount);
1344 if(iUserDefinedTransparentRegion)
1346 aWindowTreeObserver.TransparentRegionChanged(*this, *iUserDefinedTransparentRegion, iUserDefinedOpaqueRegion);
1351 CWindowElementSet& windowElementSet = Screen()->WindowElements();
1352 const TBackgroundAttributes *bElementAttr;
1353 const RArray<TPlacedAttributes> *pElementsAttr;
1355 TInt ret = windowElementSet.FindElements(*this, bElementAttr, pElementsAttr);
1358 MWsElement* element = bElementAttr->iElement;
1360 aWindowTreeObserver.ElementAdded(*this, *element);
1366 TBool CWsClientWindow::IsDSAHost() const
1368 TBool res = CWsWindow::IsDSAHost();
1370 { // check for grace period when DSA is being restarted (after aborting but before client started DSA again)
1371 res = Screen()->IsDSAClientWindow( this );
1376 void CWsClientWindow::UpdateElementExtent(const TPoint* aOffset)
1378 if (Redraw()->HasElement())
1380 Screen()->WindowElements().UpdateElementExtent(*this, aOffset);
1384 void CWsClientWindow::SetElementOpacity(TInt aOpacity)
1386 if (Redraw()->HasElement())
1388 Screen()->WindowElements().SetElementOpacity(*this,aOpacity);
1393 TRect CWsClientWindow::GetOriginalSrcElementRect() const
1395 return iOriginalSrcElementRect;
1397 TRect CWsClientWindow::GetOriginalDestElementRect() const
1399 return iOriginalDestElementRect;
1403 // Code for CWsTopClientWindow, a client window that connects to a group window //
1406 CWsTopClientWindow::CWsTopClientWindow(CWsClient* aOwner, CScreen* aScreen) : CWsClientWindow(aOwner, aScreen)
1410 void CWsTopClientWindow::ConstructL(const TWsClCmdCreateWindow &cmd, CWsWindowBase *aParent, TBool aScreenDeviceIsInvalid)
1412 iFlags|=EFlagIsTopClientWindow;
1413 CWsClientWindow::ConstructL(cmd, aParent, aScreenDeviceIsInvalid);
1416 void CWsTopClientWindow::SetInactive()
1418 iFlags&=~EFlagActive;
1422 void CWsTopClientWindow::SetScreenDeviceValidState(TBool aState)
1424 if (SetScreenDeviceValidStateFlag(aState))
1428 TBool CWsTopClientWindow::SetScreenDeviceValidStateFlag(TBool aState)
1430 TBool isSet=iFlags&EFlagScreenDeviceInvalid;
1431 if (!isSet==!aState)
1434 iFlags&=~EFlagScreenDeviceInvalid;
1436 iFlags|=EFlagScreenDeviceInvalid;
1438 MWsWindowTreeObserver* windowTreeObserver = iScreen->WindowTreeObserver();
1439 if (windowTreeObserver)
1441 windowTreeObserver->FlagChanged(*this, MWsWindowTreeObserver::EScreenDeviceValid, aState);
1449 void CWsTopClientWindow::SetOrdinalPosition(TInt aPos)
1453 OwnerPanic(EWservPanicParentDeleted);
1455 if (CheckOrdinalPositionChange(aPos))
1457 CWsWindowBase::SetOrdinalPosition(aPos);
1458 CWsTop::TriggerRedraws(RootWindow());
1462 void CWsTopClientWindow::DoMoveWindowL(TInt aIdentifier)
1464 CWsWindowGroup* group=CWsWindowGroup::WindowGroupFromIdentifierL(aIdentifier);
1467 if (group->WsOwner()!=WsOwner())
1468 User::Leave(KErrNotFound);
1469 ChangeWindowPosition(0, group);
1470 CWsTop::TriggerRedraws(RootWindow());
1473 TUint CWsTopClientWindow::RedrawPriority(TInt *aShift) const
1475 TUint ordinalPos=OrdinalPosition(EFalse);
1476 if (ordinalPos>KWinRedrawPriMaxOrdinal) // Algorithm only works for upto KWinRedrawPriMaxOrdinal windows,
1477 ordinalPos=KWinRedrawPriMaxOrdinal; // make all windows after this equal in priority
1479 *aShift=KWinRedrawPriMaxLevel;
1480 return(ordinalPos<<(KWinRedrawPriMaxLevel*KWinRedrawPriBitsPerLevel));