/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "common/system.h" #include "engines/util.h" #include "graphics/cursorman.h" #include "graphics/surface.h" #include "gui/message.h" #include "sci/sci.h" #include "sci/event.h" #include "sci/resource.h" #include "sci/engine/features.h" #include "sci/engine/state.h" #include "sci/engine/selector.h" #include "sci/engine/kernel.h" #include "sci/graphics/animate.h" #include "sci/graphics/cache.h" #include "sci/graphics/compare.h" #include "sci/graphics/controls16.h" #include "sci/graphics/cursor.h" #include "sci/graphics/palette.h" #include "sci/graphics/paint16.h" #include "sci/graphics/picture.h" #include "sci/graphics/ports.h" #include "sci/graphics/screen.h" #include "sci/graphics/text16.h" #include "sci/graphics/view.h" #ifdef ENABLE_SCI32 #include "sci/graphics/controls32.h" #include "sci/graphics/font.h" // TODO: remove once kBitmap is moved in a separate class #include "sci/graphics/text32.h" #include "sci/graphics/frameout.h" #endif namespace Sci { #ifdef ENABLE_SCI32 extern void showScummVMDialog(const Common::String &message); reg_t kIsHiRes(EngineState *s, int argc, reg_t *argv) { // Returns 0 if the screen width or height is less than 640 or 400, // respectively. if (g_system->getWidth() < 640 || g_system->getHeight() < 400) return make_reg(0, 0); return make_reg(0, 1); } // SCI32 variant, can't work like sci16 variants reg_t kCantBeHere32(EngineState *s, int argc, reg_t *argv) { // TODO // reg_t curObject = argv[0]; // reg_t listReference = (argc > 1) ? argv[1] : NULL_REG; return NULL_REG; } reg_t kAddScreenItem(EngineState *s, int argc, reg_t *argv) { if (g_sci->_gfxFrameout->findScreenItem(argv[0]) == NULL) g_sci->_gfxFrameout->kernelAddScreenItem(argv[0]); else g_sci->_gfxFrameout->kernelUpdateScreenItem(argv[0]); return s->r_acc; } reg_t kUpdateScreenItem(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelUpdateScreenItem(argv[0]); return s->r_acc; } reg_t kDeleteScreenItem(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelDeleteScreenItem(argv[0]); return s->r_acc; } reg_t kAddPlane(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelAddPlane(argv[0]); return s->r_acc; } reg_t kDeletePlane(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelDeletePlane(argv[0]); return s->r_acc; } reg_t kUpdatePlane(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelUpdatePlane(argv[0]); return s->r_acc; } reg_t kAddPicAt(EngineState *s, int argc, reg_t *argv) { reg_t planeObj = argv[0]; GuiResourceId pictureId = argv[1].toUint16(); int16 pictureX = argv[2].toSint16(); int16 pictureY = argv[3].toSint16(); g_sci->_gfxFrameout->kernelAddPicAt(planeObj, pictureId, pictureX, pictureY); return s->r_acc; } reg_t kGetHighPlanePri(EngineState *s, int argc, reg_t *argv) { return make_reg(0, g_sci->_gfxFrameout->kernelGetHighPlanePri()); } reg_t kFrameOut(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxFrameout->kernelFrameout(); return NULL_REG; } reg_t kObjectIntersect(EngineState *s, int argc, reg_t *argv) { Common::Rect objRect1 = g_sci->_gfxCompare->getNSRect(argv[0]); Common::Rect objRect2 = g_sci->_gfxCompare->getNSRect(argv[1]); return make_reg(0, objRect1.intersects(objRect2)); } // Tests if the coordinate is on the passed object reg_t kIsOnMe(EngineState *s, int argc, reg_t *argv) { uint16 x = argv[0].toUint16(); uint16 y = argv[1].toUint16(); reg_t targetObject = argv[2]; uint16 illegalBits = argv[3].offset; Common::Rect nsRect = g_sci->_gfxCompare->getNSRect(targetObject, true); // we assume that x, y are local coordinates bool contained = nsRect.contains(x, y); if (contained && illegalBits) { // If illegalbits are set, we check the color of the pixel that got clicked on // for now, we return false if the pixel is transparent // although illegalBits may get differently set, don't know yet how this really works out uint16 viewId = readSelectorValue(s->_segMan, targetObject, SELECTOR(view)); int16 loopNo = readSelectorValue(s->_segMan, targetObject, SELECTOR(loop)); int16 celNo = readSelectorValue(s->_segMan, targetObject, SELECTOR(cel)); if (g_sci->_gfxCompare->kernelIsItSkip(viewId, loopNo, celNo, Common::Point(x - nsRect.left, y - nsRect.top))) contained = false; } return make_reg(0, contained); } reg_t kCreateTextBitmap(EngineState *s, int argc, reg_t *argv) { switch (argv[0].toUint16()) { case 0: { if (argc != 4) { warning("kCreateTextBitmap(0): expected 4 arguments, got %i", argc); return NULL_REG; } reg_t object = argv[3]; Common::String text = s->_segMan->getString(readSelector(s->_segMan, object, SELECTOR(text))); debugC(kDebugLevelStrings, "kCreateTextBitmap case 0 (%04x:%04x, %04x:%04x, %04x:%04x)", PRINT_REG(argv[1]), PRINT_REG(argv[2]), PRINT_REG(argv[3])); debugC(kDebugLevelStrings, "%s", text.c_str()); uint16 maxWidth = argv[1].toUint16(); // nsRight - nsLeft + 1 uint16 maxHeight = argv[2].toUint16(); // nsBottom - nsTop + 1 return g_sci->_gfxText32->createTextBitmap(object, maxWidth, maxHeight); } case 1: { if (argc != 2) { warning("kCreateTextBitmap(1): expected 2 arguments, got %i", argc); return NULL_REG; } reg_t object = argv[1]; Common::String text = s->_segMan->getString(readSelector(s->_segMan, object, SELECTOR(text))); debugC(kDebugLevelStrings, "kCreateTextBitmap case 1 (%04x:%04x)", PRINT_REG(argv[1])); debugC(kDebugLevelStrings, "%s", text.c_str()); return g_sci->_gfxText32->createTextBitmap(object); } default: warning("CreateTextBitmap(%d)", argv[0].toUint16()); return NULL_REG; } } reg_t kDisposeTextBitmap(EngineState *s, int argc, reg_t *argv) { g_sci->_gfxText32->disposeTextBitmap(argv[0]); return s->r_acc; } reg_t kGetWindowsOption(EngineState *s, int argc, reg_t *argv) { uint16 windowsOption = argv[0].toUint16(); switch (windowsOption) { case 0: // Title bar on/off in Phantasmagoria, we return 0 (off) return NULL_REG; default: warning("GetWindowsOption: Unknown option %d", windowsOption); return NULL_REG; } } reg_t kWinHelp(EngineState *s, int argc, reg_t *argv) { switch (argv[0].toUint16()) { case 1: // Load a help file // Maybe in the future we can implement this, but for now this message should suffice showScummVMDialog("Please use an external viewer to open the game's help file: " + s->_segMan->getString(argv[1])); break; case 2: // Looks like some init function break; default: warning("Unknown kWinHelp subop %d", argv[0].toUint16()); } return s->r_acc; } /** * Used for scene transitions, replacing (but reusing parts of) the old * transition code. */ reg_t kSetShowStyle(EngineState *s, int argc, reg_t *argv) { // Can be called with 7 or 8 parameters // The style defines which transition to perform. Related to the transition // tables inside graphics/transitions.cpp uint16 showStyle = argv[0].toUint16(); // 0 - 15 reg_t planeObj = argv[1]; // the affected plane //uint16 seconds = argv[2].toUint16(); // seconds that the transition lasts //uint16 backColor = argv[3].toUint16(); // target back color(?). When fading out, it's 0x0000. When fading in, it's 0xffff //int16 priority = argv[4].toSint16(); // always 0xc8 (200) when fading in/out //uint16 animate = argv[5].toUint16(); // boolean, animate or not while the transition lasts //uint16 refFrame = argv[6].toUint16(); // refFrame, always 0 when fading in/out int16 divisions; // If the game has the pFadeArray selector, another parameter is used here, // before the optional last parameter bool hasFadeArray = g_sci->getKernel()->findSelector("pFadeArray") > 0; if (hasFadeArray) { // argv[7] divisions = (argc >= 9) ? argv[8].toSint16() : -1; // divisions (transition steps?) } else { divisions = (argc >= 8) ? argv[7].toSint16() : -1; // divisions (transition steps?) } if (showStyle > 15) { warning("kSetShowStyle: Illegal style %d for plane %04x:%04x", showStyle, PRINT_REG(planeObj)); return s->r_acc; } // GK1 calls fadeout (13) / fadein (14) with the following parameters: // seconds: 1 // backColor: 0 / -1 // fade: 200 // animate: 0 // refFrame: 0 // divisions: 0 / 20 // TODO: Check if the plane is in the list of planes to draw switch (showStyle) { //case 0: // no transition, perhaps? (like in the previous SCI versions) case 13: // fade out // TODO case 14: // fade in // TODO default: // TODO: This is all a stub/skeleton, thus we're invoking kStub() for now kStub(s, argc, argv); break; } return s->r_acc; } reg_t kCelInfo(EngineState *s, int argc, reg_t *argv) { // Used by Shivers 1, room 23601 to determine what blocks on the red door puzzle board // are occupied by pieces already switch (argv[0].toUint16()) { // subops 0 - 4 // 0 - return the view // 1 - return the loop // 2, 3 - nop case 4: { GuiResourceId viewId = argv[1].toSint16(); int16 loopNo = argv[2].toSint16(); int16 celNo = argv[3].toSint16(); int16 x = argv[4].toUint16(); int16 y = argv[5].toUint16(); byte color = g_sci->_gfxCache->kernelViewGetColorAtCoordinate(viewId, loopNo, celNo, x, y); return make_reg(0, color); } default: { kStub(s, argc, argv); return s->r_acc; } } } reg_t kScrollWindow(EngineState *s, int argc, reg_t *argv) { // Used by SQ6 and LSL6 hires for the text area in the bottom of the // screen. The relevant scripts also exist in Phantasmagoria 1, but they're // unused. This is always called by scripts 64906 (ScrollerWindow) and // 64907 (ScrollableWindow). reg_t kWindow = argv[1]; uint16 op = argv[0].toUint16(); switch (op) { case 0: // Init g_sci->_gfxFrameout->initScrollText(argv[2].toUint16()); // maxItems g_sci->_gfxFrameout->clearScrollTexts(); return argv[1]; // kWindow case 1: // Show message, called by ScrollableWindow::addString case 14: // Modify message, called by ScrollableWindow::modifyString // 5 or 6 parameters // Seems to be called with 5 parameters when the narrator speaks, and // with 6 when Roger speaks { Common::String text = s->_segMan->getString(argv[2]); uint16 x = 0;//argv[3].toUint16(); // TODO: can't be x (values are all wrong) uint16 y = 0;//argv[4].toUint16(); // TODO: can't be y (values are all wrong) // TODO: argv[5] is an optional unknown parameter (an integer set to 0) g_sci->_gfxFrameout->addScrollTextEntry(text, kWindow, x, y, (op == 14)); } break; case 2: // Clear, called by ScrollableWindow::erase g_sci->_gfxFrameout->clearScrollTexts(); break; case 3: // Page up, called by ScrollableWindow::scrollTo // TODO kStub(s, argc, argv); break; case 4: // Page down, called by ScrollableWindow::scrollTo // TODO kStub(s, argc, argv); break; case 5: // Up arrow, called by ScrollableWindow::scrollTo g_sci->_gfxFrameout->prevScrollText(); break; case 6: // Down arrow, called by ScrollableWindow::scrollTo g_sci->_gfxFrameout->nextScrollText(); break; case 7: // Home, called by ScrollableWindow::scrollTo g_sci->_gfxFrameout->firstScrollText(); break; case 8: // End, called by ScrollableWindow::scrollTo g_sci->_gfxFrameout->lastScrollText(); break; case 9: // Resize, called by ScrollableWindow::resize and ScrollerWindow::resize // TODO kStub(s, argc, argv); break; case 10: // Where, called by ScrollableWindow::where // TODO // argv[2] is an unknown integer kStub(s, argc, argv); break; case 11: // Go, called by ScrollableWindow::scrollTo // 2 extra parameters here // TODO kStub(s, argc, argv); break; case 12: // Insert, called by ScrollableWindow::insertString // 3 extra parameters here // TODO kStub(s, argc, argv); break; // case 13 (Delete) is handled below // case 14 (Modify) is handled above case 15: // Hide, called by ScrollableWindow::hide g_sci->_gfxFrameout->toggleScrollText(false); break; case 16: // Show, called by ScrollableWindow::show g_sci->_gfxFrameout->toggleScrollText(true); break; case 17: // Destroy, called by ScrollableWindow::dispose g_sci->_gfxFrameout->clearScrollTexts(); break; case 13: // Delete, unused case 18: // Text, unused case 19: // Reconstruct, unused error("kScrollWindow: Unused subop %d invoked", op); break; default: error("kScrollWindow: unknown subop %d", op); break; } return s->r_acc; } reg_t kSetFontRes(EngineState *s, int argc, reg_t *argv) { // TODO: This defines the resolution that the fonts are supposed to be displayed // in. Currently, this is only used for showing high-res fonts in GK1 Mac, but // should be extended to handle other font resolutions such as those int xResolution = argv[0].toUint16(); //int yResolution = argv[1].toUint16(); g_sci->_gfxScreen->setFontIsUpscaled(xResolution == 640 && g_sci->_gfxScreen->getUpscaledHires() != GFX_SCREEN_UPSCALED_DISABLED); return s->r_acc; } reg_t kFont(EngineState *s, int argc, reg_t *argv) { // Handle font settings for SCI2.1 switch (argv[0].toUint16()) { case 1: // Set font resolution return kSetFontRes(s, argc - 1, argv + 1); default: warning("kFont: unknown subop %d", argv[0].toUint16()); } return s->r_acc; } // TODO: Eventually, all of the kBitmap operations should be put // in a separate class #define BITMAP_HEADER_SIZE 46 reg_t kBitmap(EngineState *s, int argc, reg_t *argv) { // Used for bitmap operations in SCI2.1 and SCI3. // This is the SCI2.1 version, the functionality seems to have changed in SCI3. switch (argv[0].toUint16()) { case 0: // init bitmap surface { // 6 params, called e.g. from TextView::init() in Torin's Passage, // script 64890 and TransView::init() in script 64884 uint16 width = argv[1].toUint16(); uint16 height = argv[2].toUint16(); //uint16 skip = argv[3].toUint16(); uint16 back = argv[4].toUint16(); // usually equals skip //uint16 width2 = (argc >= 6) ? argv[5].toUint16() : 0; //uint16 height2 = (argc >= 7) ? argv[6].toUint16() : 0; //uint16 transparentFlag = (argc >= 8) ? argv[7].toUint16() : 0; // TODO: skip, width2, height2, transparentFlag // (used for transparent bitmaps) int entrySize = width * height + BITMAP_HEADER_SIZE; reg_t memoryId = s->_segMan->allocateHunkEntry("Bitmap()", entrySize); byte *memoryPtr = s->_segMan->getHunkPointer(memoryId); memset(memoryPtr, 0, BITMAP_HEADER_SIZE); // zero out the bitmap header memset(memoryPtr + BITMAP_HEADER_SIZE, back, width * height); // Save totalWidth, totalHeight // TODO: Save the whole bitmap header, like SSCI does WRITE_LE_UINT16(memoryPtr, width); WRITE_LE_UINT16(memoryPtr + 2, height); return memoryId; } break; case 1: // dispose text bitmap surface return kDisposeTextBitmap(s, argc - 1, argv + 1); case 2: // dispose bitmap surface, with extra param // 2 params, called e.g. from MenuItem::dispose in Torin's Passage, // script 64893 warning("kBitmap(2), unk1 %d, bitmap ptr %04x:%04x", argv[1].toUint16(), PRINT_REG(argv[2])); break; case 3: // tiled surface { // 6 params, called e.g. from TiledBitmap::resize() in Torin's Passage, // script 64869 reg_t hunkId = argv[1]; // obtained from kBitmap(0) // The tiled view seems to always have 2 loops. // These loops need to have 1 cel in loop 0 and 8 cels in loop 1. uint16 viewNum = argv[2].toUint16(); // vTiles selector uint16 loop = argv[3].toUint16(); uint16 cel = argv[4].toUint16(); uint16 x = argv[5].toUint16(); uint16 y = argv[6].toUint16(); byte *memoryPtr = s->_segMan->getHunkPointer(hunkId); // Get totalWidth, totalHeight uint16 totalWidth = READ_LE_UINT16(memoryPtr); uint16 totalHeight = READ_LE_UINT16(memoryPtr + 2); byte *bitmap = memoryPtr + BITMAP_HEADER_SIZE; GfxView *view = g_sci->_gfxCache->getView(viewNum); uint16 tileWidth = view->getWidth(loop, cel); uint16 tileHeight = view->getHeight(loop, cel); const byte *tileBitmap = view->getBitmap(loop, cel); uint16 width = MIN(totalWidth - x, tileWidth); uint16 height = MIN(totalHeight - y, tileHeight); for (uint16 curY = 0; curY < height; curY++) { for (uint16 curX = 0; curX < width; curX++) { bitmap[(curY + y) * totalWidth + (curX + x)] = tileBitmap[curY * tileWidth + curX]; } } } break; case 4: // add text to bitmap { // 13 params, called e.g. from TextButton::createBitmap() in Torin's Passage, // script 64894 reg_t hunkId = argv[1]; // obtained from kBitmap(0) Common::String text = s->_segMan->getString(argv[2]); uint16 textX = argv[3].toUint16(); uint16 textY = argv[4].toUint16(); //reg_t unk5 = argv[5]; //reg_t unk6 = argv[6]; //reg_t unk7 = argv[7]; // skip? //reg_t unk8 = argv[8]; // back? //reg_t unk9 = argv[9]; uint16 fontId = argv[10].toUint16(); //uint16 mode = argv[11].toUint16(); uint16 dimmed = argv[12].toUint16(); //warning("kBitmap(4): bitmap ptr %04x:%04x, font %d, mode %d, dimmed %d - text: \"%s\"", // PRINT_REG(bitmapPtr), font, mode, dimmed, text.c_str()); uint16 foreColor = 255; // TODO byte *memoryPtr = s->_segMan->getHunkPointer(hunkId); // Get totalWidth, totalHeight uint16 totalWidth = READ_LE_UINT16(memoryPtr); uint16 totalHeight = READ_LE_UINT16(memoryPtr + 2); byte *bitmap = memoryPtr + BITMAP_HEADER_SIZE; GfxFont *font = g_sci->_gfxCache->getFont(fontId); int16 charCount = 0; uint16 curX = textX, curY = textY; const char *txt = text.c_str(); while (*txt) { charCount = g_sci->_gfxText32->GetLongest(txt, totalWidth, font); if (charCount == 0) break; for (int i = 0; i < charCount; i++) { unsigned char curChar = txt[i]; font->drawToBuffer(curChar, curY, curX, foreColor, dimmed, bitmap, totalWidth, totalHeight); curX += font->getCharWidth(curChar); } curX = textX; curY += font->getHeight(); txt += charCount; while (*txt == ' ') txt++; // skip over breaking spaces } } break; case 5: // fill with color { // 6 params, called e.g. from TextView::init() and TextView::draw() // in Torin's Passage, script 64890 reg_t hunkId = argv[1]; // obtained from kBitmap(0) uint16 x = argv[2].toUint16(); uint16 y = argv[3].toUint16(); uint16 fillWidth = argv[4].toUint16(); // width - 1 uint16 fillHeight = argv[5].toUint16(); // height - 1 uint16 back = argv[6].toUint16(); byte *memoryPtr = s->_segMan->getHunkPointer(hunkId); // Get totalWidth, totalHeight uint16 totalWidth = READ_LE_UINT16(memoryPtr); uint16 totalHeight = READ_LE_UINT16(memoryPtr + 2); uint16 width = MIN(totalWidth - x, fillWidth); uint16 height = MIN(totalHeight - y, fillHeight); byte *bitmap = memoryPtr + BITMAP_HEADER_SIZE; for (uint16 curY = 0; curY < height; curY++) { for (uint16 curX = 0; curX < width; curX++) { bitmap[(curY + y) * totalWidth + (curX + x)] = back; } } } break; default: kStub(s, argc, argv); break; } return s->r_acc; } // Used for edit boxes in save/load dialogs. It's a rewritten version of kEditControl, // but it handles events on its own, using an internal loop, instead of using SCI // scripts for event management like kEditControl does. Called by script 64914, // DEdit::hilite(). reg_t kEditText(EngineState *s, int argc, reg_t *argv) { reg_t controlObject = argv[0]; if (!controlObject.isNull()) { g_sci->_gfxControls32->kernelTexteditChange(controlObject); } return s->r_acc; } reg_t kAddLine(EngineState *s, int argc, reg_t *argv) { reg_t plane = argv[0]; Common::Point startPoint(argv[1].toUint16(), argv[2].toUint16()); Common::Point endPoint(argv[3].toUint16(), argv[4].toUint16()); // argv[5] is unknown (a number, usually 200) byte color = (byte)argv[6].toUint16(); byte priority = (byte)argv[7].toUint16(); byte control = (byte)argv[8].toUint16(); // argv[9] is unknown (usually a small number, 1 or 2). Thickness, perhaps? return g_sci->_gfxFrameout->addPlaneLine(plane, startPoint, endPoint, color, priority, control); } reg_t kUpdateLine(EngineState *s, int argc, reg_t *argv) { reg_t hunkId = argv[0]; reg_t plane = argv[1]; Common::Point startPoint(argv[2].toUint16(), argv[3].toUint16()); Common::Point endPoint(argv[4].toUint16(), argv[5].toUint16()); // argv[6] is unknown (a number, usually 200) byte color = (byte)argv[7].toUint16(); byte priority = (byte)argv[8].toUint16(); byte control = (byte)argv[9].toUint16(); // argv[10] is unknown (usually a small number, 1 or 2). Thickness, perhaps? g_sci->_gfxFrameout->updatePlaneLine(plane, hunkId, startPoint, endPoint, color, priority, control); return s->r_acc; } reg_t kDeleteLine(EngineState *s, int argc, reg_t *argv) { reg_t hunkId = argv[0]; reg_t plane = argv[1]; g_sci->_gfxFrameout->deletePlaneLine(plane, hunkId); return s->r_acc; } #endif } // End of namespace Sci