diff --git a/Tutorials/TurnBasedRPG/Project.xml b/Tutorials/TurnBasedRPG/Project.xml index 381421f9f..5a54627d5 100644 --- a/Tutorials/TurnBasedRPG/Project.xml +++ b/Tutorials/TurnBasedRPG/Project.xml @@ -62,6 +62,7 @@ + diff --git a/Tutorials/TurnBasedRPG/assets/data/room-001.json b/Tutorials/TurnBasedRPG/assets/data/room-001.json index b99e25a37..bcd4969e2 100644 --- a/Tutorials/TurnBasedRPG/assets/data/room-001.json +++ b/Tutorials/TurnBasedRPG/assets/data/room-001.json @@ -1,5 +1,5 @@ { - "ogmoVersion": "3.3.0", + "ogmoVersion": "3.4.0", "width": 640, "height": 480, "offsetX": 0, @@ -16,33 +16,35 @@ "gridCellsY": 30, "entities": [ {"name": "player", "id": 0, "_eid": "40117707", "x": 128, "y": 144, "originX": 0, "originY": 0}, - {"name": "coin", "id": 1, "_eid": "40304284", "x": 320, "y": 96, "originX": 0, "originY": 0}, - {"name": "coin", "id": 2, "_eid": "40304284", "x": 352, "y": 80, "originX": 0, "originY": 0}, - {"name": "coin", "id": 3, "_eid": "40304284", "x": 192, "y": 144, "originX": 0, "originY": 0}, - {"name": "coin", "id": 4, "_eid": "40304284", "x": 224, "y": 144, "originX": 0, "originY": 0}, - {"name": "coin", "id": 5, "_eid": "40304284", "x": 256, "y": 144, "originX": 0, "originY": 0}, - {"name": "coin", "id": 6, "_eid": "40304284", "x": 288, "y": 144, "originX": 0, "originY": 0}, - {"name": "coin", "id": 7, "_eid": "40304284", "x": 304, "y": 160, "originX": 0, "originY": 0}, - {"name": "coin", "id": 8, "_eid": "40304284", "x": 304, "y": 192, "originX": 0, "originY": 0}, - {"name": "coin", "id": 9, "_eid": "40304284", "x": 304, "y": 224, "originX": 0, "originY": 0}, - {"name": "coin", "id": 10, "_eid": "40304284", "x": 304, "y": 256, "originX": 0, "originY": 0}, - {"name": "coin", "id": 11, "_eid": "40304284", "x": 304, "y": 304, "originX": 0, "originY": 0}, - {"name": "coin", "id": 12, "_eid": "40304284", "x": 320, "y": 288, "originX": 0, "originY": 0}, - {"name": "coin", "id": 13, "_eid": "40304284", "x": 336, "y": 304, "originX": 0, "originY": 0}, - {"name": "coin", "id": 14, "_eid": "40304284", "x": 288, "y": 288, "originX": 0, "originY": 0}, - {"name": "coin", "id": 15, "_eid": "40304284", "x": 272, "y": 304, "originX": 0, "originY": 0}, - {"name": "coin", "id": 16, "_eid": "40304284", "x": 256, "y": 224, "originX": 0, "originY": 0}, - {"name": "coin", "id": 17, "_eid": "40304284", "x": 256, "y": 192, "originX": 0, "originY": 0}, - {"name": "coin", "id": 18, "_eid": "40304284", "x": 208, "y": 192, "originX": 0, "originY": 0}, - {"name": "coin", "id": 19, "_eid": "40304284", "x": 208, "y": 224, "originX": 0, "originY": 0}, - {"name": "enemy", "id": 20, "_eid": "40444291", "x": 304, "y": 112, "originX": 0, "originY": 0}, - {"name": "enemy", "id": 21, "_eid": "40444291", "x": 272, "y": 208, "originX": 0, "originY": 0}, - {"name": "boss", "id": 22, "_eid": "40444462", "x": 304, "y": 288, "originX": 0, "originY": 0} + {"name": "coin", "id": 1, "_eid": "40304284", "x": 416, "y": 48, "originX": 0, "originY": 0}, + {"name": "coin", "id": 2, "_eid": "40304284", "x": 496, "y": 48, "originX": 0, "originY": 0}, + {"name": "coin", "id": 3, "_eid": "40304284", "x": 192, "y": 136, "originX": 0, "originY": 0}, + {"name": "coin", "id": 4, "_eid": "40304284", "x": 240, "y": 136, "originX": 0, "originY": 0}, + {"name": "coin", "id": 5, "_eid": "40304284", "x": 288, "y": 136, "originX": 0, "originY": 0}, + {"name": "coin", "id": 6, "_eid": "40304284", "x": 336, "y": 136, "originX": 0, "originY": 0}, + {"name": "coin", "id": 7, "_eid": "40304284", "x": 392, "y": 176, "originX": 0, "originY": 0}, + {"name": "coin", "id": 8, "_eid": "40304284", "x": 392, "y": 224, "originX": 0, "originY": 0}, + {"name": "coin", "id": 9, "_eid": "40304284", "x": 392, "y": 272, "originX": 0, "originY": 0}, + {"name": "coin", "id": 10, "_eid": "40304284", "x": 392, "y": 320, "originX": 0, "originY": 0}, + {"name": "coin", "id": 11, "_eid": "40304284", "x": 336, "y": 400, "originX": 0, "originY": 0}, + {"name": "coin", "id": 12, "_eid": "40304284", "x": 368, "y": 368, "originX": 0, "originY": 0}, + {"name": "coin", "id": 13, "_eid": "40304284", "x": 400, "y": 400, "originX": 0, "originY": 0}, + {"name": "coin", "id": 14, "_eid": "40304284", "x": 304, "y": 368, "originX": 0, "originY": 0}, + {"name": "coin", "id": 15, "_eid": "40304284", "x": 272, "y": 400, "originX": 0, "originY": 0}, + {"name": "coin", "id": 16, "_eid": "40304284", "x": 304, "y": 256, "originX": 0, "originY": 0}, + {"name": "coin", "id": 17, "_eid": "40304284", "x": 304, "y": 224, "originX": 0, "originY": 0}, + {"name": "coin", "id": 18, "_eid": "40304284", "x": 240, "y": 224, "originX": 0, "originY": 0}, + {"name": "coin", "id": 19, "_eid": "40304284", "x": 240, "y": 256, "originX": 0, "originY": 0}, + {"name": "enemy", "id": 20, "_eid": "40444291", "x": 464, "y": 64, "originX": 0, "originY": 0}, + {"name": "enemy", "id": 21, "_eid": "40444291", "x": 272, "y": 240, "originX": 0, "originY": 0}, + {"name": "boss", "id": 22, "_eid": "40444462", "x": 336, "y": 384, "originX": 0, "originY": 0}, + {"name": "coin", "id": 23, "_eid": "40304284", "x": 496, "y": 80, "originX": 0, "originY": 0}, + {"name": "coin", "id": 24, "_eid": "40304284", "x": 416, "y": 80, "originX": 0, "originY": 0} ] }, { "name": "walls", - "_eid": "40116503", + "_eid": "02788814", "offsetX": 0, "offsetY": 0, "gridCellWidth": 16, @@ -50,7 +52,7 @@ "gridCellsX": 40, "gridCellsY": 30, "tileset": "tiles", - "data": [-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 2, 2, 2, 2, 2, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 2, 2, 2, 2, 2, 2, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 2, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 2, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0, 2, 2, 2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 1, 1, 1, 1, 1, 1, 1, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 2, 2, 2, 2, 2, 2, 2, 2, 2, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1], + "data": [16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 17, 18, 18, 18, 18, 18, 18, 18, 18, 18, 19, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 16, 16, 16, 16, 16, 21, 28, 28, 22, 28, 22, 22, 28, 22, 22, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 4, 9, 9, 9, 9, 9, 9, 9, 5, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 24, 16, 16, 21, 7, 3, 1, 1, 1, 2, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 17, 18, 18, 18, 18, 18, 18, 18, 19, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 3, 1, 1, 1, 1, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 28, 28, 28, 28, 22, 22, 28, 23, 16, 16, 16, 20, 16, 16, 16, 16, 16, 16, 24, 21, 7, 1, 1, 1, 2, 1, 1, 1, 2, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 4, 5, 9, 9, 9, 9, 9, 34, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 18, 35, 7, 1, 1, 1, 1, 3, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 32, 28, 28, 28, 22, 28, 28, 28, 28, 22, 28, 28, 33, 7, 1, 30, 26, 26, 26, 26, 26, 26, 27, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 9, 9, 5, 9, 9, 9, 9, 9, 9, 9, 5, 9, 9, 8, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 2, 1, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 12, 13, 13, 13, 13, 13, 14, 30, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 31, 10, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 2, 1, 1, 23, 17, 18, 18, 18, 18, 18, 18, 18, 18, 18, 19, 21, 11, 1, 23, 24, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 2, 1, 1, 23, 21, 28, 28, 28, 28, 28, 28, 28, 22, 22, 34, 35, 7, 1, 23, 16, 16, 16, 16, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 3, 1, 1, 1, 1, 1, 23, 21, 4, 9, 9, 9, 9, 9, 9, 9, 9, 32, 33, 7, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 25, 26, 26, 26, 26, 26, 26, 26, 27, 21, 7, 1, 1, 1, 2, 1, 1, 1, 1, 9, 9, 8, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 24, 21, 7, 2, 1, 1, 1, 1, 1, 1, 1, 30, 31, 10, 1, 23, 16, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 2, 1, 23, 21, 12, 14, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 25, 26, 26, 26, 26, 26, 26, 26, 26, 26, 27, 21, 12, 14, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 17, 18, 18, 18, 18, 18, 18, 18, 35, 12, 14, 34, 19, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 24, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 22, 22, 22, 22, 29, 28, 28, 33, 12, 14, 32, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 21, 22, 22, 29, 28, 28, 28, 28, 33, 12, 14, 32, 23, 16, 24, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 4, 9, 5, 9, 9, 9, 9, 9, 8, 1, 9, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 3, 1, 1, 1, 2, 1, 23, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 20, 16, 16, 16, 16, 16, 16, 21, 11, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 21, 7, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 23, 20, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 24, 16, 21, 7, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 23, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 24, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 25, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 27, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16, 16], "exportMode": 0, "arrayMode": 0 } diff --git a/Tutorials/TurnBasedRPG/assets/data/turnBasedRPG.ogmo b/Tutorials/TurnBasedRPG/assets/data/turnBasedRPG.ogmo index ee52ccbe6..b01766817 100644 --- a/Tutorials/TurnBasedRPG/assets/data/turnBasedRPG.ogmo +++ b/Tutorials/TurnBasedRPG/assets/data/turnBasedRPG.ogmo @@ -1,6 +1,6 @@ { "name": "HaxeFlixel Tutorial", - "ogmoVersion": "3.3.0", + "ogmoVersion": "3.4.0", "levelPaths": ["."], "backgroundColor": "#282c34ff", "gridColor": "#3c4049cc", @@ -29,7 +29,7 @@ "definition": "tile", "name": "walls", "gridSize": {"x": 16, "y": 16}, - "exportID": "40116503", + "exportID": "02788814", "exportMode": 0, "arrayMode": 0, "defaultTileset": "tiles" @@ -54,7 +54,7 @@ {"x": 1, "y": 1} ] }, - "color": "#ff0000ff", + "color": "#00e3ffff", "tileX": false, "tileY": false, "tileSize": {"x": 16, "y": 16}, @@ -162,7 +162,7 @@ {"x": 1, "y": 1} ] }, - "color": "#00e3ffff", + "color": "#ff0000ff", "tileX": false, "tileY": false, "tileSize": {"x": 16, "y": 16}, @@ -182,6 +182,6 @@ } ], "tilesets": [ - {"label": "tiles", "path": "../images/tiles.png", "image": "", "tileWidth": 16, "tileHeight": 16, "tileSeparationX": 0, "tileSeparationY": 0} + {"label": "tiles", "path": "../images/tiles.png", "image": "", "tileWidth": 16, "tileHeight": 16, "tileSeparationX": 0, "tileSeparationY": 0, "tileMarginX": 0, "tileMarginY": 0} ] } \ No newline at end of file diff --git a/Tutorials/TurnBasedRPG/assets/images/bar_empty.png b/Tutorials/TurnBasedRPG/assets/images/bar_empty.png new file mode 100644 index 000000000..baec0777c Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/bar_empty.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/bar_filled.png b/Tutorials/TurnBasedRPG/assets/images/bar_filled.png new file mode 100644 index 000000000..20ca1b36b Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/bar_filled.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/boss.png b/Tutorials/TurnBasedRPG/assets/images/boss.png index 5e346d49f..5e529eeee 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/boss.png and b/Tutorials/TurnBasedRPG/assets/images/boss.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/button.png b/Tutorials/TurnBasedRPG/assets/images/button.png index 32f436866..61d6e6de2 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/button.png and b/Tutorials/TurnBasedRPG/assets/images/button.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/coin.png b/Tutorials/TurnBasedRPG/assets/images/coin.png index 2a5820aa8..5ea4008ab 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/coin.png and b/Tutorials/TurnBasedRPG/assets/images/coin.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/enemy.png b/Tutorials/TurnBasedRPG/assets/images/enemy.png index 7672375f2..07544008f 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/enemy.png and b/Tutorials/TurnBasedRPG/assets/images/enemy.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/font.png b/Tutorials/TurnBasedRPG/assets/images/font.png new file mode 100644 index 000000000..84e2038b8 Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/font.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/health.png b/Tutorials/TurnBasedRPG/assets/images/health.png index d39bd4ae8..7a549c929 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/health.png and b/Tutorials/TurnBasedRPG/assets/images/health.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/player.png b/Tutorials/TurnBasedRPG/assets/images/player.png index 6bb1e9add..e786ae632 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/player.png and b/Tutorials/TurnBasedRPG/assets/images/player.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/pointer.png b/Tutorials/TurnBasedRPG/assets/images/pointer.png index 7e0cc1ca4..38d46edf4 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/pointer.png and b/Tutorials/TurnBasedRPG/assets/images/pointer.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/small_button.png b/Tutorials/TurnBasedRPG/assets/images/small_button.png new file mode 100644 index 000000000..bd39131e9 Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/small_button.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/tiles.png b/Tutorials/TurnBasedRPG/assets/images/tiles.png index f16b622f5..429add584 100644 Binary files a/Tutorials/TurnBasedRPG/assets/images/tiles.png and b/Tutorials/TurnBasedRPG/assets/images/tiles.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/ui_section.png b/Tutorials/TurnBasedRPG/assets/images/ui_section.png new file mode 100644 index 000000000..d69ffec7d Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/ui_section.png differ diff --git a/Tutorials/TurnBasedRPG/assets/images/uiback.png b/Tutorials/TurnBasedRPG/assets/images/uiback.png new file mode 100644 index 000000000..864106cf0 Binary files /dev/null and b/Tutorials/TurnBasedRPG/assets/images/uiback.png differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/coin.wav b/Tutorials/TurnBasedRPG/assets/sounds/coin.wav index 5396aeeec..b92bdabdf 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/coin.wav and b/Tutorials/TurnBasedRPG/assets/sounds/coin.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/combat.wav b/Tutorials/TurnBasedRPG/assets/sounds/combat.wav index ec0cc74e4..135ff3d8a 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/combat.wav and b/Tutorials/TurnBasedRPG/assets/sounds/combat.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/fled.wav b/Tutorials/TurnBasedRPG/assets/sounds/fled.wav index ccd9497a3..862d8d3fe 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/fled.wav and b/Tutorials/TurnBasedRPG/assets/sounds/fled.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/hurt.wav b/Tutorials/TurnBasedRPG/assets/sounds/hurt.wav index 54491328d..5c09f1165 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/hurt.wav and b/Tutorials/TurnBasedRPG/assets/sounds/hurt.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/lose.wav b/Tutorials/TurnBasedRPG/assets/sounds/lose.wav index c543d5ec9..9d83705e5 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/lose.wav and b/Tutorials/TurnBasedRPG/assets/sounds/lose.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/miss.wav b/Tutorials/TurnBasedRPG/assets/sounds/miss.wav index 55a36b623..95d0a2da1 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/miss.wav and b/Tutorials/TurnBasedRPG/assets/sounds/miss.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/select.wav b/Tutorials/TurnBasedRPG/assets/sounds/select.wav index 6b7a9c9d7..73ec4f5f4 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/select.wav and b/Tutorials/TurnBasedRPG/assets/sounds/select.wav differ diff --git a/Tutorials/TurnBasedRPG/assets/sounds/win.wav b/Tutorials/TurnBasedRPG/assets/sounds/win.wav index a58220ca7..f62b62a77 100644 Binary files a/Tutorials/TurnBasedRPG/assets/sounds/win.wav and b/Tutorials/TurnBasedRPG/assets/sounds/win.wav differ diff --git a/Tutorials/TurnBasedRPG/source/CombatHUD.hx b/Tutorials/TurnBasedRPG/source/CombatHUD.hx deleted file mode 100644 index 07eb8b5a2..000000000 --- a/Tutorials/TurnBasedRPG/source/CombatHUD.hx +++ /dev/null @@ -1,507 +0,0 @@ -package; - -import flash.filters.ColorMatrixFilter; -import flash.geom.Matrix; -import flash.geom.Point; -import flixel.FlxG; -import flixel.FlxSprite; -import flixel.addons.effects.chainable.FlxEffectSprite; -import flixel.addons.effects.chainable.FlxWaveEffect; -import flixel.group.FlxGroup.FlxTypedGroup; -import flixel.sound.FlxSound; -import flixel.text.FlxText; -import flixel.tweens.FlxEase; -import flixel.tweens.FlxTween; -import flixel.ui.FlxBar; -import flixel.util.FlxColor; - -using flixel.util.FlxSpriteUtil; - -/** - * This enum is used to set the valid values for our outcome variable. - * Outcome can only ever be one of these 4 values and we can check for these values easily once combat is concluded. - */ -enum Outcome -{ - NONE; - ESCAPE; - VICTORY; - DEFEAT; -} - -enum Choice -{ - FIGHT; - FLEE; -} - -class CombatHUD extends FlxTypedGroup -{ - // These public variables will be used after combat has finished to help tell us what happened. - public var enemy:Enemy; // we will pass the enemySprite that the playerSprite touched to initialize combat, and this will let us also know which enemySprite to kill, etc. - public var playerHealth(default, null):Int; // when combat has finished, we will need to know how much remaining health the playerSprite has - public var outcome(default, null):Outcome; // when combat has finished, we will need to know if the playerSprite killed the enemySprite or fled - - // These are the sprites that we will use to show the combat hud interface - var background:FlxSprite; // this is the background sprite - var playerSprite:Player; // this is a sprite of the playerSprite - var enemySprite:Enemy; // this is a sprite of the enemySprite - - // These variables will be used to track the enemySprite's health - var enemyHealth:Int; - var enemyMaxHealth:Int; - var enemyHealthBar:FlxBar; // This FlxBar will show us the enemySprite's current/max health - - var playerHealthCounter:FlxText; // this will show the playerSprite's current/max health - - var damages:Array; // This array will contain 2 FlxText objects which will appear to show damage dealt (or misses) - - var pointer:FlxSprite; // This will be the pointer to show which option (Fight or Flee) the user is pointing to. - var selected:Choice; // this will track which option is selected - var choices:Map; // this map will contain the FlxTexts for our 2 options: Fight and Flee - - var results:FlxText; // this text will show the outcome of the battle for the playerSprite. - - var alpha:Float = 0; // we will use this to fade in and out our combat hud - var wait:Bool = true; // this flag will be set to true when don't want the playerSprite to be able to do anything (between turns) - - var fledSound:FlxSound; - var hurtSound:FlxSound; - var loseSound:FlxSound; - var missSound:FlxSound; - var selectSound:FlxSound; - var winSound:FlxSound; - var combatSound:FlxSound; - - var screen:FlxSprite; - - public function new() - { - super(); - - screen = new FlxSprite().makeGraphic(FlxG.width, FlxG.height, FlxColor.TRANSPARENT); - var waveEffect = new FlxWaveEffect(FlxWaveMode.ALL, 4, -1, 4); - var waveSprite = new FlxEffectSprite(screen, [waveEffect]); - add(waveSprite); - - // first, create our background. Make a black square, then draw borders onto it in white. Add it to our group. - background = new FlxSprite().makeGraphic(120, 120, FlxColor.WHITE); - background.drawRect(1, 1, 118, 44, FlxColor.BLACK); - background.drawRect(1, 46, 118, 73, FlxColor.BLACK); - background.screenCenter(); - add(background); - - // next, make a 'dummy' playerSprite that looks like our playerSprite (but can't move) and add it. - playerSprite = new Player(background.x + 36, background.y + 16); - playerSprite.animation.frameIndex = 3; - playerSprite.active = false; - playerSprite.facing = RIGHT; - add(playerSprite); - - // do the same thing for an enemySprite. We'll just use enemySprite type REGULAR for now and change it later. - enemySprite = new Enemy(background.x + 76, background.y + 16, REGULAR); - enemySprite.animation.frameIndex = 3; - enemySprite.active = false; - enemySprite.facing = LEFT; - add(enemySprite); - - // setup the playerSprite's health display and add it to the group. - playerHealthCounter = new FlxText(0, playerSprite.y + playerSprite.height + 2, 0, "3 / 3", 8); - playerHealthCounter.alignment = CENTER; - playerHealthCounter.x = playerSprite.x + 4 - (playerHealthCounter.width / 2); - add(playerHealthCounter); - - // create and add a FlxBar to show the enemySprite's health. We'll make it Red and Yellow. - enemyHealthBar = new FlxBar(enemySprite.x - 6, playerHealthCounter.y, LEFT_TO_RIGHT, 20, 10); - enemyHealthBar.createFilledBar(0xffdc143c, FlxColor.YELLOW, true, FlxColor.YELLOW); - add(enemyHealthBar); - - // create our choices and add them to the group. - choices = new Map(); - choices[FIGHT] = new FlxText(background.x + 30, background.y + 48, 85, "FIGHT", 22); - choices[FLEE] = new FlxText(background.x + 30, choices[FIGHT].y + choices[FIGHT].height + 8, 85, "FLEE", 22); - add(choices[FIGHT]); - add(choices[FLEE]); - - pointer = new FlxSprite(background.x + 10, choices[FIGHT].y + (choices[FIGHT].height / 2) - 8, AssetPaths.pointer__png); - pointer.visible = false; - add(pointer); - - // create our damage texts. We'll make them be white text with a red shadow (so they stand out). - damages = new Array(); - damages.push(new FlxText(0, 0, 40)); - damages.push(new FlxText(0, 0, 40)); - for (d in damages) - { - d.color = FlxColor.WHITE; - d.setBorderStyle(SHADOW, FlxColor.RED); - d.alignment = CENTER; - d.visible = false; - add(d); - } - - // create our results text object. We'll position it, but make it hidden for now. - results = new FlxText(background.x + 2, background.y + 9, 116, "", 18); - results.alignment = CENTER; - results.color = FlxColor.YELLOW; - results.setBorderStyle(SHADOW, FlxColor.GRAY); - results.visible = false; - add(results); - - // like we did in our HUD class, we need to set the scrollFactor on each of our children objects to 0,0. We also set alpha to 0 (so we can fade this in) - forEach(function(sprite:FlxSprite) - { - sprite.scrollFactor.set(); - sprite.alpha = 0; - }); - - // mark this object as not active and not visible so update and draw don't get called on it until we're ready to show it. - active = false; - visible = false; - - fledSound = FlxG.sound.load(AssetPaths.fled__wav); - hurtSound = FlxG.sound.load(AssetPaths.hurt__wav); - loseSound = FlxG.sound.load(AssetPaths.lose__wav); - missSound = FlxG.sound.load(AssetPaths.miss__wav); - selectSound = FlxG.sound.load(AssetPaths.select__wav); - winSound = FlxG.sound.load(AssetPaths.win__wav); - combatSound = FlxG.sound.load(AssetPaths.combat__wav); - } - - /** - * This function will be called from PlayState when we want to start combat. It will setup the screen and make sure everything is ready. - * @param playerHealth The amount of health the playerSprite is starting with - * @param enemy This links back to the Enemy we are fighting with so we can get it's health and type (to change our sprite). - */ - public function initCombat(playerHealth:Int, enemy:Enemy) - { - screen.drawFrame(); - var screenPixels = screen.framePixels; - - if (FlxG.renderBlit) - screenPixels.copyPixels(FlxG.camera.buffer, FlxG.camera.buffer.rect, new Point()); - else - screenPixels.draw(FlxG.camera.canvas, new Matrix(1, 0, 0, 1, 0, 0)); - - var rc:Float = 1 / 3; - var gc:Float = 1 / 2; - var bc:Float = 1 / 6; - screenPixels.applyFilter(screenPixels, screenPixels.rect, new Point(), - new ColorMatrixFilter([rc, gc, bc, 0, 0, rc, gc, bc, 0, 0, rc, gc, bc, 0, 0, 0, 0, 0, 1, 0])); - - combatSound.play(); - this.playerHealth = playerHealth; // we set our playerHealth variable to the value that was passed to us - this.enemy = enemy; // set our enemySprite object to the one passed to us - - updatePlayerHealth(); - - // setup our enemySprite - enemyMaxHealth = enemyHealth = if (enemy.type == REGULAR) 2 else 4; // each enemySprite will have health based on their type - enemyHealthBar.value = 100; // the enemySprite's health bar starts at 100% - enemySprite.changeType(enemy.type); // change our enemySprite's image to match their type. - - // make sure we initialize all of these before we start so nothing looks 'wrong' the second time we get - wait = true; - results.text = ""; - pointer.visible = false; - results.visible = false; - outcome = NONE; - selected = FIGHT; - movePointer(); - - visible = true; // make our hud visible (so draw gets called on it) - note, it's not active, yet! - - // do a numeric tween to fade in our combat hud when the tween is finished, call finishFadeIn - FlxTween.num(0, 1, .66, {ease: FlxEase.circOut, onComplete: finishFadeIn}, updateAlpha); - } - - /** - * This function is called by our Tween to fade in/out all the items in our hud. - */ - function updateAlpha(alpha:Float) - { - this.alpha = alpha; - forEach(function(sprite) sprite.alpha = alpha); - } - - /** - * When we've finished fading in, we set our hud to active (so it gets updates), and allow the playerSprite to interact. We show our pointer, too. - */ - function finishFadeIn(_) - { - active = true; - wait = false; - pointer.visible = true; - selectSound.play(); - } - - /** - * After we fade our hud out, we set it to not be active or visible (no update and no draw) - */ - function finishFadeOut(_) - { - active = false; - visible = false; - } - - /** - * This function is called to change the Player's health text on the screen. - */ - function updatePlayerHealth() - { - playerHealthCounter.text = playerHealth + " / 3"; - playerHealthCounter.x = playerSprite.x + 4 - (playerHealthCounter.width / 2); - } - - override public function update(elapsed:Float) - { - if (!wait) // if we're waiting, don't do any of this. - { - updateKeyboardInput(); - updateTouchInput(); - } - super.update(elapsed); - } - - function updateKeyboardInput() - { - #if FLX_KEYBOARD - // setup some simple flags to see which keys are pressed. - var up:Bool = false; - var down:Bool = false; - var fire:Bool = false; - - // check to see any keys are pressed and set the cooresponding flags. - if (FlxG.keys.anyJustReleased([SPACE, X, ENTER])) - { - fire = true; - } - else if (FlxG.keys.anyJustReleased([W, UP])) - { - up = true; - } - else if (FlxG.keys.anyJustReleased([S, DOWN])) - { - down = true; - } - - // based on which flags are set, do the specified action - if (fire) - { - selectSound.play(); - makeChoice(); // when the playerSprite chooses either option, we call this function to process their selection - } - else if (up || down) - { - // if the playerSprite presses up or down, we move the cursor up or down (with wrapping) - selected = if (selected == FIGHT) FLEE else FIGHT; - selectSound.play(); - movePointer(); - } - #end - } - - function updateTouchInput() - { - #if FLX_TOUCH - for (touch in FlxG.touches.justReleased()) - { - for (choice in choices.keys()) - { - var text = choices[choice]; - if (touch.overlaps(text)) - { - selectSound.play(); - selected = choice; - movePointer(); - makeChoice(); - return; - } - } - } - #end - } - - /** - * Call this function to place the pointer next to the currently selected choice - */ - function movePointer() - { - pointer.y = choices[selected].y + (choices[selected].height / 2) - 8; - } - - /** - * This function will process the choice the playerSprite picked - */ - function makeChoice() - { - pointer.visible = false; // hide our pointer - switch (selected) // check which item was selected when the playerSprite picked it - { - case FIGHT: - // if FIGHT was picked... - // ...the playerSprite attacks the enemySprite first - // they have an 85% chance to hit the enemySprite - if (FlxG.random.bool(85)) - { - // if they hit, deal 1 damage to the enemySprite, and setup our damage indicator - damages[1].text = "1"; - FlxTween.tween(enemySprite, {x: enemySprite.x + 4}, 0.1, { - onComplete: function(_) - { - FlxTween.tween(enemySprite, {x: enemySprite.x - 4}, 0.1); - } - }); - hurtSound.play(); - enemyHealth--; - enemyHealthBar.value = (enemyHealth / enemyMaxHealth) * 100; // change the enemySprite's health bar - } - else - { - // change our damage text to show that we missed! - damages[1].text = "MISS!"; - missSound.play(); - } - - // position the damage text over the enemySprite, and set it's alpha to 0 but it's visible to true (so that it gets draw called on it) - damages[1].x = enemySprite.x + 2 - (damages[1].width / 2); - damages[1].y = enemySprite.y + 4 - (damages[1].height / 2); - damages[1].alpha = 0; - damages[1].visible = true; - - // if the enemySprite is still alive, it will swing back! - if (enemyHealth > 0) - { - enemyAttack(); - } - - // setup 2 tweens to allow the damage indicators to fade in and float up from the sprites - FlxTween.num(damages[0].y, damages[0].y - 12, 1, {ease: FlxEase.circOut}, updateDamageY); - FlxTween.num(0, 1, .2, {ease: FlxEase.circInOut, onComplete: doneDamageIn}, updateDamageAlpha); - - case FLEE: - // if the playerSprite chose to FLEE, we'll give them a 50/50 chance to escape - if (FlxG.random.bool(50)) - { - // if they succeed, we show the 'escaped' message and trigger it to fade in - outcome = ESCAPE; - results.text = "ESCAPED!"; - fledSound.play(); - results.visible = true; - results.alpha = 0; - FlxTween.tween(results, {alpha: 1}, .66, {ease: FlxEase.circInOut, onComplete: doneResultsIn}); - } - else - { - // if they fail to escape, the enemySprite will get a free-swing - enemyAttack(); - FlxTween.num(damages[0].y, damages[0].y - 12, 1, {ease: FlxEase.circOut}, updateDamageY); - FlxTween.num(0, 1, .2, {ease: FlxEase.circInOut, onComplete: doneDamageIn}, updateDamageAlpha); - } - } - - // regardless of what happens, we need to set our 'wait' flag so that we can show what happened before moving on - wait = true; - } - - /** - * This function is called anytime we want the enemySprite to swing at the playerSprite - */ - function enemyAttack() - { - // first, lets see if the enemySprite hits or not. We'll give him a 30% chance to hit - if (FlxG.random.bool(30)) - { - // if we hit, flash the screen white, and deal one damage to the playerSprite - then update the playerSprite's health - FlxG.camera.flash(FlxColor.WHITE, .2); - FlxG.camera.shake(0.01, 0.2); - hurtSound.play(); - damages[0].text = "1"; - playerHealth--; - updatePlayerHealth(); - } - else - { - // if the enemySprite misses, show it on the screen - damages[0].text = "MISS!"; - missSound.play(); - } - - // setup the combat text to show up over the playerSprite and fade in/raise up - damages[0].x = playerSprite.x + 2 - (damages[0].width / 2); - damages[0].y = playerSprite.y + 4 - (damages[0].height / 2); - damages[0].alpha = 0; - damages[0].visible = true; - } - - /** - * This function is called from our Tweens to move the damage displays up on the screen - */ - function updateDamageY(damageY:Float) - { - damages[0].y = damages[1].y = damageY; - } - - /** - * This function is called from our Tweens to fade in/out the damage text - */ - function updateDamageAlpha(damageAlpha:Float) - { - damages[0].alpha = damages[1].alpha = damageAlpha; - } - - /** - * This function is called when our damage texts have finished fading in - it will trigger them to start fading out again, after a short delay - */ - function doneDamageIn(_) - { - FlxTween.num(1, 0, .66, {ease: FlxEase.circInOut, startDelay: 1, onComplete: doneDamageOut}, updateDamageAlpha); - } - - /** - * This function is triggered when our results text has finished fading in. If we're not defeated, we will fade out the entire hud after a short delay - */ - function doneResultsIn(_) - { - FlxTween.num(1, 0, .66, {ease: FlxEase.circOut, onComplete: finishFadeOut, startDelay: 1}, updateAlpha); - } - - /** - * This function is triggered when the damage texts have finished fading out again. They will clear and reset them for next time. - * It will also check to see what we're supposed to do next - if the enemySprite is dead, we trigger victory, if the playerSprite is dead we trigger defeat, otherwise we reset for the next round. - */ - function doneDamageOut(_) - { - damages[0].visible = false; - damages[1].visible = false; - damages[0].text = ""; - damages[1].text = ""; - - if (playerHealth <= 0) - { - // if the playerSprite's health is 0, we show the defeat message on the screen and fade it in - outcome = DEFEAT; - loseSound.play(); - results.text = "DEFEAT!"; - results.visible = true; - results.alpha = 0; - FlxTween.tween(results, {alpha: 1}, 0.66, {ease: FlxEase.circInOut, onComplete: doneResultsIn}); - } - else if (enemyHealth <= 0) - { - // if the enemySprite's health is 0, we show the victory message - outcome = VICTORY; - winSound.play(); - results.text = "VICTORY!"; - results.visible = true; - results.alpha = 0; - FlxTween.tween(results, {alpha: 1}, 0.66, {ease: FlxEase.circInOut, onComplete: doneResultsIn}); - } - else - { - // both are still alive, so we reset and have the playerSprite pick their next action - wait = false; - pointer.visible = true; - } - } -} diff --git a/Tutorials/TurnBasedRPG/source/GameOverState.hx b/Tutorials/TurnBasedRPG/source/GameOverState.hx deleted file mode 100644 index f824af731..000000000 --- a/Tutorials/TurnBasedRPG/source/GameOverState.hx +++ /dev/null @@ -1,100 +0,0 @@ -package; - -import flixel.FlxG; -import flixel.FlxSprite; -import flixel.FlxState; -import flixel.text.FlxText; -import flixel.ui.FlxButton; -import flixel.util.FlxAxes; -import flixel.util.FlxColor; - -class GameOverState extends FlxState -{ - var titleText:FlxText; // the title text - var messageText:FlxText; // the final score message text - var scoreIcon:FlxSprite; // sprite for a coin icon - var scoreText:FlxText; // text of the score - var highscoreText:FlxText; // text to show the highscore - var mainMenuButton:FlxButton; // button to go to main menu - - /** - * Called from PlayState, this will set our win and score variables - * @param win true if the player beat the boss, false if they died - * @param score the number of coins collected - */ - public function new(win:Bool, score:Int) - { - super(); - - #if FLX_MOUSE - FlxG.mouse.visible = true; - #end - - // create and add each of our items - - titleText = new FlxText(0, 20, 0, if (win) "You Win!" else "Game Over!", 22); - titleText.alignment = CENTER; - titleText.screenCenter(FlxAxes.X); - add(titleText); - - messageText = new FlxText(0, (FlxG.height / 2) - 18, 0, "Final Score:", 8); - messageText.alignment = CENTER; - messageText.screenCenter(FlxAxes.X); - add(messageText); - - scoreIcon = new FlxSprite((FlxG.width / 2) - 8, 0, AssetPaths.coin__png); - scoreIcon.screenCenter(FlxAxes.Y); - add(scoreIcon); - - scoreText = new FlxText((FlxG.width / 2), 0, 0, Std.string(score), 8); - scoreText.screenCenter(FlxAxes.Y); - add(scoreText); - - // we want to see what the highscore is - var highscore = checkHighscore(score); - - highscoreText = new FlxText(0, (FlxG.height / 2) + 10, 0, "Highscore: " + highscore, 8); - highscoreText.alignment = CENTER; - highscoreText.screenCenter(FlxAxes.Y); - add(highscoreText); - - mainMenuButton = new FlxButton(0, FlxG.height - 32, "Main Menu", switchToMainMenu); - mainMenuButton.screenCenter(FlxAxes.X); - mainMenuButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); - add(mainMenuButton); - - FlxG.camera.fade(FlxColor.BLACK, 0.33, true); - } - - /** - * This function will compare the new score with the saved highscore. - * If the new score is higher, it will save it as the new highscore, otherwise, it will return the saved highscore. - * @param score The new score - * @return the highscore - */ - function checkHighscore(score:Int):Int - { - var highscore:Int = score; - if (FlxG.save.data.highscore != null && FlxG.save.data.highscore > highscore) - { - highscore = FlxG.save.data.highscore; - } - else - { - // data is less or there is no data; save current score - FlxG.save.data.highscore = highscore; - } - return highscore; - } - - /** - * When the user hits the main menu button, it should fade out and then take them back to the MenuState - */ - function switchToMainMenu():Void - { - FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() - { - FlxG.switchState(MenuState.new); - }); - } -} diff --git a/Tutorials/TurnBasedRPG/source/HUD.hx b/Tutorials/TurnBasedRPG/source/HUD.hx deleted file mode 100644 index 09db963d2..000000000 --- a/Tutorials/TurnBasedRPG/source/HUD.hx +++ /dev/null @@ -1,46 +0,0 @@ -package; - -import flixel.FlxG; -import flixel.FlxSprite; -import flixel.group.FlxGroup; -import flixel.text.FlxText; -import flixel.util.FlxColor; - -using flixel.util.FlxSpriteUtil; - -class HUD extends FlxTypedGroup -{ - var background:FlxSprite; - var healthCounter:FlxText; - var moneyCounter:FlxText; - var healthIcon:FlxSprite; - var moneyIcon:FlxSprite; - - public function new() - { - super(); - background = new FlxSprite().makeGraphic(FlxG.width, 20, FlxColor.BLACK); - background.drawRect(0, 19, FlxG.width, 1, FlxColor.WHITE); - healthCounter = new FlxText(16, 2, 0, "3 / 3", 8); - healthCounter.setBorderStyle(SHADOW, FlxColor.GRAY, 1, 1); - moneyCounter = new FlxText(0, 2, 0, "0", 8); - moneyCounter.setBorderStyle(SHADOW, FlxColor.GRAY, 1, 1); - healthIcon = new FlxSprite(4, healthCounter.y + (healthCounter.height / 2) - 4, AssetPaths.health__png); - moneyIcon = new FlxSprite(FlxG.width - 12, moneyCounter.y + (moneyCounter.height / 2) - 4, AssetPaths.coin__png); - moneyCounter.alignment = RIGHT; - moneyCounter.x = moneyIcon.x - moneyCounter.width - 4; - add(background); - add(healthIcon); - add(moneyIcon); - add(healthCounter); - add(moneyCounter); - forEach(function(sprite) sprite.scrollFactor.set(0, 0)); - } - - public function updateHUD(health:Int, money:Int) - { - healthCounter.text = health + " / 3"; - moneyCounter.text = Std.string(money); - moneyCounter.x = moneyIcon.x - moneyCounter.width - 4; - } -} diff --git a/Tutorials/TurnBasedRPG/source/Main.hx b/Tutorials/TurnBasedRPG/source/Main.hx index 44c432989..ad37778ca 100644 --- a/Tutorials/TurnBasedRPG/source/Main.hx +++ b/Tutorials/TurnBasedRPG/source/Main.hx @@ -4,6 +4,7 @@ import flixel.FlxG; import flixel.FlxGame; import flixel.util.FlxSave; import openfl.display.Sprite; +import states.MenuState; class Main extends Sprite { diff --git a/Tutorials/TurnBasedRPG/source/MenuState.hx b/Tutorials/TurnBasedRPG/source/MenuState.hx deleted file mode 100644 index 29a71272d..000000000 --- a/Tutorials/TurnBasedRPG/source/MenuState.hx +++ /dev/null @@ -1,78 +0,0 @@ -package; - -import flixel.FlxG; -import flixel.FlxState; -import flixel.text.FlxText; -import flixel.ui.FlxButton; -import flixel.util.FlxColor; - -class MenuState extends FlxState -{ - var titleText:FlxText; - var playButton:FlxButton; - var optionsButton:FlxButton; - #if desktop - var exitButton:FlxButton; - #end - - override public function create() - { - titleText = new FlxText(20, 0, 0, "HaxeFlixel\nTutorial\nGame", 22); - titleText.alignment = CENTER; - titleText.screenCenter(X); - add(titleText); - - playButton = new FlxButton(0, 0, "Play", clickPlay); - playButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); - playButton.x = (FlxG.width / 2) - 10 - playButton.width; - playButton.y = FlxG.height - playButton.height - 10; - add(playButton); - - optionsButton = new FlxButton(0, 0, "Options", clickOptions); - optionsButton.x = (FlxG.width / 2) + 10; - optionsButton.y = FlxG.height - optionsButton.height - 10; - add(optionsButton); - - #if desktop - exitButton = new FlxButton(FlxG.width - 28, 8, "X", clickExit); - exitButton.loadGraphic(AssetPaths.button__png, true, 20, 20); - add(exitButton); - #end - - if (FlxG.sound.music == null) // don't restart the music if it's already playing - { - #if flash - FlxG.sound.playMusic(AssetPaths.HaxeFlixel_Tutorial_Game__mp3, 1, true); - #else - FlxG.sound.playMusic(AssetPaths.HaxeFlixel_Tutorial_Game__ogg, 1, true); - #end - } - - FlxG.camera.fade(FlxColor.BLACK, 0.33, true); - - super.create(); - } - - function clickPlay() - { - FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() - { - FlxG.switchState(PlayState.new); - }); - } - - function clickOptions() - { - FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() - { - FlxG.switchState(OptionsState.new); - }); - } - - #if desktop - function clickExit() - { - Sys.exit(0); - } - #end -} diff --git a/Tutorials/TurnBasedRPG/source/Coin.hx b/Tutorials/TurnBasedRPG/source/objects/Coin.hx similarity index 74% rename from Tutorials/TurnBasedRPG/source/Coin.hx rename to Tutorials/TurnBasedRPG/source/objects/Coin.hx index 70e2994e4..4f8d39dad 100644 --- a/Tutorials/TurnBasedRPG/source/Coin.hx +++ b/Tutorials/TurnBasedRPG/source/objects/Coin.hx @@ -1,4 +1,4 @@ -package; +package objects; import flixel.FlxSprite; import flixel.tweens.FlxEase; @@ -9,7 +9,9 @@ class Coin extends FlxSprite public function new(x:Float, y:Float) { super(x, y); - loadGraphic(AssetPaths.coin__png, false, 8, 8); + loadGraphic(AssetPaths.coin__png, true, 12, 12); + animation.add("idle", [0, 1], 4); + animation.play("idle"); } override function kill() diff --git a/Tutorials/TurnBasedRPG/source/Enemy.hx b/Tutorials/TurnBasedRPG/source/objects/Enemy.hx similarity index 70% rename from Tutorials/TurnBasedRPG/source/Enemy.hx rename to Tutorials/TurnBasedRPG/source/objects/Enemy.hx index db26ef240..0e5ea9bd6 100644 --- a/Tutorials/TurnBasedRPG/source/Enemy.hx +++ b/Tutorials/TurnBasedRPG/source/objects/Enemy.hx @@ -1,9 +1,10 @@ -package; +package objects; import flixel.FlxG; import flixel.FlxSprite; import flixel.math.FlxPoint; import flixel.math.FlxVelocity; +import flixel.tile.FlxTilemap; import flixel.sound.FlxSound; using flixel.util.FlxSpriteUtil; @@ -16,8 +17,8 @@ enum EnemyType class Enemy extends FlxSprite { - static inline var WALK_SPEED:Float = 40; - static inline var CHASE_SPEED:Float = 70; + static inline var WALK_SPEED:Float = 50; + static inline var CHASE_SPEED:Float = 90; var brain:FSM; var idleTimer:Float; @@ -27,15 +28,19 @@ class Enemy extends FlxSprite public var type(default, null):EnemyType; public var seesPlayer:Bool; public var playerPosition:FlxPoint; + public var maxHP:Int; + public var hp:Int; public function new(x:Float, y:Float, type:EnemyType) { super(x, y); - this.type = type; - var graphic = if (type == BOSS) AssetPaths.boss__png else AssetPaths.enemy__png; - loadGraphic(graphic, true, 16, 16); - setFacingFlip(LEFT, false, false); - setFacingFlip(RIGHT, true, false); + + changeType(type); + maxHP = type == REGULAR ? 2 : 4; + hp = maxHP; + + setFacingFlip(LEFT, true, false); + setFacingFlip(RIGHT, false, false); animation.add("d_idle", [0]); animation.add("lr_idle", [3]); animation.add("u_idle", [6]); @@ -43,9 +48,8 @@ class Enemy extends FlxSprite animation.add("lr_walk", [3, 4, 3, 5], 6); animation.add("u_walk", [6, 7, 6, 8], 6); drag.x = drag.y = 10; - setSize(8, 8); - offset.x = 4; - offset.y = 8; + setSize(12, 12); + offset.set(6, 12); brain = new FSM(idle); idleTimer = 0; @@ -55,11 +59,18 @@ class Enemy extends FlxSprite stepSound = FlxG.sound.load(AssetPaths.step__wav, 0.4); stepSound.proximity(x, y, FlxG.camera.target, FlxG.width * 0.6); } + + public function hurt(damage:Int) + { + hp -= damage; + } - override public function update(elapsed:Float) + override function update(elapsed:Float) { if (this.isFlickering()) + { return; + } var action = "idle"; if (velocity.x != 0 || velocity.y != 0) @@ -67,37 +78,31 @@ class Enemy extends FlxSprite action = "walk"; if (Math.abs(velocity.x) > Math.abs(velocity.y)) { - if (velocity.x < 0) - facing = LEFT; - else - facing = RIGHT; + facing = (velocity.x < 0) ? LEFT : RIGHT; } else { - if (velocity.y < 0) - facing = UP; - else - facing = DOWN; + facing = (velocity.y < 0) ? UP : DOWN; } - - stepSound.setPosition(x + frameWidth / 2, y + height); + + stepSound.setPosition(x + width / 2, y + height); stepSound.play(); } - + switch (facing) { case LEFT, RIGHT: animation.play("lr_" + action); - + case UP: animation.play("u_" + action); - + case DOWN: animation.play("d_" + action); - + case _: } - + brain.update(elapsed); super.update(elapsed); } @@ -125,7 +130,9 @@ class Enemy extends FlxSprite idleTimer = FlxG.random.int(1, 4); } else + { idleTimer -= elapsed; + } } function chase(elapsed:Float) @@ -139,6 +146,14 @@ class Enemy extends FlxSprite FlxVelocity.moveTowardsPoint(this, playerPosition, CHASE_SPEED); } } + + public function checkVision(player:Player, walls:FlxTilemap) + { + // Store the player position + player.getMidpoint(playerPosition); + // Cast a ray from here to the player and see if a wall is blocking + seesPlayer = walls.ray(getMidpoint(FlxPoint.weak()), playerPosition); + } public function changeType(type:EnemyType) { @@ -146,7 +161,7 @@ class Enemy extends FlxSprite { this.type = type; var graphic = if (type == BOSS) AssetPaths.boss__png else AssetPaths.enemy__png; - loadGraphic(graphic, true, 16, 16); + loadGraphic(graphic, true, 24, 24); } } } diff --git a/Tutorials/TurnBasedRPG/source/FSM.hx b/Tutorials/TurnBasedRPG/source/objects/FSM.hx similarity index 91% rename from Tutorials/TurnBasedRPG/source/FSM.hx rename to Tutorials/TurnBasedRPG/source/objects/FSM.hx index 9a7c4f116..8a6dd49c4 100644 --- a/Tutorials/TurnBasedRPG/source/FSM.hx +++ b/Tutorials/TurnBasedRPG/source/objects/FSM.hx @@ -1,3 +1,5 @@ +package objects; + class FSM { public var activeState:Float->Void; diff --git a/Tutorials/TurnBasedRPG/source/Player.hx b/Tutorials/TurnBasedRPG/source/objects/Player.hx similarity index 56% rename from Tutorials/TurnBasedRPG/source/Player.hx rename to Tutorials/TurnBasedRPG/source/objects/Player.hx index 32a3186e6..5ed1a8c8e 100644 --- a/Tutorials/TurnBasedRPG/source/Player.hx +++ b/Tutorials/TurnBasedRPG/source/objects/Player.hx @@ -1,4 +1,4 @@ -package; +package objects; import flixel.FlxG; import flixel.FlxSprite; @@ -7,16 +7,21 @@ import flixel.sound.FlxSound; class Player extends FlxSprite { - static inline var SPEED:Float = 100; + static inline var SPEED:Float = 110; + /** Reaches top speed in 0.15 seconds */ + static inline var ACCEL:Float = SPEED / 0.15; - var stepSound:FlxSound; + public final maxHP:Int = 3; + public var hp:Int; public function new(x:Float = 0, y:Float = 0) { + hp = maxHP; super(x, y); - loadGraphic(AssetPaths.player__png, true, 16, 16); - setFacingFlip(LEFT, false, false); - setFacingFlip(RIGHT, true, false); + + loadGraphic(AssetPaths.player__png, true, 24, 24); + setFacingFlip(LEFT, true, false); + setFacingFlip(RIGHT, false, false); animation.add("d_idle", [0]); animation.add("lr_idle", [3]); animation.add("u_idle", [6]); @@ -25,20 +30,44 @@ class Player extends FlxSprite animation.add("u_walk", [6, 7, 6, 8], 6); drag.x = drag.y = 800; - setSize(8, 8); - offset.set(4, 8); - - stepSound = FlxG.sound.load(AssetPaths.step__wav); + maxVelocity.x = maxVelocity.y = SPEED; + setSize(12, 12); + offset.set(6, 12); + } + + public function hurt(damage:Int) + { + hp -= damage; } override function update(elapsed:Float) { - updateMovement(); super.update(elapsed); + + updateMovement(); } function updateMovement() { + var action = "idle"; + // check if the player is moving, and not walking into walls + if (velocity.x != 0 || velocity.y != 0) + { + // FlxG.sound.play(AssetPaths.step__wav) + action = "walk"; + } + + switch (facing) + { + case LEFT, RIGHT: + animation.play("lr_" + action); + case UP: + animation.play("u_" + action); + case DOWN: + animation.play("d_" + action); + case _: + } + var up:Bool = false; var down:Bool = false; var left:Bool = false; @@ -58,65 +87,48 @@ class Player extends FlxSprite left = left || virtualPad.buttonLeft.pressed; right = right || virtualPad.buttonRight.pressed; #end - + + // Cancel out opposing directions if (up && down) + { up = down = false; + } + if (left && right) + { left = right = false; - - if (up || down || left || right) + } + + acceleration.set(0, 0); + if (right) { - var newAngle:Float = 0; - if (up) - { - newAngle = -90; - if (left) - newAngle -= 45; - else if (right) - newAngle += 45; - facing = UP; - } - else if (down) - { - newAngle = 90; - if (left) - newAngle += 45; - else if (right) - newAngle -= 45; - facing = DOWN; - } - else if (left) - { - newAngle = 180; - facing = LEFT; - } - else if (right) - { - newAngle = 0; - facing = RIGHT; - } - - // determine our velocity based on angle and speed - velocity.setPolarDegrees(SPEED, newAngle); + facing = RIGHT; + acceleration.x = ACCEL; } - - var action = "idle"; - // check if the player is moving, and not walking into walls - if ((velocity.x != 0 || velocity.y != 0) && touching == NONE) + else if (left) { - stepSound.play(); - action = "walk"; + facing = LEFT; + acceleration.x = -ACCEL; } - - switch (facing) + + if (down) { - case LEFT, RIGHT: - animation.play("lr_" + action); - case UP: - animation.play("u_" + action); - case DOWN: - animation.play("d_" + action); - case _: + facing = DOWN; + acceleration.y = ACCEL; + } + else if (up) + { + facing = UP; + acceleration.y = -ACCEL; + } + + // Prevent faster speeds on diagonal movement + var magnitude = velocity.length; + if (magnitude > SPEED) + { + // Reduce speed to SPEED but maintain direction + velocity.x *= SPEED / magnitude; + velocity.y *= SPEED / magnitude; } } } diff --git a/Tutorials/TurnBasedRPG/source/states/GameOverState.hx b/Tutorials/TurnBasedRPG/source/states/GameOverState.hx new file mode 100644 index 000000000..d5758c50f --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/states/GameOverState.hx @@ -0,0 +1,106 @@ +package states; + +import flixel.util.FlxTimer; +import flixel.tweens.FlxEase; +import flixel.tweens.FlxTween; +import flixel.FlxG; +import flixel.FlxSprite; +import flixel.FlxState; +import flixel.text.FlxBitmapText; +import flixel.text.FlxText; +import flixel.ui.FlxButton; +import flixel.util.FlxAxes; +import flixel.util.FlxColor; +import ui.LargeText; + +class GameOverState extends FlxState +{ + /** + * Called from PlayState, this will set our win and score variables + * @param win Whether the player beat the boss, or died + * @param score The number of coins collected + */ + public function new(win:Bool, score:Int) + { + super(); + + #if FLX_MOUSE + FlxG.mouse.visible = true; + #end + + // create and add each of our items + + var titleText = new LargeText(0, 20, if (win) "You Win!" else "Game Over!"); + titleText.screenCenter(FlxAxes.X); + add(titleText); + + var messageText = new FlxText(0, (FlxG.height / 2) - 18, 0, "Final Score: 0", 8); + messageText.screenCenter(FlxAxes.X); + add(messageText); + + // Fade the camera from black + FlxG.camera.fade(FlxColor.BLACK, 0.33, true); + + // Count up the points for dramatic effect + FlxTween.num(0, score, 1.0, // from 0 to score in 1.0 second + { + startDelay: 0.33,// wait for the fade to complete + ease: FlxEase.circOut, + onComplete: function (tween:FlxTween) + { + // Wait 1 second and then show the highscore + new FlxTimer().start(0.5, (_)->showHighscore(score)); + } + }, + function updateText(tweenedScore) + { + messageText.text = "Final Score: " + Math.floor(tweenedScore); + } + ); + } + + function showHighscore(score:Int) + { + // Get previous highscore + var highscore = 0; + if (FlxG.save.data.highscore != null) + { + highscore = FlxG.save.data.highscore; + } + + var highscoreText = new FlxText(0, (FlxG.height / 2) + 10, 0, "Highscore: " + highscore, 8); + add(highscoreText); + + // New high score + if (score > highscore) + { + FlxG.save.data.highscore = score; + highscoreText.text = "New Highscore!"; + } + + highscoreText.screenCenter(FlxAxes.XY); + + // Wait a second then show the + new FlxTimer().start(1.0, (_)->showButton()); + } + + function showButton() + { + var mainMenuButton = new FlxButton(0, FlxG.height - 32, "Main Menu", switchToMainMenu); + mainMenuButton.loadGraphic(AssetPaths.button__png, true, 80, 20); + mainMenuButton.screenCenter(FlxAxes.X); + mainMenuButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); + add(mainMenuButton); + } + + /** + * When the user hits the main menu button, it should fade out and then take them back to the MenuState + */ + function switchToMainMenu():Void + { + FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() + { + FlxG.switchState(MenuState.new); + }); + } +} diff --git a/Tutorials/TurnBasedRPG/source/states/MenuState.hx b/Tutorials/TurnBasedRPG/source/states/MenuState.hx new file mode 100644 index 000000000..ea92059e3 --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/states/MenuState.hx @@ -0,0 +1,111 @@ +package states; + +import ui.OptionsSubState; +import ui.LargeText; +import flixel.FlxG; +import flixel.FlxState; +import flixel.text.FlxBitmapText; +import flixel.text.FlxText; +import flixel.ui.FlxButton; +import flixel.util.FlxColor; +import flixel.addons.editors.ogmo.FlxOgmo3Loader; + +class MenuState extends FlxState +{ + override public function create() + { + // NOTE: differs from tutorial! + var map = new FlxOgmo3Loader(AssetPaths.turnBasedRPG__ogmo, AssetPaths.room_001__json); + var walls = map.loadTilemap(AssetPaths.tiles__png, "walls"); + walls.x -= 123; + walls.y -= 10; + add(walls); + + // Use FlxBitmapText for crisper edges on large text + var titleText = new LargeText(0, 16, "DUNGEON\nCRAWLER"); + titleText.alignment = CENTER; + titleText.setBorderStyle(OUTLINE, 0xFF3f2631); + titleText.screenCenter(X); + add(titleText); + + // TUTORIAL VERSION: + // var titleText = new FlxText(0, 16, 0, "DUNGEON\nCRAWLER", 32); + // titleText.alignment = CENTER; + // titleText.screenCenter(X); + // add(titleText); + + var playButton = new FlxButton(0, 0, "Play", clickPlay); + playButton.loadGraphic(AssetPaths.button__png, true, 80, 20); + playButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); + playButton.x = (FlxG.width / 2) - 10 - playButton.width; + playButton.y = FlxG.height - playButton.height - 10; + add(playButton); + + var optionsButton = new FlxButton(0, 0, "Options", clickOptions); + optionsButton.loadGraphic(AssetPaths.button__png, true, 80, 20); + optionsButton.x = (FlxG.width / 2) + 10; + optionsButton.y = FlxG.height - optionsButton.height - 10; + add(optionsButton); + + #if desktop + var exitButton = new FlxButton(FlxG.width - 28, 8, "X", clickExit); + exitButton.loadGraphic(AssetPaths.small_button__png, true, 20, 20); + add(exitButton); + #end + + if (FlxG.sound.music == null) // don't restart the music if it's already playing + { + initSound(); + } + + FlxG.camera.fade(FlxColor.BLACK, 0.33, true); + + super.create(); + } + + function initSound() + { + var volumes:{ music:Float, sound:Float } = null; + if (FlxG.save.data.volumes != null) + { + volumes = FlxG.save.data.volumes; + } + else + { + volumes = { music:0.5, sound:1.0 }; + } + + FlxG.sound.defaultMusicGroup.volume = volumes.music; + FlxG.sound.defaultSoundGroup.volume = volumes.sound; + + #if flash + FlxG.sound.playMusic(AssetPaths.HaxeFlixel_Tutorial_Game__mp3, 1.0, true); + #else + FlxG.sound.playMusic(AssetPaths.HaxeFlixel_Tutorial_Game__ogg, 1.0, true); + #end + } + + function clickPlay() + { + FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() + { + FlxG.switchState(PlayState.new); + }); + } + + function clickOptions() + { + openSubState(new OptionsSubState()); + // FlxG.camera.fade(FlxColor.BLACK, 0.33, false, function() + // { + // FlxG.switchState(OptionsState.new); + // }); + } + + #if desktop + function clickExit() + { + Sys.exit(0); + } + #end +} diff --git a/Tutorials/TurnBasedRPG/source/OptionsState.hx b/Tutorials/TurnBasedRPG/source/states/OptionsState.hx similarity index 76% rename from Tutorials/TurnBasedRPG/source/OptionsState.hx rename to Tutorials/TurnBasedRPG/source/states/OptionsState.hx index 4a567f211..7c1d2e0b2 100644 --- a/Tutorials/TurnBasedRPG/source/OptionsState.hx +++ b/Tutorials/TurnBasedRPG/source/states/OptionsState.hx @@ -1,8 +1,9 @@ -package; +package states; import flixel.FlxG; import flixel.FlxState; import flixel.text.FlxText; +import flixel.text.FlxBitmapText; import flixel.ui.FlxBar; import flixel.ui.FlxButton; import flixel.util.FlxAxes; @@ -11,49 +12,41 @@ import flixel.util.FlxColor; class OptionsState extends FlxState { // define our screen elements - var titleText:FlxText; var volumeBar:FlxBar; - var volumeText:FlxText; var volumeAmountText:FlxText; - var volumeDownButton:FlxButton; - var volumeUpButton:FlxButton; - var clearDataButton:FlxButton; - var backButton:FlxButton; #if desktop var fullscreenButton:FlxButton; #end - + override public function create():Void { // setup and add our objects to the screen - titleText = new FlxText(0, 20, 0, "Options", 22); - titleText.alignment = CENTER; - titleText.screenCenter(FlxAxes.X); + var titleText = new LargeText(0, 32, "Options"); + titleText.screenCenter(X); add(titleText); - - volumeText = new FlxText(0, titleText.y + titleText.height + 10, 0, "Volume", 8); + + var volumeText = new FlxText(0, titleText.y + titleText.height + 10, 0, "Volume", 8); volumeText.alignment = CENTER; - volumeText.screenCenter(FlxAxes.X); + volumeText.screenCenter(X); add(volumeText); - + // the volume buttons will be smaller than 'default' buttons - volumeDownButton = new FlxButton(8, volumeText.y + volumeText.height + 2, "-", - clickVolumeDown); - volumeDownButton.loadGraphic(AssetPaths.button__png, true, 20, 20); + var volumeDownButton = new FlxButton(8, volumeText.y + volumeText.height + 2, "-", clickVolumeDown); + volumeDownButton.loadGraphic(AssetPaths.small_button__png, true, 20, 20); volumeDownButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); add(volumeDownButton); - - volumeUpButton = new FlxButton(FlxG.width - 28, volumeDownButton.y, "+", clickVolumeUp); - volumeUpButton.loadGraphic(AssetPaths.button__png, true, 20, 20); + + var volumeUpButton = new FlxButton(FlxG.width - 28, volumeDownButton.y, "+", clickVolumeUp); + volumeUpButton.loadGraphic(AssetPaths.small_button__png, true, 20, 20); volumeUpButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); add(volumeUpButton); - + volumeBar = new FlxBar(volumeDownButton.x + volumeDownButton.width + 4, volumeDownButton.y, LEFT_TO_RIGHT, Std.int(FlxG.width - 64), Std.int(volumeUpButton.height)); volumeBar.createFilledBar(0xff464646, FlxColor.WHITE, true, FlxColor.WHITE); add(volumeBar); - + volumeAmountText = new FlxText(0, 0, 200, (FlxG.sound.volume * 100) + "%", 8); volumeAmountText.alignment = CENTER; volumeAmountText.borderStyle = FlxTextBorderStyle.OUTLINE; @@ -61,23 +54,25 @@ class OptionsState extends FlxState volumeAmountText.y = volumeBar.y + (volumeBar.height / 2) - (volumeAmountText.height / 2); volumeAmountText.screenCenter(FlxAxes.X); add(volumeAmountText); - + #if desktop fullscreenButton = new FlxButton(0, volumeBar.y + volumeBar.height + 8, FlxG.fullscreen ? "FULLSCREEN" : "WINDOWED", clickFullscreen); + fullscreenButton.loadGraphic(AssetPaths.button__png, true, 80, 20); fullscreenButton.screenCenter(FlxAxes.X); add(fullscreenButton); #end - - clearDataButton = new FlxButton((FlxG.width / 2) - 90, FlxG.height - 28, "Clear Data", - clickClearData); + + var clearDataButton = new FlxButton((FlxG.width / 2) - 90, FlxG.height - 28, "Clear Data", clickClearData); + clearDataButton.loadGraphic(AssetPaths.button__png, true, 80, 20); clearDataButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); add(clearDataButton); - - backButton = new FlxButton((FlxG.width / 2) + 10, FlxG.height - 28, "Back", clickBack); + + var backButton = new FlxButton((FlxG.width / 2) + 10, FlxG.height - 28, "Back", clickBack); + backButton.loadGraphic(AssetPaths.button__png, true, 80, 20); backButton.onUp.sound = FlxG.sound.load(AssetPaths.select__wav); add(backButton); - + // update our bar to show the current volume level updateVolume(); diff --git a/Tutorials/TurnBasedRPG/source/PlayState.hx b/Tutorials/TurnBasedRPG/source/states/PlayState.hx similarity index 52% rename from Tutorials/TurnBasedRPG/source/PlayState.hx rename to Tutorials/TurnBasedRPG/source/states/PlayState.hx index b2a51962b..9b6d0dcaf 100644 --- a/Tutorials/TurnBasedRPG/source/PlayState.hx +++ b/Tutorials/TurnBasedRPG/source/states/PlayState.hx @@ -1,101 +1,91 @@ -package; +package states; import flixel.FlxG; import flixel.FlxState; import flixel.addons.editors.ogmo.FlxOgmo3Loader; -import flixel.group.FlxGroup.FlxTypedGroup; +import flixel.group.FlxGroup; import flixel.sound.FlxSound; import flixel.tile.FlxTilemap; import flixel.util.FlxColor; #if mobile import flixel.ui.FlxVirtualPad; #end +import objects.Coin; +import objects.Enemy; +import objects.Player; +import ui.CombatSubState; +import ui.HUD; using flixel.util.FlxSpriteUtil; class PlayState extends FlxState { var player:Player; - var map:FlxOgmo3Loader; var walls:FlxTilemap; var coins:FlxTypedGroup; var enemies:FlxTypedGroup; - + var hud:HUD; var money:Int = 0; - var health:Int = 3; - - var inCombat:Bool = false; - var combatHud:CombatHUD; - - var ending:Bool; - var won:Bool; - - var coinSound:FlxSound; - + #if mobile public static var virtualPad:FlxVirtualPad; #end - + override public function create() { #if FLX_MOUSE FlxG.mouse.visible = false; #end - - map = new FlxOgmo3Loader(AssetPaths.turnBasedRPG__ogmo, AssetPaths.room_001__json); + + var map = new FlxOgmo3Loader(AssetPaths.turnBasedRPG__ogmo, AssetPaths.room_001__json); walls = map.loadTilemap(AssetPaths.tiles__png, "walls"); walls.follow(); - walls.setTileProperties(1, NONE); - walls.setTileProperties(2, ANY); + walls.setTileProperties(0, NONE, null, null, 16); + walls.setTileProperties(16, ANY, null, null, 20); add(walls); - + coins = new FlxTypedGroup(); add(coins); - + enemies = new FlxTypedGroup(); add(enemies); - + player = new Player(); map.loadEntities(placeEntities, "entities"); add(player); - + FlxG.camera.follow(player, TOPDOWN, 1); - + hud = new HUD(); add(hud); - - combatHud = new CombatHUD(); - add(combatHud); - - coinSound = FlxG.sound.load(AssetPaths.coin__wav); - + #if mobile virtualPad = new FlxVirtualPad(FULL, NONE); add(virtualPad); #end - + FlxG.camera.fade(FlxColor.BLACK, 0.33, true); - + super.create(); } - + function placeEntities(entity:EntityData) { var x = entity.x; var y = entity.y; - + switch (entity.name) { case "player": player.setPosition(x, y); - + case "coin": coins.add(new Coin(x + 4, y + 4)); - + case "enemy": enemies.add(new Enemy(x + 4, y, REGULAR)); - + case "boss": enemies.add(new Enemy(x + 4, y, BOSS)); } @@ -104,62 +94,12 @@ class PlayState extends FlxState override public function update(elapsed:Float) { super.update(elapsed); - - if (ending) - { - return; - } - - if (inCombat) - { - if (!combatHud.visible) - { - health = combatHud.playerHealth; - hud.updateHUD(health, money); - if (combatHud.outcome == DEFEAT) - { - ending = true; - FlxG.camera.fade(FlxColor.BLACK, 0.33, false, doneFadeOut); - } - else - { - if (combatHud.outcome == VICTORY) - { - combatHud.enemy.kill(); - if (combatHud.enemy.type == BOSS) - { - won = true; - ending = true; - FlxG.camera.fade(FlxColor.BLACK, 0.33, false, doneFadeOut); - } - } - else - { - combatHud.enemy.flicker(); - } - inCombat = false; - player.active = true; - enemies.active = true; - - #if mobile - virtualPad.visible = true; - #end - } - } - } - else - { - FlxG.collide(player, walls); - FlxG.overlap(player, coins, playerTouchCoin); - FlxG.collide(enemies, walls); - enemies.forEachAlive(checkEnemyVision); - FlxG.overlap(player, enemies, playerTouchEnemy); - } - } - - function doneFadeOut() - { - FlxG.switchState(() -> new GameOverState(won, money)); + + FlxG.collide(player, walls); + FlxG.overlap(player, coins, playerTouchCoin); + FlxG.collide(enemies, walls); + enemies.forEachAlive(checkEnemyVision); + FlxG.overlap(player, enemies, playerTouchEnemy); } function playerTouchCoin(player:Player, coin:Coin) @@ -168,22 +108,14 @@ class PlayState extends FlxState { coin.kill(); money++; - hud.updateHUD(health, money); - coinSound.play(true); + hud.updateMoney(money); + FlxG.sound.play(AssetPaths.coin__wav); } } function checkEnemyVision(enemy:Enemy) { - if (walls.ray(enemy.getMidpoint(), player.getMidpoint())) - { - enemy.seesPlayer = true; - enemy.playerPosition = player.getMidpoint(); - } - else - { - enemy.seesPlayer = false; - } + enemy.checkVision(player, walls); } function playerTouchEnemy(player:Player, enemy:Enemy) @@ -196,13 +128,40 @@ class PlayState extends FlxState function startCombat(enemy:Enemy) { - inCombat = true; - player.active = false; - enemies.active = false; - combatHud.initCombat(health, enemy); - #if mobile virtualPad.visible = false; #end + + FlxG.sound.play(AssetPaths.combat__wav); + openSubState(new CombatSubState(player, enemy, (outcome)->handleCombatOutcome(outcome, enemy))); + } + + function handleCombatOutcome(outcome:CombatOutcome, enemy:Enemy) + { + hud.updateHealth(player.hp); + switch(outcome) + { + case VICTORY: + enemy.kill(); + if (enemy.type == BOSS) + { + fadeToGameOver(true); + } + case ESCAPED: + enemy.flicker(); + case DEFEAT: + player.alive = false; + + fadeToGameOver(false); + } + } + + function fadeToGameOver(won:Bool) + { + function onComplete() + { + FlxG.switchState(()->new GameOverState(won, money)); + } + FlxG.camera.fade(FlxColor.BLACK, 0.33, false, onComplete); } } diff --git a/Tutorials/TurnBasedRPG/source/ui/CombatSubState.hx b/Tutorials/TurnBasedRPG/source/ui/CombatSubState.hx new file mode 100644 index 000000000..c144676bf --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/ui/CombatSubState.hx @@ -0,0 +1,528 @@ +package ui; + +import flixel.FlxG; +import flixel.FlxSprite; +import flixel.group.FlxGroup; +import flixel.group.FlxSpriteGroup; +import flixel.math.FlxRect; +import flixel.text.FlxText; +import flixel.tweens.FlxEase; +import flixel.tweens.FlxTween; +import flixel.ui.FlxBar; +import flixel.util.FlxAxes; +import flixel.util.FlxColor; +import flixel.util.FlxTimer; +import flixel.util.FlxDirectionFlags; +import flixel.addons.display.FlxSliceSprite; +import flixel.addons.effects.chainable.FlxEffectSprite; +import flixel.addons.effects.chainable.FlxWaveEffect; + +import objects.Player; +import objects.Enemy; + +import openfl.filters.ColorMatrixFilter; +import openfl.geom.Matrix; +import openfl.geom.Point; + +enum CombatOutcome +{ + ESCAPED; + VICTORY; + DEFEAT; +} + +private enum Choice +{ + FIGHT; + FLEE; +} + +/** + * A SubState that pauses the main game to show the combat UI + */ +class CombatSubState extends flixel.FlxSubState +{ + /** The UI handling the actual battling */ + var ui:CombatUI; + + public function new (player:Player, enemy:Enemy, callback:(CombatOutcome)->Void) + { + super(); + + // Adds a neat wave effect to the game, this helps separate the game from the UI + var waveEffect = new CombatWaveEffect(4, 4); + add(waveEffect); + + /** Called when the UI determines an outcome of the battle */ + function outcomeReceived(outcome:CombatOutcome) + { + // disable the UI, tween it away, then pass the outcome to the callback + ui.active = false; + FlxTween.tween(ui, { y:FlxG.height }, 0.5, { ease: FlxEase.backIn, onComplete: + function (_) + { + // send the result to the play state and close + callback(outcome); + close(); + } + }); + waveEffect.fadeOut(0.5); + } + + // Create the UI + ui = new CombatUI(player, enemy, outcomeReceived); + ui.screenCenter(); + // ignore camera scroll to maintain screen position + ui.scrollFactor.set(0, 0); + add(ui); + + // store target y and then hide UI below the screen + var targetY = ui.y; + ui.y = FlxG.height; + + // disable the UI, tween it up from the bottom, then enable it + ui.active = false; + function onTweenComplete(tween:FlxTween) + { + ui.active = true; + } + FlxG.camera.flash(FlxColor.WHITE, .2); + FlxTween.tween(ui, { y:targetY }, 0.5, + { + startDelay: 0.2,// wait for flash + ease: FlxEase.backOut, + onComplete: onTweenComplete + } + ); + waveEffect.fadeIn(1.0); + } +} + +/** + * The UI that controls combat, extends `FlxSpriteGroup` which changes its members + * when certain properties are changed, namely `x`, `y`, `alpha` and `scrollFactor`. + */ +class CombatUI extends FlxSpriteGroup +{ + /** A function that handles the outcome of this match */ + var callback:(CombatOutcome)->Void; + + /** A reference to the player sprite in the main game */ + var player:Player; + /** A sprite representing our hero */ + var playerIcon:FlxSprite; + /** Used to show the result of attacks */ + var playerDmg:FlxText; + /** Displays the player's health */ + var playerHealthBar:FlxBar; + + /** A reference to our current opponent in the main game */ + var enemy:Enemy; + /** A sprite representing our enemy */ + var enemyIcon:FlxSprite; + /** Used to show the result of attacks */ + var enemyDmg:FlxText; + /** Displays the enemy's health */ + var enemyHealthBar:FlxBar; + + /** A group of texts that show damage or missed attacks */ + var attackResults:FlxTypedGroup; + + /** Shows the currently selected option (FIGHT or FLEE). */ + var pointer:FlxSprite; + /** Tracks which option is selected */ + var selected:Choice = FIGHT; + /** Contains text displaying our 2 options: FIGHT and FLEE */ + var choices:Map; + + public function new (player:Player, enemy:Enemy, callback:(CombatOutcome)->Void) + { + this.callback = callback; + this.player = player; + this.enemy = enemy; + super(); + + // Make a background to visially separate the ui from the game underneath + var bg = new FlxSliceSprite(AssetPaths.uiback__png, new FlxRect(16, 16, 16, 16), 128, 128); + // Stretch the sections rather than tiling for better performance + bg.stretchTop + = bg.stretchRight + = bg.stretchLeft + = bg.stretchBottom + = bg.stretchCenter + = true; + add(bg); + + final border = 6; + var section = new FlxSliceSprite(AssetPaths.ui_section__png, new FlxRect(16, 16, 16, 16), bg.width - border * 2, 72); + section.x = border; + section.y = bg.height - section.height - border; + // Stretch the sections rather than tiling for better performance + section.stretchTop + = bg.stretchRight + = bg.stretchLeft + = bg.stretchBottom + = section.stretchCenter + = true; + add(section); + + // Make a 'dummy' player and health bar + playerIcon = createAvatar(24, 8, player, player.maxHP); + playerIcon.facing = RIGHT; + + // Make a 'dummy' enemy and health bar + enemyIcon = createAvatar(80, 8, enemy, enemy.maxHP); + enemyIcon.facing = LEFT; + + attackResults = new FlxTypedGroup(); + for (i in 0...2) + { + var text = new FlxText(0, 0); + // Use same color as the pointer + text.color = 0xFFe5e5e5; + text.setBorderStyle(SHADOW, FlxColor.RED); + attackResults.add(text); + add(text); + text.kill(); + } + + // create our choices and add them to the group. + choices = new Map(); + choices[FIGHT] = new LargeText(40, 64, "FIGHT", 2); + choices[FLEE] = new LargeText(40, choices[FIGHT].y + choices[FIGHT].height + 10, "FLEE", 2); + add(choices[FIGHT]); + add(choices[FLEE]); + + pointer = new FlxSprite(16, choices[FIGHT].y + (choices[FIGHT].height / 2) - 8, AssetPaths.pointer__png); + add(pointer); + } + + /** + * Creates and adds an Icon and health bar for the target fighter + */ + function createAvatar(x:Float, y:Float, target:FlxSprite, maxHealth:Float) + { + // Create an "dummy" icon of the target + var icon = new FlxSprite(x, y); + // Use the target's graphic to have the same frames + icon.loadGraphicFromSprite(target); + icon.animation.play("lr_idle"); + + icon.setFacingFlip(LEFT, true, false); + icon.setFacingFlip(RIGHT, false, false); + add(icon); + + // create a health bar + var bar = new FlxBar(0, icon.y + icon.height + 2, LEFT_TO_RIGHT, 30, 10); + bar.createImageBar("assets/images/bar_empty.png", "assets/images/bar_filled.png", 0x0, 0x0); + bar.setRange(0, maxHealth); + // tracks the target's health automatically + bar.parent = target; + bar.parentVariable = "hp"; + bar.update(0);//redraw + // yellow bar, yellow border, red underneath + centerOn(bar, icon, X); + add(bar); + + return icon; + } + + override function update(elapsed:Float) + { + super.update(elapsed); + + // Only check input between turns when the pointer is showing + if (pointer.exists) + { + #if FLX_KEYBOARD + updateKeyboardInput(); + #end + + #if FLX_TOUCH + updateTouchInput(); + #end + } + } + + /** + * Call this function to place the pointer next to the currently selected choice + */ + inline function hilightChoice(choice:Choice) + { + selected = choice; + centerOn(pointer, choices[selected], Y); + } + + #if FLX_KEYBOARD + function updateKeyboardInput() + { + // Create helper func with shorter name + var justPressed = FlxG.keys.anyJustPressed; + var justReleased = FlxG.keys.anyJustReleased; + if (justPressed([SPACE, X, ENTER])) + { + FlxG.sound.play(AssetPaths.select__wav); + onChoose(selected); + } + // if the playerSprite presses up or down (but not both) + else if (justReleased([W, UP]) != justReleased([S, DOWN])) + { + FlxG.sound.play(AssetPaths.select__wav); + // Move the cursor up or down (with wrapping) + hilightChoice(selected == FIGHT ? FLEE : FIGHT); + } + } + #end + + #if FLX_TOUCH + function updateTouchInput() + { + for (touch in FlxG.touches.justReleased()) + { + for (choice in choices.keys()) + { + var text = choices[choice]; + if (touch.overlaps(text)) + { + FlxG.sound.play(AssetPaths.select__wav); + hilightChoice(choice); + onChoose(choice); + return; + } + } + } + } + #end + + function onChoose(selected:Choice) + { + // kill texts from last round + attackResults.killMembers(); + // hide our pointer + pointer.exists = false; + // check which item was selected when the playerSprite picked it + switch (selected) + { + case FIGHT: + playerAttack(); + + case FLEE: + // 50% chance to flee + if (FlxG.random.bool(50)) + { + // Success + FlxG.sound.play(AssetPaths.fled__wav); + showOutcome(ESCAPED); + } + else + { + // Failure, enemy gets a free attack + enemyAttack(); + } + } + } + + function showAttackResult(icon:FlxSprite, msg:String, ?onComplete:()->Void) + { + // recycle dead text + var text = attackResults.recycle(); + text.text = msg; + centerOn(text, icon, XY); + // text.alpha = 0; + + var callback = onComplete != null ? (_)->onComplete() : null; + FlxTween.tween(text, { alpha: 2 }, 0.5, { ease: FlxEase.circOut, onComplete: callback }); + FlxTween.tween(text, { y: text.y - 12 }, 1, { ease: FlxEase.circOut, onComplete:(_)->text.kill() }); + } + + function playerAttack() + { + // Move the player towards the enemy then back + FlxTween.tween(playerIcon, {x: playerIcon.x + 12}, 0.1) + .then(FlxTween.tween(playerIcon, {x: playerIcon.x}, 0.1)); + + var result:String; + // 85% chance for player to hit + if (FlxG.random.bool(85)) + { + // Success + FlxG.sound.play(AssetPaths.hurt__wav); + // Move the enemy, then move it back + FlxTween.tween(enemyIcon, {x: enemyIcon.x + 4}, 0.1, {startDelay: 0.1}) + .then(FlxTween.tween(enemyIcon, {x: enemyIcon.x}, 0.1)); + + // Deal 1 damage to the enemy + enemy.hurt(1); + result = "1"; + } + else + { + // We missed + FlxG.sound.play(AssetPaths.miss__wav); + result = "MISS!"; + } + + // Show the attack result then it's the enemy's turn + if (enemy.hp > 0) + { + showAttackResult(enemyIcon, result, enemyAttack); + } + else + { + showAttackResult(enemyIcon, result, roundEnd); + } + } + + function enemyAttack() + { + // Move the enemy towards the player, then move it back + FlxTween.tween(enemyIcon, {x: enemyIcon.x - 12}, 0.1) + .then(FlxTween.tween(enemyIcon, {x: enemyIcon.x}, 0.1)); + + var result:String; + // 30% chance to hit + if (FlxG.random.bool(30)) + { + // Flash the screen white + FlxG.camera.flash(FlxColor.WHITE, .2); + FlxG.camera.shake(0.01, 0.2); + + // Move the enemy, then move it back + FlxTween.tween(playerIcon, {x: playerIcon.x - 4}, 0.1, {startDelay: 0.1}) + .then(FlxTween.tween(playerIcon, {x: playerIcon.x}, 0.1)); + // Deal 1 damage + player.hurt(1); + FlxG.sound.play(AssetPaths.hurt__wav); + result = "1"; + } + else + { + // Missed + FlxG.sound.play(AssetPaths.miss__wav); + result = "MISS!"; + } + + showAttackResult(playerIcon, result, roundEnd); + } + + function roundEnd() + { + if (player.hp <= 0) + { + showOutcome(DEFEAT); + } + else if (enemy.hp <= 0) + { + showOutcome(VICTORY); + } + else + { + // Enables UI for net turn + pointer.exists = true; + } + } + + function showOutcome(outcome:CombatOutcome) + { + var text = new LargeText(0, 0, outcome.getName(), 3); + text.color = FlxColor.YELLOW; + text.setBorderStyle(SHADOW, FlxColor.GRAY); + // Adding a sprite to a sprite group will change the x/y by the group's, so add first + add(text); + text.screenCenter(); + + // Store desired y, then tween to it + var targetY = text.y; + text.y = -text.height; + FlxTween.tween(text, { y: targetY }, 1, { ease: FlxEase.backOut, onComplete: + function (_) + { + // Hold it there for a sec, then start the outro + FlxTimer.wait(1.0, ()->callback(outcome)); + } + }); + } + + /** + * Centers the first sprite's so it's center x mathes the center x of the target + */ + static inline function centerOn(sprite:FlxSprite, target:FlxSprite, axes:FlxAxes = XY) + { + if (axes.x) + { + sprite.x = target.x + (target.width - sprite.width) / 2; + } + + if (axes.y) + { + sprite.y = target.y + (target.height - sprite.height) / 2; + } + } +} + +/** + * Helper class that handles the wave effect + */ +private class CombatWaveEffect extends FlxEffectSprite +{ + /** + * [Description] + * @param strength How strong you want the effect + * @param speed How fast you want the effect to move, higher values = faster + */ + public function new(strength = 10, speed = 3.0) + { + var screen = new FlxSprite().makeGraphic(FlxG.width, FlxG.height, FlxColor.TRANSPARENT); + super(screen, [new FlxWaveEffect(FlxWaveMode.ALL, strength, -1, speed)]); + scrollFactor.set(0, 0); + + // Draw the camera to a bitmap + screen.drawFrame(); + var screenPixels = screen.framePixels; + if (FlxG.renderBlit) + screenPixels.copyPixels(FlxG.camera.buffer, FlxG.camera.buffer.rect, new Point()); + else + screenPixels.draw(FlxG.camera.canvas, new Matrix(1, 0, 0, 1, 0, 0)); + + // Apply desaturating color matrix + var rc:Float = 1 / 3; + var gc:Float = 1 / 2; + var bc:Float = 1 / 6; + screenPixels.applyFilter(screenPixels, screenPixels.rect, new Point(), + new ColorMatrixFilter([rc, gc, bc, 0, 0, rc, gc, bc, 0, 0, rc, gc, bc, 0, 0, 0, 0, 0, 1, 0])); + } + + /** + * Tweens the `alpha` and `strength` from 0 + * @param time The duration of the fade + */ + public function fadeIn(time:Float) + { + var waveEffect:FlxWaveEffect = cast this.effects[0]; + var strength = waveEffect.strength; + // Fade effect in + // alpha = 0; + FlxTween.num(0.0, 1.0, time, (n)-> + { + alpha = n; + waveEffect.strength = Math.floor(strength * FlxEase.circOut(Math.max(n - 0.5, 0) * 2)); + }); + } + + /** + * Tweens the `alpha` and `strength` to 0 + * @param time The duration of the fade + */ + public function fadeOut(time:Float) + { + var waveEffect:FlxWaveEffect = cast this.effects[0]; + var strength = waveEffect.strength; + // Fade effect in + // alpha = 1.0; + FlxTween.num(1.0, 0.0, time, (n)-> + { + alpha = n; + waveEffect.strength = Math.floor(strength * FlxEase.circOut(Math.max(n - 0.5, 0) * 2)); + }); + } +} \ No newline at end of file diff --git a/Tutorials/TurnBasedRPG/source/ui/HUD.hx b/Tutorials/TurnBasedRPG/source/ui/HUD.hx new file mode 100644 index 000000000..b240d19a1 --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/ui/HUD.hx @@ -0,0 +1,52 @@ +package ui; + +import flixel.FlxG; +import flixel.FlxSprite; +import flixel.group.FlxGroup.FlxTypedGroup; +import flixel.text.FlxText; +import flixel.util.FlxColor; +import objects.Coin; + +using flixel.util.FlxSpriteUtil; + +class HUD extends FlxTypedGroup +{ + var healthCounter:FlxText; + var moneyCounter:FlxText; + + public function new() + { + super(); + + var healthIcon = new FlxSprite(4, 4, AssetPaths.health__png); + healthCounter = new FlxText(0, 0, 0, "3 / 3", 8); + healthCounter.setBorderStyle(SHADOW, FlxColor.GRAY, 1, 1); + healthCounter.x = healthIcon.x + healthIcon.width; + healthCounter.y = healthIcon.y + (healthIcon.height - healthCounter.height) / 2; + add(healthIcon); + add(healthCounter); + + var moneyIcon = new Coin(0, 4); + moneyIcon.solid = false; + moneyCounter = new FlxText(0, 0, 0, "00", 8); + moneyCounter.setBorderStyle(SHADOW, FlxColor.GRAY, 1, 1); + moneyIcon.x = FlxG.width - 4 - moneyIcon.width - moneyCounter.width; + moneyCounter.x = moneyIcon.x + moneyIcon.width; + moneyCounter.y = moneyIcon.y + (moneyIcon.height - moneyCounter.height) / 2; + moneyCounter.text = "0"; + add(moneyIcon); + add(moneyCounter); + + forEach(function(sprite) sprite.scrollFactor.set(0, 0)); + } + + public function updateHealth(health:Int) + { + healthCounter.text = health + " / 3"; + } + + public function updateMoney(money:Int) + { + moneyCounter.text = Std.string(money); + } +} diff --git a/Tutorials/TurnBasedRPG/source/ui/LargeText.hx b/Tutorials/TurnBasedRPG/source/ui/LargeText.hx new file mode 100644 index 000000000..dd3f87120 --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/ui/LargeText.hx @@ -0,0 +1,64 @@ +package ui; + +import flixel.FlxG; +import flixel.math.FlxPoint; +import flixel.math.FlxRect; +import flixel.text.FlxBitmapFont; +import flixel.text.FlxBitmapText; + +/** + * Note: This is not in the tutorial, the tutorial uses FlxText everywhere, which is easier to teach. + * Feel free to use this in your games + * Font created by Rick Hoppmann: https://tinyworlds.itch.io/free-pixel-font-thaleah + */ +@:forward +abstract LargeText(FlxBitmapText) to FlxBitmapText +{ + static function getDefaultFont() + { + final graphic = FlxG.bitmap.add(AssetPaths.font__png); + final font = FlxBitmapFont.findFont(graphic.imageFrame.frame); + if (font != null) + { + return font; + } + + final chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ!"; + final widths = ['I'.code=>3, '!'.code=>3, 'T'.code=>7]; + final defaultWidth = 8; + final defaultHeight = 8; + final font = FlxBitmapFont.fromMonospace(graphic, "", new FlxPoint(defaultWidth, defaultHeight)); + + var x:Int = 0; + for (i in 0...chars.length) + { + final charCode = chars.charCodeAt(i); + final width = if (widths.exists(charCode)) widths[charCode] else defaultWidth; + final frame = FlxRect.get(x, 0, width, defaultHeight); + @:privateAccess + font.addCharFrame(charCode, frame, FlxPoint.weak(), width); + x += width; + } + return font; + } + + public function new (x = 0.0, y = 0.0, text = "", scale = 4) + { + this = new FlxBitmapText(x, y, text, getDefaultFont()); + this.text = text; + this.autoUpperCase = true; + setScale(scale); + } + + function setScale(scale:Int) + { + this.scale.set(scale, scale); + this.updateHitbox(); + } + + inline public function setBorderStyle(style, color = 0x0, size = 1, quality = 1) + { + this.setBorderStyle(style, color, size, quality); + this.updateHitbox(); + } +} \ No newline at end of file diff --git a/Tutorials/TurnBasedRPG/source/ui/OptionsSubState.hx b/Tutorials/TurnBasedRPG/source/ui/OptionsSubState.hx new file mode 100644 index 000000000..a17242f2f --- /dev/null +++ b/Tutorials/TurnBasedRPG/source/ui/OptionsSubState.hx @@ -0,0 +1,272 @@ +package ui; + +import flixel.FlxSprite; +import flixel.FlxG; +import flixel.addons.display.FlxSliceSprite; +import flixel.group.FlxGroup; +import flixel.group.FlxSpriteGroup; +import flixel.math.FlxRect; +import flixel.text.FlxText; +import flixel.ui.FlxBar; +import flixel.ui.FlxButton; +import flixel.util.FlxColor; + +/** + * A SubState that pauses the main menu to show the options UI + */ +class OptionsSubState extends flixel.FlxSubState +{ + public function new () + { + super(); + + // black-out the main menu + var back = new FlxSprite(); + back.makeGraphic(1, 1, 0x80000000); + back.setGraphicSize(FlxG.width, FlxG.height); + back.updateHitbox(); + add(back); + + // add the UI + var ui = new OptionsUI(()->close()); + add(ui); + } +} + +/** + * The UI that controls options, extends `FlxSpriteGroup` which changes its members + * when certain properties are changed, namely `x`, `y`, `alpha` and `scrollFactor`. + */ +class OptionsUI extends FlxGroup +{ + + // define our screen elements + var musicBar:VolumeBar; + var soundBar:VolumeBar; + var fullscreenButton:FlxButton; + + public function new (onClose:()->Void) + { + super(); + + if (FlxG.save.data.volumes == null) + { + initSave(); + } + + // Make a background to visially separate the ui from the game underneath + var bg = new FlxSliceSprite(AssetPaths.uiback__png, new FlxRect(16, 16, 16, 16), 200, 160); + // Stretch the sections rather than tiling them for better performance + bg.stretchTop + = bg.stretchRight + = bg.stretchLeft + = bg.stretchBottom + = bg.stretchCenter + = true; + bg.screenCenter(XY); + add(bg); + + // place the group in the screen center, this will move the bg, and anything added later + + var titleText = new LargeText(bg.x, bg.y + 4, "Options", 2); + titleText.screenCenter(X); + add(titleText); + + var gap = 8; + var barX = bg.x + gap; + var barY = titleText.y + titleText.height + gap; + var barWidth = bg.width - gap * 2; + + musicBar = new VolumeBar(barX, barY, barWidth, "Music", FlxG.sound.defaultMusicGroup.volume, updateMusic); + add(musicBar); + barY += musicBar.height + gap; + + soundBar = new VolumeBar(barX, barY, barWidth, "Sound", FlxG.sound.defaultSoundGroup.volume, updateSound); + add(soundBar); + // barY += soundBar.height + gap; + + fullscreenButton = new Button(0, soundBar.y + soundBar.height + 8, FlxG.fullscreen ? "Windowed" : "Fullscreen", clickFullscreen); + fullscreenButton.screenCenter(X); + add(fullscreenButton); + + var clearDataButton = new Button(bg.x + 10, 0, "Clear Data", clickClearData); + clearDataButton.y = bg.y + bg.height - clearDataButton.height - 10; + add(clearDataButton); + + var backButton = new Button(0, clearDataButton.y, "Back", onClose); + backButton.x = bg.x + bg.width - backButton.width - 10; + add(backButton); + } + + function clickFullscreen() + { + fullscreenButton.text = FlxG.fullscreen ? "Windowed" : "Fullscreen"; + FlxG.fullscreen = !FlxG.fullscreen; + FlxG.save.data.fullscreen = FlxG.fullscreen; + FlxG.save.flush(); + } + + function initSave() + { + FlxG.save.data.volumes = + { + music: FlxG.sound.defaultMusicGroup.volume, + sound: FlxG.sound.defaultSoundGroup.volume + }; + FlxG.sound.muted = false; + FlxG.sound.volume = 1.0; + FlxG.save.data.fullscreen = FlxG.fullscreen; + FlxG.save.flush(); + trace(FlxG.save.data); + } + + /** + * The user wants to clear the saved data - we just call erase on our save object and then reset the volume to .5 + */ + function clickClearData() + { + FlxG.save.erase(); + initSave(); + musicBar.setVolume(0.5, true); + soundBar.setVolume(1.0, true); + } + + /** + * Whenever we want to show the value of volume, we call this to change the bar and the amount text + */ + function updateMusic(volume:Float) + { + FlxG.sound.defaultMusicGroup.volume = volume; + FlxG.save.data.volumes.music = volume; + FlxG.save.flush(); + } + + /** + * Whenever we want to show the value of volume, we call this to change the bar and the amount text + */ + function updateSound(volume:Float) + { + FlxG.sound.defaultSoundGroup.volume = volume; + FlxG.save.data.volumes.sound = volume; + FlxG.save.flush(); + } +} + +class VolumeBar extends FlxSpriteGroup +{ + var bar:FlxBar; + var amountText:FlxText; + var label:String; + var onChange:(amount:Float)->Void; + + /** + * + * @param x + * @param y + * @param width + * @param label + * @param onChange + * @return + */ + public function new (x:Float, y:Float, width:Float, label:String, volume:Float, onChange:(amount:Float)->Void) + { + this.label = label; + this.onChange = onChange; + super(); + + // var label = new FlxText(0, 0, 0, label, 8); + // label.x = (width - label.width) / 2; + // add(label); + + // the volume buttons will be smaller than 'default' buttons + var downButton = new SmallButton(0, 0, "-", clickDown); + add(downButton); + + var upButton = new SmallButton(0, downButton.y, "+", clickUp); + upButton.x = width - upButton.width; + add(upButton); + + var barWidth = Std.int(width - (4 + upButton.width) * 2); + var barHeight = Std.int(upButton.height); + bar = new FlxBar(downButton.x + downButton.width + 4, downButton.y, LEFT_TO_RIGHT, barWidth, barHeight); + bar.createFilledBar(0xff464646, FlxColor.WHITE, true, FlxColor.WHITE); + add(bar); + + amountText = new FlxText(0, 0, 200, "100%", 8); + amountText.alignment = CENTER; + amountText.setBorderStyle(OUTLINE, 0xff464646); + amountText.x = bar.x + (bar.width - amountText.width) / 2; + amountText.y = bar.y + (bar.height - amountText.height) / 2; + add(amountText); + + // + this.x = x; + this.y = y; + setVolume(volume); + } + + function clickDown() + { + setVolumeHelper(bar.value - 10); + } + + function clickUp() + { + setVolumeHelper(bar.value + 10); + } + + function setVolumeHelper(volume:Float, dispatch = true) + { + bar.value = Math.round(volume); // Note: bar.value is automatically clamped between 0 and 100 + amountText.text = label + " " + bar.value + "%"; + + if (dispatch) + { + onChange(bar.value / 100); + } + } + + public function setVolume(volume:Float, dispatch = false) + { + setVolumeHelper(volume * 100, dispatch); + } + + override function destroy() + { + // remove references but do not destroy + bar = null; + amountText = null; + onChange = null; + + // this will actually destroy them + super.destroy(); + } +} + +/** + * Helper class for creating an 80x20 button with a custom graphic + */ +class Button extends FlxButton +{ + public function new (x, y, label, onClick) + { + super(x, y, label, onClick); + + loadGraphic(AssetPaths.button__png, true, 80, 20); + onUp.sound = FlxG.sound.load(AssetPaths.select__wav); + } +} + +/** + * Helper class for creating small 20x20 buttons + */ +class SmallButton extends FlxButton +{ + public function new (x, y, label, onClick) + { + super(x, y, label, onClick); + + loadGraphic(AssetPaths.small_button__png, true, 20, 20); + onUp.sound = FlxG.sound.load(AssetPaths.select__wav); + } +} \ No newline at end of file