/* 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 3 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, see . * */ #include "engines/metaengine.h" #include "base/plugins.h" #include "base/version.h" #include "common/events.h" #include "common/system.h" #include "common/translation.h" #include "common/util.h" #include "graphics/surface.h" #include "graphics/fonts/amigafont.h" #include "gui/about.h" #include "gui/gui-manager.h" #include "gui/ThemeEval.h" namespace GUI { class EE; class EEHandler { public: bool handleKeyDown(Common::KeyState &state); }; enum { kScrollStartDelay = 1500, kScrollMillisPerPixel = 60 }; // Every Line should start with a letter followed by a digit. Currently those can be // (all subject to change) // Letter: // C, L, R -- set center/left/right alignment // A -- ASCII text to replace the next (latin1) line // Digit: // 0 - 2 -- set a custom color: // 0 normal text // 1 highlighted text // 2 disabled text // TODO: Maybe add a tab/indent feature; that is, make it possible to specify // an amount by which that line shall be indented (the indent of course would have // to be considered while performing any word wrapping, too). // // TODO: Add different font sizes (for bigger headlines) // TODO: Allow color change in the middle of a line... static const char *const copyright_text[] = { "", "C0""Copyright (C) 2001-2024 The ScummVM Team", "C0""https://www.scummvm.org", "", "C0""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 binary.", "", }; static const char *const gpl_text[] = { "", "C0""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 3 of the License, or (at your option) any later version.", "C0""", "C0""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.", "", "C0""You should have received a copy of the GNU General Public License along with this program. If not, see .", "", }; #include "gui/credits.h" AboutDialog::AboutDialog() : Dialog(10, 20, 300, 174), _scrollPos(0), _scrollTime(0), _willClose(false), _autoScroll(true) { reflowLayout(); int i; for (i = 0; i < 1; i++) _lines.push_back(Common::U32String()); Common::String version("C0""ScummVM "); version += gScummVMVersion; addLine(version); // I18N: built on with Common::U32String date = Common::U32String::format(_("(built on %s with %s)"), gScummVMBuildDate, gScummVMCompiler); addLine(Common::U32String("C2") + date); for (i = 0; i < ARRAYSIZE(copyright_text); i++) addLine(Common::U32String(copyright_text[i])); Common::U32String features("C1"); features += _("Features compiled in:"); addLine(features); Common::String featureList("C0"); featureList += gScummVMFeatures; addLine(featureList); _lines.push_back(Common::U32String()); Common::U32String extensionSupportString[3] = { _("not supported"), _("disabled"), _("enabled") }; byte sse2Support = 0; byte avx2Support = 0; byte neonSupport = 0; #ifdef SCUMMVM_SSE2 ++sse2Support; if (g_system->hasFeature(OSystem::kFeatureCpuSSE2)) ++sse2Support; #endif #ifdef SCUMMVM_AVX2 ++avx2Support; if (g_system->hasFeature(OSystem::kFeatureCpuAVX2)) ++avx2Support; #endif #ifdef SCUMMVM_NEON ++neonSupport; if (g_system->hasFeature(OSystem::kFeatureCpuNEON)) ++neonSupport; #endif Common::U32String extensionsInfo("C1"); // I18N: CPU extensions are sets of extra processor instructions used to speed up operations. See Intel AVX2, ARM NEON, etc. extensionsInfo += _("CPU extensions support:"); addLine(extensionsInfo); Common::U32String compiledExtensionsList("C0"); compiledExtensionsList += Common::U32String::format("SSE2(%S) AVX2(%S) NEON(%S)", extensionSupportString[sse2Support].c_str(), extensionSupportString[avx2Support].c_str(), extensionSupportString[neonSupport].c_str()); addLine(compiledExtensionsList); _lines.push_back(Common::U32String()); Common::U32String engines("C1"); engines += _("Available engines:"); addLine(engines); const PluginList &plugins = EngineMan.getPlugins(PLUGIN_TYPE_ENGINE); PluginList::const_iterator iter = plugins.begin(); for (; iter != plugins.end(); ++iter) { Common::String str; const Plugin *p = EngineMan.findPlugin((*iter)->getName()); if (!p) { warning("Cannot find plugin for %s", (*iter)->getName()); continue; } str = "C0"; str += p->get().getEngineName(); addLine(str); str = "C2"; str += p->get().getOriginalCopyright(); addLine(str); //addLine(""); } for (i = 0; i < ARRAYSIZE(gpl_text); i++) addLine(Common::U32String(gpl_text[i])); _lines.push_back(Common::U32String()); for (i = 0; i < ARRAYSIZE(credits); i++) addLine(Common::U32String(credits[i], Common::kUtf8)); } void AboutDialog::addLine(const Common::U32String &str) { Common::U32String::const_iterator strBeginItr = str.begin(); if (*strBeginItr == 0) { _lines.push_back(Common::U32String()); } else { Common::U32String format(str.begin(), str.begin() + 2); strBeginItr += 2; Common::U32String renderStr(strBeginItr, str.end()); Common::U32StringArray wrappedLines; g_gui.getFont().wordWrapText(renderStr, _w - 2 * _xOff, wrappedLines); for (Common::U32StringArray::const_iterator i = wrappedLines.begin(); i != wrappedLines.end(); ++i) { _lines.push_back(format + *i); } } } void AboutDialog::open() { _scrollTime = g_system->getMillis() + kScrollStartDelay; _scrollPos = 0; _willClose = false; Dialog::open(); } void AboutDialog::close() { Dialog::close(); } void AboutDialog::drawDialog(DrawLayer layerToDraw) { Dialog::drawDialog(layerToDraw); setTextDrawableArea(Common::Rect(_x, _y, _x + _w, _y + _h)); // Draw text // TODO: Add a "fade" effect for the top/bottom text lines // TODO: Maybe prerender all of the text into another surface, // and then simply compose that over the screen surface // in the right way. Should be even faster... const int firstLine = _scrollPos / _lineHeight; const int lastLine = MIN((_scrollPos + _h) / _lineHeight + 1, (uint32)_lines.size()); int y = _y + _yOff - (_scrollPos % _lineHeight); for (int line = firstLine; line < lastLine; line++) { Common::U32String str = _lines[line]; Common::U32String::const_iterator strLineItrBegin = _lines[line].begin(); Common::U32String::const_iterator strLineItrEnd = _lines[line].end(); Graphics::TextAlign align = Graphics::kTextAlignCenter; ThemeEngine::WidgetStateInfo state = ThemeEngine::kStateEnabled; if (strLineItrBegin != strLineItrEnd) { switch (*strLineItrBegin) { case 'C': align = Graphics::kTextAlignCenter; break; case 'L': align = Graphics::kTextAlignLeft; break; case 'R': align = Graphics::kTextAlignRight; break; default: error("Unknown scroller opcode '%c'", str[0]); break; } switch (*(strLineItrBegin + 1)) { case '0': state = ThemeEngine::kStateEnabled; break; case '1': state = ThemeEngine::kStateHighlight; break; case '2': state = ThemeEngine::kStateDisabled; break; case '3': warning("Need state for color 3"); // color = g_gui._shadowcolor; break; case '4': warning("Need state for color 4"); // color = g_gui._bgcolor; break; default: error("Unknown color type '%c'", str[1]); } strLineItrBegin += 2; } // Trim leading whitespaces if center mode is on if (align == Graphics::kTextAlignCenter) while (strLineItrBegin != strLineItrEnd && *strLineItrBegin == ' ') strLineItrBegin++; Common::U32String renderStr(strLineItrBegin, strLineItrEnd); if (!renderStr.empty()) g_gui.theme()->drawText(Common::Rect(_x + _xOff, y, _x + _w - _xOff, y + g_gui.theme()->getFontHeight()), renderStr, state, align, ThemeEngine::kTextInversionNone, 0, false, ThemeEngine::kFontStyleBold, ThemeEngine::kFontColorNormal, true, _textDrawableArea); y += _lineHeight; } } void AboutDialog::handleTickle() { const uint32 t = g_system->getMillis(); int scrollOffset = ((int)t - (int)_scrollTime) / kScrollMillisPerPixel; if (_autoScroll && scrollOffset > 0) { int modifiers = g_system->getEventManager()->getModifierState(); // Scroll faster when shift is pressed if (modifiers & Common::KBD_SHIFT) scrollOffset *= 4; // Reverse scrolling when alt is pressed if (modifiers & Common::KBD_ALT) scrollOffset *= -1; _scrollPos += scrollOffset; _scrollTime = t; if (_scrollPos < 0) { _scrollPos = 0; } else if ((uint32)_scrollPos > _lines.size() * _lineHeight) { _scrollPos = 0; _scrollTime += kScrollStartDelay; } drawDialog(kDrawLayerForeground); } } void AboutDialog::handleMouseUp(int x, int y, int button, int clickCount) { // Close upon any mouse click close(); } void AboutDialog::handleMouseWheel(int x, int y, int direction) { const int stepping = 5 * _lineHeight * direction; if (stepping == 0) return; _autoScroll = false; int newScrollPos = _scrollPos + stepping; if (newScrollPos < 0) { _scrollPos = 0; } else if ((uint32)newScrollPos < _lines.size() * _lineHeight) { _scrollPos = newScrollPos; } drawDialog(kDrawLayerForeground); } void AboutDialog::handleKeyDown(Common::KeyState state) { EEHandler eeHandler; if (eeHandler.handleKeyDown(state)) { reflowLayout(); return; } if (state.ascii) _willClose = true; } void AboutDialog::handleKeyUp(Common::KeyState state) { if (state.ascii && _willClose) close(); } void AboutDialog::reflowLayout() { Dialog::reflowLayout(); int i; const int screenW = g_system->getOverlayWidth(); const int screenH = g_system->getOverlayHeight(); _xOff = g_gui.xmlEval()->getVar("Globals.About.XOffset", 5); _yOff = g_gui.xmlEval()->getVar("Globals.About.YOffset", 5); int outerBorder = g_gui.xmlEval()->getVar("Globals.About.OuterBorder"); _w = screenW - 2 * outerBorder; _h = screenH - 2 * outerBorder; _lineHeight = g_gui.getFontHeight() + 3; // Heuristic to compute 'optimal' dialog width int maxW = _w - 2*_xOff; _w = 0; for (i = 0; i < ARRAYSIZE(credits); i++) { int tmp = g_gui.getStringWidth(credits[i]) + 5; if (_w < tmp && tmp <= maxW) { _w = tmp; } } _w += 2*_xOff; // Center the dialog _x = (screenW - _w) / 2; _y = (screenH - _h) / 2; } //////////////////////////////// //// Here we go //////////////////////////////// enum { kDirUp, kDirLeft, kDirRight }; enum { kModeMenu, kModePlay, kModePause }; enum { kPlComp, kPlHuman }; enum { kSndPoint, kSndHit, kSndLoss }; enum { kSpL1, kSpL2, kSpL3, kSpR1, kSpR2, kSpR3, kSpB1, kSpB2, kSpB3, kSpB4, kSp0, kSp1, kSp2, kSp3, kSp4, kSp5, kSp6, kSp7, kSp8, kSp9, kSpCode1, kSpCode1h = kSpCode1 + 6, kSpNum0 = kSpCode1h + 6, kSpStar = kSpNum0 + 10, kSpSpace, kSpLast }; class EE { public: EE(); ~EE(); void run(); private: Graphics::Surface _back; int _hits; bool _rmode; // animation after loosing int _score[2]; bool _soundEnabled; bool _shouldQuit; int _rcnt; //last 20 cycles of round int _winner; //who wins the point or serving int _tvelx, _tvely; int _x[2], _y[2], _px[2]; int _frame[2], _frameindex[2]; int _rebound[2]; int _bx, _by, _pbx, _pby, _tbx, _tby, _bvelx, _bvely; int _serve, _server, _servevel, _compserve; int _hitter, _starter, _sstage; int _bytop; int _rnd; int _olx, _oly, _orx, _ory, _obx, _oby; bool _air; bool _oCoords; int _mode; int _keymove[2][3]; int _opts; int _opt; Graphics::AmigaFont _font; Graphics::Surface _sp[kSpLast]; Graphics::PixelFormat _format; int _windowX, _windowY, _windowW, _windowH; float _scale; bool _inited; uint32 _colorBlack, _colorBlue, _colorOrange, _colorKey; void cls(bool update = true); void loadSounds(); void processKeyUp(Common::Event &e); void processKeyDown(Common::Event &e); void getPos(); void setwinner(); void resetpt(); void calcscore(); int destination(int pl, int destx, int tol); bool moveball(); void docollisions(); void computer0(); void computer1(); void init(); void draw(int sn, int x, int y); void doMenu(Common::Event &e); void putshapes(); void game(); void playSound(int d); void setupGraphics(); void genField(); }; bool EEHandler::handleKeyDown(Common::KeyState &state) { if (state.ascii == 'v') { EE *ee = new EE(); ee->run(); delete ee; g_gui.redrawFull(); return true; } return false; } EE::EE() { init(); _scale = 1.0; _windowW = 320; _windowH = 200; _windowX = _windowY = 0; _format = g_system->getOverlayFormat(); _colorBlack = _colorBlue = _colorOrange = _colorKey = 0; } EE::~EE() { for (int i = 0; i < kSpLast; i++) _sp[i].free(); } void EE::cls(bool update) { _back.fillRect(Common::Rect(0, 0, _windowW, _windowH), _colorBlack); if (update) g_system->copyRectToOverlay(_back.getPixels(), _back.pitch, _windowX, _windowY, _windowW, _windowH); } void EE::run() { if (!_inited) return; _shouldQuit = false; setupGraphics(); init(); while (!_shouldQuit) { Common::Event event; while (g_system->getEventManager()->pollEvent(event)) { switch (event.type) { case Common::EVENT_QUIT: case Common::EVENT_RETURN_TO_LAUNCHER: _shouldQuit = true; break; case Common::EVENT_SCREEN_CHANGED: if (g_gui.checkScreenChange()) setupGraphics(); break; case Common::EVENT_LBUTTONDOWN: break; case Common::EVENT_KEYDOWN: processKeyDown(event); break; case Common::EVENT_KEYUP: processKeyUp(event); break; default: break; } } game(); putshapes(); g_system->updateScreen(); g_system->delayMillis(10); } } const int polecol[] = {0, 1, 2, 3, 3, 4, 6, 7, 9, 14}; const int jump[] = {-4, -4, -3, -3, -3, -3, -2, -2, -2, -2, -2, -1, -1, -1, -1, -1, -1, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4}; void EE::loadSounds() { #if 0 if ($.isEmptyObject(sounds)) { for (i = 0; i < snNames.length; i++) { s = snNames[i]; sounds[s] = new Audio(); sounds[s].src = 'sounds/' + s + '.wav'; } } #endif } void EE::processKeyUp(Common::Event &e) { switch (e.kbd.keycode) { case Common::KEYCODE_LEFT: // left _keymove[1][kDirLeft] = 0; break; case Common::KEYCODE_RIGHT: // right _keymove[1][kDirRight] = 0; break; case Common::KEYCODE_UP: // top _keymove[1][kDirUp] = 0; break; case Common::KEYCODE_a: //a _keymove[0][kDirLeft] = 0; break; case Common::KEYCODE_d: // d _keymove[0][kDirRight] = 0; break; case Common::KEYCODE_w: // w _keymove[0][kDirUp] = 0; break; default: break; } } void EE::processKeyDown(Common::Event &e) { if (_mode == kModeMenu) { doMenu(e); return; } switch (e.kbd.keycode) { case Common::KEYCODE_LEFT: // left _keymove[1][kDirLeft] = -2; break; case Common::KEYCODE_RIGHT: // right _keymove[1][kDirRight] = 2; break; case Common::KEYCODE_UP: // top _keymove[1][kDirUp]= 1; break; case Common::KEYCODE_a: //a _keymove[0][kDirLeft] = -2; break; case Common::KEYCODE_d: // d _keymove[0][kDirRight] = 2; break; case Common::KEYCODE_w: // w _keymove[0][kDirUp] = 1; break; case Common::KEYCODE_ESCAPE: _mode = kModeMenu; break; case Common::KEYCODE_p: // p _keymove[0][kDirRight] = _keymove[0][kDirLeft] = _keymove[0][kDirUp] = 0; _mode = (_mode == kModePlay) ? kModePause : kModePlay; break; case Common::KEYCODE_r: // r if (_mode == kModePlay) init(); break; case Common::KEYCODE_s: // s if (!_soundEnabled) loadSounds(); _soundEnabled = !_soundEnabled; break; default: break; } } void EE::getPos() { for (int i = 0; i < 2; i++) { if (_keymove[i][kDirUp] && _frameindex[i] == -1) _frameindex[i] = 0; int velx = _keymove[i][kDirLeft] + _keymove[i][kDirRight]; int tx = _x[i] + velx; int bnd = 3 + i * 155; if (velx > 0) { if (tx < bnd + 119) _x[i] = tx; else _x[i] = bnd + 119; } else if (tx > bnd) { _x[i] = tx; } else { _x[i] = bnd; } if (_frameindex[i] == -2) { _y[i] = 173; _frame[i] = 0; _frameindex[i] = -1; } if (_frameindex[i] == -1) { if (velx != 0) { if (abs(_px[i] - _x[i]) > 4) { _frame[i] ^= 1; _px[i] = _x[i]; } } else { _frame[i] = 0; } } else { _frame[i] = 2 + (_frameindex[i] > 18); if (_frameindex[i] == 19) _y[i] -= 4; _y[i] += jump[_frameindex[i]++]; if (_frameindex[i] > 37) _frameindex[i] = -2; } } } void EE::setwinner() { if (_hits > 2) { _winner = 1 - _hitter; } else { _winner = _tbx < 150 ? 1 : 0; } _tvely = abs(_bvely) >> 3; _tvelx = abs(_bvelx) >> 3; } void EE::resetpt() { _keymove[0][kDirUp] = 0; _keymove[1][kDirUp] = 0; getPos(); if (abs(_bvelx) > _tvelx) { if (_bvelx < 0) _bvelx = -_tvelx; else _bvelx = _tvelx; } if (abs(_bvely) > _tvely) { if (_bvely < 0) _bvely = -_tvely; else _bvely = _tvely; } docollisions(); moveball(); putshapes(); } void EE::calcscore() { if (_winner == _server) { if (_soundEnabled && _mode != kModeMenu) { playSound(kSndPoint); } _score[_winner]++; if (_score[_winner] > 14 && _score[_winner] - _score[1 - _winner] > 1) { init(); return; } } else { if (_soundEnabled && _mode != kModeMenu) { playSound(kSndLoss); } _server = _winner; } _bx = (_tbx = _pbx = (64 + _winner * 165)) << 6; _by = (_tby = _pby = 135) << 6; _bvelx = _bvely = _hits = _rebound[0] = _rebound[1] = 0; _serve = _servevel = 1; _hitter = 2; _compserve = abs(_rnd) % 5; if (_score[_server] == 14) _compserve = 5; _sstage = 0; } int EE::destination(int pl, int destx, int tol) { int xp = _x[pl]; if (abs(xp - destx) < tol) { _keymove[pl][kDirLeft] = 0; _keymove[pl][kDirRight] = 0; return 1; } if (xp < destx) { _keymove[pl][kDirLeft] = 0; _keymove[pl][kDirRight] = 2; } else { _keymove[pl][kDirLeft] = -2; _keymove[pl][kDirRight] = 0; } return 0; } int reset = false; bool EE::moveball() { if (!reset) { _bx = 4096; _by = 8640; _bvelx = 125; _bvely = -259; reset = true; } int rbvelx = _bvelx; int rbvely = _bvely; if (rbvelx > 319) rbvelx = 319; if (rbvelx < -319) rbvelx = -319; if (rbvely > 319) rbvely = 319; if (rbvely < -319) rbvely = -319; _pbx = _tbx; _pby = _tby; _bx += rbvelx; _by += rbvely; if (_bx < 320) { _bx = 320; rbvelx = -rbvelx; rbvelx -= rbvelx >> 4; rbvely -= rbvely >> 4; if (_hitter == 1) { _hitter = 2; _hits = 0; } } if (_bx > 18112) { _bx = 18112; rbvelx = -rbvelx; rbvelx -= rbvelx >> 4; rbvely -= rbvely >> 4; if (_hitter == 0) { _hitter = 2; _hits = 0; } } if (_by < 832) { _by = 832; rbvely = -rbvely; rbvelx -= rbvelx >> 4; rbvely -= rbvely >> 4; } bool hitfloor; if (_by > 11392) { _by = 11392; rbvely = -rbvely; hitfloor = false; } else { hitfloor = true; } //if (rbvely > 0) // Checked with original, this is how it is // rbvely += 1; //else rbvely += 1; _tbx = _bx >> 6; _tby = _by >> 6; _bvelx = rbvelx; _bvely = rbvely; return hitfloor; } void EE::docollisions() { for (int i = 0; i < 2; i++) { int dx = _tbx - _x[i] - i * 7; int dy = (_tby - _y[i]) >> 1; int dist = (dx >> 2) * dx + dy * dy; if (dist < 110) { int rndoff = 8 - (_rnd & 15); if (_frameindex[i] > -1) _bvely = -abs(_bvely) + (jump[_frameindex[i]] << (3 << _servevel)); else _bvely = -abs(_bvely); _bvely += rndoff; _bvelx += dx * abs(dx) + ((_keymove[i][kDirRight] + _keymove[i][kDirLeft]) << (4 + _servevel)) + rndoff; if (!_rebound[i]) { if (_mode != kModeMenu) { if (_soundEnabled && _mode != kModeMenu) { playSound(kSndHit); } } _bytop = 200; _serve = 0; _rebound[i] = 1; if (_hitter != i) { _hitter = i; _hits = 0; } else { _hits++; } } } else if (_rebound[i]) { _rebound[i] = _servevel = 0; } } int i = 1; if (_tby > 91) { if (_pbx < 128 && _tbx > 127) { _bvelx = -abs(_bvelx) >> 1; _bx = 127 * 64; i = 0; } else if (_pbx > 159 && _tbx < 160) { _bvelx = abs(_bvelx) >> 1; _bx = 160 * 64; i = 0; } } if (i && _tby > 81 && _tbx > 127 && _tbx < 160) { if (_tby > 91) { if (_tbx < 148) _bvelx = -abs(_bvelx); else _bvelx = abs(_bvelx); } else if ((_tbx > 147 && 161 - _tbx >= polecol[91 - _tby]) || (_tbx < 148 && _tbx - 133 >= polecol[91 - _tby])) { if (_bvely > 0) { int dx = _tbx - 145; if (dx < -5) _bvelx = -abs(_bvelx); if (dx > 5) _bvelx = abs(_bvelx); _bvely = -abs(_bvely); } if (abs(_bvelx) > 32) _bvelx = _bvelx >> 1; if (abs(_bvely) > 32) _bvely = _bvely >> 1; } } } void EE::computer0() { _keymove[0][kDirUp] = 0; if (_tby < _bytop) _bytop = _tby; int rndoff = 5 - _rnd % 10; int dest = 0; if (_serve && ((_server & 1) == 0)) { switch (_compserve) { case 0: dest = destination(0, 55, 2); break; case 1: dest = destination(0, 84, 2); break; case 2: dest = destination(0, 80, 2); break; case 3: if (_sstage == 0) { if ((dest = destination(0, 44, 2))) _sstage = 1; } else { destination(0, 58, 2); dest = 1; } break; case 4: if (_sstage == 0) { if ((dest = destination(0, 90, 2))) _sstage = 1; } else { destination(0, 58, 2); dest = 1; } break; case 5: if (_sstage == 0) { if (destination(0, 3, 2)) _sstage = 1; dest = 0; } else { destination(0, 8 + _sstage++, 1); dest = 1; } break; default: break; } _keymove[0][kDirUp] = dest; } else if (_bvely > 0 && _tbx < 140) { int ystep, destx; if (_bvely >> 6 == 0) ystep = 0; else ystep = (140 - _tby) / (_bvely >> 6); if (ystep < 1 || (_bvelx >> 6) == 0) destx = _tbx; else destx = _tbx + (_bvelx >> 6) * ystep - 4; int dx = _x[0] - _tbx; if (abs(_bvelx) < 128 && _bytop < 75) { if ((_tby < 158) ^ (_bvelx < 0)) destination(0, _tbx - 15, 3); else destination(0, _tbx + 15, 3); } else { if (_tby > 130) { if (abs(dx) > 6 && abs(_bvelx) < 1024) { destination(0, _tbx - ((_x[0] - 60) >> 3), 3); } else { destination(0, _tbx + rndoff + ((_x[0] - 60) >> 3), 10); _keymove[0][kDirUp] = _x[0] < 105 && (_hitter != 0 || _hits < 2); } } else { if (destx < 3) destx = 6 - destx; if (destx > 123) destx = 246 - destx; destination(0, destx + rndoff, 3); } } } else destination(0, 56, 8); } void EE::computer1() { _keymove[1][kDirUp] = 0; if (_tby < _bytop) _bytop = _tby; int rndoff = 5 - _rnd % 10; if (_serve && ((_server & 1) == 1)) { int dest = 0; switch (_compserve) { case 0: dest = destination(1, 232, 2); break; case 1: dest = destination(1, 202, 2); break; case 2: dest = destination(1, 208, 2); break; case 3: if (_sstage == 0) { if ((dest = destination(1, 250, 2))) _sstage = 1; } else { destination(1, 220, 2); dest = 1; } break; case 4: if (_sstage == 0) { if ((dest = destination(1, 190, 2))) _sstage = 1; } else { destination(1, 230, 2); dest = 1; } break; case 5: if (_sstage == 0) { if (destination(1, 277, 2)) _sstage = 1; dest = 0; } else { destination(1, 272 - _sstage++, 1); dest = 1; } break; default: break; } _keymove[1][kDirUp] = dest; } else if (_bvely > 0 && _tbx > 125) { int ystep, destx; if (_bvely >> 6 == 0) ystep = 0; else ystep = (140 - _tby) / (_bvely >> 6); if (ystep < 1 || (_bvelx >> 6) == 0) destx = _tbx; else destx = _tbx + (_bvelx >> 6) * ystep - 4; int dx = _x[1] - _tbx; if (abs(_bvelx) < 128 && _bytop < 75) { if ((_tby < 158) ^ (_bvelx < 0)) destination(1, _tbx + 15, 3); else destination(1, _tbx - 15, 3); } else { if (_tby > 130) { if (abs(dx) > 6 && abs(_bvelx) < 1024) destination(1, _tbx + ((_x[1] - 218) >> 3), 3); else { destination(1, _tbx - rndoff - ((_x[1] - 218) >> 3), 10); _keymove[1][kDirUp] = _x[1] > 175 && (_hitter != 1 || _hits < 2); } } else { if (destx < 158) destx = 316 - destx; if (destx > 277) destx = 554 - destx; destination(1, destx - rndoff, 3); } } } else destination(1, 211, 8); } void EE::init() { _rnd = 0; _starter = _winner = _hits = 0; _bvelx = _bvely = 0; _tbx = 200; _tby = 20; _bytop = 200; _x[0] = 64; _x[1] = 226; _air = false; _rcnt = 0; _tvelx = _tvely = 0; for (int i = 0; i < 2; i++) { _px[i] = _x[i]; _y[i] = 173; _frameindex[i] = -1; _rebound[i] = 0; _score[i] = 0; _keymove[i][kDirRight] = _keymove[i][kDirLeft] = _keymove[i][kDirUp] = 0; _frame[i] = 0; } _oCoords = false; _opts = 3; _opt = 0; _mode = kModeMenu; _soundEnabled = false; _olx = _oly = _orx = _ory = _obx = _oby = 0; _compserve = 0; _sstage = 0; _tbx = 64 + _starter * 165; _pbx = _tbx; _bx = _tbx << 6; _tby = 135; _pby = 135; _by = _tby << 6; _server = 2 + _starter; _hitter = 2; _serve = _servevel = 1; _rmode = false; _shouldQuit = false; _inited = true; } void EE::draw(int sn, int x1, int y1) { int x = x1 * _scale; int y = y1 * _scale; _back.copyRectToSurfaceWithKey(_sp[sn].getPixels(), _sp[sn].pitch, x, y, _sp[sn].w, _sp[sn].h, _colorKey); g_system->copyRectToOverlay(_back.getBasePtr(x, y), _back.pitch, _windowX + x, _windowY + y, _sp[sn].w, _sp[sn].h); } void EE::putshapes() { int sprite; cls(false); if (_oCoords) { int obx = _obx * _scale, oby = _oby * _scale; int olx = _olx * _scale, oly = _oly * _scale; int orx = _orx * _scale, ory = _ory * _scale; g_system->copyRectToOverlay(_back.getBasePtr(obx, oby), _back.pitch, _windowX + obx, _windowY + oby, _sp[kSpB1].w, _sp[kSpB1].h); g_system->copyRectToOverlay(_back.getBasePtr(olx, oly), _back.pitch, _windowX + olx, _windowY + oly, _sp[kSpL1].w, _sp[kSpL1].h); g_system->copyRectToOverlay(_back.getBasePtr(orx, ory), _back.pitch, _windowX + orx, _windowY + ory, _sp[kSpR1].w, _sp[kSpR1].h); } sprite = kSpB1 + (_tbx / 16) % 4; draw(sprite, _tbx, _tby); _obx = _tbx; _oby = _tby; if (_y[0] < 173) { sprite = kSpL3; } else { if ((_x[0] / 8) % 2) sprite = kSpL1; else sprite = kSpL2; } draw(sprite, _x[0], _y[0]); _olx = _x[0]; _oly = _y[0]; if (_y[1] < 173) { sprite = kSpR3; } else { if ((_x[1] / 8) % 2) sprite = kSpR1; else sprite = kSpR2; } draw(sprite, _x[1], _y[1]); _orx = _x[1]; _ory = _y[1]; _oCoords = true; const int frames[] = { 153, 106, 160, 200, 4, 13, 6, 200, 310, 13, 312, 200, 4, 13, 312, 14, 4, 199, 312, 200, 154, 107, 159, 199 }; uint32 color1 = _back.format.RGBToColor(5 * 16, 7 * 16, 8 * 16); uint32 color2 = _back.format.RGBToColor(6 * 16, 8 * 16, 12 * 16); const int *ptr = frames; for (int i = 0; i < 6; i++, ptr += 4) { Common::Rect r(ptr[0] * _scale, ptr[1] * _scale, ptr[2] * _scale, ptr[3] * _scale); _back.fillRect(r, (i < 5 ? color1 : color2)); g_system->copyRectToOverlay(_back.getBasePtr(r.left, r.top), _back.pitch, _windowX + r.left, _windowY + r.top, r.width(), r.height()); } for (int i = 0; i < 2; i++) { int startx = i ? 264 : 32; draw(kSpNum0 + _score[i] / 10, startx, 1); startx += 8; draw(kSpNum0 + _score[i] % 10, startx, 1); startx += 8; draw(_server == i ? kSpStar : kSpSpace, startx, 1); } for (int i = 0; i < 6; i++) { int sx = i == 0 ? 92 : 80; int sy = i == 0 ? 2 : 20; int code = i == 0 ? kSpCode1 : i - 1 == _opt ? kSpCode1h + i : kSpCode1 + i; draw(code, sx, sy + i * 10); if (_mode != kModeMenu) break; } } void EE::doMenu(Common::Event &e) { switch (e.kbd.keycode) { case Common::KEYCODE_UP: _opt--; if (_opt < 0) _opt = 4; break; case Common::KEYCODE_DOWN: _opt = (_opt + 1) % 5; break; case Common::KEYCODE_RETURN: if (_opt < 4) { _opts = _opt; _mode = kModePlay; cls(); } else { _shouldQuit = true; } break; case Common::KEYCODE_ESCAPE: _shouldQuit = true; break; default: break; } } void EE::game() { if (_mode == kModePlay || _mode == kModeMenu) { _rnd = (_rnd * 5 + 1) % 32767; if (_opts & 1) computer0(); if (_opts & 2) computer1(); if (_rmode) { if (_rcnt-- > 0 || _frameindex[0] != -1 || _frameindex[1] != -1) { resetpt(); } else { _rmode = false; calcscore(); } return; } getPos(); if (_serve) { docollisions(); _air = true; } else { if (_air) { docollisions(); _air = moveball(); } } if (!_air || _hits > 2) { setwinner(); _rmode = true; _rcnt = 20; } } } void EE::playSound(int sound) { // FIXME } static const uint32 ball[21] = { 0x0000fd00, 0x00074ac0, 0x000afde0, 0x001703e8, 0x0038f7d8, 0x0056f73c, 0x00aef6fc, 0x00cf79fb, 0x01afbbf7, 0x01b7bdcd, 0x0077dd3a, 0x013beef3, 0x01bbef6a, 0x015df78b, 0x00defbea, 0x00ef7dd8, 0x00779eb4, 0x0029ef68, 0x001ef4a0, 0x000673c0, 0x0000ae00 }; static const uint32 head[38] = { 0xbb400000, 0x0000000f, 0xea650000, 0x000003af, 0xfaaa6000, 0x00000e7f, 0xfea5a400, 0x03ff2af0, 0x3eaa6900, 0x3ea969c0, 0x3ea99940, 0xeaaa69c8, 0x3a669540, 0xea995540, 0x5a555550, 0xa9aa999a, 0x65a65550, 0x2aa69a5a, 0x59555650, 0x02aaaa55, 0x6aa66550, 0x000aa950, 0x6a569950, 0x000a9500, 0xa9595580, 0x00015001, 0xa9a55540, 0x00000015, 0xa9a9a500, 0x0000015a, 0x59695800, 0x00001659, 0x65659000, 0x00000a96, 0xa65a8000, 0x000002aa, 0xa6500000, 0x0000002a }; static const uint32 legs[42] = { 0xa0000000, 0x00000000, 0x80000000, 0x00000002, 0x80000000, 0x00000002, 0xa0000000, 0x00000000, 0x50000000, 0x00000000, 0xf0000000, 0x00000003, 0xfc000000, 0x000003ff, 0xa0000000, 0x00000002, 0x0a000000, 0x0000000a, 0x02400000, 0x00000028, 0x00700000, 0x00000028, 0x00fc0000, 0x00000014, 0x03c00000, 0x000000fc, 0x0f000000, 0x0000ffff, 0xa0000000, 0x00000000, 0x80000000, 0x00000002, 0xa8000000, 0x00000002, 0x29c00000, 0x00000000, 0x01f00000, 0x00000000, 0x0fc00000, 0x00000000, 0xfc000000, 0x00000000 }; static const int spcolors[10 * 3] = { 0, 0, 0, 8, 15, 8, 6, 12, 6, 12, 13, 12, 0, 0, 0, 15, 8, 8, 12, 6, 6, 13, 12, 12, 0, 0, 0, 12, 12, 12 }; const char *const codes = "Dvhgkm#Ztrsm|ffrs(#$%&'#$%&O}pes&}{1$M{tiq$%&'#$M{tiq${y5(Fsrv||hv%&'#$" "Hutxxxjx'~v2%N|udr%&'#Gtsw}wiw&}{1$Hutxxxjx'#$%&'(#$%W|qw$%&'(#$%&'"; void EE::setupGraphics() { // Determine scale factor float scaleX = g_system->getOverlayWidth() * 0.9 / 320; float scaleY = g_system->getOverlayHeight() * 0.9 / 200; _scale = MIN(scaleX, scaleY); _windowW = 320 * _scale; _windowH = 200 * _scale; _windowX = (g_system->getOverlayWidth() > 320) ? (g_system->getOverlayWidth() - _windowW) / 2 : 0; _windowY = (g_system->getOverlayHeight() > 200) ? (g_system->getOverlayHeight() - _windowH) / 2 : 0; _format = g_system->getOverlayFormat(); _back.create(_windowW, _windowH, _format); _colorBlack = _format.RGBToColor( 0 * 16, 0 * 16, 0 * 16); _colorBlue = _format.RGBToColor( 5 * 16, 7 * 16, 8 * 16); _colorOrange = _format.RGBToColor(15 * 16, 7 * 16, 8 * 16); _colorKey = _colorBlack; cls(); uint32 palette[12]; for (int i = 0; i < 10 * 3; i += 3) palette[i / 3] = _back.format.RGBToColor(spcolors[i] * 16, spcolors[i + 1] * 16, spcolors[i + 2] * 16); Graphics::Surface tmp; int w = 25, h = 21; tmp.create(w, h, g_system->getOverlayFormat()); for (int s = 0; s < 4; s++) { int posy = 0, dy = 1; if (s & 2) { posy = 20; dy = -1; } for (int y = 0; y < h; y++, posy += dy) { uint32 pixels = ball[y]; int posx = 0, dx = 1; if (s & 1) { posx = 24; dx = -1; } for (int x = 0; x < w; x++, posx += dx) { int color = pixels & 1; pixels >>= 1; tmp.setPixel(posx, posy, palette[color + 8]); } } Graphics::Surface *tmp2 = tmp.scale(w * _scale, h * _scale); _sp[kSpB1 + s].copyFrom(*tmp2); tmp2->free(); delete tmp2; } tmp.free(); w = 32; h = 26; tmp.create(w, h, g_system->getOverlayFormat()); for (int s = 0; s < 6; s++) { for (int y = 0; y < h; y++) { const uint32 *ptr = (y < 19) ? &head[y * 2] : &legs[(y - 19 + (s % 3) * 7) * 2]; uint32 pixels = *ptr++; int posx = 0, dx = 1; if (s > 2) { posx = 31; dx = -1; } for (int x = 0; x < w; x++, posx += dx) { int color = pixels & 3; pixels >>= 2; if (x == 15) pixels = *ptr; tmp.setPixel(posx, y, palette[color + 4 * (s / 3)]); } } Graphics::Surface *tmp2 = tmp.scale(w * _scale, h * _scale); _sp[kSpL1 + s].copyFrom(*tmp2); tmp2->free(); delete tmp2; } tmp.free(); w = 23 * 8; h = 10; tmp.create(w, h, g_system->getOverlayFormat()); for (int hl = 0; hl < 2; hl++) { for (int i = 0; i < 6; i++) { tmp.fillRect(Common::Rect(0, 0, w, h), hl == 1 ? _colorBlue : _colorKey); char buf[100]; int c; for (c = 0; c < 23; c++) buf[c] = codes[c + 23 * i] - 3 - c % 6; buf[c] = 0; int c1 = i == 0 ? _colorOrange : hl == 1 ? _colorKey : _colorBlue; _font.drawString(&tmp, buf, 0, 1, w, c1, Graphics::kTextAlignLeft, 0, false); Graphics::Surface *tmp2 = tmp.scale(w * _scale, h * _scale, true); _sp[kSpCode1 + hl * 6 + i].copyFrom(*tmp2); tmp2->free(); delete tmp2; } } tmp.free(); w = 8; h = 10; tmp.create(w, h, g_system->getOverlayFormat()); for (int i = 0; i < 12; i++) { tmp.fillRect(Common::Rect(0, 0, w, h), _colorKey); char buf[2]; buf[0] = i == 10 ? '*' : i == 11 ? ' ' : '0' + i; buf[1] = 0; int c = i > 9 ? _colorBlue : _colorOrange; _font.drawString(&tmp, buf, 0, 1, w, c, Graphics::kTextAlignLeft, 0, false); Graphics::Surface *tmp2 = tmp.scale(w * _scale, h * _scale, true); _sp[kSpNum0 + i].copyFrom(*tmp2); tmp2->free(); delete tmp2; } tmp.free(); } } // End of namespace GUI