sl@0
|
1 |
// Copyright (c) 2006-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 |
// SCREEN_REDRAW.CPP
|
sl@0
|
15 |
//
|
sl@0
|
16 |
//
|
sl@0
|
17 |
|
sl@0
|
18 |
#include "ScreenRedraw.h"
|
sl@0
|
19 |
|
sl@0
|
20 |
#include <hal.h>
|
sl@0
|
21 |
|
sl@0
|
22 |
#include "debugbar.h"
|
sl@0
|
23 |
#include "screen.h"
|
sl@0
|
24 |
#include "inifile.h"
|
sl@0
|
25 |
#include "offscreenbitmap.h"
|
sl@0
|
26 |
#include "wspluginmanager.h"
|
sl@0
|
27 |
#include "pointer.h"
|
sl@0
|
28 |
#include "rootwin.h"
|
sl@0
|
29 |
#include "walkwindowtree.h"
|
sl@0
|
30 |
#include "wstop.h"
|
sl@0
|
31 |
#include "WsMemMgr.h"
|
sl@0
|
32 |
#include "Graphics/WsRenderStageFactory.h"
|
sl@0
|
33 |
#include "Graphics/WsRenderStage.h"
|
sl@0
|
34 |
#include "EVENT.H"
|
sl@0
|
35 |
|
sl@0
|
36 |
GLREF_D CDebugLogBase *wsDebugLog;
|
sl@0
|
37 |
|
sl@0
|
38 |
#ifdef USE_DEBUG_REGIONS
|
sl@0
|
39 |
# define DEBUG_REGION(col,fill,reg) DebugRegion(col,fill,reg)
|
sl@0
|
40 |
# define DEBUG_RECT(col,fill,rect) DebugRect(col,fill,rect)
|
sl@0
|
41 |
#else
|
sl@0
|
42 |
# define DEBUG_REGION(col,fill,reg)
|
sl@0
|
43 |
# define DEBUG_RECT(col,fill,rect)
|
sl@0
|
44 |
#endif
|
sl@0
|
45 |
|
sl@0
|
46 |
#if defined(__WINS__) && defined(_DEBUG)
|
sl@0
|
47 |
# define DEBUGOSB { CWsOffScreenBitmap * ofb = iScreen.OffScreenBitmap(); if (ofb) ofb->Update(); }
|
sl@0
|
48 |
#else
|
sl@0
|
49 |
# define DEBUGOSB
|
sl@0
|
50 |
#endif
|
sl@0
|
51 |
|
sl@0
|
52 |
#ifdef _DEBUG
|
sl@0
|
53 |
# define LOG_SCREEN_REDRAW_START {if (wsDebugLog) {_LIT(KLogScreenRedrawStart, ">> CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawStart);}}
|
sl@0
|
54 |
# define LOG_SCREEN_REDRAW_END {if (wsDebugLog) {_LIT(KLogScreenRedrawEnd, "<< CScreenRedraw::OnAnimation()"); wsDebugLog->MiscMessage(CDebugLogBase::ELogEverything, KLogScreenRedrawEnd);}}
|
sl@0
|
55 |
#else
|
sl@0
|
56 |
# define LOG_SCREEN_REDRAW_START
|
sl@0
|
57 |
# define LOG_SCREEN_REDRAW_END
|
sl@0
|
58 |
#endif
|
sl@0
|
59 |
|
sl@0
|
60 |
CScreenRedraw::TTimedRect::TTimedRect(const TRect& aRect, const TTime& aTime):
|
sl@0
|
61 |
iRect(aRect), iTime(aTime)
|
sl@0
|
62 |
{
|
sl@0
|
63 |
}
|
sl@0
|
64 |
|
sl@0
|
65 |
TInt CScreenRedraw::TTimedRect::Compare(const TTimedRect& aOne,const TTimedRect& aOther)
|
sl@0
|
66 |
{
|
sl@0
|
67 |
if(aOne.iTime < aOther.iTime)
|
sl@0
|
68 |
return -1;
|
sl@0
|
69 |
else if(aOne.iTime > aOther.iTime)
|
sl@0
|
70 |
return 1;
|
sl@0
|
71 |
else
|
sl@0
|
72 |
return 0;
|
sl@0
|
73 |
}
|
sl@0
|
74 |
|
sl@0
|
75 |
CScreenRedraw * CScreenRedraw::NewL(CScreen& aScreen)
|
sl@0
|
76 |
{
|
sl@0
|
77 |
CScreenRedraw * self = new (ELeave) CScreenRedraw(aScreen);
|
sl@0
|
78 |
CleanupStack::PushL(self);
|
sl@0
|
79 |
self->ConstructL();
|
sl@0
|
80 |
CleanupStack::Pop(self);
|
sl@0
|
81 |
return self;
|
sl@0
|
82 |
}
|
sl@0
|
83 |
|
sl@0
|
84 |
CScreenRedraw::CScreenRedraw(CScreen& aScreen): iScreen(aScreen)
|
sl@0
|
85 |
{
|
sl@0
|
86 |
}
|
sl@0
|
87 |
|
sl@0
|
88 |
CScreenRedraw::~CScreenRedraw()
|
sl@0
|
89 |
{
|
sl@0
|
90 |
CWsRenderStage * stage = iRenderStages;
|
sl@0
|
91 |
while (stage!=NULL)
|
sl@0
|
92 |
{
|
sl@0
|
93 |
CWsRenderStage *next=stage->Next();
|
sl@0
|
94 |
delete stage;
|
sl@0
|
95 |
stage=next;
|
sl@0
|
96 |
}
|
sl@0
|
97 |
iTimedDrawRect.Close();
|
sl@0
|
98 |
iInvalid.Close();
|
sl@0
|
99 |
iTopLayer.Close();
|
sl@0
|
100 |
iBannedRegion.Close();
|
sl@0
|
101 |
}
|
sl@0
|
102 |
|
sl@0
|
103 |
void CScreenRedraw::ConstructL()
|
sl@0
|
104 |
{
|
sl@0
|
105 |
CWsPluginManager * pluginMgr = CWsTop::WindowServer()->PluginManager();
|
sl@0
|
106 |
|
sl@0
|
107 |
// Setup the render stages for this screen:
|
sl@0
|
108 |
_LIT(KDefaultRenderStages, "std");
|
sl@0
|
109 |
_LIT(KDefaultFlickerFreeRenderStages, "flickerbuffer std");
|
sl@0
|
110 |
_LIT(KRenderStages,"RENDERSTAGES");
|
sl@0
|
111 |
TPtrC stagesString;
|
sl@0
|
112 |
const TBool customStages = WsIniFile->FindVar(iScreen.ScreenNumber(),KRenderStages,stagesString);
|
sl@0
|
113 |
|
sl@0
|
114 |
// If noone specifies stages for this screen, assume the standard implementation:
|
sl@0
|
115 |
const TDesC * stages;
|
sl@0
|
116 |
if (customStages)
|
sl@0
|
117 |
stages = &stagesString;
|
sl@0
|
118 |
else if (iScreen.OffScreenBitmap())
|
sl@0
|
119 |
stages = &KDefaultFlickerFreeRenderStages();
|
sl@0
|
120 |
else
|
sl@0
|
121 |
stages = &KDefaultRenderStages();
|
sl@0
|
122 |
|
sl@0
|
123 |
CWsRenderStage * lastStage = 0;
|
sl@0
|
124 |
|
sl@0
|
125 |
// Parse the string for implementation IDs:
|
sl@0
|
126 |
TLex lex(*stages);
|
sl@0
|
127 |
while(true)
|
sl@0
|
128 |
{
|
sl@0
|
129 |
TPtrC ptr = lex.NextToken();
|
sl@0
|
130 |
if (ptr.Length() > 0)
|
sl@0
|
131 |
{
|
sl@0
|
132 |
TInt err = KErrNone;
|
sl@0
|
133 |
MWsRenderStageFactory * factory = pluginMgr->FindNamedImplementation<MWsRenderStageFactory>(ptr);
|
sl@0
|
134 |
if (factory)
|
sl@0
|
135 |
{
|
sl@0
|
136 |
CWsRenderStage * stage = 0;
|
sl@0
|
137 |
TRAP(err, stage = factory->CreateStageL(static_cast<MWsScreen*>(&iScreen), this));
|
sl@0
|
138 |
if (err == KErrNone)
|
sl@0
|
139 |
{
|
sl@0
|
140 |
if (!stage)
|
sl@0
|
141 |
{
|
sl@0
|
142 |
err = KErrNotFound;
|
sl@0
|
143 |
}
|
sl@0
|
144 |
else
|
sl@0
|
145 |
{
|
sl@0
|
146 |
if (lastStage)
|
sl@0
|
147 |
lastStage->SetNext(stage);
|
sl@0
|
148 |
else
|
sl@0
|
149 |
iRenderStages = stage;
|
sl@0
|
150 |
lastStage = stage;
|
sl@0
|
151 |
}
|
sl@0
|
152 |
}
|
sl@0
|
153 |
}
|
sl@0
|
154 |
else
|
sl@0
|
155 |
{
|
sl@0
|
156 |
err = KErrNotFound;
|
sl@0
|
157 |
}
|
sl@0
|
158 |
|
sl@0
|
159 |
if (wsDebugLog)
|
sl@0
|
160 |
{
|
sl@0
|
161 |
TBuf<64> buf;
|
sl@0
|
162 |
if (err == KErrNone)
|
sl@0
|
163 |
{
|
sl@0
|
164 |
_LIT(KAddedRenderStage,"Added render stage: ");
|
sl@0
|
165 |
buf.Append(KAddedRenderStage);
|
sl@0
|
166 |
buf.Append(ptr);
|
sl@0
|
167 |
wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf);
|
sl@0
|
168 |
}
|
sl@0
|
169 |
else
|
sl@0
|
170 |
{
|
sl@0
|
171 |
_LIT(KMissingRenderStage,"Failed to add render stage (%d): ");
|
sl@0
|
172 |
buf.Append(KMissingRenderStage);
|
sl@0
|
173 |
buf.Append(ptr);
|
sl@0
|
174 |
wsDebugLog->MiscMessage(CDebugLogBase::ELogImportant,buf,err);
|
sl@0
|
175 |
}
|
sl@0
|
176 |
}
|
sl@0
|
177 |
}
|
sl@0
|
178 |
else
|
sl@0
|
179 |
{
|
sl@0
|
180 |
break;
|
sl@0
|
181 |
}
|
sl@0
|
182 |
}
|
sl@0
|
183 |
}
|
sl@0
|
184 |
|
sl@0
|
185 |
const TTime& CScreenRedraw::Now() const
|
sl@0
|
186 |
{
|
sl@0
|
187 |
if(!iAnimating)
|
sl@0
|
188 |
{
|
sl@0
|
189 |
iNow.UniversalTime();
|
sl@0
|
190 |
}
|
sl@0
|
191 |
return iNow;
|
sl@0
|
192 |
}
|
sl@0
|
193 |
|
sl@0
|
194 |
void CScreenRedraw::ScheduleRender(const TTimeIntervalMicroSeconds& aFromNow)
|
sl@0
|
195 |
{
|
sl@0
|
196 |
iRenderScheduled = ETrue;
|
sl@0
|
197 |
TTime then(Now() + aFromNow);
|
sl@0
|
198 |
if ((!iScheduled) || then < iNext)
|
sl@0
|
199 |
iNext = then;
|
sl@0
|
200 |
iScheduled = ETrue;
|
sl@0
|
201 |
CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
|
sl@0
|
202 |
}
|
sl@0
|
203 |
|
sl@0
|
204 |
void CScreenRedraw::ScheduleRedraw()
|
sl@0
|
205 |
{
|
sl@0
|
206 |
iNext = Now();
|
sl@0
|
207 |
iScheduled = ETrue;
|
sl@0
|
208 |
|
sl@0
|
209 |
// The other scheduler also removes future animations which this one encompasses.
|
sl@0
|
210 |
// We choose not to do the region calculations needed to achieve that here.
|
sl@0
|
211 |
MWsAnimationScheduler* animSched=CWsTop::WindowServer()->AnimationScheduler();
|
sl@0
|
212 |
if (animSched)
|
sl@0
|
213 |
{
|
sl@0
|
214 |
animSched->ScheduleRedraw(iScreen,iNext);
|
sl@0
|
215 |
}
|
sl@0
|
216 |
}
|
sl@0
|
217 |
|
sl@0
|
218 |
void CScreenRedraw::ScheduleAnimation(const TRect& aRect,const TTimeIntervalMicroSeconds& aFromNow,const TTimeIntervalMicroSeconds& /*aFreq*/,const TTimeIntervalMicroSeconds& /*aStop*/)
|
sl@0
|
219 |
{
|
sl@0
|
220 |
TRect test(aRect);
|
sl@0
|
221 |
test.Intersection(iScreen.DrawableArea());
|
sl@0
|
222 |
if(!test.IsEmpty())
|
sl@0
|
223 |
{
|
sl@0
|
224 |
const TTime then(Now() + aFromNow);
|
sl@0
|
225 |
TTimedRect tRect(aRect, then);
|
sl@0
|
226 |
|
sl@0
|
227 |
const TInt error = iTimedDrawRect.InsertInOrderAllowRepeats(tRect,TTimedRect::Compare);
|
sl@0
|
228 |
if (KErrNone == error)
|
sl@0
|
229 |
{
|
sl@0
|
230 |
if (iScheduled)
|
sl@0
|
231 |
{
|
sl@0
|
232 |
if (then < iNext)
|
sl@0
|
233 |
{
|
sl@0
|
234 |
iNext = then;
|
sl@0
|
235 |
}
|
sl@0
|
236 |
}
|
sl@0
|
237 |
else
|
sl@0
|
238 |
{
|
sl@0
|
239 |
iNext = then;
|
sl@0
|
240 |
iScheduled = ETrue;
|
sl@0
|
241 |
}
|
sl@0
|
242 |
// remove further futures that are completely contained
|
sl@0
|
243 |
TInt count = iTimedDrawRect.Count();
|
sl@0
|
244 |
for(TInt i=0; i<count; i++)
|
sl@0
|
245 |
{
|
sl@0
|
246 |
const TTimedRect& future = iTimedDrawRect[i];
|
sl@0
|
247 |
if(future.iTime.Int64() > then.Int64())
|
sl@0
|
248 |
{
|
sl@0
|
249 |
TRect rect(aRect);
|
sl@0
|
250 |
rect.BoundingRect(future.iRect);
|
sl@0
|
251 |
if(rect == aRect) // future is completely contained within aRect
|
sl@0
|
252 |
{
|
sl@0
|
253 |
iTimedDrawRect.Remove(i);
|
sl@0
|
254 |
count--;
|
sl@0
|
255 |
i--;
|
sl@0
|
256 |
}
|
sl@0
|
257 |
}
|
sl@0
|
258 |
}
|
sl@0
|
259 |
CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen,iNext);
|
sl@0
|
260 |
|
sl@0
|
261 |
// Blue rectangles for scheduled animations
|
sl@0
|
262 |
DEBUG_RECT(TRgb(0x00, 0x00, 0xFF),TRgb(0x00, 0x00, 0xFF, 0x20),&aRect);
|
sl@0
|
263 |
}
|
sl@0
|
264 |
}
|
sl@0
|
265 |
}
|
sl@0
|
266 |
|
sl@0
|
267 |
// This adds a region to the stored invalid region.
|
sl@0
|
268 |
// The invalid region is the area of the screen that needs to be redrawn in addition to any animations.
|
sl@0
|
269 |
// The draw region is the area of the screen on which only the top window needs to be redrawn.
|
sl@0
|
270 |
// If the top window has transparency, this can only be true when it has been made newly visible.
|
sl@0
|
271 |
// The value of aSchedule could be determined automatically from iAnimating, but passing it this way
|
sl@0
|
272 |
// allows us to have the assert, which is a very valuable assert.
|
sl@0
|
273 |
void CScreenRedraw::AddRedrawRegion(const TRegion& aRegion, TBool aSchedule, TRedrawDepth aDepth)
|
sl@0
|
274 |
{
|
sl@0
|
275 |
WS_ASSERT_DEBUG(!iAnimating || !aSchedule, EWsPanicScheduledRedraw);
|
sl@0
|
276 |
|
sl@0
|
277 |
if(aRegion.CheckError())
|
sl@0
|
278 |
{
|
sl@0
|
279 |
iInvalid.ForceError();
|
sl@0
|
280 |
|
sl@0
|
281 |
if (aSchedule)
|
sl@0
|
282 |
ScheduleRedraw();
|
sl@0
|
283 |
}
|
sl@0
|
284 |
else if(aRegion.Count()) // often called despite window not being visible
|
sl@0
|
285 |
{
|
sl@0
|
286 |
if (aDepth == ERedrawAll)
|
sl@0
|
287 |
{
|
sl@0
|
288 |
// red lines for an invalid region which is ready to be drawn
|
sl@0
|
289 |
DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x20),&aRegion);
|
sl@0
|
290 |
|
sl@0
|
291 |
iInvalid.Union(aRegion);
|
sl@0
|
292 |
|
sl@0
|
293 |
if (aSchedule)
|
sl@0
|
294 |
ScheduleRedraw();
|
sl@0
|
295 |
}
|
sl@0
|
296 |
else
|
sl@0
|
297 |
{
|
sl@0
|
298 |
// yellow lines for a valid region which we will draw on top of
|
sl@0
|
299 |
DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x20),&aRegion);
|
sl@0
|
300 |
|
sl@0
|
301 |
iTopLayer.Union(aRegion);
|
sl@0
|
302 |
|
sl@0
|
303 |
if (aSchedule)
|
sl@0
|
304 |
ScheduleRedraw();
|
sl@0
|
305 |
}
|
sl@0
|
306 |
}
|
sl@0
|
307 |
}
|
sl@0
|
308 |
|
sl@0
|
309 |
// This causes any asynchronously scheduled redraw to happen immediately
|
sl@0
|
310 |
// It should be avoided where possible for performance reasons, but is
|
sl@0
|
311 |
// needed whenever the redraw store is discarded for a window which still
|
sl@0
|
312 |
// has a redraw region pending.
|
sl@0
|
313 |
void CScreenRedraw::DoRedrawNow()
|
sl@0
|
314 |
{
|
sl@0
|
315 |
if(!iAnimating)
|
sl@0
|
316 |
CWsTop::WindowServer()->AnimationScheduler()->DoRedrawNow(iScreen);
|
sl@0
|
317 |
}
|
sl@0
|
318 |
|
sl@0
|
319 |
#ifdef USE_DEBUG_REGIONS
|
sl@0
|
320 |
void CScreenRedraw::DebugRect(TRgb aColor, TRgb aFill, const TRect* aRect)
|
sl@0
|
321 |
{
|
sl@0
|
322 |
if (aRect)
|
sl@0
|
323 |
{
|
sl@0
|
324 |
CFbsBitGc * gc = iScreen.GetBitGc();
|
sl@0
|
325 |
gc->SetPenColor(aColor);
|
sl@0
|
326 |
gc->SetPenStyle(CGraphicsContext::ESolidPen);
|
sl@0
|
327 |
gc->SetPenSize(TSize(2,2));
|
sl@0
|
328 |
gc->SetBrushColor(aFill);
|
sl@0
|
329 |
gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
|
sl@0
|
330 |
TRect smaller = *aRect;
|
sl@0
|
331 |
smaller.iBr.iX -= 1;
|
sl@0
|
332 |
smaller.iBr.iY -= 1;
|
sl@0
|
333 |
gc->DrawRect(smaller);
|
sl@0
|
334 |
iScreen.Update();
|
sl@0
|
335 |
}
|
sl@0
|
336 |
}
|
sl@0
|
337 |
|
sl@0
|
338 |
void CScreenRedraw::DebugRegion(TRgb aColor, TRgb aFill, const TRegion * aRegion)
|
sl@0
|
339 |
{
|
sl@0
|
340 |
if (aRegion)
|
sl@0
|
341 |
{
|
sl@0
|
342 |
CFbsBitGc * gc = iScreen.GetBitGc();
|
sl@0
|
343 |
gc->SetPenColor(aColor);
|
sl@0
|
344 |
gc->SetPenStyle(CGraphicsContext::ESolidPen);
|
sl@0
|
345 |
gc->SetPenSize(TSize(2,2));
|
sl@0
|
346 |
gc->SetBrushColor(aFill);
|
sl@0
|
347 |
gc->SetBrushStyle(CGraphicsContext::ESolidBrush);
|
sl@0
|
348 |
for (const TRect *rect = aRegion->RectangleList(); rect - aRegion->RectangleList() < aRegion->Count(); ++rect)
|
sl@0
|
349 |
{
|
sl@0
|
350 |
TRect smaller = *rect;
|
sl@0
|
351 |
smaller.iBr.iX -= 1;
|
sl@0
|
352 |
smaller.iBr.iY -= 1;
|
sl@0
|
353 |
gc->DrawRect(smaller);
|
sl@0
|
354 |
}
|
sl@0
|
355 |
iScreen.Update();
|
sl@0
|
356 |
}
|
sl@0
|
357 |
}
|
sl@0
|
358 |
#endif
|
sl@0
|
359 |
|
sl@0
|
360 |
void CScreenRedraw::OnAnimation()
|
sl@0
|
361 |
{
|
sl@0
|
362 |
LOG_SCREEN_REDRAW_START
|
sl@0
|
363 |
ASSERT(!iAnimating);
|
sl@0
|
364 |
ASSERT(iScheduled);
|
sl@0
|
365 |
iAnimating = ETrue;
|
sl@0
|
366 |
iScheduled = EFalse;
|
sl@0
|
367 |
TBool futureAnimationRequired = EFalse;
|
sl@0
|
368 |
|
sl@0
|
369 |
CWsActiveScheduler::Static()->PrepareDraw();
|
sl@0
|
370 |
|
sl@0
|
371 |
// Calculate any updates required by region changes:
|
sl@0
|
372 |
RegionUpdate();
|
sl@0
|
373 |
|
sl@0
|
374 |
// Add the timed rectangles to the invalid region:
|
sl@0
|
375 |
iNow.UniversalTime();
|
sl@0
|
376 |
TInt count(iTimedDrawRect.Count());
|
sl@0
|
377 |
while (0 < count)
|
sl@0
|
378 |
{
|
sl@0
|
379 |
if(iTimedDrawRect[0].iTime.Int64() <= iNow.Int64())
|
sl@0
|
380 |
{
|
sl@0
|
381 |
iInvalid.AddRect(iTimedDrawRect[0].iRect);
|
sl@0
|
382 |
iTimedDrawRect.Remove(0);
|
sl@0
|
383 |
count--;
|
sl@0
|
384 |
}
|
sl@0
|
385 |
else
|
sl@0
|
386 |
{
|
sl@0
|
387 |
futureAnimationRequired = ETrue;
|
sl@0
|
388 |
break;
|
sl@0
|
389 |
}
|
sl@0
|
390 |
}
|
sl@0
|
391 |
|
sl@0
|
392 |
// Animating rectangles could cause iInvalid to overlap iTopLayer,
|
sl@0
|
393 |
// in which case iTopLayer won't work.
|
sl@0
|
394 |
iTopLayer.SubRegion(iInvalid);
|
sl@0
|
395 |
iTopLayer.Intersect(iScreen.RootWindow()->WindowArea());
|
sl@0
|
396 |
iTopLayer.Tidy();
|
sl@0
|
397 |
// Anything in the top layer is implcitly invalid
|
sl@0
|
398 |
iInvalid.Union(iTopLayer);
|
sl@0
|
399 |
iInvalid.Intersect(iScreen.RootWindow()->WindowArea());
|
sl@0
|
400 |
|
sl@0
|
401 |
/*
|
sl@0
|
402 |
if (const CDebugBar* dbg = iScreen.DebugBar())
|
sl@0
|
403 |
{
|
sl@0
|
404 |
iTopLayer.SubRect(dbg->Rect());
|
sl@0
|
405 |
iInvalid.SubRect(dbg->Rect());
|
sl@0
|
406 |
}
|
sl@0
|
407 |
*/
|
sl@0
|
408 |
|
sl@0
|
409 |
iInvalid.Tidy();
|
sl@0
|
410 |
|
sl@0
|
411 |
if(iInvalid.CheckError()) //Will: agree with Andy, want bounding rects instead!
|
sl@0
|
412 |
{
|
sl@0
|
413 |
iTopLayer.Clear();
|
sl@0
|
414 |
iInvalid.Clear();
|
sl@0
|
415 |
iInvalid.Copy(iScreen.RootWindow()->WindowArea()); // assumed cannot fail, all regions can contain at least 1 rect..
|
sl@0
|
416 |
}
|
sl@0
|
417 |
|
sl@0
|
418 |
iInvalid.SubRegion( iBannedRegion );
|
sl@0
|
419 |
iInvalid.Tidy();
|
sl@0
|
420 |
iTopLayer.SubRegion( iBannedRegion );
|
sl@0
|
421 |
iTopLayer.Tidy();
|
sl@0
|
422 |
|
sl@0
|
423 |
STACK_REGION invalidCopy;
|
sl@0
|
424 |
invalidCopy.Copy(iInvalid);
|
sl@0
|
425 |
TWalkWindowTreeScheduleRegions regionScheduler(&invalidCopy, iTopLayer);
|
sl@0
|
426 |
TWalkWindowTreeScheduleFallback fallbackScheduler(iScreen.FallbackMap());
|
sl@0
|
427 |
TWalkWindowTreeSchedule * scheduler = ®ionScheduler;
|
sl@0
|
428 |
|
sl@0
|
429 |
// At this point, if the DEBUG_REGION is being used:
|
sl@0
|
430 |
// Red represents invalid regions that need to be redrawn completely.
|
sl@0
|
431 |
// Yellow represents regions that only need the top window to be drawn.
|
sl@0
|
432 |
// Blue represents regions which are being animated server side.
|
sl@0
|
433 |
if (iRenderScheduled || !iInvalid.IsEmpty())
|
sl@0
|
434 |
{
|
sl@0
|
435 |
iRenderScheduled = EFalse;
|
sl@0
|
436 |
// invalidCopy.ForceError(); //### DEBUG
|
sl@0
|
437 |
|
sl@0
|
438 |
iScreen.RootWindow()->WalkWindowTree(regionScheduler,EWalkChildren);
|
sl@0
|
439 |
if (!regionScheduler.ScheduledRegionsOk())
|
sl@0
|
440 |
{
|
sl@0
|
441 |
// our region calculations for what to draw failed at some point.
|
sl@0
|
442 |
// From this point on we MUST NOT rely on allocating memory
|
sl@0
|
443 |
// Andy - localRedrawRegion allocates
|
sl@0
|
444 |
// Andy - setPenSize allocates (even if you don't call it)
|
sl@0
|
445 |
// Andy - all draw commands add to a gdi dirty region (which allocates)
|
sl@0
|
446 |
// Andy - combining client clipping regions with window clipping regions allocates
|
sl@0
|
447 |
scheduler = &fallbackScheduler;
|
sl@0
|
448 |
iScreen.FallbackMap()->Prepare();
|
sl@0
|
449 |
iScreen.RootWindow()->WalkWindowTree(fallbackScheduler,EWalkChildren);
|
sl@0
|
450 |
}
|
sl@0
|
451 |
|
sl@0
|
452 |
CWsActiveScheduler::Static()->StartDraw();
|
sl@0
|
453 |
CWsMemoryManager::Static()->EnableReserve();
|
sl@0
|
454 |
|
sl@0
|
455 |
if (&fallbackScheduler == scheduler)
|
sl@0
|
456 |
iAnimationRegion = iScreen.FallbackMap()->Region();
|
sl@0
|
457 |
else
|
sl@0
|
458 |
iAnimationRegion = &iInvalid;
|
sl@0
|
459 |
|
sl@0
|
460 |
// Redraw debug regions more brightly than before:
|
sl@0
|
461 |
DEBUG_REGION(TRgb(0xFF, 0x00, 0x00),TRgb(0xFF, 0x00, 0x00, 0x80),&iInvalid);
|
sl@0
|
462 |
DEBUG_REGION(TRgb(0xFF, 0xFF, 0x00),TRgb(0xFF, 0xFF, 0x00, 0x80),&iTopLayer);
|
sl@0
|
463 |
|
sl@0
|
464 |
RWsRegion accumulatedDrawing;
|
sl@0
|
465 |
|
sl@0
|
466 |
// Pipe the drawing into the first render stage:
|
sl@0
|
467 |
CFbsBitGc * stageGc = iRenderStages->Begin();
|
sl@0
|
468 |
|
sl@0
|
469 |
for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
|
sl@0
|
470 |
{
|
sl@0
|
471 |
const TRegion * targetRegion = scheduler->Region(win);
|
sl@0
|
472 |
const TRect * screenRect = 0;
|
sl@0
|
473 |
if ((&fallbackScheduler == scheduler) && !targetRegion->IsContainedBy(iScreen.RootWindow()->Abs()))
|
sl@0
|
474 |
{
|
sl@0
|
475 |
screenRect = &iScreen.RootWindow()->Abs();
|
sl@0
|
476 |
}
|
sl@0
|
477 |
if (!screenRect)
|
sl@0
|
478 |
{
|
sl@0
|
479 |
// Purple regions are about to be drawn
|
sl@0
|
480 |
DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),targetRegion);
|
sl@0
|
481 |
// Do the drawing
|
sl@0
|
482 |
stageGc->Reset();
|
sl@0
|
483 |
win->Render(stageGc, *targetRegion);
|
sl@0
|
484 |
accumulatedDrawing.Union(*targetRegion);
|
sl@0
|
485 |
// Green regions have been drawn
|
sl@0
|
486 |
DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),targetRegion);
|
sl@0
|
487 |
}
|
sl@0
|
488 |
else
|
sl@0
|
489 |
{
|
sl@0
|
490 |
// Our region overlaps the edges of the screen, and we have no memory
|
sl@0
|
491 |
// to create a clipped one, so we will use a single-rect region for each rect
|
sl@0
|
492 |
// and call Draw multiple times.
|
sl@0
|
493 |
TRegionFix<1> rectRegion;
|
sl@0
|
494 |
for (const TRect * rect = targetRegion->RectangleList() + targetRegion->Count() - 1; rect >= targetRegion->RectangleList(); --rect)
|
sl@0
|
495 |
{
|
sl@0
|
496 |
rectRegion.Clear();
|
sl@0
|
497 |
TRect combined(*screenRect);
|
sl@0
|
498 |
combined.Intersection(*rect);
|
sl@0
|
499 |
rectRegion.AddRect(combined);
|
sl@0
|
500 |
// Purple regions are about to be drawn
|
sl@0
|
501 |
DEBUG_REGION(TRgb(0x80, 0x00, 0x80),TRgb(0x80, 0x00, 0x80, 0x80),&rectRegion);
|
sl@0
|
502 |
// Do the drawing
|
sl@0
|
503 |
stageGc->Reset();
|
sl@0
|
504 |
win->Render(stageGc, rectRegion);
|
sl@0
|
505 |
accumulatedDrawing.AddRect(combined);
|
sl@0
|
506 |
// Green regions have been drawn
|
sl@0
|
507 |
DEBUG_REGION(TRgb(0x00, 0xFF, 0x00),TRgb(0x00, 0xFF, 0x00, 0x80),&rectRegion);
|
sl@0
|
508 |
}
|
sl@0
|
509 |
}
|
sl@0
|
510 |
DEBUGOSB
|
sl@0
|
511 |
}
|
sl@0
|
512 |
|
sl@0
|
513 |
DEBUGOSB
|
sl@0
|
514 |
iScreen.SpriteManager()->DrawFloatingSprites(stageGc,accumulatedDrawing); // we limit sprite drawing over only actually affected ares but necessary to all "planned" for redraw
|
sl@0
|
515 |
if (!accumulatedDrawing.CheckError() && iScreen.SpriteManager()->SpriteCount() == 0)
|
sl@0
|
516 |
{
|
sl@0
|
517 |
iAnimationRegion = &accumulatedDrawing;
|
sl@0
|
518 |
}
|
sl@0
|
519 |
|
sl@0
|
520 |
// Tell the render stage we've finished:
|
sl@0
|
521 |
iRenderStages->End();
|
sl@0
|
522 |
|
sl@0
|
523 |
// We nolonger need the regions
|
sl@0
|
524 |
for (CWsWindow * win = scheduler->HeadWindow(); win; win = win->NextScheduled())
|
sl@0
|
525 |
{
|
sl@0
|
526 |
win->ClearScheduledRegion();
|
sl@0
|
527 |
}
|
sl@0
|
528 |
|
sl@0
|
529 |
CWsMemoryManager::Static()->DisableReserve();
|
sl@0
|
530 |
|
sl@0
|
531 |
if (const CDebugBar* dbg = iScreen.DebugBar())
|
sl@0
|
532 |
{
|
sl@0
|
533 |
dbg->RedrawDebugBar();
|
sl@0
|
534 |
}
|
sl@0
|
535 |
|
sl@0
|
536 |
// At this point, if the DEBUG_REGION is being used, there should usually be only green regions
|
sl@0
|
537 |
// displayed. If we see red or yellow, then something didn't get redrawn that should have been.
|
sl@0
|
538 |
// If we see purple then a window has disobeyed the const setting on the region to draw.
|
sl@0
|
539 |
// Red or yellow is valid - a window can decide it isn't ready to draw yet - purple is very bad.
|
sl@0
|
540 |
iScreen.Update();
|
sl@0
|
541 |
// At this point, if the DEBUG_REGION is being used, there should be no regions visible
|
sl@0
|
542 |
// of any colour. If we see green, then it means an area of the screen was drawn to which
|
sl@0
|
543 |
// wasn't invalid, or the screen update call failed. The former is more likely.
|
sl@0
|
544 |
// If we still see red or yellow it is a region that is not yet ready to draw.
|
sl@0
|
545 |
|
sl@0
|
546 |
const TRect* rect = iAnimationRegion->RectangleList();
|
sl@0
|
547 |
TInt pixels = 0;
|
sl@0
|
548 |
for(TInt r = iAnimationRegion->Count(); r>0; r--)
|
sl@0
|
549 |
{
|
sl@0
|
550 |
pixels += (rect->Width()*rect->Height());
|
sl@0
|
551 |
rect++;
|
sl@0
|
552 |
}
|
sl@0
|
553 |
CWsActiveScheduler::Static()->StopDraw(pixels);
|
sl@0
|
554 |
|
sl@0
|
555 |
TWindowServerEvent::NotifyScreenDrawingEvent(iAnimationRegion);
|
sl@0
|
556 |
|
sl@0
|
557 |
iAnimationRegion = 0;
|
sl@0
|
558 |
accumulatedDrawing.Close();
|
sl@0
|
559 |
iInvalid.Clear();
|
sl@0
|
560 |
}
|
sl@0
|
561 |
else
|
sl@0
|
562 |
{
|
sl@0
|
563 |
CWsActiveScheduler::Static()->CancelPrepare();
|
sl@0
|
564 |
}
|
sl@0
|
565 |
|
sl@0
|
566 |
iInvalid.Clear();
|
sl@0
|
567 |
iTopLayer.Clear();
|
sl@0
|
568 |
invalidCopy.Close();
|
sl@0
|
569 |
iAnimating = EFalse;
|
sl@0
|
570 |
|
sl@0
|
571 |
if (futureAnimationRequired && iTimedDrawRect.Count() > 0 && !iScheduled)
|
sl@0
|
572 |
{
|
sl@0
|
573 |
// If this flag is set then it means there were already some animations scheduled when we ran,
|
sl@0
|
574 |
// but they themselves didn't run. We need to make sure we have _something_ scheduled.
|
sl@0
|
575 |
CWsTop::WindowServer()->AnimationScheduler()->ScheduleAnimation(iScreen, iTimedDrawRect[0].iTime);
|
sl@0
|
576 |
iScheduled = ETrue;
|
sl@0
|
577 |
}
|
sl@0
|
578 |
|
sl@0
|
579 |
if(iObserver)
|
sl@0
|
580 |
{
|
sl@0
|
581 |
iObserver->ScreenUpdated(iScreen.ScreenNumber());
|
sl@0
|
582 |
iObserver=NULL; //once signalled we are never going to call it again
|
sl@0
|
583 |
}
|
sl@0
|
584 |
LOG_SCREEN_REDRAW_END
|
sl@0
|
585 |
}
|
sl@0
|
586 |
|
sl@0
|
587 |
//
|
sl@0
|
588 |
void CScreenRedraw::DiscardAllSchedules()
|
sl@0
|
589 |
{
|
sl@0
|
590 |
ASSERT(!iAnimating);
|
sl@0
|
591 |
|
sl@0
|
592 |
iTimedDrawRect.Reset();
|
sl@0
|
593 |
iInvalid.Clear();
|
sl@0
|
594 |
iInvalid.Tidy();
|
sl@0
|
595 |
}
|
sl@0
|
596 |
|
sl@0
|
597 |
/**
|
sl@0
|
598 |
Indicates that a window has moved or changed ordinal position so that the visible regions
|
sl@0
|
599 |
of all windows needs to be recalculated
|
sl@0
|
600 |
*/
|
sl@0
|
601 |
void CScreenRedraw::ScheduleRegionUpdate(const TRegion* aDefinitelyDirty)
|
sl@0
|
602 |
{
|
sl@0
|
603 |
iRegionUpdateScheduled = ETrue;
|
sl@0
|
604 |
ScheduleRedraw();
|
sl@0
|
605 |
if(aDefinitelyDirty)
|
sl@0
|
606 |
{
|
sl@0
|
607 |
iInvalid.Union(*aDefinitelyDirty);
|
sl@0
|
608 |
// Cyan regions for invalidations caused by this:
|
sl@0
|
609 |
DEBUG_REGION(TRgb(0x00, 0xFF, 0xFF),TRgb(0x00, 0xFF, 0xFF, 0x20),aDefinitelyDirty);
|
sl@0
|
610 |
}
|
sl@0
|
611 |
}
|
sl@0
|
612 |
|
sl@0
|
613 |
/**
|
sl@0
|
614 |
Recalculates visible regions and schedules redraws or queues redraw events in response to
|
sl@0
|
615 |
any changes
|
sl@0
|
616 |
*/
|
sl@0
|
617 |
void CScreenRedraw::RegionUpdate()
|
sl@0
|
618 |
{
|
sl@0
|
619 |
if (iRegionUpdateScheduled)
|
sl@0
|
620 |
{
|
sl@0
|
621 |
iRegionUpdateScheduled = EFalse;
|
sl@0
|
622 |
|
sl@0
|
623 |
TWalkWindowTreeUpdateRegions updater(iScreen);
|
sl@0
|
624 |
updater.Walk();
|
sl@0
|
625 |
|
sl@0
|
626 |
WsPointer::ReLogCurrentWindow();
|
sl@0
|
627 |
CWsTop::TriggerRedraws(iScreen.RootWindow());
|
sl@0
|
628 |
}
|
sl@0
|
629 |
}
|
sl@0
|
630 |
|
sl@0
|
631 |
void CScreenRedraw::SetObserver(MWsScreenRedrawObserver* aObserver)
|
sl@0
|
632 |
{
|
sl@0
|
633 |
iObserver = aObserver;
|
sl@0
|
634 |
}
|
sl@0
|
635 |
|
sl@0
|
636 |
TBool CScreenRedraw::IsUpdatePending()
|
sl@0
|
637 |
{
|
sl@0
|
638 |
if(iScheduled || iAnimating)
|
sl@0
|
639 |
return ETrue;
|
sl@0
|
640 |
else
|
sl@0
|
641 |
return EFalse;
|
sl@0
|
642 |
}
|
sl@0
|
643 |
|
sl@0
|
644 |
/**
|
sl@0
|
645 |
Overidding MWsObjectProvider
|
sl@0
|
646 |
*/
|
sl@0
|
647 |
TAny* CScreenRedraw::ResolveObjectInterface(TUint aTypeId)
|
sl@0
|
648 |
{
|
sl@0
|
649 |
TAny* interface = NULL;
|
sl@0
|
650 |
|
sl@0
|
651 |
switch (aTypeId)
|
sl@0
|
652 |
{
|
sl@0
|
653 |
case KWsScreenRedraw:
|
sl@0
|
654 |
interface = static_cast<MWsScreenRedraw*>(this);
|
sl@0
|
655 |
break;
|
sl@0
|
656 |
}
|
sl@0
|
657 |
|
sl@0
|
658 |
if (!interface)
|
sl@0
|
659 |
{
|
sl@0
|
660 |
interface = iRenderStages->ResolveObjectInterface(aTypeId);
|
sl@0
|
661 |
}
|
sl@0
|
662 |
|
sl@0
|
663 |
return interface;
|
sl@0
|
664 |
}
|
sl@0
|
665 |
|
sl@0
|
666 |
const TRegion * CScreenRedraw::AnimationRegion() const
|
sl@0
|
667 |
{
|
sl@0
|
668 |
if (iAnimating)
|
sl@0
|
669 |
return iAnimationRegion;
|
sl@0
|
670 |
else
|
sl@0
|
671 |
return 0;
|
sl@0
|
672 |
}
|
sl@0
|
673 |
|
sl@0
|
674 |
void CScreenRedraw::UpdateDevice()
|
sl@0
|
675 |
{
|
sl@0
|
676 |
iScreen.Update();
|
sl@0
|
677 |
}
|
sl@0
|
678 |
|
sl@0
|
679 |
void CScreenRedraw::BanThisRegionUpdate( TRegion& aForbiddenRegion )
|
sl@0
|
680 |
{
|
sl@0
|
681 |
iBannedRegion.Union( aForbiddenRegion );
|
sl@0
|
682 |
iBannedRegion.Tidy();
|
sl@0
|
683 |
}
|
sl@0
|
684 |
|
sl@0
|
685 |
void CScreenRedraw::LiftRegionUpdateBan( TRegion& aFormerForbiddenRegion )
|
sl@0
|
686 |
{
|
sl@0
|
687 |
iBannedRegion.SubRegion( aFormerForbiddenRegion );
|
sl@0
|
688 |
iBannedRegion.Tidy();
|
sl@0
|
689 |
}
|
sl@0
|
690 |
|
sl@0
|
691 |
void CScreenRedraw::AcceptFadeRequest( CWsWindow* aWin, TBool aFadeOn, TBool aFadeBehind, TBool aIncludeChildren )
|
sl@0
|
692 |
{
|
sl@0
|
693 |
if ( aFadeOn )
|
sl@0
|
694 |
{
|
sl@0
|
695 |
TWalkWindowTreeScheduleFadeNoRedraw walkerFadeNoRedraw;
|
sl@0
|
696 |
if ( aFadeBehind )
|
sl@0
|
697 |
{
|
sl@0
|
698 |
aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkBehind );
|
sl@0
|
699 |
}
|
sl@0
|
700 |
else
|
sl@0
|
701 |
{
|
sl@0
|
702 |
if(aWin->WinType() != EWinTypeGroup)
|
sl@0
|
703 |
{
|
sl@0
|
704 |
ScheduleRegionUpdate( aWin->VisibleRegionIfValid() );
|
sl@0
|
705 |
}
|
sl@0
|
706 |
|
sl@0
|
707 |
if ( aIncludeChildren )
|
sl@0
|
708 |
{
|
sl@0
|
709 |
aWin->WalkWindowTree( walkerFadeNoRedraw, EWalkChildren );
|
sl@0
|
710 |
}
|
sl@0
|
711 |
}
|
sl@0
|
712 |
}
|
sl@0
|
713 |
else
|
sl@0
|
714 |
{ // fade off, just initiate redraw
|
sl@0
|
715 |
TWalkWindowTreeScheduleRedraws walkerRedraw( TWalkWindowTreeScheduleRedraws::ERedrawFilterOmitDSA ) ;
|
sl@0
|
716 |
if ( aFadeBehind )
|
sl@0
|
717 |
{
|
sl@0
|
718 |
aWin->WalkWindowTree( walkerRedraw, EWalkBehind );
|
sl@0
|
719 |
}
|
sl@0
|
720 |
else
|
sl@0
|
721 |
{ // fade this win not behind
|
sl@0
|
722 |
if ( !aWin->IsDSAHost() )
|
sl@0
|
723 |
{
|
sl@0
|
724 |
AddRedrawRegion( aWin->VisibleRegion() );
|
sl@0
|
725 |
}
|
sl@0
|
726 |
if ( aIncludeChildren )
|
sl@0
|
727 |
{
|
sl@0
|
728 |
aWin->WalkWindowTree( walkerRedraw, EWalkChildren );
|
sl@0
|
729 |
}
|
sl@0
|
730 |
}
|
sl@0
|
731 |
}
|
sl@0
|
732 |
ScheduleRegionUpdate(NULL);
|
sl@0
|
733 |
}
|