|
1 /* This file is part of the KDE project. |
|
2 |
|
3 Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). |
|
4 |
|
5 This library is free software: you can redistribute it and/or modify |
|
6 it under the terms of the GNU Lesser General Public License as published by |
|
7 the Free Software Foundation, either version 2.1 or 3 of the License. |
|
8 |
|
9 This library is distributed in the hope that it will be useful, |
|
10 but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
12 GNU Lesser General Public License for more details. |
|
13 |
|
14 You should have received a copy of the GNU Lesser General Public License |
|
15 along with this library. If not, see <http://www.gnu.org/licenses/>. |
|
16 */ |
|
17 |
|
18 #ifndef PHONON_MEDIAOBJECT_H |
|
19 #define PHONON_MEDIAOBJECT_H |
|
20 |
|
21 #include <phonon/mediaobjectinterface.h> |
|
22 #include <phonon/addoninterface.h> |
|
23 |
|
24 #include <QtCore/QHash> |
|
25 #include <QtCore/QObject> |
|
26 #include <QtCore/QQueue> |
|
27 #include <QtCore/QBasicTimer> |
|
28 #include <QtCore/QMutex> |
|
29 #include <QtCore/QThread> |
|
30 |
|
31 #include "backendnode.h" |
|
32 #include "mediagraph.h" |
|
33 |
|
34 QT_BEGIN_NAMESPACE |
|
35 |
|
36 namespace Phonon |
|
37 { |
|
38 class MediaSource; |
|
39 |
|
40 namespace DS9 |
|
41 { |
|
42 class VideoWidget; |
|
43 class AudioOutput; |
|
44 |
|
45 class QWinWaitCondition |
|
46 { |
|
47 public: |
|
48 QWinWaitCondition() : m_handle(::CreateEvent(0,0,0,0)) |
|
49 { |
|
50 } |
|
51 |
|
52 ~QWinWaitCondition() |
|
53 { |
|
54 ::CloseHandle(m_handle); |
|
55 } |
|
56 |
|
57 void reset() |
|
58 { |
|
59 //will block |
|
60 ::ResetEvent(m_handle); |
|
61 } |
|
62 |
|
63 void set() |
|
64 { |
|
65 //will unblock |
|
66 ::SetEvent(m_handle); |
|
67 } |
|
68 |
|
69 operator HANDLE() |
|
70 { |
|
71 return m_handle; |
|
72 } |
|
73 |
|
74 operator HEVENT() |
|
75 { |
|
76 return reinterpret_cast<HEVENT>(m_handle); |
|
77 } |
|
78 |
|
79 |
|
80 private: |
|
81 HANDLE m_handle; |
|
82 }; |
|
83 |
|
84 class WorkerThread : public QThread |
|
85 { |
|
86 Q_OBJECT |
|
87 public: |
|
88 WorkerThread(); |
|
89 ~WorkerThread(); |
|
90 |
|
91 virtual void run(); |
|
92 |
|
93 //wants to know as soon as the state is set |
|
94 void addStateChangeRequest(Graph graph, OAFilterState, QList<Filter> = QList<Filter>()); |
|
95 |
|
96 quint16 addSeekRequest(Graph graph, qint64 time); |
|
97 quint16 addUrlToRender(const QString &url); |
|
98 quint16 addFilterToRender(const Filter &filter); |
|
99 |
|
100 void replaceGraphForEventManagement(Graph newGraph, Graph oldGraph); |
|
101 |
|
102 void abortCurrentRender(qint16 renderId); |
|
103 |
|
104 //tells the thread to stop processing |
|
105 void signalStop(); |
|
106 |
|
107 Q_SIGNALS: |
|
108 void asyncRenderFinished(quint16, HRESULT, Graph); |
|
109 void asyncSeekingFinished(quint16, qint64); |
|
110 void stateReady(Graph, Phonon::State); |
|
111 void eventReady(Graph, long eventCode, long param1); |
|
112 |
|
113 private: |
|
114 |
|
115 enum Task |
|
116 { |
|
117 None, |
|
118 Render, |
|
119 Seek, |
|
120 ChangeState, |
|
121 ReplaceGraph //just updates recalls WaitForMultipleObject |
|
122 }; |
|
123 |
|
124 struct Work |
|
125 { |
|
126 Work() : task(None), id(0), time(0) { } |
|
127 Task task; |
|
128 quint16 id; |
|
129 Graph graph; |
|
130 Graph oldGraph; |
|
131 Filter filter; |
|
132 QString url; |
|
133 union |
|
134 { |
|
135 qint64 time; |
|
136 OAFilterState state; |
|
137 }; |
|
138 QList<Filter> decoders; //for the state change requests |
|
139 }; |
|
140 void handleTask(); |
|
141 |
|
142 Work m_currentWork; |
|
143 QQueue<Work> m_queue; |
|
144 bool m_finished; |
|
145 quint16 m_currentWorkId; |
|
146 QWinWaitCondition m_waitCondition; |
|
147 QMutex m_mutex; // mutex for the m_queue, m_finished and m_currentWorkId |
|
148 |
|
149 //this is for WaitForMultipleObjects |
|
150 struct |
|
151 { |
|
152 Graph graph; |
|
153 HANDLE handle; |
|
154 } m_graphHandle[FILTER_COUNT]; |
|
155 }; |
|
156 |
|
157 |
|
158 class MediaObject : public BackendNode, public Phonon::MediaObjectInterface |
|
159 #ifndef QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
160 , public Phonon::AddonInterface |
|
161 #endif //QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
162 { |
|
163 friend class Stream; |
|
164 Q_OBJECT |
|
165 Q_INTERFACES(Phonon::MediaObjectInterface |
|
166 #ifndef QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
167 Phonon::AddonInterface |
|
168 #endif //QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
169 ) |
|
170 public: |
|
171 MediaObject(QObject *parent); |
|
172 ~MediaObject(); |
|
173 Phonon::State state() const; |
|
174 bool hasVideo() const; |
|
175 bool isSeekable() const; |
|
176 qint64 currentTime() const; |
|
177 qint32 tickInterval() const; |
|
178 |
|
179 void setTickInterval(qint32 newTickInterval); |
|
180 void play(); |
|
181 void pause(); |
|
182 void stop(); |
|
183 void ensureStopped(); |
|
184 void seek(qint64 time); |
|
185 |
|
186 QString errorString() const; |
|
187 Phonon::ErrorType errorType() const; |
|
188 |
|
189 #ifndef QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
190 bool hasInterface(Interface) const; |
|
191 QVariant interfaceCall(Interface iface, int command, const QList<QVariant> ¶ms); |
|
192 #endif //QT_NO_PHONON_ABSTRACTMEDIASTREAM |
|
193 |
|
194 qint64 totalTime() const; |
|
195 qint32 prefinishMark() const; |
|
196 void setPrefinishMark(qint32 newPrefinishMark); |
|
197 |
|
198 qint32 transitionTime() const; |
|
199 void setTransitionTime(qint32); |
|
200 |
|
201 qint64 remainingTime() const; |
|
202 |
|
203 MediaSource source() const; |
|
204 void setSource(const MediaSource &source); |
|
205 void setNextSource(const MediaSource &source); |
|
206 |
|
207 |
|
208 //COM error management |
|
209 bool catchComError(HRESULT hr); |
|
210 |
|
211 void grabNode(BackendNode *node); |
|
212 bool connectNodes(BackendNode *source, BackendNode *sink); |
|
213 bool disconnectNodes(BackendNode *source, BackendNode *sink); |
|
214 |
|
215 void switchFilters(int index, Filter oldFilter, Filter newFilter); |
|
216 |
|
217 WorkerThread *workerThread(); |
|
218 void loadingFinished(MediaGraph *mg); |
|
219 void seekingFinished(MediaGraph *mg); |
|
220 MediaGraph *currentGraph() const; |
|
221 |
|
222 //this is used by the backend only |
|
223 Phonon::State transactionState; |
|
224 |
|
225 private Q_SLOTS: |
|
226 void switchToNextSource(); |
|
227 void slotStateReady(Graph, Phonon::State); |
|
228 void handleEvents(Graph, long eventCode, long param1); |
|
229 void finishLoading(quint16 workId, HRESULT hr, Graph); |
|
230 void finishSeeking(quint16 workId, qint64 time); |
|
231 |
|
232 Q_SIGNALS: |
|
233 void stateChanged(Phonon::State newstate, Phonon::State oldstate); |
|
234 void tick(qint64 time); |
|
235 void metaDataChanged(QMultiMap<QString, QString>); |
|
236 void seekableChanged(bool); |
|
237 void hasVideoChanged(bool); |
|
238 void bufferStatus(int); |
|
239 |
|
240 // AddonInterface: |
|
241 void titleChanged(int); |
|
242 void availableTitlesChanged(int); |
|
243 void chapterChanged(int); |
|
244 void availableChaptersChanged(int); |
|
245 void angleChanged(int); |
|
246 void availableAnglesChanged(int); |
|
247 |
|
248 void finished(); |
|
249 void prefinishMarkReached(qint32); |
|
250 void aboutToFinish(); |
|
251 void totalTimeChanged(qint64 length) const; |
|
252 void currentSourceChanged(const MediaSource &); |
|
253 |
|
254 protected: |
|
255 void setState(Phonon::State); |
|
256 void timerEvent(QTimerEvent *e); |
|
257 |
|
258 private: |
|
259 #ifndef QT_NO_PHONON_VIDEO |
|
260 void updateVideoGeometry(); |
|
261 #endif // QT_NO_PHONON_VIDEO |
|
262 void handleComplete(IGraphBuilder *graph); |
|
263 MediaGraph *nextGraph() const; |
|
264 |
|
265 void updateTargetTick(); |
|
266 void updateStopPosition(); |
|
267 |
|
268 mutable QString m_errorString; |
|
269 mutable Phonon::ErrorType m_errorType; |
|
270 |
|
271 Phonon::State m_state; |
|
272 Phonon::State m_nextState; |
|
273 qint32 m_transitionTime; |
|
274 |
|
275 qint32 m_prefinishMark; |
|
276 |
|
277 QBasicTimer m_tickTimer; |
|
278 qint32 m_tickInterval; |
|
279 |
|
280 //the graph(s) |
|
281 MediaGraph* m_graphs[FILTER_COUNT]; |
|
282 |
|
283 //...the videowidgets in the graph |
|
284 QList<VideoWidget*> m_videoWidgets; |
|
285 QList<AudioOutput*> m_audioOutputs; |
|
286 |
|
287 bool m_buffering:1; |
|
288 bool m_oldHasVideo:1; |
|
289 bool m_prefinishMarkSent:1; |
|
290 bool m_aboutToFinishSent:1; |
|
291 bool m_nextSourceReadyToStart:1; |
|
292 |
|
293 //for TitleInterface (and commands) |
|
294 #ifndef QT_NO_PHONON_MEDIACONTROLLER |
|
295 bool m_autoplayTitles:1; |
|
296 QList<qint64> m_titles; |
|
297 int m_currentTitle; |
|
298 int _iface_availableTitles() const; |
|
299 int _iface_currentTitle() const; |
|
300 void _iface_setCurrentTitle(int title, bool bseek = true); |
|
301 void setTitles(const QList<qint64> &titles); |
|
302 qint64 titleAbsolutePosition(int title) const; |
|
303 #endif //QT_NO_PHONON_MEDIACONTROLLER |
|
304 qint64 m_targetTick; |
|
305 |
|
306 WorkerThread m_thread; |
|
307 }; |
|
308 } |
|
309 } |
|
310 |
|
311 QT_END_NAMESPACE |
|
312 |
|
313 #endif // PHONON_MEDIAOBJECT_H |