ScummVM API documentation
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
qt_decoder.h
1 /* ScummVM - Graphic Adventure Engine
2  *
3  * ScummVM is the legal property of its developers, whose names
4  * are too numerous to list here. Please refer to the COPYRIGHT
5  * file distributed with this source distribution.
6  *
7  * This program is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see <http://www.gnu.org/licenses/>.
19  *
20  */
21 
22 //
23 // Partially based on ffmpeg code.
24 //
25 // Copyright (c) 2001 Fabrice Bellard.
26 // First version by Francois Revol revol@free.fr
27 // Seek function by Gael Chardon gael.dev@4now.net
28 //
29 
30 #ifndef VIDEO_QT_DECODER_H
31 #define VIDEO_QT_DECODER_H
32 
33 #include "audio/decoders/quicktime_intern.h"
34 #include "common/keyboard.h"
35 #include "common/scummsys.h"
36 
37 #include "graphics/palette.h"
38 #include "graphics/transform_tools.h"
39 
40 #include "video/video_decoder.h"
41 
42 namespace Common {
43 class Archive;
44 class Rational;
45 }
46 
47 namespace Graphics {
48 class Cursor;
49 struct PixelFormat;
50 }
51 
52 namespace Image {
53 class Codec;
54 }
55 
56 namespace Video {
57 
67 public:
69  virtual ~QuickTimeDecoder();
70 
71  bool loadFile(const Common::Path &filename);
72  bool loadStream(Common::SeekableReadStream *stream);
73  void close();
74  uint16 getWidth() const { return _width; }
75  uint16 getHeight() const { return _height; }
76  const Graphics::Surface *decodeNextFrame();
77  Audio::Timestamp getDuration() const { return Audio::Timestamp(0, _duration, _timeScale); }
78 
79  void enableEditListBoundsCheckQuirk(bool enable) { _enableEditListBoundsCheckQuirk = enable; }
80  Common::String getAliasPath();
81 
83  // QTVR stuff
85  void setTargetSize(uint16 w, uint16 h);
86 
87  void handleMouseMove(int16 x, int16 y);
88  void handleMouseButton(bool isDown, int16 x = -1, int16 y = -1, bool repeat = false);
89  void handleKey(Common::KeyState &state, bool down, bool repeat = false);
90 
91  Common::Point getLastClick() { return _mouseDrag; }
92 
93  float getPanAngle() const { return _panAngle; }
94  void setPanAngle(float panAngle);
95  float getTiltAngle() const { return _tiltAngle; }
96  void setTiltAngle(float tiltAngle);
97  float getFOV() const { return _fov; }
98  bool setFOV(float fov);
99  int getCurrentNodeID() { return _currentSample == -1 ? 0 : _panoTrack->panoSamples[_currentSample].hdr.nodeID; }
100  Common::String getCurrentNodeName();
101 
102  int getCurrentRow() { return _nextVideoTrack->getCurFrame() / _nav.columns; }
103  void setCurrentRow(int row);
104  int getCurrentColumn() { return _nextVideoTrack->getCurFrame() % _nav.columns; }
105  void setCurrentColumn(int column);
106 
107  int getZoomState() { return _zoomState; }
108 
109  const PanoHotSpot *getRolloverHotspot() { return _rolloverHotspot; }
110  int getRolloverHotspotID() { return _rolloverHotspotID; }
111  const PanoHotSpot *getClickedHotspot() { return _clickedHotspot; }
112  int getClickedHotspotID() { return _clickedHotspotID; }
113  Common::Point getPanLoc(int16 x, int16 y);
114  Graphics::FloatPoint getPanAngles(int16 x, int16 y);
115 
116  Common::String getHotSpotName(int id);
117  void setClickedHotSpot(int id);
118  const PanoHotSpot *getHotSpotByID(int id);
119  const PanoNavigation *getHotSpotNavByID(int id);
120 
121  void nudge(const Common::String &direction);
122 
123  bool isVR() const { return _isVR; }
124  QTVRType getQTVRType() const { return _qtvrType; }
125 
126  int getWarpMode() const { return _warpMode; }
127  void setWarpMode(int warpMode);
128  float getQuality() const { return _quality; }
129  void setQuality(float quality);
130  Common::String getTransitionMode() const { return _transitionMode == kTransitionModeNormal ? "normal" : "swing"; }
131  void setTransitionMode(Common::String mode);
132  float getTransitionSpeed() const { return _transitionSpeed; }
133  void setTransitionSpeed(float speed);
134  Common::String getUpdateMode() const;
135  void setUpdateMode(Common::String mode);
136 
137  void renderHotspots(bool mode);
138 
139  struct NodeData {
140  uint32 nodeID;
141 
142  float defHPan;
143  float defVPan;
144  float defZoom;
145 
146  float minHPan;
147  float minVPan;
148  float maxHPan;
149  float maxVPan;
150  float minZoom;
151 
152  Common::String name;
153  };
154 
155  NodeData getNodeData(uint32 nodeID);
156  void goToNode(uint32 nodeID);
157 
158 protected:
159  Common::QuickTimeParser::SampleDesc *readSampleDesc(Common::QuickTimeParser::Track *track, uint32 format, uint32 descSize);
160  Common::QuickTimeParser::SampleDesc *readPanoSampleDesc(Common::QuickTimeParser::Track *track, uint32 format, uint32 descSize);
161 
162 private:
163  void init();
164 
165  void updateAudioBuffer();
166 
167  void handleObjectMouseMove(int16 x, int16 y);
168  void handleObjectMouseButton(bool isDown, int16 x, int16 y, bool repeat);
169  void handlePanoMouseMove(int16 x, int16 y);
170  void handlePanoMouseButton(bool isDown, int16 x, int16 y, bool repeat);
171 
172  void handleObjectKey(Common::KeyState &state, bool down, bool repeat);
173  void handlePanoKey(Common::KeyState &state, bool down, bool repeat);
174 
175  void closeQTVR();
176  void updateAngles();
177  void lookupHotspot(int16 x, int16 y);
178  void updateQTVRCursor(int16 x, int16 y);
179  void setCursor(int curId);
180  void cleanupCursors();
181  void computeInteractivityZones();
182 
183  uint16 _width, _height;
184 
185 public:
186  int _currentSample = -1;
187  Common::Point _prevMouse;
188  bool _isMouseButtonDown = false;
189  Common::Point _mouseDrag;
190 
191  bool _isKeyDown = false;
192  Common::KeyState _lastKey;
193 
194  enum {
195  kZoomNone,
196  kZoomQuestion,
197  kZoomIn,
198  kZoomOut,
199  kZoomLimit,
200 
201  kTransitionModeNormal,
202  kTransitionModeSwing,
203 
204  kUpdateModeNormal,
205  kUpdateModeUpdateBoth,
206  kUpdateModeOffscreenOnly,
207  kUpdateModeFromOffscreen,
208  kUpdateModeDirectToScreen,
209  };
210 
211 private:
212  Common::Rect _curBbox;
213 
214  int _currentQTVRCursor = -1;
215  Common::Archive *_dataBundle = nullptr;
216  Graphics::Cursor **_cursorCache = nullptr;
217  int _cursorDirMap[256];
218 
219  bool _isVR = false;
220 
221  uint8 _warpMode = 2; // (2 | 1 | 0) for 2-d, 1-d or no warping
222  float _quality = 0.0f;
223  int _transitionMode = kTransitionModeNormal;
224  float _transitionSpeed = 1.0f;
225  int _updateMode = kUpdateModeNormal;
226 
227  float _panAngle = 0.0f;
228  float _tiltAngle = 0.0f;
229  float _fov = 56.0f;
230  float _hfov = 56.0f;
231  int _zoomState = kZoomNone;
232  bool _repeatTimerActive = false;
233 
234  const PanoHotSpot *_rolloverHotspot = nullptr;
235  int _rolloverHotspotID = 0;
236  const PanoHotSpot *_clickedHotspot = nullptr;
237  int _clickedHotspotID = 0;
238  bool _renderHotspots = false;
239 
240  Graphics::Surface *_scaledSurface;
241  void scaleSurface(const Graphics::Surface *src, Graphics::Surface *dst,
242  const Common::Rational &scaleFactorX, const Common::Rational &scaleFactorY);
243 
244  bool _enableEditListBoundsCheckQuirk;
245 
246  class VideoSampleDesc : public Common::QuickTimeParser::SampleDesc {
247  public:
248  VideoSampleDesc(Common::QuickTimeParser::Track *parentTrack, uint32 codecTag);
249  ~VideoSampleDesc();
250 
251  void initCodec();
252 
253  // TODO: Make private in the long run
254  uint16 _bitsPerSample;
255  char _codecName[32];
256  uint16 _colorTableId;
257  Graphics::Palette _palette;
258  Image::Codec *_videoCodec;
259  };
260 
261  // The AudioTrackHandler is currently just a wrapper around some
262  // QuickTimeDecoder functions.
263  class AudioTrackHandler : public SeekableAudioTrack {
264  public:
265  AudioTrackHandler(QuickTimeDecoder *decoder, QuickTimeAudioTrack *audioTrack);
266 
267  void updateBuffer();
268 
269  protected:
270  Audio::SeekableAudioStream *getSeekableAudioStream() const;
271 
272  private:
273  QuickTimeDecoder *_decoder;
274  QuickTimeAudioTrack *_audioTrack;
275  };
276 
277  class PanoSampleDesc : public Common::QuickTimeParser::SampleDesc {
278  public:
279  PanoSampleDesc(Common::QuickTimeParser::Track *parentTrack, uint32 codecTag);
280  ~PanoSampleDesc();
281 
282  int16 _majorVersion; // must be zero, also observed to be 1
283  int16 _minorVersion; // must be zero, also observed to be 1
284 
285  int32 _sceneTrackID; // ID of video track that contains panoramic scene
286  int32 _loResSceneTrackID; // ID of video track that contains low res panoramic scene
287  byte _reserved3[4 * 6]; // must be zero
288  int32 _hotSpotTrackID; // ID of video track that contains hotspot image
289  byte _reserved4[4 * 9]; // must be zero
290 
291  float _hPanStart; // horizontal pan range start angle (e.g. 0)
292  float _hPanEnd; // horizontal pan range end angle (e.g. 360)
293  float _vPanTop; // vertical pan range top angle (e.g. 42.5)
294  float _vPanBottom; // vertical pan range bottom angle (e.g. -42.5)
295  float _minimumZoom; // minimum zoom angle (e.g. 5; use 0 for default)
296  float _maximumZoom; // maximum zoom angle (e.g. 65; use 0 for default)
297 
298  // info for the highest res version of scene track
299  uint32 _sceneSizeX; // pixel width of the panorama (e.g. 768)
300  uint32 _sceneSizeY; // pixel height of the panorama (e.g. 2496)
301  uint32 _numFrames; // number of diced frames (e.g. 24)
302  int16 _reserved5; // must be zero
303  int16 _sceneNumFramesX; // diced frames wide (e.g. 1)
304  int16 _sceneNumFramesY; // diced frames high (e.g. 24)
305  int16 _sceneColorDepth; // bit depth of the scene track (e.g. 32)
306 
307  // info for the highest rest version of hotSpot track
308  int32 _hotSpotSizeX; // pixel width of the hot spot panorama (e.g. 768)
309  int32 _hotSpotSizeY; // pixel height of the hot spot panorama (e.g. 2496)
310  int16 _reserved6; // must be zero
311  int16 _hotSpotNumFramesX; // diced frames wide (e.g. 1)
312  int16 _hotSpotNumFramesY; // diced frames high (e.g. 24)
313  int16 _hotSpotColorDepth; // must be 8
314  };
315 
316  // The VideoTrackHandler is the bridge between the time of playback
317  // and the media for the given track. It calculates when to start
318  // tracks and at what rate to play the media using the edit list.
319  class VideoTrackHandler : public VideoTrack {
320  public:
321  VideoTrackHandler(QuickTimeDecoder *decoder, Common::QuickTimeParser::Track *parent);
322  ~VideoTrackHandler();
323 
324  bool endOfTrack() const;
325  bool isSeekable() const { return true; }
326  bool seek(const Audio::Timestamp &time);
327  Audio::Timestamp getDuration() const;
328 
329  uint16 getWidth() const;
330  uint16 getHeight() const;
331  Graphics::PixelFormat getPixelFormat() const;
332  bool setOutputPixelFormat(const Graphics::PixelFormat &format);
333  int getCurFrame() const { return _curFrame; }
334  void setCurFrame(int32 curFrame) { _curFrame = curFrame; }
335  int getFrameCount() const;
336  uint32 getNextFrameStartTime() const; // milliseconds
337  const Graphics::Surface *decodeNextFrame();
338  Audio::Timestamp getFrameTime(uint frame) const;
339  const byte *getPalette() const;
340  bool hasDirtyPalette() const { return _curPalette; }
341  bool setReverse(bool reverse);
342  bool isReversed() const { return _reversed; }
343  bool canDither() const;
344  void setDither(const byte *palette);
345 
346  Common::Rational getScaledWidth() const;
347  Common::Rational getScaledHeight() const;
348 
349  const Graphics::Surface *bufferNextFrame();
350 
351  private:
352  QuickTimeDecoder *_decoder;
354  uint32 _curEdit;
355  int32 _curFrame;
356  int32 _delayedFrameToBufferTo;
357  uint32 _nextFrameStartTime; // media time
358  Graphics::Surface *_scaledSurface;
359  int32 _durationOverride; // media time
360  const byte *_curPalette;
361  mutable bool _dirtyPalette;
362  bool _reversed;
363 
364  // Forced dithering of frames
365  Graphics::Palette _forcedDitherPalette;
366  byte *_ditherTable;
367  Graphics::Surface *_ditherFrame;
368  const Graphics::Surface *forceDither(const Graphics::Surface &frame);
369 
370  Common::SeekableReadStream *getNextFramePacket(uint32 &descId);
371  uint32 getCurFrameDuration(); // media time
372  uint32 findKeyFrame(uint32 frame) const;
373  bool isEmptyEdit() const;
374  void enterNewEditListEntry(bool bufferFrames, bool intializingTrack = false);
375  uint32 getRateAdjustedFrameTime() const; // media time
376  uint32 getCurEditTimeOffset() const; // media time
377  uint32 getCurEditTrackDuration() const; // media time
378  bool atLastEdit() const;
379  bool endOfCurEdit() const;
380  void checkEditListBounds();
381  };
382 
383  class PanoTrackHandler : public VideoTrack {
384  public:
385  PanoTrackHandler(QuickTimeDecoder *decoder, Common::QuickTimeParser::Track *parent);
386  ~PanoTrackHandler();
387 
388  bool endOfTrack() const { return false; }
389  uint16 getWidth() const;
390  uint16 getHeight() const;
391  int getCurFrame() const { return 1; }
392  uint32 getNextFrameStartTime() const { return 0; }
393  Graphics::PixelFormat getPixelFormat() const;
394  const Graphics::Surface *decodeNextFrame();
395 
396  Common::Rational getScaledWidth() const;
397  Common::Rational getScaledHeight() const;
398 
399  void initPanorama();
400  void constructPanorama();
401  Graphics::Surface *constructMosaic(VideoTrackHandler *track, uint w, uint h, Common::String fname);
402 
403  Common::Point projectPoint(int16 x, int16 y);
404 
405  void setDirty() { _dirty = true; }
406 
407  private:
408  QuickTimeDecoder *_decoder;
410 
411  void projectPanorama();
412 
413  const Graphics::Surface *bufferNextFrame();
414 
415  public:
416  Graphics::Surface *_constructedPano;
417  Graphics::Surface *_constructedHotspots;
418  Graphics::Surface *_projectedPano;
419  Graphics::Surface *_planarProjection;
420 
421  private:
422  bool _isPanoConstructed;
423 
424  bool _dirty;
425  };
426 };
427 
428 } // End of namespace Video
429 
430 #endif
Definition: quicktime_intern.h:65
Definition: str.h:59
Definition: surface.h:67
Definition: quicktime.h:263
Definition: qt_decoder.h:66
Definition: pixelformat.h:138
Definition: quicktime.h:134
Definition: rect.h:144
Definition: path.h:52
Definition: timestamp.h:83
Definition: stream.h:745
Definition: rational.h:40
Definition: quicktime.h:200
Definition: quicktime.h:298
Definition: audiostream.h:212
Definition: archive.h:141
Definition: codec.h:57
Definition: quicktime_intern.h:47
Definition: atari-cursor.h:38
Definition: video_decoder.h:53
Definition: cursor.h:42
Audio::Timestamp getDuration() const
Definition: qt_decoder.h:77
Definition: algorithm.h:29
Definition: formatinfo.h:28
Definition: rect.h:45
uint16 getWidth() const
Definition: qt_decoder.h:74
Definition: transform_tools.h:32
Definition: qt_decoder.h:139
Definition: keyboard.h:294
Simple class for handling a palette data.
Definition: palette.h:51
Definition: avi_frames.h:36
Definition: video_decoder.h:844
Definition: movie_decoder.h:32
uint16 getHeight() const
Definition: qt_decoder.h:75
Definition: video_decoder.h:589