From defbaa3f604b003dd285dd6c99e93f8c74c9d6ac Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Thu, 29 Jan 2015 18:03:00 +0300 Subject: [PATCH] improve workspace operations; add upgrade dependencies and refresh project commans; add Tetris project as a sample --- src/dlangide.d | 2 +- src/dlangide/builders/builder.d | 5 +- src/dlangide/ui/commands.d | 4 + src/dlangide/ui/frame.d | 52 +- src/dlangide/ui/outputpanel.d | 18 + src/dlangide/workspace/project.d | 17 + src/dlangide/workspace/workspace.d | 3 +- views/res/i18n/en.ini | 2 + workspaces/tetris/dub.json | 17 + workspaces/tetris/src/gui.d | 599 ++++++++++++++++++ workspaces/tetris/src/model.d | 444 +++++++++++++ workspaces/tetris/src/tetris.d | 50 ++ workspaces/tetris/views/res/arrow-down.png | Bin 0 -> 843 bytes workspaces/tetris/views/res/arrow-left.png | Bin 0 -> 834 bytes workspaces/tetris/views/res/arrow-right.png | Bin 0 -> 810 bytes workspaces/tetris/views/res/dtetris-logo1.png | Bin 0 -> 971 bytes workspaces/tetris/views/res/levelup.png | Bin 0 -> 1220 bytes workspaces/tetris/views/res/pause.png | Bin 0 -> 1145 bytes .../tetris/views/res/popup_background.9.png | Bin 0 -> 374 bytes workspaces/tetris/views/res/rotate.png | Bin 0 -> 1934 bytes .../tetris/views/res/tetris_logo_big.png | Bin 0 -> 3509 bytes workspaces/tetris/views/res/tx_fabric.jpg | Bin 0 -> 9061 bytes workspaces/tetris/views/resources.list | 10 + 23 files changed, 1211 insertions(+), 12 deletions(-) create mode 100644 workspaces/tetris/dub.json create mode 100644 workspaces/tetris/src/gui.d create mode 100644 workspaces/tetris/src/model.d create mode 100644 workspaces/tetris/src/tetris.d create mode 100644 workspaces/tetris/views/res/arrow-down.png create mode 100644 workspaces/tetris/views/res/arrow-left.png create mode 100644 workspaces/tetris/views/res/arrow-right.png create mode 100644 workspaces/tetris/views/res/dtetris-logo1.png create mode 100644 workspaces/tetris/views/res/levelup.png create mode 100644 workspaces/tetris/views/res/pause.png create mode 100644 workspaces/tetris/views/res/popup_background.9.png create mode 100644 workspaces/tetris/views/res/rotate.png create mode 100644 workspaces/tetris/views/res/tetris_logo_big.png create mode 100644 workspaces/tetris/views/res/tx_fabric.jpg create mode 100644 workspaces/tetris/views/resources.list diff --git a/src/dlangide.d b/src/dlangide.d index cf9338d..f7af2d7 100644 --- a/src/dlangide.d +++ b/src/dlangide.d @@ -39,7 +39,7 @@ extern (C) int UIAppMain(string[] args) { // open home screen tab frame.showHomeScreen(); // for testing: load workspace at startup - frame.openFileOrWorkspace(appendPath(exePath, "../workspaces/sample1/sample1.dlangidews")); + //frame.openFileOrWorkspace(appendPath(exePath, "../workspaces/sample1/sample1.dlangidews")); // show window window.show(); diff --git a/src/dlangide/builders/builder.d b/src/dlangide/builders/builder.d index ae5f679..6f07359 100644 --- a/src/dlangide/builders/builder.d +++ b/src/dlangide/builders/builder.d @@ -63,9 +63,12 @@ class Builder : BackgroundOperationWatcher { params ~= "clean".dup; } else if (_buildOp == BuildOperation.Run) { params ~= "run".dup; + } else if (_buildOp == BuildOperation.Upgrade) { + params ~= "upgrade".dup; + params ~= "--force-remove".dup; } - if (_buildOp != BuildOperation.Clean) { + if (_buildOp != BuildOperation.Clean && _buildOp != BuildOperation.Upgrade) { switch (_buildConfig) { default: case BuildConfiguration.Debug: diff --git a/src/dlangide/ui/commands.d b/src/dlangide/ui/commands.d index c68952f..fbbec86 100644 --- a/src/dlangide/ui/commands.d +++ b/src/dlangide/ui/commands.d @@ -22,6 +22,8 @@ enum IDEActions : int { BuildProject, RebuildProject, CleanProject, + RefreshProject, + UpdateProjectDependencies, SetStartupProject, ProjectSettings, DebugStart, @@ -52,6 +54,8 @@ const Action ACTION_PROJECT_REBUILD = new Action(IDEActions.RebuildProject, "MEN const Action ACTION_PROJECT_CLEAN = new Action(IDEActions.CleanProject, "MENU_BUILD_PROJECT_CLEAN"c, null); const Action ACTION_PROJECT_SET_STARTUP = new Action(IDEActions.SetStartupProject, "MENU_PROJECT_SET_AS_STARTUP"c, null); const Action ACTION_PROJECT_SETTINGS = new Action(IDEActions.ProjectSettings, "MENU_PROJECT_SETTINGS"c, null); +const Action ACTION_PROJECT_REFRESH = new Action(IDEActions.RefreshProject, "MENU_PROJECT_REFRESH"c); +const Action ACTION_PROJECT_UPDATE_DEPENDENCIES = new Action(IDEActions.UpdateProjectDependencies, "MENU_PROJECT_UPDATE_DEPENDENCIES"c); const Action ACTION_DEBUG_START = new Action(IDEActions.DebugStart, "MENU_DEBUG_START_DEBUGGING"c, "debug-run"c, KeyCode.F5, 0); const Action ACTION_DEBUG_START_NO_DEBUG = new Action(IDEActions.DebugStartNoDebug, "MENU_DEBUG_START_NO_DEBUGGING"c, null, KeyCode.F5, KeyFlag.Control); const Action ACTION_DEBUG_CONTINUE = new Action(IDEActions.DebugContinue, "MENU_DEBUG_CONTINUE"c); diff --git a/src/dlangide/ui/frame.d b/src/dlangide/ui/frame.d index fc4eec0..cb498dd 100644 --- a/src/dlangide/ui/frame.d +++ b/src/dlangide/ui/frame.d @@ -298,7 +298,7 @@ class IDEFrame : AppFrame { editItem.add(new Action(20, "MENU_EDIT_PREFERENCES")); MenuItem projectItem = new MenuItem(new Action(21, "MENU_PROJECT")); - projectItem.add(ACTION_PROJECT_SET_STARTUP, ACTION_PROJECT_SETTINGS); + projectItem.add(ACTION_PROJECT_SET_STARTUP, ACTION_PROJECT_REFRESH, ACTION_PROJECT_UPDATE_DEPENDENCIES, ACTION_PROJECT_SETTINGS); MenuItem buildItem = new MenuItem(new Action(22, "MENU_BUILD")); buildItem.add(ACTION_WORKSPACE_BUILD, ACTION_WORKSPACE_REBUILD, ACTION_WORKSPACE_CLEAN, @@ -416,6 +416,14 @@ class IDEFrame : AppFrame { buildProject(BuildOperation.Run); //setBackgroundOperation(new BackgroundOperationWatcherTest(this)); return true; + case IDEActions.UpdateProjectDependencies: + buildProject(BuildOperation.Upgrade); + //setBackgroundOperation(new BackgroundOperationWatcherTest(this)); + return true; + case IDEActions.RefreshProject: + refreshWorkspace(); + //setBackgroundOperation(new BackgroundOperationWatcherTest(this)); + return true; case IDEActions.WindowCloseAllDocuments: askForUnsavedEdits(delegate() { closeAllDocuments(); @@ -454,15 +462,20 @@ class IDEFrame : AppFrame { return; } } else if (filename.isProjectFile) { + _logPanel.clear(); + _logPanel.logLine("Trying to open project from " ~ filename); Project project = new Project(); if (!project.load(filename)) { + _logPanel.logLine("Cannot read project file " ~ filename); window.showMessageBox(UIString("Cannot open project"d), UIString("Error occured while opening project"d)); return; } + _logPanel.logLine("Project file is opened ok"); string defWsFile = project.defWorkspaceFile; if (currentWorkspace) { Project existing = currentWorkspace.findProject(project.filename); if (existing) { + _logPanel.logLine("This project already exists in current workspace"); window.showMessageBox(UIString("Open project"d), UIString("Project is already in workspace"d)); return; } @@ -470,28 +483,43 @@ class IDEFrame : AppFrame { [ACTION_ADD_TO_CURRENT_WORKSPACE, ACTION_CREATE_NEW_WORKSPACE, ACTION_CANCEL], 0, delegate(const Action result) { if (result.id == IDEActions.CreateNewWorkspace) { // new ws - Workspace ws = new Workspace(); - ws.name = project.name; - ws.description = project.description; - ws.addProject(project); - ws.save(defWsFile); - setWorkspace(ws); + createNewWorkspaceForExistingProject(project); } else if (result.id == IDEActions.AddToCurrentWorkspace) { // add to current currentWorkspace.addProject(project); currentWorkspace.save(); - _wsPanel.reloadItems(); + refreshWorkspace(); } return true; }); } else { // new workspace file + createNewWorkspaceForExistingProject(project); } } else { + _logPanel.logLine("File is not recognized as DlangIDE project or workspace file"); window.showMessageBox(UIString("Invalid workspace file"d), UIString("This file is not a valid workspace or project file"d)); } } + void refreshWorkspace() { + _logPanel.logLine("Refreshing workspace"); + _wsPanel.reloadItems(); + } + + void createNewWorkspaceForExistingProject(Project project) { + string defWsFile = project.defWorkspaceFile; + _logPanel.logLine("Creating new workspace " ~ defWsFile); + // new ws + Workspace ws = new Workspace(); + ws.name = project.name; + ws.description = project.description; + ws.addProject(project); + ws.save(defWsFile); + setWorkspace(ws); + _logPanel.logLine("Done"); + } + //bool loadWorkspace(string path) { // // testing workspace loader // Workspace ws = new Workspace(); @@ -505,11 +533,17 @@ class IDEFrame : AppFrame { closeAllDocuments(); currentWorkspace = ws; _wsPanel.workspace = ws; + if (ws.startupProject && ws.startupProject.mainSourceFile) { + openSourceFile(ws.startupProject.mainSourceFile.filename); + _tabs.setFocus(); + } } void buildProject(BuildOperation buildOp) { - if (!currentWorkspace || !currentWorkspace.startupProject) + if (!currentWorkspace || !currentWorkspace.startupProject) { + _logPanel.logLine("No project is opened"); return; + } Builder op = new Builder(this, currentWorkspace.startupProject, _logPanel, currentWorkspace.buildConfiguration, buildOp, false); setBackgroundOperation(op); } diff --git a/src/dlangide/ui/outputpanel.d b/src/dlangide/ui/outputpanel.d index 26f26e5..c6937a4 100644 --- a/src/dlangide/ui/outputpanel.d +++ b/src/dlangide/ui/outputpanel.d @@ -4,6 +4,8 @@ import dlangui.all; import dlangide.workspace.workspace; import dlangide.workspace.project; +import std.utf; + class OutputPanel : DockWindow { protected LogWidget _logWidget; @@ -24,6 +26,22 @@ class OutputPanel : DockWindow { _logWidget.appendText(msg); } + void logLine(string category, dstring msg) { + appendText(category, msg ~ "\n"); + } + + void logLine(dstring msg) { + logLine(null, msg); + } + + void logLine(string category, string msg) { + appendText(category, toUTF32(msg ~ "\n")); + } + + void logLine(string msg) { + logLine(null, msg); + } + void clear(string category = null) { _logWidget.text = ""d; } diff --git a/src/dlangide/workspace/project.d b/src/dlangide/workspace/project.d index 1d7abd1..b5c22bc 100644 --- a/src/dlangide/workspace/project.d +++ b/src/dlangide/workspace/project.d @@ -209,6 +209,7 @@ class Project : WorkspaceItem { protected Workspace _workspace; protected bool _opened; protected ProjectFolder _items; + protected ProjectSourceFile _mainSourceFile; this(string fname = null) { super(fname); _items = new ProjectFolder(fname); @@ -220,6 +221,7 @@ class Project : WorkspaceItem { return buildNormalizedPath(_dir, path); } + @property ProjectSourceFile mainSourceFile() { return _mainSourceFile; } @property ProjectFolder items() { return _items; } @@ -244,6 +246,19 @@ class Project : WorkspaceItem { return folder; } + void findMainSourceFile() { + string n = toUTF8(name); + string[] mainnames = ["app.d", "main.d", n ~ ".d"]; + foreach(sname; mainnames) { + _mainSourceFile = findSourceFileItem(buildNormalizedPath(_dir, "src", sname)); + if (_mainSourceFile) + break; + _mainSourceFile = findSourceFileItem(buildNormalizedPath(_dir, "source", sname)); + if (_mainSourceFile) + break; + } + } + /// tries to find source file in project, returns found project source file item, or null if not found ProjectSourceFile findSourceFileItem(ProjectItem dir, string filename) { for (int i = 0; i < dir.childCount; i++) { @@ -266,6 +281,7 @@ class Project : WorkspaceItem { } override bool load(string fname = null) { + _mainSourceFile = null; if (fname.length > 0) filename = fname; if (!exists(filename) || !isFile(filename)) { @@ -282,6 +298,7 @@ class Project : WorkspaceItem { Log.d(" project description: ", _description); _items = findItems(); + findMainSourceFile(); } catch (JSONException e) { Log.e("Cannot parse json", e); return false; diff --git a/src/dlangide/workspace/workspace.d b/src/dlangide/workspace/workspace.d index 9e0924d..dbd5cba 100644 --- a/src/dlangide/workspace/workspace.d +++ b/src/dlangide/workspace/workspace.d @@ -12,7 +12,8 @@ enum BuildOperation { Build, Clean, Rebuild, - Run + Run, + Upgrade } enum BuildConfiguration { diff --git a/views/res/i18n/en.ini b/views/res/i18n/en.ini index fa69540..f49e621 100644 --- a/views/res/i18n/en.ini +++ b/views/res/i18n/en.ini @@ -28,6 +28,8 @@ MENU_BUILD_PROJECT_CLEAN=Clean Project MENU_PROJECT=&PROJECT MENU_PROJECT_SET_AS_STARTUP=Set as Startup Project MENU_PROJECT_SETTINGS=Project Settings +MENU_PROJECT_REFRESH=Refresh Workspace Items +MENU_PROJECT_UPDATE_DEPENDENCIES=Upgrade Dependencies MENU_DEBUG=&DEBUG MENU_DEBUG_START_DEBUGGING=Start Debugging MENU_DEBUG_START_NO_DEBUGGING=Start Without Debugging diff --git a/workspaces/tetris/dub.json b/workspaces/tetris/dub.json new file mode 100644 index 0000000..efbd2d8 --- /dev/null +++ b/workspaces/tetris/dub.json @@ -0,0 +1,17 @@ +{ + "name": "tetris", + "description": "dlangui library example: Tetris game", + "homepage": "https://github.com/buggins/dlangui", + "license": "Boost", + "authors": ["Vadim Lopatin"], + + "targetPath": "bin", + "targetType": "executable", + "targetName": "tetris", + + "stringImportPaths": ["views", "views/res", "views/res/i18n", "views/res/mdpi"], + + "dependencies": { + "dlangui:dlanguilib": "~master" + } +} diff --git a/workspaces/tetris/src/gui.d b/workspaces/tetris/src/gui.d new file mode 100644 index 0000000..61e19ba --- /dev/null +++ b/workspaces/tetris/src/gui.d @@ -0,0 +1,599 @@ +module gui; + +import model; + +import dlangui.all; + +/// game action codes +enum TetrisAction : int { + MoveLeft = 10000, + MoveRight, + RotateCCW, + FastDown, + Pause, + LevelUp, +} + +const Action ACTION_MOVE_LEFT = (new Action(TetrisAction.MoveLeft, KeyCode.LEFT)).addAccelerator(KeyCode.KEY_A).iconId("arrow-left"); +const Action ACTION_MOVE_RIGHT = (new Action(TetrisAction.MoveRight, KeyCode.RIGHT)).addAccelerator(KeyCode.KEY_D).iconId("arrow-right"); +const Action ACTION_ROTATE = (new Action(TetrisAction.RotateCCW, KeyCode.UP)).addAccelerator(KeyCode.KEY_W).iconId("rotate"); +const Action ACTION_FAST_DOWN = (new Action(TetrisAction.FastDown, KeyCode.SPACE)).addAccelerator(KeyCode.KEY_S).iconId("arrow-down"); +const Action ACTION_PAUSE = (new Action(TetrisAction.Pause, KeyCode.ESCAPE)).addAccelerator(KeyCode.PAUSE).iconId("pause"); +const Action ACTION_LEVEL_UP = (new Action(TetrisAction.LevelUp, KeyCode.ADD)).addAccelerator(KeyCode.INS).iconId("levelup"); + +const Action[] CUP_ACTIONS = [ACTION_PAUSE, ACTION_ROTATE, ACTION_LEVEL_UP, + ACTION_MOVE_LEFT, ACTION_FAST_DOWN, ACTION_MOVE_RIGHT]; + +/// about dialog +Widget createAboutWidget() +{ + LinearLayout res = new VerticalLayout(); + res.padding(Rect(10,10,10,10)); + res.addChild(new TextWidget(null, "DLangUI Tetris demo app"d)); + res.addChild(new TextWidget(null, "(C) Vadim Lopatin, 2014"d)); + res.addChild(new TextWidget(null, "http://github.com/buggins/dlangui"d)); + Button closeButton = new Button("close", "Close"d); + closeButton.onClickListener = delegate(Widget src) { + Log.i("Closing window"); + res.window.close(); + return true; + }; + res.addChild(closeButton); + return res; +} + +/// Cup States +enum CupState : int { + /// New figure appears + NewFigure, + /// Game is paused + Paused, + /// Figure is falling + FallingFigure, + /// Figure is hanging - pause between falling by one row + HangingFigure, + /// destroying complete rows + DestroyingRows, + /// falling after some rows were destroyed + FallingRows, + /// Game is over + GameOver, +} + +/// Cup widget +class CupWidget : Widget { + /// cup columns count + int _cols; + /// cup rows count + int _rows; + /// cup data + Cup _cup; + + + /// Level 1..10 + int _level; + /// Score + int _score; + /// Single cell movement duration for current level, in 1/10000000 of seconds + long _movementDuration; + /// When true, figure is falling down fast + bool _fastDownFlag; + /// animation helper for fade and movement in different states + AnimationHelper _animation; + /// GameOver popup + private PopupWidget _gameOverPopup; + /// Status widget + private StatusWidget _status; + /// Current state + protected CupState _state; + + protected int _totalRowsDestroyed; + + static const int[10] LEVEL_SPEED = [15000000, 10000000, 7000000, 6000000, 5000000, 4000000, 300000, 2000000, 1500000, 1000000]; + + static const int RESERVED_ROWS = 5; // reserved for next figure + + /// set difficulty level 1..10 + void setLevel(int level) { + if (level > 10) + return; + _level = level; + _movementDuration = LEVEL_SPEED[level - 1]; + _status.setLevel(_level); + } + + static const int MIN_FAST_FALLING_INTERVAL = 600000; + + static const int ROWS_FALLING_INTERVAL = 1200000; + + /// change game state, init state animation when necessary + void setCupState(CupState state) { + int animationIntervalPercent = 100; + switch (state) { + case CupState.FallingFigure: + animationIntervalPercent = _fastDownFlag ? 10 : 25; + break; + case CupState.HangingFigure: + animationIntervalPercent = 75; + break; + case CupState.NewFigure: + animationIntervalPercent = 100; + break; + case CupState.FallingRows: + animationIntervalPercent = 25; + break; + case CupState.DestroyingRows: + animationIntervalPercent = 50; + break; + default: + // no animation for other states + animationIntervalPercent = 0; + break; + } + _state = state; + if (animationIntervalPercent) { + long interval = _movementDuration * animationIntervalPercent / 100; + if (_fastDownFlag && falling && interval > MIN_FAST_FALLING_INTERVAL) + interval = MIN_FAST_FALLING_INTERVAL; + if (_state == CupState.FallingRows) + interval = ROWS_FALLING_INTERVAL; + _animation.start(interval, 255); + } + invalidate(); + } + + void addScore(int score) { + _score += score; + _status.setScore(_score); + } + + /// returns true if figure is in falling - movement state + @property bool falling() { + return _state == CupState.FallingFigure; + } + + /// Turn on / off fast falling down + bool handleFastDown(bool fast) { + if (fast == true) { + if (_fastDownFlag) + return false; + // handle turn on fast down + if (falling) { + _fastDownFlag = true; + // if already falling, just increase speed + _animation.interval = _movementDuration * 10 / 100; + if (_animation.interval > MIN_FAST_FALLING_INTERVAL) + _animation.interval = MIN_FAST_FALLING_INTERVAL; + return true; + } else if (_state == CupState.HangingFigure) { + _fastDownFlag = true; + setCupState(CupState.FallingFigure); + return true; + } else { + return false; + } + } + _fastDownFlag = fast; + return true; + } + + static const int[] NEXT_LEVEL_SCORE = [0, 20, 50, 100, 200, 350, 500, 750, 1000, 1500, 2000]; + + /// try start next figure + protected void nextFigure() { + if (!_cup.dropNextFigure()) { + // Game Over + setCupState(CupState.GameOver); + Widget popupWidget = new TextWidget("popup", "Game Over!"d); + popupWidget.padding(Rect(30, 30, 30, 30)).backgroundImageId("popup_background").alpha(0x40).fontWeight(800).fontSize(30); + _gameOverPopup = window.showPopup(popupWidget, this); + } else { + setCupState(CupState.NewFigure); + if (_level < 10 && _totalRowsDestroyed >= NEXT_LEVEL_SCORE[_level]) + setLevel(_level + 1); // level up + } + } + + protected void destroyFullRows() { + setCupState(CupState.DestroyingRows); + } + + protected void onAnimationFinished() { + switch (_state) { + case CupState.NewFigure: + _fastDownFlag = false; + _cup.genNextFigure(); + setCupState(CupState.HangingFigure); + break; + case CupState.FallingFigure: + if (_cup.isPositionFreeBelow()) { + _cup.move(0, -1, false); + if (_fastDownFlag) + setCupState(CupState.FallingFigure); + else + setCupState(CupState.HangingFigure); + } else { + // At bottom of cup + _cup.putFigure(); + _fastDownFlag = false; + if (_cup.hasFullRows) { + destroyFullRows(); + } else { + nextFigure(); + } + } + break; + case CupState.HangingFigure: + setCupState(CupState.FallingFigure); + break; + case CupState.DestroyingRows: + int rowsDestroyed = _cup.destroyFullRows(); + _totalRowsDestroyed += rowsDestroyed; + _status.setRowsDestroyed(_totalRowsDestroyed); + int scorePerRow = 0; + for (int i = 0; i < rowsDestroyed; i++) { + scorePerRow += 10; + addScore(scorePerRow); + } + if (_cup.markFallingCells()) { + setCupState(CupState.FallingRows); + } else { + nextFigure(); + } + break; + case CupState.FallingRows: + if (_cup.moveFallingCells()) { + // more cells to fall + setCupState(CupState.FallingRows); + } else { + // no more cells to fall, next figure + if (_cup.hasFullRows) { + // new full rows were constructed: destroy + destroyFullRows(); + } else { + // next figure + nextFigure(); + } + } + break; + default: + break; + } + } + + /// start new game + void newGame() { + setLevel(1); + init(_cols, _rows); + _cup.dropNextFigure(); + setCupState(CupState.NewFigure); + if (window && _gameOverPopup) { + window.removePopup(_gameOverPopup); + _gameOverPopup = null; + } + _score = 0; + _status.setScore(0); + _totalRowsDestroyed = 0; + _status.setRowsDestroyed(0); + } + + /// init cup + void init(int cols, int rows) { + _cup.init(cols, rows); + _cols = cols; + _rows = rows; + } + + protected Rect cellRect(Rect rc, int col, int row) { + int dx = rc.width / _cols; + int dy = rc.height / (_rows + RESERVED_ROWS); + int dd = dx; + if (dd > dy) + dd = dy; + int x0 = rc.left + (rc.width - dd * _cols) / 2 + dd * col; + int y0 = rc.bottom - (rc.height - dd * (_rows + RESERVED_ROWS)) / 2 - dd * row - dd; + return Rect(x0, y0, x0 + dd, y0 + dd); + } + + /// Handle keys + override bool onKeyEvent(KeyEvent event) { + if (event.action == KeyAction.KeyDown && _state == CupState.GameOver) { + // restart game + newGame(); + return true; + } + if (event.action == KeyAction.KeyDown && _state == CupState.NewFigure) { + // stop new figure fade in if key is pressed + onAnimationFinished(); + } + if (event.keyCode == KeyCode.DOWN) { + if (event.action == KeyAction.KeyDown) { + handleFastDown(true); + } else if (event.action == KeyAction.KeyUp) { + handleFastDown(false); + } + return true; + } + if ((event.action == KeyAction.KeyDown || event.action == KeyAction.KeyUp) && event.keyCode != KeyCode.SPACE) + handleFastDown(false); // don't stop fast down on Space key KeyUp + return super.onKeyEvent(event); + } + + /// draw cup cell + protected void drawCell(DrawBuf buf, Rect cellRc, uint color, int offset = 0) { + cellRc.top += offset; + cellRc.bottom += offset; + + cellRc.right--; + cellRc.bottom--; + + int w = cellRc.width / 6; + buf.drawFrame(cellRc, color, Rect(w,w,w,w)); + cellRc.shrink(w, w); + color = addAlpha(color, 0xC0); + buf.fillRect(cellRc, color); + } + + /// draw figure + protected void drawFigure(DrawBuf buf, Rect rc, FigurePosition figure, int dy, uint alpha = 0) { + uint color = addAlpha(_figureColors[figure.index - 1], alpha); + FigureShape shape = figure.shape; + foreach(cell; shape.cells) { + Rect cellRc = cellRect(rc, figure.x + cell.dx, figure.y + cell.dy); + cellRc.top += dy; + cellRc.bottom += dy; + drawCell(buf, cellRc, color); + } + } + + //================================================================================================= + // Overrides of Widget methods + + /// returns true is widget is being animated - need to call animate() and redraw + override @property bool animating() { + switch (_state) { + case CupState.NewFigure: + case CupState.FallingFigure: + case CupState.HangingFigure: + case CupState.DestroyingRows: + case CupState.FallingRows: + return true; + default: + return false; + } + } + + /// animates window; interval is time left from previous draw, in hnsecs (1/10000000 of second) + override void animate(long interval) { + _animation.animate(interval); + if (_animation.finished) { + onAnimationFinished(); + } + } + + /// Draw widget at its position to buffer + override void onDraw(DrawBuf buf) { + super.onDraw(buf); + Rect rc = _pos; + applyMargins(rc); + auto saver = ClipRectSaver(buf, rc, alpha); + applyPadding(rc); + + Rect topLeft = cellRect(rc, 0, _rows - 1); + Rect bottomRight = cellRect(rc, _cols - 1, 0); + Rect cupRc = Rect(topLeft.left, topLeft.top, bottomRight.right, bottomRight.bottom); + + int fw = 7; + int dw = 0; + uint fcl = 0xA0606090; + buf.fillRect(cupRc, 0xC0A0C0B0); + buf.fillRect(Rect(cupRc.left - dw - fw, cupRc.top, cupRc.left - dw, cupRc.bottom + dw), fcl); + buf.fillRect(Rect(cupRc.right + dw, cupRc.top, cupRc.right + dw + fw, cupRc.bottom + dw), fcl); + buf.fillRect(Rect(cupRc.left - dw - fw, cupRc.bottom + dw, cupRc.right + dw + fw, cupRc.bottom + dw + fw), fcl); + + int fallingCellOffset = 0; + if (_state == CupState.FallingRows) { + fallingCellOffset = _animation.getProgress(topLeft.height); + } + + for (int row = 0; row < _rows; row++) { + uint cellAlpha = 0; + if (_state == CupState.DestroyingRows && _cup.isRowFull(row)) + cellAlpha = _animation.progress; + for (int col = 0; col < _cols; col++) { + + int value = _cup[col, row]; + Rect cellRc = cellRect(rc, col, row); + + Point middle = cellRc.middle; + buf.fillRect(Rect(middle.x - 1, middle.y - 1, middle.x + 1, middle.y + 1), 0x80404040); + + if (value != EMPTY) { + uint cl = addAlpha(_figureColors[value - 1], cellAlpha); + int offset = fallingCellOffset > 0 && _cup.isCellFalling(col, row) ? fallingCellOffset : 0; + drawCell(buf, cellRc, cl, offset); + } + } + } + + // draw current figure falling + if (_state == CupState.FallingFigure || _state == CupState.HangingFigure) { + int dy = 0; + if (falling && _cup.isPositionFreeBelow()) + dy = _animation.getProgress(topLeft.height); + drawFigure(buf, rc, _cup.currentFigure, dy, 0); + } + + // draw next figure + if (_cup.hasNextFigure) { + //auto shape = _nextFigure.shape; + uint nextFigureAlpha = 0; + if (_state == CupState.NewFigure) { + nextFigureAlpha = _animation.progress; + drawFigure(buf, rc, _cup.currentFigure, 0, 255 - nextFigureAlpha); + } + if (_state != CupState.GameOver) { + drawFigure(buf, rc, _cup.nextFigure, 0, blendAlpha(0xA0, nextFigureAlpha)); + } + } + + } + + /// override to handle specific actions + override bool handleAction(const Action a) { + switch (a.id) { + case TetrisAction.MoveLeft: + _cup.move(-1, 0, falling); + return true; + case TetrisAction.MoveRight: + _cup.move(1, 0, falling); + return true; + case TetrisAction.RotateCCW: + _cup.rotate(1, falling); + return true; + case TetrisAction.FastDown: + handleFastDown(true); + return true; + case TetrisAction.Pause: + // TODO: implement pause + return true; + case TetrisAction.LevelUp: + setLevel(_level + 1); + return true; + default: + if (parent) // by default, pass to parent widget + return parent.handleAction(a); + return false; + } + } + + /// Measure widget according to desired width and height constraints. (Step 1 of two phase layout). + override void measure(int parentWidth, int parentHeight) { + measuredContent(parentWidth, parentHeight, parentWidth * 3 / 5, parentHeight); + } + + this(StatusWidget status) { + super("CUP"); + this._status = status; + layoutWidth(FILL_PARENT).layoutHeight(FILL_PARENT).layoutWeight(2).setState(State.Default).focusable(true).padding(Rect(20, 20, 20, 20)); + + _cols = 10; + _rows = 18; + newGame(); + + focusable = true; + + acceleratorMap.add(CUP_ACTIONS); + } +} + +/// Panel to show game status +class StatusWidget : VerticalLayout { + private TextWidget _level; + private TextWidget _rowsDestroyed; + private TextWidget _score; + private CupWidget _cup; + void setCup(CupWidget cup) { + _cup = cup; + } + TextWidget createTextWidget(dstring str, uint color) { + TextWidget res = new TextWidget(null, str); + res.layoutWidth(FILL_PARENT).alignment(Align.Center).fontSize(25).textColor(color); + return res; + } + + Widget createControls() { + TableLayout res = new TableLayout(); + res.colCount = 3; + foreach(const Action a; CUP_ACTIONS) { + ImageButton btn = new ImageButton(a); + btn.focusable = false; + res.addChild(btn); + } + res.layoutWidth(WRAP_CONTENT).layoutHeight(WRAP_CONTENT).margins(Rect(10, 10, 10, 10)).alignment(Align.Center); + return res; + } + + this() { + super("CUP_STATUS"); + + addChild(new VSpacer()); + + ImageWidget image = new ImageWidget(null, "tetris_logo_big"); + image.layoutWidth(FILL_PARENT).alignment(Align.Center).clickable(true); + image.onClickListener = delegate(Widget src) { + _cup.handleAction(ACTION_PAUSE); + // about dialog when clicking on image + Window wnd = Platform.instance.createWindow("About...", window, WindowFlag.Modal); + wnd.mainWidget = createAboutWidget(); + wnd.show(); + return true; + }; + addChild(image); + + addChild(new VSpacer()); + addChild(createTextWidget("Level:"d, 0x008000)); + addChild((_level = createTextWidget(""d, 0x008000))); + addChild(new VSpacer()); + addChild(createTextWidget("Rows:"d, 0x202080)); + addChild((_rowsDestroyed = createTextWidget(""d, 0x202080))); + addChild(new VSpacer()); + addChild(createTextWidget("Score:"d, 0x800000)); + addChild((_score = createTextWidget(""d, 0x800000))); + addChild(new VSpacer()); + addChild(createControls()); + addChild(new VSpacer()); + + layoutWidth(FILL_PARENT).layoutHeight(FILL_PARENT).layoutWeight(2).padding(Rect(10, 10, 10, 10)); + } + + void setLevel(int level) { + _level.text = toUTF32(to!string(level)); + } + + void setScore(int score) { + _score.text = toUTF32(to!string(score)); + } + + void setRowsDestroyed(int rows) { + _rowsDestroyed.text = toUTF32(to!string(rows)); + } + + override bool handleAction(const Action a) { + return _cup.handleAction(a); + } +} + +/// Cup page: cup widget + status widget +class CupPage : HorizontalLayout { + CupWidget _cup; + StatusWidget _status; + this() { + super("CUP_PAGE"); + layoutWidth(FILL_PARENT).layoutHeight(FILL_PARENT); + _status = new StatusWidget(); + _cup = new CupWidget(_status); + _status.setCup(_cup); + addChild(_cup); + addChild(_status); + } + /// Measure widget according to desired width and height constraints. (Step 1 of two phase layout). + override void measure(int parentWidth, int parentHeight) { + super.measure(parentWidth, parentHeight); + /// fixed size + measuredContent(parentWidth, parentHeight, 600, 550); + } +} + +// +class GameWidget : FrameLayout { + + CupPage _cupPage; + this() { + super("GAME"); + _cupPage = new CupPage(); + addChild(_cupPage); + //showChild(_cupPage.id, Visibility.Invisible, true); + backgroundImageId = "tx_fabric.tiled"; + } +} diff --git a/workspaces/tetris/src/model.d b/workspaces/tetris/src/model.d new file mode 100644 index 0000000..b7eee79 --- /dev/null +++ b/workspaces/tetris/src/model.d @@ -0,0 +1,444 @@ +module model; + +import std.random : uniform; + +/// Cell codes +enum : int { + WALL = -1, + EMPTY = 0, + FIGURE1, + FIGURE2, + FIGURE3, + FIGURE4, + FIGURE5, + FIGURE6, + FIGURE7, +} + +/// Orientations +enum : int { + ORIENTATION0, + ORIENTATION90, + ORIENTATION180, + ORIENTATION270 +} + + +/// Cell offset +struct FigureCell { + // horizontal offset + int dx; + // vertical offset + int dy; + this(int[2] v) { + dx = v[0]; + dy = v[1]; + } +} + +/// Single figure shape for some particular orientation - 4 cells +struct FigureShape { + /// by cell index 0..3 + FigureCell[4] cells; + /// lowest y coordinate - to show next figure above cup + int extent; + /// upper y coordinate - initial Y offset to place figure to cup + int y0; + /// Init cells (cell 0 is [0,0]) + this(int[2] c2, int[2] c3, int[2] c4) { + cells[0] = FigureCell([0, 0]); + cells[1] = FigureCell(c2); + cells[2] = FigureCell(c3); + cells[3] = FigureCell(c4); + extent = y0 = 0; + foreach (cell; cells) { + if (extent > cell.dy) + extent = cell.dy; + if (y0 < cell.dy) + y0 = cell.dy; + } + } +} + +/// Figure data - shapes for 4 orientations +struct Figure { + FigureShape[4] shapes; // by orientation + this(FigureShape[4] v) { + shapes = v; + } +} + +/// All shapes +const Figure[7] FIGURES = [ + // FIGURE1 =========================================== + // ## #### + // 00## 00## + // ## + Figure([FigureShape([1, 0], [ 1, 1], [0,-1]), + FigureShape([0, 1], [-1, 1], [1, 0]), + FigureShape([1, 0], [ 1, 1], [0,-1]), + FigureShape([0, 1], [-1, 1], [1, 0])]), + // FIGURE2 =========================================== + // ## #### + // 00## ##00 + // ## + Figure([FigureShape([1, 0], [0, 1], [ 1,-1]), + FigureShape([0, 1], [1, 1], [-1, 0]), + FigureShape([1, 0], [0, 1], [ 1,-1]), + FigureShape([0, 1], [1, 1], [-1, 0])]), + // FIGURE3 =========================================== + // ## ## #### + // ##00## 00 ##00## 00 + // ## #### ## + Figure([FigureShape([1, 0], [-1, 0], [-1,-1]), + FigureShape([0, 1], [ 0,-1], [ 1,-1]), + FigureShape([1, 0], [-1, 0], [ 1, 1]), + FigureShape([0, 1], [-1, 1], [ 0,-1])]), + // FIGURE4 =========================================== + // #### ## ## + // ##00## 00 ##00## 00 + // ## ## #### + Figure([FigureShape([1, 0], [-1, 0], [ 1,-1]), + FigureShape([0, 1], [ 0,-1], [ 1, 1]), + FigureShape([1, 0], [-1, 0], [-1, 1]), + FigureShape([0, 1], [-1,-1], [ 0,-1])]), + // FIGURE5 =========================================== + // #### + // 00## + // + Figure([FigureShape([1, 0], [0, 1], [ 1, 1]), + FigureShape([1, 0], [0, 1], [ 1, 1]), + FigureShape([1, 0], [0, 1], [ 1, 1]), + FigureShape([1, 0], [0, 1], [ 1, 1])]), + // FIGURE6 =========================================== + // ## + // ## + // 00 ##00#### + // ## + Figure([FigureShape([0, 1], [0, 2], [ 0,-1]), + FigureShape([1, 0], [2, 0], [-1, 0]), + FigureShape([0, 1], [0, 2], [ 0,-1]), + FigureShape([1, 0], [2, 0], [-1, 0])]), + // FIGURE7 =========================================== + // ## ## ## + // ##00## 00## ##00## ##00 + // ## ## ## + Figure([FigureShape([1, 0], [-1,0], [ 0,-1]), + FigureShape([0, 1], [0,-1], [ 1, 0]), + FigureShape([1, 0], [-1,0], [ 0, 1]), + FigureShape([0, 1], [0,-1], [-1, 0])]), +]; + +/// colors for different figure types +const uint[7] _figureColors = [0xC00000, 0x80A000, 0xA00080, 0x0000C0, 0x800020, 0x408000, 0x204000]; + +/// Figure type, orientation and position container +struct FigurePosition { + int index; + int orientation; + int x; + int y; + this(int index, int orientation, int x, int y) { + this.index = index; + this.orientation = orientation; + this.x = x; + this.y = y; + } + /// return rotated position CCW for angle=1, CW for angle=-1 + FigurePosition rotate(int angle) { + int newOrientation = (orientation + 4 + angle) & 3; + return FigurePosition(index, newOrientation, x, y); + } + /// return moved position + FigurePosition move(int dx, int dy = 0) { + return FigurePosition(index, orientation, x + dx, y + dy); + } + /// return shape for figure orientation + @property FigureShape shape() const { + return FIGURES[index - 1].shapes[orientation]; + } + /// return color for figure + @property uint color() const { + return _figureColors[index - 1]; + } + /// return true if figure index is not initialized + @property empty() const { + return index == 0; + } + /// clears content + void reset() { + index = 0; + } +} + +/** +Cup content + +Coordinates are relative to bottom left corner. +*/ +struct Cup { + private int[] _cup; + private int _cols; + private int _rows; + private bool[] _destroyedFullRows; + private int[] _cellGroups; + + private FigurePosition _currentFigure; + /// current figure index, orientation, position + @property ref FigurePosition currentFigure() { return _currentFigure; } + + private FigurePosition _nextFigure; + /// next figure + @property ref FigurePosition nextFigure() { return _nextFigure; } + + /// returns number of columns + @property int cols() { + return _cols; + } + /// returns number of columns + @property int rows() { + return _rows; + } + /// inits empty cup of specified size + void init(int cols, int rows) { + _cols = cols; + _rows = rows; + _cup = new int[_cols * _rows]; + _destroyedFullRows = new bool[_rows]; + _cellGroups = new int[_cols * _rows]; + } + /// returns cell content at specified position + int opIndex(int col, int row) { + if (col < 0 || row < 0 || col >= _cols || row >= _rows) + return WALL; + return _cup[row * _cols + col]; + } + /// set cell value + void opIndexAssign(int value, int col, int row) { + if (col < 0 || row < 0 || col >= _cols || row >= _rows) + return; // ignore modification of cells outside cup + _cup[row * _cols + col] = value; + } + /// put current figure into cup at current position and orientation + void putFigure() { + FigureShape shape = _currentFigure.shape; + foreach(cell; shape.cells) { + this[_currentFigure.x + cell.dx, _currentFigure.y + cell.dy] = _currentFigure.index; + } + } + + /// check if all cells where specified figure is located are free + bool isPositionFree(in FigurePosition pos) { + FigureShape shape = pos.shape; + foreach(cell; shape.cells) { + int value = this[pos.x + cell.dx, pos.y + cell.dy]; + if (value != 0) // occupied + return false; + } + return true; + } + /// returns true if specified row is full + bool isRowFull(int row) { + for (int i = 0; i < _cols; i++) + if (this[i, row] == EMPTY) + return false; + return true; + } + /// returns true if at least one row is full + @property bool hasFullRows() { + for (int i = 0; i < _rows; i++) + if (isRowFull(i)) + return true; + return false; + } + /// destroy all full rows, saving flags for destroyed rows; returns count of destroyed rows, 0 if no rows destroyed + int destroyFullRows() { + int res = 0; + for (int i = 0; i < _rows; i++) { + if (isRowFull(i)) { + _destroyedFullRows[i] = true; + res++; + for (int col = 0; col < _cols; col++) + this[col, i] = EMPTY; + } else { + _destroyedFullRows[i] = false; + } + } + return res; + } + + /// check if all cells where current figire is located are free + bool isPositionFree() { + return isPositionFree(_currentFigure); + } + + /// check if all cells where current figire is located are free + bool isPositionFreeBelow() { + return isPositionFree(_currentFigure.move(0, -1)); + } + + /// try to rotate current figure, returns true if figure rotated + bool rotate(int angle, bool falling) { + FigurePosition newpos = _currentFigure.rotate(angle); + if (isPositionFree(newpos)) { + if (falling) { + // special handling for fall animation + if (!isPositionFree(newpos.move(0, -1))) { + if (isPositionFreeBelow()) + return false; + } + } + _currentFigure = newpos; + return true; + } else if (isPositionFree(newpos.move(0, -1))) { + _currentFigure = newpos.move(0, -1); + return true; + } + return false; + } + + /// try to move current figure, returns true if figure rotated + bool move(int deltaX, int deltaY, bool falling) { + FigurePosition newpos = _currentFigure.move(deltaX, deltaY); + if (isPositionFree(newpos)) { + if (falling && !isPositionFree(newpos.move(0, -1))) { + if (isPositionFreeBelow()) + return false; + } + _currentFigure = newpos; + return true; + } + return false; + } + + /// random next figure + void genNextFigure() { + _nextFigure.index = uniform(FIGURE1, FIGURE7 + 1); + _nextFigure.orientation = ORIENTATION0; + _nextFigure.x = _cols / 2; + _nextFigure.y = _rows - _nextFigure.shape.extent + 1; + } + + /// New figure: put it on top of cup + bool dropNextFigure() { + if (_nextFigure.empty) + genNextFigure(); + _currentFigure = _nextFigure; + _currentFigure.x = _cols / 2; + _currentFigure.y = _rows - 1 - _currentFigure.shape.y0; + return isPositionFree(); + } + + /// get cell group / falling cell value + private int cellGroup(int col, int row) { + if (col < 0 || row < 0 || col >= _cols || row >= _rows) + return 0; + return _cellGroups[col + row * _cols]; + } + + /// set cell group / falling cells value + private void setCellGroup(int value, int col, int row) { + _cellGroups[col + row * _cols] = value; + } + + /// recursive fill occupied area of cells with group id + private void fillCellGroup(int x, int y, int value) { + if (x < 0 || y < 0 || x >= _cols || y >= _rows) + return; + if (this[x, y] != EMPTY && cellGroup(x, y) == 0) { + setCellGroup(value, x, y); + fillCellGroup(x + 1, y, value); + fillCellGroup(x - 1, y, value); + fillCellGroup(x, y + 1, value); + fillCellGroup(x, y - 1, value); + } + } + + /// 1 == next cell below is occupied, 2 == one empty cell + private int distanceToOccupiedCellBelow(int col, int row) { + for (int y = row - 1; y >= -1; y--) { + if (this[col, y] != EMPTY) + return row - y; + } + return 1; + } + + /// mark cells in _cellGroups[] matrix which can fall down (value > 0 is distance to fall) + bool markFallingCells() { + _cellGroups = new int[_cols * _rows]; + int groupId = 1; + for (int y = 0; y < _rows; y++) { + for (int x = 0; x < _cols; x++) { + if (this[x, y] != EMPTY && cellGroup(x, y) == 0) { + fillCellGroup(x, y, groupId); + groupId++; + } + } + } + // check space below each group - can it fall down? + int[] spaceBelowGroup = new int[groupId]; + for (int y = 0; y < _rows; y++) { + for (int x = 0; x < _cols; x++) { + int group = cellGroup(x, y); + if (group > 0) { + if (y == 0) + spaceBelowGroup[group] = 1; + else if (this[x, y - 1] != EMPTY && cellGroup(x, y - 1) != group) + spaceBelowGroup[group] = 1; + else if (this[x, y - 1] == EMPTY) { + int dist = distanceToOccupiedCellBelow(x, y); + if (spaceBelowGroup[group] == 0 || spaceBelowGroup[group] > dist) + spaceBelowGroup[group] = dist; + } + } + } + } + // replace group IDs with distance to fall (0 == cell cannot fall) + for (int y = 0; y < _rows; y++) { + for (int x = 0; x < _cols; x++) { + int group = cellGroup(x, y); + if (group > 0) { + // distance to fall + setCellGroup(spaceBelowGroup[group] - 1, x, y); + } + } + } + bool canFall = false; + for (int i = 1; i < groupId; i++) + if (spaceBelowGroup[i] > 1) + canFall = true; + return canFall; + } + + /// moves all falling cells one cell down + /// returns true if there are more cells to fall + bool moveFallingCells() { + bool res = false; + for (int y = 0; y < _rows - 1; y++) { + for (int x = 0; x < _cols; x++) { + int dist = cellGroup(x, y + 1); + if (dist > 0) { + // move cell down, decreasing distance + setCellGroup(dist - 1, x, y); + this[x, y] = this[x, y + 1]; + setCellGroup(0, x, y + 1); + this[x, y + 1] = EMPTY; + if (dist > 1) + res = true; + } + } + } + return res; + } + + /// return true if cell is currently falling + bool isCellFalling(int col, int row) { + return cellGroup(col, row) > 0; + } + + /// returns true if next figure is generated + @property bool hasNextFigure() { + return !_nextFigure.empty; + } +} + diff --git a/workspaces/tetris/src/tetris.d b/workspaces/tetris/src/tetris.d new file mode 100644 index 0000000..564fc27 --- /dev/null +++ b/workspaces/tetris/src/tetris.d @@ -0,0 +1,50 @@ +// Written in the D programming language. + +/** +This app is a Tetris demo for DlangUI library. + +Synopsis: + +---- + dub run dlangui:tetris +---- + +Copyright: Vadim Lopatin, 2014 +License: Boost License 1.0 +Authors: Vadim Lopatin, coolreader.org@gmail.com + */ +module main; + +import dlangui.all; +import model; +import gui; + +/// Required for Windows platform: DMD cannot find WinMain if it's in library +mixin APP_ENTRY_POINT; + +/// entry point for dlangui based application +extern (C) int UIAppMain(string[] args) { + + //auto power2 = delegate(int X) { return X * X; }; + auto power2 = (int X) => X * X; + + // embed resources listed in views/resources.list into executable + embeddedResourceList.addResources(embedResourcesFromList!("resources.list")()); + + // select translation file - for english language + Platform.instance.uiLanguage = "en"; + // load theme from file "theme_default.xml" + Platform.instance.uiTheme = "theme_default"; + + // create window + Window window = Platform.instance.createWindow("DLangUI: Tetris game example"d, null, WindowFlag.Modal); + + window.mainWidget = new GameWidget(); + + window.windowIcon = drawableCache.getImage("dtetris-logo1"); + + window.show(); + + // run message loop + return Platform.instance.enterMessageLoop(); +} diff --git a/workspaces/tetris/views/res/arrow-down.png b/workspaces/tetris/views/res/arrow-down.png new file mode 100644 index 0000000000000000000000000000000000000000..88a570db8d178137a90f939d3f6b0ab14d868cfb GIT binary patch literal 843 zcmV-R1GM~!P)4Jt7tj;&i1a!<>gDtvOnPn)8>1$o1T2Uw2(p3(@uw;Uf`Vx)2&hy8@-O~^ z?oQ`54jmIwV%<%e^i967d-Oi@e)G*NSN|GdPAlBEveGCMGbavK9Z8j&J zn6T23k#RaUHo=}P)MBwwr_=q3Xr+~v4O*|)X`|6V;h9y9R;yK7E|=-a$qBXFXQ*ZX z%ggKZ^t5qjSgBMxrp02B777JgDwS9=ngg7Q05Ihcc-J1F*=*8UtrNQl0VoFG&<{XN z0R?FeaCUYEfCni6N>LPj0M5_PvHJmV2Xk|CLseBp=>yPiw`r@@!iu>up7lu}6#7ab z(HzBfeSNJDfGE+t(Wqk#^#wozx3{;L2%HhR0`L}KQ(_`s2OyF7NxEriY3+eIgc0OZ zf#;|8z}%ewd4Oc{``dUtS&v4yX)N}qHa4Qv?+>b?`NhQrE+t_sn{;y0qBT-Fg z$1`_dKI-$$M@TQ)+S>V$PG@O1@z}4fu5cz`)qvn_z&l!j4avb+Rq5hlkS$>0{j&hO zyE|_pk+@Q=*3>QlI18}rkpK=O7{Uzo=sl52rBtuiO9uvI@~pz)EfR@r_cMq#AiC^& zKs=U$cu%01d0KV5y~_N&-zZ5g(vwoD9Er!DhQJYw6Y=u$lBWO&0CZ|^Z;zRK6j}Bg zhlV~AuAU)cvE)5OU4-DruhjMiT}=R4O%91mLYe3cM9a0hP&Q zR7r9pc5eQ1WwXBs8$$#8AZ?dQWibf=pcO~~tw0jYJgZJmON!Sk8wUsf%3MrhpKNYM z48h=PyHF^qVhRA@H)JliuUf5kWoE`}bh!q}t12GFKs~(gAB=N-5<-8h?FqoVtKw1?T%~cGGNLO3R0TKYUhuOJkf!OGopa=Ur2&9} zwQcvq=XuHAdqPAC0kkQ{14D$zt(F;><}n@}oy?f#*wOQ+cyjV%6_S)nsnBklO1<9lU3U=pzK^f3uSgKj z&(HYy_=uba9TSNBE&UEQ9*=QW0S<=)bX^A>$41+>vDxgzC)a9eP%a-T`}+sJZM#8u zdk`Q&yuH2Q%gYNb5J`hfbwTGz)9Ez4kH)*ZI~)uMz@^1Q7$NFWfNZt^g@Rbow4#4| z`!ige0eJL0FA;3_|mn%nm}5 zblrfxy}TmR@UO3L=4?l@xK1XMF|{vkiS&Eh?(XK20Z6r42XeWRqN;h1J0eb-o7sRK zCa;^ZrjZNM#far|u~ZVPFi-7v7fK~v$>&S-9f*a<=Zgp*l&V%?u8xm&5W^^2kFZuV&SBA#2>_|lIQ&9mwOWNwed=HJ8)*&zFi^AIM*si- M07*qoM6N<$f<3>4BLDyZ literal 0 HcmV?d00001 diff --git a/workspaces/tetris/views/res/arrow-right.png b/workspaces/tetris/views/res/arrow-right.png new file mode 100644 index 0000000000000000000000000000000000000000..accf5ff6b01f13ecea78f661deec61c990fa990c GIT binary patch literal 810 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 ij+kdg00002b3#c}2nbc| zMg#x=010qNS#tmY19kua19ky@)q>0b000?uMObuGZ)S9NVRB^vcXxL#X>MzCV_|S* zE^l&Yo9;Xs000B{Nkl^Bw_qq@)F2oWehM>5RxKr%Hm6}!%+eJ5;P+bTi3lS*+iMTL0p`m}$nM`IT^X~QY zaG4NCay|}k4)2`reDBVC=eR(>eVF$r-4qj)1*i$D;c~M0PMbdKBbY0P8D8 zNRZe0(#=lS4j`Bv4AY$T#V)(F>5$@qgbX?I43H&hGVkVQTK`M%-dOl9R8v}PlTfBY zi6Kx_XPp{e(V)oOvODwsMjyfSW8sRgbZCm;7>{s>LF)-A9bt!GSz(YOIg)jEYVNmv z1n0)Wim$Y|!zjl&!d>KX1_-1=S4uXx!c9g*BZ^}mZ1f1;9t>A~RnmD}-Nz`9B_lW? z0AARoLt71gX2l*QS$4-iYC!^!PWx(`EtOXpXIQuMLe5;j%?k_(ck>KSD6@^PrqfZu zTjj9UEjB3e5=Y{bXGp9+X2H-k6+2RKUOr5==^p;LmOwhwEw*SeNk!$kNFa(HGKK-u zRZ!`3w1jRkodNeoK5Xumws?wX8B=*yiC?HsOV_htc8-SPJfF~|t$W)nvB)6?>h9je zpGZp`B~LIKJ2`?1ZgB&mdA7LD28rp&2~X$v@KS(GB*rNxdAP4PAj zxg#Mfs#4O^iJuSw4-k|P?C3C|O?k9NW=@#Snlh&M2Y27mKA zEr?d9!3_>$)>9G;NvH)J4GS>GWR~PWe;o5}V-&T-|4X2-ix-T>&Ur*`wm-HEFKilt zuO(p>asSD#JyU84ISQ2I(R#6!$GlZj6jqaPDd^{#{VCl@kf+FqJX$X{f|z$rRf;Yp zVTpQ%9ad=4p(}V53{#Rv>&23686AG2DFW(Y$pv2iG`yw>$eTRG5QX?gI{s@akKaO# zckR=1pMCXu0%0~Z#l`}6XweZQ5l`4(Y|?dCIiC~I12n@dB=B{O`HcAf;wlaCw*?)k z!5dP8Cbjkdfj`KLfO%Dt1z?WJOtrxGI3}@zeM2oyMbus8-%>#3%c|NNsxNA&2Aj$E z^C}f`g1y`TNQISea-KDt%T7aeVQ;7!g~ocQ=#cPC@ifIam82kQ`jKyNcEl<_wXkoj zo4qiG7K=BbLmcH%?&C0yTk3pGXxP9L6QWZHY-kGiAc z{7Q*?!3S*fYvdn(Ik0uVe&i9$)QSF0LB)kP-tSS8J1_2WE$Gy>GW&!1mnYcdF%~fe z+%(wV#&AH=!R()alfxQSF4luPToe^0rfHb4q=ipvRezS{xnZJKH}B`+w;fY9sjg%w+4c1M)Z02P$$Bb`3$E{etUWjGFrT5s-78uQ zQ`I{hHJ0CJ-F5h!P2!agN9UfKu<~GKCQlOIrfVG!Ya;3xJZIV--Fv9H9~c4*p00i_ I>zopr0L#9LQ2+n{ literal 0 HcmV?d00001 diff --git a/workspaces/tetris/views/res/rotate.png b/workspaces/tetris/views/res/rotate.png new file mode 100644 index 0000000000000000000000000000000000000000..57fad56f3c30e69862a449381101ed547eabaf38 GIT binary patch literal 1934 zcmV;92XXj`P)0b000McNliru-V7QH2N&BsXJG&U2OCL5 zK~z}7#aC-+l-U`6-t%2Emt1BNtEo<$vSo<2wX#2iE&Y){1%EDN%es`-AN^5McWV>8 z)OtycY``nnB1PzrSVYiRsuW9wDvG#Lp{*@m)3`Am#!Ql#FO!*XGT(R3cTRuo%#gTA z%yt*{fx~y;eCK_i_jb;Epp?Sr8HWxXqR+FRlu}p;`uqDe0N)LV!|yaVH@_8&#oAk1 zTH@hwm}X~Z3y$MVIF9qT>FMbUob$i-_xFpH$R(u|Zrr%>&1^PX+_r7o*dmHUMD#Nv z>RYvHRou31*tU&GB!X}_3|-eDgg_7k@H`KW((I>iD0FnstYE9VHkMy=n=Bn z?CEeg+}q#ZAH4b>t4%!5Ln4ttb93`9#@M6w_Vzs+H*Q2I6oQoUGbfjn5}xOwv$GS4 zMB;-?CUb53_U+n|1f-PUoFkb`V$-Hgsnx4jgLA&D62=&GU0?Pe&-2jH(Sc+#`90tF zfA=Z@y*3GgpuS2|T3cHYi9{f!T(HmgeWX$; zq|@mib$56Fdw6*G#DW|c7#JvYc6K&ODM3Vw#?)~fjE;`N_x*Z4)U2G3_V#uplgUM8 zR4NsmKYzX$jYbofFJGSdlm}`;Ye`g#<%<_DFf}z*U!R(qg5x+Sl}csb_aUVWMIsR- zlSyPU8B9-4W5b3GXlQ6yK)^Ijq*AHqg9i_O3*g;)4k)D%1VO!?=g-H+#*oY9z!-z; zx)>Q5sT7OFlZIiWZr{EgzIX3lxV5$Q+d&Zg`u_d<-uU=9JkP_!hY!IyhY(`H*t~f& z3WdVkU0q%Ff+Q3|KuTFB5Cj48`8@LZJctNE5WsPqLQ6}_n^&)1?Yna2%Cp+uYuBzl zxOMB+2dPx*tFG%l$mMc^b1p|mN0pRP&X>s;Lo%5(iRcG)0-W;&qh>u3)!nG;I%2U{ zlVw>yU#j*mUAmNOZ*TvOl=6iTqTo2re~Bn_*cN zOw;`3sZ*!Ez7&Z_Byv~R^$7skxw*Ls-}f_|^9<)a%Q>IYG_B-$o~l;?=lng-^Zvj& z@2KTMwU!kTDNWO$X&M=Z!E{|$ZEbA@DdidfBa0Cj92_k6^z>u_G&9EL%H^^!48sTD z0|_kpxh2UDpTT1F!*b z*L5w!Fw}yBsD|$DZeonVFbrC=W)0c4P5FGD{pIg(vK~OL&;{!Z!zf89=c-C;1S|mD z_kGha3~#|XP{Z)>u)28hqKe1kDjJO{(=^fA+Dhee+4{||+Fz@A&z?O7fHzFjoMnua ziD(8u0e}M_4ZxY1nVAWNLjPHmKn+54u z%1+ZXD?*485zP`&nTTcqysQ!_004IE*g=M2n3iR=1wjDLxyu;iQc9!hdIi8xN*SDUqf{zsiA3UY z{We?-y}i9u1*$|;HLEF|PQ$is=KDUgEQ>WZHfl;KJsOSLl}e>aDHT#m8A>TbN*O4n zOr?}1rKEH^txrx)PTaY3=O4>@fQWG7#0j~3_ij3R^k|*szJ2@9(a`}SlDe*gi0W4~ z=bRIfM?|PvA27yrDJ54*aVe#q&1NYUi~VasKkzE{?%hj52pm6te91=ez<~o)%^?E7 zlv1o}X+zaosg%;Bl$uh?0AN0U{#>l2^qh~^`o&*SXbv+8wCXSL3mQc6~> z763Q^PDC02MnoDBiDI!RJkN9ID_=GTR`^(8QRQEV2VTX%zyOs>CE}c`p5!U@#{25^Z^JJW3|}(-2ZFiQ UU?h#cxBvhE07*qoM6N<$g86EwJOBUy literal 0 HcmV?d00001 diff --git a/workspaces/tetris/views/res/tetris_logo_big.png b/workspaces/tetris/views/res/tetris_logo_big.png new file mode 100644 index 0000000000000000000000000000000000000000..54175c091df245d7e216b36a33725181230d2860 GIT binary patch literal 3509 zcmZu!XHXN|5>1FC5JZZARKcK>(5pb`1nDFK5<}5|geX0L^eQS~!vNBxs&oiS?*tK% z8kAl`Q|V19g23bZX5NoC^M33-bI#1YyR&!Z>})LB+<+O%3k3iG%tnSti&Km^4IBgb z^c)HbGCBpYqlp0$5C>obur=k}k3Cfwy$rEF007hZe*y?dOXofnA-+at`VbnBgF%GH z378yqDhb@Py6LOu>F$pA@I9pffSxzr!58l&?WEV}#VP3K(BY z5B9Yhs-uL{Qae*G>&iz*Cq_ry(5}6f&H1vp5^U!f7&Btf(1>A;$8F3FKaCAo8RBbg zPU7q=D6L(DaTKoEJesqB?!#G#WAN(fo3$Jq98a!3@lp-cPEk!r4LwQebV}*Dobd7? z<0{RjVz@{3aJ6EursXJ|wbE}VDKgFh$ zcD^61&XrhJO(sfIxyz{)Zj1o5_OQd^S@dx=+VT&7FC_)p@YNvaR=jKSb;zr3 z-sKgYOCxcO4sH8gRU`6XbPlV5|C%F}NoI2^&rFVKzM`b?w5YP6OuTf)9Nnkyj&~g{ zDU6WuKhl&IkcuUM+71HtbyZ66OD01+9AKAcfli!!T`fxNIeB53%Q7Op~+E(O-VHU>JV&$mtEAgeoN0rI(xAC$`EAVVJCQF zs5DD1s%d4ooy;SThmQ1Q(S8rmt$%GBY?IQ4eh=s&VC&%gVP3Xap1@@L&;1{D0iUJq z42PD7%=CYPSX@c|(rcYem%H_StuM}~&isV)J7WS>;T%s)57~sd!v@UE@ZiPSdTk8w zv@PPB1`+2EcenOxrk`XqxOH(zePKPORDLaU^YpW7lwGV6Q)t#e!b+FQ;7lvqxjKA)MN{N0c=>VdP~D zUr~d%t)HWlfq2kys|R|o;*JPe_szQXgZN;mN_$f> zYMe-d%!O+$?Y!V*%lfnyzbux~Pw*EV8Tk&K5s6=_0D#R?SP{gMqN2C=FEZ+3wF_s% zE7nTTvZE}yKSXZEu8bEUV>w3P-)|Uecn?<@Ki4V-!B`k-`|870>_+dVC7MMeXprNnTHS6jwbd;Hdx z(T}RCL+jYb<|ErNflXz+9(xn;cyG;aXypymSZ9K3($L^wCb1zrHJdxOJArRL{9&Ta zU!40|U-SzB*ZsJIgQz8C#0TnpS9e`yHelu!&d7Nj&?S%jg{8nccB0<*~4V++)Ng#F5U!Nzgo}jMrm$S6KZa9U2y%Xs& zqxh1hXcVDI@y1nhg9qSn$CER3R;4zaw|iUiC&VlZMcKrk*%}`!OKvMXXgwmAl$;l^ zDGJ+7r4g{D^ce(kq_qb*>)>FQ{4h_sRJX{w)X_zs9-zm>lxlswwYpsX> zxr4(HS67-nDJ%SK%(EMtDGuC!{xv|+tTH^V*!*t)r&tp6r38`o7e}ET=XMo0SPEci zBIXGbbK!%n{>fv0?p8bvpDw_2n>k3AH9<`Z8a{*iZK6A(g#hTvI#8IT%i1luku`z$IvRVSir01Pc$YIq;LC{~KpXM)9pUn>ea`Q>j z?q-jJP23NYJWyw<2*t$nf=$2KCj^5STHFmpWoYz%$>V+n0#P4WYL%0mA~3{s>E4(} zv--m9@098u-t%bNS8Tas^g%+RaT@ctI&|eARUx%LL{-?crAbxRuD-H8B0oOI!o?G7 zW~O7i=jL9*PHp#OTS*i`D=B+B9r`I#GrEW4$r$_*oh>zchY~!y z;Yx{V`AG7azU;x*ux-Un^K{*Jd@<9YMg~)@T09e^#;p|*qSEHsoXVGVx6ffMWPjF4 zUj2;eyQfPf1kgN|=`W$sJzR(7?ZcujdayHsIPn;xP z>;DdmzZlMyGTo=`&F1w^Y9)|N1=l04hpn7`0^&FBa{RMEq#fB{+t$`Sro_~ea%o0* zhARutNp}^XV3^se&QZmsbUFciukK`ezY$Cv+bw6xrAlk8_L@@r`t&~u^0=NA zV#?(`qiMZq{6&^a>ciS?SEGCXJOBU7(t1Rz1u665OM^daL6Jil*8YUyA)Be*knWx? zUKInRbF3Hh7j*UoT&78rfP3(^vN93lXKlSPMuOPzlyT@GVN?$>{ic`@pi7Izc87ZQ-B(O}SyOBnHRXHb z=f8n6Gc-dFFddOU-wMk0{t;$wQ@X^TWJKX7{qR4RCGhLZ)k?Z|LWqXDWDAa5h6^X6 ztOX5R%KHQMU48_~6%|(&03_nVELnb_=U42Pma1nEt8jrcB4!1W+@*?+?^WxZ)TT9G39gG@jjzd1W6534b-E73txRU&YwB8L!RPkQ=Pb6Aq=0YBJ z8xohl__^A+VM-p}{$vKUMrYG=-0vfV5fu#Csh_vfd@b(I>A=f9pq@EbWau5^4%?2U z395lFCErWa({YX|=!oAFtvqOy(5^qOKqNgK8*BfS7*8+W^PO0OkfjmPQ(tMHyi50P zf7aLhi*f3RLpa<_urE?p>yx=XEG=YgJ=}CB(Pgb2RaB5O+b$#ha=n?KVX%F=wum#L zuI@3Z>QWVrrOY?ItC#2e4~ZQyfdJQ9i)A0Qek5giVm@m0sU}#~jB1QCeKK&!-_dkA zVSGj9l@>6}F^$O}=Aoyxu}DmT+}9~eQ=}cGjH8Maer^aNRHoLKa}`H0LuY2!b_pWvKnpyLleP;H0o|%1S=A2BN%ro{Fg!^~_ z00tukFaiKT2b=}b0iaW4d|LVbg{@Ap1c>$@ewtIvMf2a-=RWu!e#cXP*=a?e)<5I_ z-O>XmMYQ2sw7{uxP=|IdgIPBHDjaKb76WAp&+zj&O+ z{SVi@Q~YnP$EWyz&@}(TO$?`x9-KoC)`g=N+zW}O3%p5%FZb)DlU0iTK4RDO>JF$Lt|6( ztM-o0uI`>*O5f<%_=k_5CMG}6FDz1*mRG*6e%spK+5Ne<|M$V+A6=l+wf?vLb1D8O zU7V-7Xux1vFvA~RAeyj0!a2co!V2_IJxc~xESHF43?u9Uv7q`L6H>|Q2e%t;gqa7W zJTJQSN7_G>{jUj&{XbInA7THk>oWkM1)VM)EhnG@9C4>Wmh8^Ob;nV&*c9Kig)A!b znx?UmlnQA;$uhTR-s(^6@0U2$z#csbQ}d*smZJxK4ZYXRczA34t@n1c9eEkU`(WN4 zQr32!VJ!dq!GV!~T>7XQkB-Wf3Ch5AYGWCHcg>$4c;@a6d<{dCC*=49mP+v){?0ES-X7&M`q0-L{-2i!o^n?*+!x#Gl zVDeoEc4`g1!%G`sj`X*P?%-Uky%rRC>o}Gyrbv@jm7r8me7q{{Iy$i?rKlV?HebhTj<>e#H`xp zXh7gxoI;)>Z}tTytDqp5)O5|_lsZGDV$`oOlm!bq(XEXR)%0Z2XpnGUC^u5TTjFN! z!h*gC1r^N%Ou|>n=1AT2e|M#Y$26nu9pc*VCN5$l zr%qMBo*GwF?fEX1BR`elZK-i_!IccJSdm;W{SXxbPJL$l1RzK#wvPd1qrSBrBQ-Dw z&K$Ls-mP)m1WxiV`DNXj=#_?}V@BEzMgBxYo&YYAp61$rxBcjy#F9n=~STSX!bk_oKKcsAXD_K3Y!Ar6<5p-pDL_&FVp#!2A6(P$h-i($U60G}%k_ zW_JJ#uY0`p?OzYYaR|bB)w?>5A!^$*XMGyc6JiE3DkLMJM&ze8Q^g48ry;NS^t8-O zKxt_|zxT|?=(oFEY8St3AOmT%(tt?JC!shgr^$)B$p17`dI_ZqQB4j0UESk%I7T*C zj!&Ytm+4VEb?4@`-fa--WY(DE8d;ZnwYtO8836oUHuSsI-Ti06!GUL`F~Q3CV#V^N zekgiHU?s0~j=rOHDc<&C8V$y+D`HOK=OVAp(Jl1*L{1dRkRlq_i&k4n#q=O~e?F~M z`-P~37fTt)kJzz(D_q8-$i8`f zya){=MNkq24M+UCqddvoHJ4aSa?`#O1jEUze=~&tU_5wOV*e-t>#QE;#^o(@VUDD> ziy)Wgon~oJ>5!FH?IMkH#(X5z{n_DrNb({7P<2#x`yw6GG%Xr*nxj8g@@_TsS?P{m zw9U>E{6dvLxliw{+H|^I5LMYq2UwTCUh|;do_f`I0z6NrHtu8wPW~W?52Quw4F%(g zA8y_K`GdxN<1|SrGG3SMbc;A+^6sG9P5|xk19#OkgDhRh-*eC*SWs|VH#U}Dm!Do^ zY*d)4nIS&lX)&t8z^kLz_Vv7`&FP1~d>x%CsL;Dp{V;2%iJS`w?J)-Lvka3>?(W*T zg=u%CVs|2g)VnS*S3JKkCYf@(ye>Y*_MP~*aE6c1BYX!cMC&ZK>Xe-E0q^=AJ|^cM z9*(D+djoeiUp}X&l~^p=Yp_;}tJ8ga z?pzHHZN(z6a$p0Y$xl|2Jg^h<-rfDC2Uq`|YcbNCdY>uNb`p`{L0zXMv6mf5q!JYf z5ooraeodoH8YHj0L&UJJ_Ml>c(18_g8z=s$t5D<}Zsw^#oO?rE)Nsb#^=$gf1FMvt z3qE6xJ=s?14SQaZY1SCVqEwv+Ijl^kTn4^&YH@_2{DMZ8w|{*(uX7ZVSRt?_^)C9i z5&yG;EMucZN|#N>=*WWZUWQO6pWE^9O}FsxEx#8^720dQfSpkGv;AO)CTbPxdf0Zp zK0fVUz;@k(ztVOVR^7VU_jOB6WQa`>0o0^%NhZ-Bd^j`;9}2d)dZRvJaC8P4s1j=B zzi8hG#NiXDy9LDTev`d81{oWDmQ9Kkh#xy_%p^u~gRRhaA8DH4=DTnDkeUE=#5?Cc{!J z8x^xNzNQ$JA1v zwEooOZZ1fKnj5la>sr@{#Mag7-)ntC?qOJl&%YY2s4ov{7ip{#2&se}rR-JBrG1Ja zCIhu9-=$X1NCrF)8d;c?fv+rUjYhS0Y`^X-n=yVcICq`gv{E6X=+N`baBvTUlB4=J z2uBNsfu9te055Ic%&bWDUdzq2IGEx5ru1nmpJg+Pog@8?N2>B$PbsFJ^BPC9qmN}r z`^qoA6iQX6kBs1!XKn-#Tc!yg@`*;-vXk*|u6=ytc#m94zO!$9WlU;@zfh4_j$T&f z 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 literal 0 HcmV?d00001 diff --git a/workspaces/tetris/views/resources.list b/workspaces/tetris/views/resources.list new file mode 100644 index 0000000..03c36c3 --- /dev/null +++ b/workspaces/tetris/views/resources.list @@ -0,0 +1,10 @@ +res/arrow-down.png +res/arrow-left.png +res/arrow-right.png +res/dtetris-logo1.png +res/levelup.png +res/pause.png +res/popup_background.9.png +res/rotate.png +res/tetris_logo_big.png +res/tx_fabric.jpg