aboutsummaryrefslogtreecommitdiff
path: root/engines/mads
diff options
context:
space:
mode:
authorPaul Gilbert2014-03-18 22:37:33 -0400
committerPaul Gilbert2014-03-18 22:37:33 -0400
commit5b4b7a5ca972f533c7a6129e84fd5d26e3b2ebb6 (patch)
treec815932473b1e5a32ce22a0d70cf9f41fe6a1f8f /engines/mads
parent4875c83f1023585ff7c807f5633aac84eb5e3c2b (diff)
downloadscummvm-rg350-5b4b7a5ca972f533c7a6129e84fd5d26e3b2ebb6.tar.gz
scummvm-rg350-5b4b7a5ca972f533c7a6129e84fd5d26e3b2ebb6.tar.bz2
scummvm-rg350-5b4b7a5ca972f533c7a6129e84fd5d26e3b2ebb6.zip
MADS: Finished UISlots::draw method
Diffstat (limited to 'engines/mads')
-rw-r--r--engines/mads/scene.cpp4
-rw-r--r--engines/mads/sprites.h4
-rw-r--r--engines/mads/user_interface.cpp144
-rw-r--r--engines/mads/user_interface.h33
4 files changed, 169 insertions, 16 deletions
diff --git a/engines/mads/scene.cpp b/engines/mads/scene.cpp
index 60b33853a4..79ac2ceab1 100644
--- a/engines/mads/scene.cpp
+++ b/engines/mads/scene.cpp
@@ -379,8 +379,8 @@ void Scene::doFrame() {
_kernelMessages.update();
}
- _userInterface._uiSlots.draw(_vm->_game->_abortTimers2 == kTransitionFadeIn ? 0xff : 0,
- _vm->_game->_abortTimers2);
+ _userInterface._uiSlots.draw(_vm->_game->_abortTimers2 == kTransitionFadeIn,
+ _vm->_game->_abortTimers2 != 0);
// Write any text needed by the interface
if (_vm->_game->_abortTimers2)
diff --git a/engines/mads/sprites.h b/engines/mads/sprites.h
index 6118924de7..2017961d29 100644
--- a/engines/mads/sprites.h
+++ b/engines/mads/sprites.h
@@ -31,8 +31,8 @@
namespace MADS {
enum SlotType {
- ST_NONE = 0, ST_FOREGROUND = 1, ST_BACKGROUND = -4, ST_MINUS3 = -3,
- ST_FULL_SCREEN_REFRESH = -2, ST_EXPIRED = -1
+ ST_NONE = 0, ST_FOREGROUND = 1, ST_MINUS5 = -5, ST_BACKGROUND = -4,
+ ST_MINUS3 = -3, ST_FULL_SCREEN_REFRESH = -2, ST_EXPIRED = -1
};
class MADSEngine;
diff --git a/engines/mads/user_interface.cpp b/engines/mads/user_interface.cpp
index f34d68c6ce..d11c1220df 100644
--- a/engines/mads/user_interface.cpp
+++ b/engines/mads/user_interface.cpp
@@ -57,13 +57,136 @@ void UISlots::add(const Common::Point &pt, int frameNumber, int spritesIndex) {
push_back(ie);
}
-void UISlots::draw(int v1, int v2) {
-
+void UISlots::draw(bool updateFlag, bool delFlag) {
+ Scene &scene = _vm->_game->_scene;
+ UserInterface &userInterface = scene._userInterface;
+ DirtyArea *dirtyAreaPtr = nullptr;
+
+ // Loop through setting up the dirty areas
+ for (uint idx = 0; idx < size(); ++idx) {
+ DirtyArea &dirtyArea = userInterface._dirtyAreas[idx];
+ UISlot &slot = (*this)[idx];
+
+ if (slot._slotType >= ST_NONE) {
+ dirtyArea._active = false;
+ } else {
+ dirtyArea.setUISlot(&slot);
+ dirtyArea._textActive = true;
+ if (slot._field2 == 200 && slot._slotType == ST_MINUS5) {
+ dirtyArea._active = false;
+ dirtyAreaPtr = &dirtyArea;
+ }
+ }
+ }
+
+ userInterface._dirtyAreas.merge(1, userInterface._uiSlots.size());
+ if (dirtyAreaPtr)
+ dirtyAreaPtr->_active = true;
+
+ // Main draw loop
+ for (uint idx = 0; idx < size(); ++idx) {
+ DirtyArea &dirtyArea = userInterface._dirtyAreas[idx];
+ UISlot &slot = (*this)[idx];
+
+ if (dirtyArea._active && dirtyArea._bounds.width() > 0
+ && dirtyArea._bounds.height() > 0 && slot._slotType >= -20) {
+
+ // TODO: Figure out the difference between two copy methods used
+ if (slot._slotType >= ST_EXPIRED) {
+ userInterface._surface.copyTo(&userInterface, dirtyArea._bounds,
+ Common::Point(dirtyArea._bounds.left, dirtyArea._bounds.top));
+ } else {
+ userInterface._surface.copyTo(&userInterface, dirtyArea._bounds,
+ Common::Point(dirtyArea._bounds.left, dirtyArea._bounds.top));
+ }
+ }
+ }
+
+ for (uint idx = 0; idx < size(); ++idx) {
+ DirtyArea &dirtyArea = userInterface._dirtyAreas[idx];
+ UISlot &slot = (*this)[idx];
+
+ int slotType = slot._slotType;
+ if (slotType >= ST_NONE) {
+ dirtyArea.setUISlot(&slot);
+ if (!updateFlag)
+ slotType &= ~0x40;
+
+ dirtyArea._textActive = slotType > 0;
+ slot._slotType &= 0x40;
+ }
+ }
+
+ userInterface._dirtyAreas.merge(1, userInterface._uiSlots.size());
+
+ for (uint idx = 0; idx < size(); ++idx) {
+ DirtyArea &dirtyArea = userInterface._dirtyAreas[idx];
+ UISlot &slot = (*this)[idx];
+
+ if (slot._slotType >= ST_NONE && !(slot._slotType & 0x40)) {
+ if (!dirtyArea._active) {
+ error("TODO: Original code here doesn't make sense!");
+ }
+
+ if (dirtyArea._textActive) {
+ SpriteAsset *asset = scene._sprites[slot._spritesIndex];
+
+ if (slot._field2 == 200) {
+ MSprite *sprite = asset->getFrame(slot._frameNumber & 0x7F);
+ sprite->copyTo(&userInterface, slot._position);
+ } else {
+ MSprite *sprite = asset->getFrame(slot._frameNumber - 1);
+ sprite->copyTo(&userInterface, slot._position);
+ }
+ }
+ }
+ }
+
+ // Mark areas of the screen surface for updating
+ if (updateFlag) {
+ _vm->_screen.setPointer(&userInterface);
+ userInterface.setBounds(Common::Rect(0, scene._interfaceY,
+ MADS_SCREEN_WIDTH - 1, userInterface.h + scene._interfaceY - 1));
+ warning("TODO: sub_111C8 / sub_1146C");
+
+ for (uint idx = 0; idx < size(); ++idx) {
+ DirtyArea &dirtyArea = userInterface._dirtyAreas[idx];
+ UISlot &slot = (*this)[idx];
+
+ if (dirtyArea._active && dirtyArea._textActive &&
+ dirtyArea._bounds.width() > 0 && dirtyArea._bounds.height() > 0) {
+ // Flag area of screen as needing update
+ Common::Rect r = dirtyArea._bounds;
+ r.translate(0, scene._interfaceY);
+ _vm->_screen.copyRectToScreen(r);
+ }
+ }
+
+ warning("TODO: sub 115A2 / sub111D3");
+ }
+
+ // Post-processing to remove slots no longer needed
+ for (int idx = (int)size() - 1; idx >= 0; --idx) {
+ UISlot &slot = (*this)[idx];
+
+ if (slot._slotType < ST_NONE) {
+ if (delFlag || updateFlag)
+ remove_at(idx);
+ else if (slot._slotType >= -20)
+ slot._slotType -= 20;
+ } else {
+ if (updateFlag)
+ slot._slotType &= 0xBF;
+ else
+ slot._slotType |= 0x40;
+ }
+ }
}
/*------------------------------------------------------------------------*/
-UserInterface::UserInterface(MADSEngine *vm) : _vm(vm) {
+UserInterface::UserInterface(MADSEngine *vm) : _vm(vm), _dirtyAreas(vm),
+ _uiSlots(vm) {
_invSpritesIndex = -1;
_invFrameNumber = 1;
_category = CAT_NONE;
@@ -77,7 +200,9 @@ UserInterface::UserInterface(MADSEngine *vm) : _vm(vm) {
_v1A = -1;
_v1C = -1;
_v1E = -1;
+ _dirtyAreas.resize(50);
+ // Map the user interface to the bottom of the game's screen surface
byte *pData = _vm->_screen.getBasePtr(0, MADS_SCENE_HEIGHT);
setPixels(pData, MADS_SCREEN_WIDTH, MADS_INTERFACE_HEIGHT);
}
@@ -101,9 +226,10 @@ void UserInterface::load(const Common::String &resName) {
}
delete palStream;
- // set the size for the interface
+ // Read in the surface data
Common::SeekableReadStream *pixelsStream = madsPack.getItemStream(1);
- pixelsStream->read(getData(), MADS_SCREEN_WIDTH * MADS_INTERFACE_HEIGHT);
+ _surface.setSize(MADS_SCREEN_WIDTH, MADS_INTERFACE_HEIGHT);
+ pixelsStream->read(_surface.getData(), MADS_SCREEN_WIDTH * MADS_INTERFACE_HEIGHT);
delete pixelsStream;
}
@@ -124,7 +250,9 @@ void UserInterface::setup(int id) {
resName += "A";
resName += ".INT";
+
load(resName);
+ _surface.copyTo(this);
}
scene._screenObjects._v832EC = id;
@@ -139,7 +267,7 @@ void UserInterface::setup(int id) {
copyTo(&_surface);
if (_vm->_game->_v1 == 5)
- scene._userInterface._uiSlots.draw(0, 0);
+ scene._userInterface._uiSlots.draw(false, false);
scene._action.clear();
drawTextElements();
@@ -273,7 +401,7 @@ void UserInterface::writeVocab(ScrCategory category, int id) {
}
void UserInterface::setBounds(const Common::Rect &r) {
- _bounds = r;
+ _drawBounds = r;
}
void UserInterface::loadElements() {
@@ -465,7 +593,7 @@ void UserInterface::noInventoryAnim() {
void UserInterface::refresh() {
_uiSlots.clear();
_uiSlots.fullRefresh();
- _uiSlots.draw(0, 0);
+ _uiSlots.draw(false, false);
drawTextElements();
}
diff --git a/engines/mads/user_interface.h b/engines/mads/user_interface.h
index a474fe1fa1..9d2af0425d 100644
--- a/engines/mads/user_interface.h
+++ b/engines/mads/user_interface.h
@@ -27,6 +27,7 @@
#include "common/rect.h"
#include "common/str.h"
#include "mads/msurface.h"
+#include "mads/screen.h"
namespace MADS {
@@ -47,18 +48,42 @@ public:
UISlot();
};
+/**
+ * Sprite list for the user interface
+ */
class UISlots : public Common::Array<UISlot> {
+private:
+ MADSEngine *_vm;
public:
+ /**
+ * Constructor
+ */
+ UISlots(MADSEngine *vm) : _vm(vm) {}
+
+ /**
+ * Add a sprite to the list
+ */
void add(const Common::Point &pt, int frameNumber, int spritesIndex);
+
+ /**
+ * Adds a special entry for full refresh of the user interface
+ */
void fullRefresh();
- void draw(int v1, int v2);
+ /**
+ * Draw all the sprites in the list on the user interface.
+ * @param updateFlag Flag drawn areas to be updated on physical screen
+ * @param delFlag Controls how used slots are deleted after drawing
+ */
+ void draw(bool updateFlag, bool delFlag);
};
class UserInterface : public MSurface {
+ friend class UISlots;
private:
MADSEngine *_vm;
+ MSurface _surface;
int _invSpritesIndex;
int _invFrameNumber;
@@ -114,11 +139,12 @@ private:
*/
void inventoryAnim();
public:
+ UISlots _uiSlots;
+ DirtyAreas _dirtyAreas;
ScrCategory _category;
int _screenObjectsCount;
- Common::Rect _bounds;
+ Common::Rect _drawBounds;
Common::Rect *_rectP;
- MSurface _surface;
int _inventoryTopIndex;
int _objectY;
int _selectedInvIndex;
@@ -128,7 +154,6 @@ public:
int _v1A;
int _v1C;
int _v1E;
- UISlots _uiSlots;
public:
/**
* Constructor