1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
|
/*
* Copyright (C) 2005-2018 Team Kodi
* This file is part of Kodi - https://kodi.tv
*
* SPDX-License-Identifier: GPL-2.0-or-later
* See LICENSES/README.md for more information.
*/
#pragma once
#include "DirtyRegionTracker.h"
#include "GUIWindow.h"
#include "IMsgTargetCallback.h"
#include "IWindowManagerCallback.h"
#include "guilib/WindowIDs.h"
#include "messaging/IMessageTarget.h"
#include <list>
#include <unordered_map>
#include <utility>
#include <vector>
class CGUIDialog;
class CGUIMediaWindow;
#ifdef TARGET_WINDOWS_STORE
#pragma pack(push, 8)
#endif
enum class DialogModalityType;
#ifdef TARGET_WINDOWS_STORE
#pragma pack(pop)
#endif
namespace KODI
{
namespace MESSAGING
{
class CApplicationMessenger;
}
}
#define WINDOW_ID_MASK 0xffff
/*!
\ingroup winman
\brief
*/
class CGUIWindowManager : public KODI::MESSAGING::IMessageTarget
{
friend CGUIDialog;
friend CGUIMediaWindow;
public:
CGUIWindowManager();
~CGUIWindowManager() override;
bool SendMessage(CGUIMessage& message);
bool SendMessage(int message, int senderID, int destID, int param1 = 0, int param2 = 0);
bool SendMessage(CGUIMessage& message, int window);
void Initialize();
void Add(CGUIWindow* pWindow);
void AddUniqueInstance(CGUIWindow *window);
void AddCustomWindow(CGUIWindow* pWindow);
void Remove(int id);
void Delete(int id);
void ActivateWindow(int iWindowID, const std::string &strPath = "");
void ForceActivateWindow(int iWindowID, const std::string &strPath = "");
void ChangeActiveWindow(int iNewID, const std::string &strPath = "");
void ActivateWindow(int iWindowID, const std::vector<std::string>& params, bool swappingWindows = false, bool force = false);
void PreviousWindow();
/**
* \brief Switch window to fullscreen
*
* \param force enforce fullscreen switch
* \return True if switch is made to fullscreen, otherwise False
*/
bool SwitchToFullScreen(bool force = false);
void CloseDialogs(bool forceClose = false) const;
void CloseInternalModalDialogs(bool forceClose = false) const;
void OnApplicationMessage(KODI::MESSAGING::ThreadMessage* pMsg) override;
int GetMessageMask() override;
// OnAction() runs through our active dialogs and windows and sends the message
// off to the callbacks (application, python, playlist player) and to the
// currently focused window(s). Returns true only if the message is handled.
bool OnAction(const CAction &action) const;
/*! \brief Process active controls allowing them to animate before rendering.
*/
void Process(unsigned int currentTime);
/*! \brief Mark the screen as dirty, forcing a redraw at the next Render()
*/
void MarkDirty();
/*! \brief Mark a region as dirty, forcing a redraw at the next Render()
*/
void MarkDirty(const CRect& rect);
/*! \brief Rendering of the current window and any dialogs
Render is called every frame to draw the current window and any dialogs.
It should only be called from the application thread.
Returns true only if it has rendered something.
*/
bool Render();
void RenderEx() const;
/*! \brief Do any post render activities.
*/
void AfterRender();
/*! \brief Per-frame updating of the current window and any dialogs
FrameMove is called every frame to update the current window and any dialogs
on screen. It should only be called from the application thread.
*/
void FrameMove();
/*! \brief Return whether the window manager is initialized.
The window manager is initialized on skin load - if the skin isn't yet loaded,
no windows should be able to be initialized.
\return true if the window manager is initialized, false otherwise.
*/
bool Initialized() const { return m_initialized; }
/*! \brief Create and initialize all windows and dialogs
*/
void CreateWindows();
/*! \brief Destroy and remove all windows and dialogs
*
* \return true on success, false if destruction fails for any window
*/
bool DestroyWindows();
/*! \brief Destroy and remove the window or dialog with the given id
*
*\param id the window id
*/
void DestroyWindow(int id);
/*! \brief Return the window of type \code{T} with the given id or
* null if no window exists with the given id.
*
* \tparam T the window class type
* \param id the window id
* \return the window with for the given type \code{T} or null
*/
template<typename T,
typename std::enable_if<std::is_base_of<CGUIWindow, T>::value>::type* = nullptr>
T* GetWindow(int id) const
{
return dynamic_cast<T*>(GetWindow(id));
}
/*! \brief Return the window with the given id or null.
*
* \param id the window id
* \return the window with the given id or null
*/
CGUIWindow* GetWindow(int id) const;
/*! \brief Return the dialog window with the given id or null.
*
* \param id the dialog window id
* \return the dialog window with the given id or null
*/
CGUIDialog* GetDialog(int id) const;
void SetCallback(IWindowManagerCallback& callback);
void DeInitialize();
/*! \brief Register a dialog as active dialog
*
* \param dialog The dialog to register as active dialog
*/
void RegisterDialog(CGUIWindow* dialog);
void RemoveDialog(int id);
/*! \brief Get the ID of the topmost dialog
*
* \param ignoreClosing ignore dialog is closing
* \return the ID of the topmost dialog or WINDOW_INVALID if no dialog is active
*/
int GetTopmostDialog(bool ignoreClosing = false) const;
/*! \brief Get the ID of the topmost modal dialog
*
* \param ignoreClosing ignore dialog is closing
* \return the ID of the topmost modal dialog or WINDOW_INVALID if no modal dialog is active
*/
int GetTopmostModalDialog(bool ignoreClosing = false) const;
void SendThreadMessage(CGUIMessage& message, int window = 0);
void DispatchThreadMessages();
// method to removed queued messages with message id in the requested message id list.
// pMessageIDList: point to first integer of a 0 ends integer array.
int RemoveThreadMessageByMessageIds(int *pMessageIDList);
void AddMsgTarget( IMsgTargetCallback* pMsgTarget );
int GetActiveWindow() const;
int GetActiveWindowOrDialog() const;
bool HasModalDialog(bool ignoreClosing) const;
bool HasVisibleModalDialog() const;
bool IsDialogTopmost(int id, bool modal = false) const;
bool IsDialogTopmost(const std::string &xmlFile, bool modal = false) const;
bool IsModalDialogTopmost(int id) const;
bool IsModalDialogTopmost(const std::string &xmlFile) const;
bool IsWindowActive(int id, bool ignoreClosing = true) const;
bool IsWindowVisible(int id) const;
bool IsWindowActive(const std::string &xmlFile, bool ignoreClosing = true) const;
bool IsWindowVisible(const std::string &xmlFile) const;
/*! \brief Checks if the given window is an addon window.
*
* \return true if the given window is an addon window, otherwise false.
*/
bool IsAddonWindow(int id) const { return (id >= WINDOW_ADDON_START && id <= WINDOW_ADDON_END); }
/*! \brief Checks if the given window is a python window.
*
* \return true if the given window is a python window, otherwise false.
*/
bool IsPythonWindow(int id) const
{
return (id >= WINDOW_PYTHON_START && id <= WINDOW_PYTHON_END);
}
bool HasVisibleControls();
#ifdef _DEBUG
void DumpTextureUse();
#endif
private:
void RenderPass() const;
void LoadNotOnDemandWindows();
void UnloadNotOnDemandWindows();
void AddToWindowHistory(int newWindowID);
/*!
\brief Check if the given window id is in the window history, and if so, remove this
window and all overlying windows from the history so that we always have a predictable
"Back" behaviour for each window.
\param windowID the window id to remove from the window history
*/
void RemoveFromWindowHistory(int windowID);
void ClearWindowHistory();
void CloseWindowSync(CGUIWindow *window, int nextWindowID = 0);
int GetTopmostDialog(bool modal, bool ignoreClosing) const;
friend class KODI::MESSAGING::CApplicationMessenger;
/*! \brief Activate the given window.
*
* \param windowID The window ID to activate.
* \param params Parameter
* \param swappingWindows True if the window should be swapped with the previous window instead of put it in the window history, otherwise false
* \param force True to ignore checks which refuses opening the window, otherwise false
*/
void ActivateWindow_Internal(int windowID, const std::vector<std::string> ¶ms, bool swappingWindows, bool force = false);
bool ProcessRenderLoop(bool renderOnly);
bool HandleAction(const CAction &action) const;
std::unordered_map<int, CGUIWindow*> m_mapWindows;
std::vector<CGUIWindow*> m_vecCustomWindows;
std::vector<CGUIWindow*> m_activeDialogs;
std::vector<CGUIWindow*> m_deleteWindows;
std::deque<int> m_windowHistory;
IWindowManagerCallback* m_pCallback;
std::list< std::pair<CGUIMessage*,int> > m_vecThreadMessages;
CCriticalSection m_critSection;
std::vector<IMsgTargetCallback*> m_vecMsgTargets;
int m_iNested;
bool m_initialized;
mutable bool m_touchGestureActive{false};
mutable bool m_inhibitTouchGestureEvents{false};
CDirtyRegionList m_dirtyregions;
CDirtyRegionTracker m_tracker;
};
|