-rw-r--r-- | noncore/multimedia/opieplayer2/xinevideowidget.cpp | 52 |
1 files changed, 23 insertions, 29 deletions
diff --git a/noncore/multimedia/opieplayer2/xinevideowidget.cpp b/noncore/multimedia/opieplayer2/xinevideowidget.cpp index 7f59085..78ebe19 100644 --- a/noncore/multimedia/opieplayer2/xinevideowidget.cpp +++ b/noncore/multimedia/opieplayer2/xinevideowidget.cpp | |||
@@ -1,97 +1,96 @@ | |||
1 | |||
2 | /* | 1 | /* |
3 | This file is part of the Opie Project | 2 | This file is part of the Opie Project |
4 | 3 | ||
5 | Copyright (c) 2002 Max Reiss <harlekin@handhelds.org> | 4 | Copyright (c) 2002 Max Reiss <harlekin@handhelds.org> |
6 | Copyright (c) 2002 L. Potter <ljp@llornkcor.com> | 5 | Copyright (c) 2002 L. Potter <ljp@llornkcor.com> |
7 | Copyright (c) 2002 Holger Freyther <zecke@handhelds.org> | 6 | Copyright (c) 2002 Holger Freyther <zecke@handhelds.org> |
8 | =. | 7 | =. |
9 | .=l. | 8 | .=l. |
10 | .>+-= | 9 | .>+-= |
11 | _;:, .> :=|. This program is free software; you can | 10 | _;:, .> :=|. This program is free software; you can |
12 | .> <`_, > . <= redistribute it and/or modify it under | 11 | .> <`_, > . <= redistribute it and/or modify it under |
13 | :`=1 )Y*s>-.-- : the terms of the GNU General Public | 12 | :`=1 )Y*s>-.-- : the terms of the GNU General Public |
14 | .="- .-=="i, .._ License as published by the Free Software | 13 | .="- .-=="i, .._ License as published by the Free Software |
15 | - . .-<_> .<> Foundation; either version 2 of the License, | 14 | - . .-<_> .<> Foundation; either version 2 of the License, |
16 | ._= =} : or (at your option) any later version. | 15 | ._= =} : or (at your option) any later version. |
17 | .%`+i> _;_. | 16 | .%`+i> _;_. |
18 | .i_,=:_. -<s. This program is distributed in the hope that | 17 | .i_,=:_. -<s. This program is distributed in the hope that |
19 | + . -:. = it will be useful, but WITHOUT ANY WARRANTY; | 18 | + . -:. = it will be useful, but WITHOUT ANY WARRANTY; |
20 | : .. .:, . . . without even the implied warranty of | 19 | : .. .:, . . . without even the implied warranty of |
21 | =_ + =;=|` MERCHANTABILITY or FITNESS FOR A | 20 | =_ + =;=|` MERCHANTABILITY or FITNESS FOR A |
22 | _.=:. : :=>`: PARTICULAR PURPOSE. See the GNU | 21 | _.=:. : :=>`: PARTICULAR PURPOSE. See the GNU |
23 | ..}^=.= = ; Library General Public License for more | 22 | ..}^=.= = ; Library General Public License for more |
24 | ++= -. .` .: details. | 23 | ++= -. .` .: details. |
25 | : = ...= . :.=- | 24 | : = ...= . :.=- |
26 | -. .:....=;==+<; You should have received a copy of the GNU | 25 | -. .:....=;==+<; You should have received a copy of the GNU |
27 | -_. . . )=. = Library General Public License along with | 26 | -_. . . )=. = Library General Public License along with |
28 | -- :-=` this library; see the file COPYING.LIB. | 27 | -- :-=` this library; see the file COPYING.LIB. |
29 | If not, write to the Free Software Foundation, | 28 | If not, write to the Free Software Foundation, |
30 | Inc., 59 Temple Place - Suite 330, | 29 | Inc., 59 Temple Place - Suite 330, |
31 | Boston, MA 02111-1307, USA. | 30 | Boston, MA 02111-1307, USA. |
32 | 31 | ||
33 | */ | 32 | */ |
34 | 33 | ||
35 | #include "xinevideowidget.h" | 34 | #include "xinevideowidget.h" |
36 | #include <opie2/odebug.h> | 35 | #include <opie2/odebug.h> |
37 | 36 | ||
38 | #include <qimage.h> | 37 | #include <qimage.h> |
39 | #include <qdirectpainter_qws.h> | 38 | #include <qdirectpainter_qws.h> |
40 | #include <qgfx_qws.h> | 39 | #include <qgfx_qws.h> |
41 | #include <qsize.h> | 40 | #include <qsize.h> |
42 | #include <qapplication.h> | 41 | #include <qapplication.h> |
43 | 42 | ||
44 | #include <qpe/resource.h> | 43 | #include <qpe/resource.h> |
45 | 44 | ||
46 | #include <pthread.h> | 45 | #include <pthread.h> |
47 | 46 | ||
48 | 47 | ||
49 | 48 | ||
50 | // 0 deg rot: copy a line from src to dst (use libc memcpy) | 49 | // 0 deg rot: copy a line from src to dst (use libc memcpy) |
51 | 50 | ||
52 | // 180 deg rot: copy a line from src to dst reversed | 51 | // 180 deg rot: copy a line from src to dst reversed |
53 | 52 | ||
54 | /* | 53 | /* |
55 | * This code relies the len be a multiply of 16bit | 54 | * This code relies the len be a multiply of 16bit |
56 | */ | 55 | */ |
57 | static inline void memcpy_rev ( void *_dst, void *_src, size_t len ) | 56 | static inline void memcpy_rev ( void *_dst, void *_src, size_t len ) |
58 | { | 57 | { |
59 | 58 | ||
60 | /* | 59 | /* |
61 | * move the source to the end | 60 | * move the source to the end |
62 | */ | 61 | */ |
63 | char *src_c = static_cast<char*>(_src) + len; | 62 | char *src_c = static_cast<char*>(_src) + len; |
64 | 63 | ||
65 | /* | 64 | /* |
66 | * as we copy by 16bit and not 8bit | 65 | * as we copy by 16bit and not 8bit |
67 | * devide the length by two | 66 | * devide the length by two |
68 | */ | 67 | */ |
69 | len >>= 1; | 68 | len >>= 1; |
70 | 69 | ||
71 | short int* dst = static_cast<short int*>( _dst ); | 70 | short int* dst = static_cast<short int*>( _dst ); |
72 | short int* src = reinterpret_cast<short int*>( src_c ); | 71 | short int* src = reinterpret_cast<short int*>( src_c ); |
73 | 72 | ||
74 | /* | 73 | /* |
75 | * Increment dst after assigning | 74 | * Increment dst after assigning |
76 | * Decrement src before assigning becase we move backwards | 75 | * Decrement src before assigning becase we move backwards |
77 | */ | 76 | */ |
78 | while ( len-- ) | 77 | while ( len-- ) |
79 | *dst++ = *--src; | 78 | *dst++ = *--src; |
80 | 79 | ||
81 | } | 80 | } |
82 | 81 | ||
83 | // 90 deg rot: copy a column from src to dst | 82 | // 90 deg rot: copy a column from src to dst |
84 | 83 | ||
85 | static inline void memcpy_step ( void *_dst, void *_src, size_t len, size_t step ) | 84 | static inline void memcpy_step ( void *_dst, void *_src, size_t len, size_t step ) |
86 | { | 85 | { |
87 | short int *dst = static_cast<short int*>( _dst ); | 86 | short int *dst = static_cast<short int*>( _dst ); |
88 | short int *src = static_cast<short int*>( _src ); | 87 | short int *src = static_cast<short int*>( _src ); |
89 | 88 | ||
90 | len >>= 1; | 89 | len >>= 1; |
91 | 90 | ||
92 | /* | 91 | /* |
93 | * Copy 16bit from src to dst and move to the next address | 92 | * Copy 16bit from src to dst and move to the next address |
94 | */ | 93 | */ |
95 | while ( len-- ) { | 94 | while ( len-- ) { |
96 | *dst++ = *src; | 95 | *dst++ = *src; |
97 | src = reinterpret_cast<short int*>(reinterpret_cast<char*>(src)+step); | 96 | src = reinterpret_cast<short int*>(reinterpret_cast<char*>(src)+step); |
@@ -122,218 +121,213 @@ XineVideoWidget::XineVideoWidget ( QWidget* parent, const char* name ) | |||
122 | setBackgroundMode ( NoBackground ); | 121 | setBackgroundMode ( NoBackground ); |
123 | 122 | ||
124 | m_logo = 0; | 123 | m_logo = 0; |
125 | m_buff = 0; | 124 | m_buff = 0; |
126 | m_bytes_per_line_fb = qt_screen-> linestep ( ); | 125 | m_bytes_per_line_fb = qt_screen-> linestep ( ); |
127 | m_bytes_per_pixel = ( qt_screen->depth() + 7 ) / 8; | 126 | m_bytes_per_pixel = ( qt_screen->depth() + 7 ) / 8; |
128 | m_rotation = 0; | 127 | m_rotation = 0; |
129 | m_lastsize = 0; | 128 | m_lastsize = 0; |
130 | } | 129 | } |
131 | 130 | ||
132 | 131 | ||
133 | XineVideoWidget::~XineVideoWidget ( ) | 132 | XineVideoWidget::~XineVideoWidget ( ) |
134 | { | 133 | { |
135 | ThreadUtil::AutoLock a(m_bufmutex); | 134 | ThreadUtil::AutoLock a(m_bufmutex); |
136 | if (m_buff) { | 135 | if (m_buff) { |
137 | delete[]m_buff; | 136 | delete[]m_buff; |
138 | m_lastsize=0; | 137 | m_lastsize=0; |
139 | m_buff = 0; | 138 | m_buff = 0; |
140 | } | 139 | } |
141 | if (m_logo) { | 140 | if (m_logo) { |
142 | delete m_logo; | 141 | delete m_logo; |
143 | } | 142 | } |
144 | } | 143 | } |
145 | 144 | ||
146 | void XineVideoWidget::clear ( ) | 145 | void XineVideoWidget::clear ( ) |
147 | { | 146 | { |
148 | ThreadUtil::AutoLock a(m_bufmutex); | 147 | ThreadUtil::AutoLock a(m_bufmutex); |
149 | if (m_buff) { | 148 | if (m_buff) { |
150 | delete[]m_buff; | 149 | delete[]m_buff; |
151 | m_lastsize=0; | 150 | m_lastsize=0; |
152 | m_buff = 0; | 151 | m_buff = 0; |
153 | } | 152 | } |
154 | repaint ( false ); | 153 | repaint ( false ); |
155 | } | 154 | } |
156 | 155 | ||
157 | QSize XineVideoWidget::videoSize() const | 156 | QSize XineVideoWidget::videoSize() const |
158 | { | 157 | { |
159 | QSize s = size(); | 158 | QSize s = size(); |
160 | bool fs = ( s == qApp->desktop()->size() ); | 159 | bool fs = ( s == qApp->desktop()->size() ); |
161 | 160 | ||
162 | // if we are in fullscreen mode, do not rotate the video | 161 | // if we are in fullscreen mode, do not rotate the video |
163 | // (!! the paint routine uses m_rotation + qt_screen-> transformOrientation() !!) | 162 | // (!! the paint routine uses m_rotation + qt_screen-> transformOrientation() !!) |
164 | m_rotation = fs ? - qt_screen->transformOrientation() : 0; | 163 | m_rotation = fs ? - qt_screen->transformOrientation() : 0; |
165 | 164 | ||
166 | if ( fs && qt_screen->isTransformed() ) | 165 | if ( fs && qt_screen->isTransformed() ) |
167 | s = qt_screen->mapToDevice( s ); | 166 | s = qt_screen->mapToDevice( s ); |
168 | 167 | ||
169 | return s; | 168 | return s; |
170 | } | 169 | } |
171 | 170 | ||
172 | void XineVideoWidget::paintEvent ( QPaintEvent * ) | 171 | void XineVideoWidget::paintEvent ( QPaintEvent * ) |
173 | { | 172 | { |
174 | if (m_bufmutex.isLocked()) { | 173 | if (m_bufmutex.isLocked()) { |
175 | return; | 174 | return; |
176 | } | 175 | } |
177 | ThreadUtil::AutoLock a(m_bufmutex); | 176 | ThreadUtil::AutoLock a(m_bufmutex); |
178 | QPainter p ( this ); | 177 | QPainter p ( this ); |
179 | p. fillRect ( rect (), black ); | 178 | p. fillRect ( rect (), black ); |
180 | if (m_logo) | 179 | if (m_logo) |
181 | p. drawImage ( 0, 0, *m_logo ); | 180 | p. drawImage ( 0, 0, *m_logo ); |
182 | } | 181 | } |
183 | 182 | ||
184 | void XineVideoWidget::paintEvent2 ( QPaintEvent * ) | 183 | void XineVideoWidget::paintEvent2 ( QPaintEvent * ) |
185 | { | 184 | { |
186 | ThreadUtil::AutoLock a(m_bufmutex); | 185 | ThreadUtil::AutoLock a(m_bufmutex); |
187 | QPainter p (this); | 186 | QPainter p (this); |
188 | if ( m_buff == 0 ) { | 187 | if ( m_buff == 0 ) { |
189 | p. fillRect ( rect ( ), black ); | 188 | p. fillRect ( rect ( ), black ); |
190 | if ( m_logo ) | 189 | if ( m_logo ) |
191 | p. drawImage ( 0, 0, *m_logo ); | 190 | p. drawImage ( 0, 0, *m_logo ); |
192 | } else if (m_lastsize){ | 191 | } else if (m_lastsize){ |
193 | // Qt needs to be notified which areas were really updated .. strange | 192 | // Qt needs to be notified which areas were really updated .. strange |
194 | QArray <QRect> qt_bug_workaround_clip_rects; | 193 | QArray <QRect> qt_bug_workaround_clip_rects; |
195 | 194 | ||
196 | { | 195 | { |
197 | QDirectPainter dp ( this ); | 196 | QDirectPainter dp ( this ); |
198 | 197 | ||
199 | int rot = dp. transformOrientation ( ) + m_rotation; // device rotation + custom rotation | 198 | int rot = dp. transformOrientation ( ) + m_rotation; // device rotation + custom rotation |
200 | 199 | ||
201 | bool rot90 = (( -m_rotation ) & 1 ); | 200 | bool rot90 = (( -m_rotation ) & 1 ); |
202 | 201 | ||
203 | int _vw,_vh; | 202 | int _vw,_vh; |
204 | switch (rot90) { | 203 | switch (rot90) { |
205 | case true: | 204 | case true: |
206 | _vh = m_framesize.width(); | 205 | _vh = m_framesize.width(); |
207 | _vw = m_framesize.height(); | 206 | _vw = m_framesize.height(); |
208 | break; | 207 | break; |
209 | default: | 208 | default: |
210 | _vw = m_framesize.width(); | 209 | _vw = m_framesize.width(); |
211 | _vh = m_framesize.height(); | 210 | _vh = m_framesize.height(); |
212 | break; | 211 | break; |
213 | break; | 212 | break; |
214 | } | 213 | } |
215 | int middle_w = _vw/2; | 214 | int middle_w = _vw/2; |
216 | int middle_h = _vh/2; | 215 | int middle_h = _vh/2; |
217 | m_thisframe.setRect(width()/2-middle_w,height()/2-middle_h,_vw,_vh); | 216 | m_thisframe.setRect(width()/2-middle_w,height()/2-middle_h,_vw,_vh); |
218 | QRect topFill,rightFill,leftFill,downFill; | 217 | |
219 | topFill. setRect(0,0 ,width(),m_thisframe.top()); | 218 | { |
220 | downFill.setCoords(0,m_thisframe.bottom(),width(),height()); | 219 | // is this stuff realy needed? it seems working without, too. |
221 | rightFill.setCoords(m_thisframe.right(),0,width(),height()); | 220 | QRect topFill,rightFill,leftFill,downFill; |
222 | leftFill.setCoords(0,0,m_thisframe.left(),height()); | 221 | topFill. setRect(0,0 ,width(),m_thisframe.top()); |
223 | 222 | downFill.setCoords(0,m_thisframe.bottom(),width(),height()); | |
224 | if (topFill.isValid()) { | 223 | rightFill.setCoords(m_thisframe.right(),0,width(),height()); |
225 | p.fillRect(topFill,black); | 224 | leftFill.setCoords(0,0,m_thisframe.left(),height()); |
226 | } | 225 | |
227 | if (downFill.isValid()) { | 226 | if (topFill.isValid()) { |
228 | p.fillRect(downFill,black); | 227 | p.fillRect(topFill,black); |
229 | } | 228 | } |
230 | if (rightFill.isValid()) { | 229 | if (downFill.isValid()) { |
231 | p.fillRect(rightFill,black); | 230 | p.fillRect(downFill,black); |
232 | } | 231 | } |
233 | if (leftFill.isValid()) { | 232 | if (rightFill.isValid()) { |
234 | p.fillRect(leftFill,black); | 233 | p.fillRect(rightFill,black); |
234 | } | ||
235 | if (leftFill.isValid()) { | ||
236 | p.fillRect(leftFill,black); | ||
237 | } | ||
238 | |||
235 | } | 239 | } |
236 | uchar *fb = dp. frameBuffer ( ); | 240 | uchar *fb = dp. frameBuffer ( ); |
237 | uchar *frame = m_buff; | 241 | uchar *frame = m_buff; |
238 | 242 | ||
239 | // where is the video frame in fb coordinates | 243 | // where is the video frame in fb coordinates |
240 | QRect framerect = qt_screen-> mapToDevice ( QRect ( mapToGlobal ( m_thisframe. topLeft ( )), m_thisframe. size ( )), QSize ( qt_screen-> width ( ), | 244 | QRect framerect = qt_screen-> mapToDevice ( QRect ( mapToGlobal ( m_thisframe. topLeft ( )), m_thisframe. size ( )), QSize ( qt_screen-> width ( ), |
241 | qt_screen-> height ( ))); | 245 | qt_screen-> height ( ))); |
242 | 246 | ||
243 | uchar * src = frame; | 247 | uchar * src = frame; |
244 | uchar * dst = fb+framerect.y()*m_bytes_per_line_fb+framerect.x()*m_bytes_per_pixel; | 248 | uchar * dst = fb+framerect.y()*m_bytes_per_line_fb+framerect.x()*m_bytes_per_pixel; |
245 | 249 | ||
246 | for (int y=framerect.top();y<=framerect.bottom();++y) { | 250 | for (int y=framerect.top();y<=framerect.bottom();++y) { |
247 | switch (rot) { | 251 | switch (rot) { |
248 | case 0: | 252 | case 0: |
249 | memcpy(dst,src,m_bytes_per_line_frame); | 253 | memcpy(dst,src,m_bytes_per_line_frame); |
250 | src+=m_bytes_per_line_frame; | 254 | src+=m_bytes_per_line_frame; |
251 | break; | 255 | break; |
252 | case 1: | 256 | case 1: |
253 | memcpy_step ( dst, src, m_thisframe.size().width(), m_bytes_per_line_frame ); | 257 | memcpy_step ( dst, src, m_thisframe.size().width(), m_bytes_per_line_frame ); |
254 | src -= m_bytes_per_pixel; | 258 | src -= m_bytes_per_pixel; |
255 | break; | 259 | break; |
256 | case 2: | 260 | case 2: |
257 | memcpy_rev ( dst, src, m_thisframe.size().width()); | 261 | memcpy_rev ( dst, src, m_thisframe.size().width()); |
258 | src-=m_bytes_per_line_frame; | 262 | src-=m_bytes_per_line_frame; |
259 | break; | 263 | break; |
260 | case 3: | 264 | case 3: |
261 | memcpy_step_rev ( dst, src, m_thisframe.size().width(), m_bytes_per_line_frame ); | 265 | memcpy_step_rev ( dst, src, m_thisframe.size().width(), m_bytes_per_line_frame ); |
262 | src += m_bytes_per_pixel; | 266 | src += m_bytes_per_pixel; |
263 | break; | 267 | break; |
264 | default: | 268 | default: |
265 | break; | 269 | break; |
266 | } | 270 | } |
267 | dst += m_bytes_per_line_fb; | 271 | dst += m_bytes_per_line_fb; |
268 | } | 272 | } |
269 | } | 273 | } |
270 | |||
271 | |||
272 | { | ||
273 | // QVFB hack by Martin Jones | ||
274 | // We need to "touch" all affected clip rects with a normal QPainter in addition to the QDirectPainter | ||
275 | p.fillRect(QRect(mapFromGlobal (m_thisframe. topLeft ( )), m_thisframe.size ()),QBrush(NoBrush)); | ||
276 | } | ||
277 | } | 274 | } |
278 | } | 275 | } |
279 | 276 | ||
280 | QImage *XineVideoWidget::logo ( ) const | 277 | QImage *XineVideoWidget::logo ( ) const |
281 | { | 278 | { |
282 | return m_logo; | 279 | return m_logo; |
283 | } | 280 | } |
284 | 281 | ||
285 | 282 | ||
286 | void XineVideoWidget::setLogo ( QImage* logo ) | 283 | void XineVideoWidget::setLogo ( QImage* logo ) |
287 | { | 284 | { |
288 | delete m_logo; | 285 | delete m_logo; |
289 | m_logo = logo; | 286 | m_logo = logo; |
290 | } | 287 | } |
291 | 288 | ||
292 | void XineVideoWidget::setVideoFrame ( uchar* img, int w, int h, int bpl ) | 289 | void XineVideoWidget::setVideoFrame ( uchar* img, int w, int h, int bpl ) |
293 | { | 290 | { |
294 | // mutex area for AutoLock | 291 | // mutex area for AutoLock |
295 | { | 292 | { |
296 | if (m_bufmutex.isLocked()||w>width()||h>height()) { | 293 | if (m_bufmutex.isLocked()||w>width()||h>height()) { |
297 | // no time to wait - drop frame | 294 | // no time to wait - drop frame |
298 | return; | 295 | return; |
299 | } | 296 | } |
300 | ThreadUtil::AutoLock a(m_bufmutex); | 297 | ThreadUtil::AutoLock a(m_bufmutex); |
301 | bool rot90 = (( -m_rotation ) & 1 ); | 298 | bool rot90 = (( -m_rotation ) & 1 ); |
302 | int l = h*bpl; | 299 | int l = h*bpl; |
303 | if (l!=m_lastsize) { | 300 | if (l!=m_lastsize) { |
304 | if (m_buff) { | 301 | if (m_buff) { |
305 | delete[]m_buff; | 302 | delete[]m_buff; |
306 | } | 303 | } |
307 | if (l>0) { | 304 | if (l>0) { |
308 | m_buff = new uchar[l]; | 305 | m_buff = new uchar[l]; |
309 | odebug << "Point to: " << (unsigned long)m_buff << oendl; | 306 | odebug << "Point to: " << (unsigned long)m_buff << oendl; |
310 | m_lastsize=l; | 307 | m_lastsize=l; |
311 | } else { | 308 | } else { |
312 | m_buff = 0; | 309 | m_buff = 0; |
313 | } | 310 | } |
314 | } else if (l==0){ | ||
315 | delete[] m_buff; | ||
316 | m_buff = 0; | ||
317 | } | 311 | } |
318 | m_lastsize = l; | 312 | m_lastsize = l; |
319 | m_framesize = QSize(w,h); | 313 | m_framesize = QSize(w,h); |
320 | if (m_buff && m_lastsize) { | 314 | if (m_buff && m_lastsize) { |
321 | memcpy(m_buff,img,m_lastsize); | 315 | memcpy(m_buff,img,m_lastsize); |
322 | } | 316 | } |
323 | m_bytes_per_line_frame = bpl; | 317 | m_bytes_per_line_frame = bpl; |
324 | } // Release Mutex | 318 | } // Release Mutex |
325 | 319 | ||
326 | paintEvent2(0); | 320 | if (m_buff) paintEvent2(0); |
327 | } | 321 | } |
328 | 322 | ||
329 | void XineVideoWidget::resizeEvent ( QResizeEvent * ) | 323 | void XineVideoWidget::resizeEvent ( QResizeEvent * ) |
330 | { | 324 | { |
331 | emit videoResized( videoSize() ); | 325 | emit videoResized( videoSize() ); |
332 | } | 326 | } |
333 | 327 | ||
334 | 328 | ||
335 | void XineVideoWidget::mouseReleaseEvent ( QMouseEvent * /*me*/ ) | 329 | void XineVideoWidget::mouseReleaseEvent ( QMouseEvent * /*me*/ ) |
336 | { | 330 | { |
337 | emit clicked(); | 331 | emit clicked(); |
338 | } | 332 | } |
339 | 333 | ||