kaggle-environments 1.17.12__py2.py3-none-any.whl → 1.18.0__py2.py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of kaggle-environments might be problematic. Click here for more details.
- kaggle_environments/__init__.py +1 -1
- kaggle_environments/api.py +5 -13
- kaggle_environments/envs/cabt/cabt.js +164 -0
- kaggle_environments/envs/cabt/cabt.json +28 -0
- kaggle_environments/envs/cabt/cabt.py +119 -0
- kaggle_environments/envs/cabt/cg/__init__.py +0 -0
- kaggle_environments/envs/cabt/cg/cg.dll +0 -0
- kaggle_environments/envs/cabt/cg/game.py +70 -0
- kaggle_environments/envs/cabt/cg/libcg.so +0 -0
- kaggle_environments/envs/cabt/cg/sim.py +44 -0
- kaggle_environments/envs/open_spiel/games/chess/chess.js +25 -22
- kaggle_environments/envs/open_spiel/open_spiel.py +53 -1
- kaggle_environments/envs/open_spiel/test_open_spiel.py +85 -1
- kaggle_environments/helpers.py +126 -86
- kaggle_environments/main.py +29 -44
- kaggle_environments/static/player.html +71 -31
- kaggle_environments/utils.py +8 -12
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/METADATA +2 -71
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/RECORD +23 -15
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/WHEEL +1 -1
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/entry_points.txt +0 -0
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/licenses/LICENSE +0 -0
- {kaggle_environments-1.17.12.dist-info → kaggle_environments-1.18.0.dist-info}/top_level.txt +0 -0
kaggle_environments/__init__.py
CHANGED
|
@@ -20,7 +20,7 @@ from .core import *
|
|
|
20
20
|
from .main import http_request
|
|
21
21
|
from . import errors
|
|
22
22
|
|
|
23
|
-
__version__ = "1.
|
|
23
|
+
__version__ = "1.18.0"
|
|
24
24
|
|
|
25
25
|
__all__ = ["Agent", "environments", "errors", "evaluate", "http_request",
|
|
26
26
|
"make", "register", "utils", "__version__",
|
kaggle_environments/api.py
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import requests
|
|
2
2
|
|
|
3
|
-
from typing import
|
|
3
|
+
from typing import List
|
|
4
4
|
|
|
5
5
|
base_url = "https://www.kaggle.com/requests/EpisodeService/"
|
|
6
6
|
get_url = base_url + "GetEpisodeReplay"
|
|
@@ -8,30 +8,22 @@ list_url = base_url + "ListEpisodes"
|
|
|
8
8
|
|
|
9
9
|
|
|
10
10
|
def get_episode_replay(episode_id: int):
|
|
11
|
-
body = {
|
|
12
|
-
"EpisodeId": episode_id
|
|
13
|
-
}
|
|
11
|
+
body = {"EpisodeId": episode_id}
|
|
14
12
|
|
|
15
13
|
response = requests.post(get_url, json=body)
|
|
16
14
|
return response.json()
|
|
17
15
|
|
|
18
16
|
|
|
19
17
|
def list_episodes(episode_ids: List[int]):
|
|
20
|
-
return __list_episodes({
|
|
21
|
-
"Ids": episode_ids
|
|
22
|
-
})
|
|
18
|
+
return __list_episodes({"Ids": episode_ids})
|
|
23
19
|
|
|
24
20
|
|
|
25
21
|
def list_episodes_for_team(team_id: int):
|
|
26
|
-
return __list_episodes({
|
|
27
|
-
"TeamId": team_id
|
|
28
|
-
})
|
|
22
|
+
return __list_episodes({"TeamId": team_id})
|
|
29
23
|
|
|
30
24
|
|
|
31
25
|
def list_episodes_for_submission(submission_id: int):
|
|
32
|
-
return __list_episodes({
|
|
33
|
-
"SubmissionId": submission_id
|
|
34
|
-
})
|
|
26
|
+
return __list_episodes({"SubmissionId": submission_id})
|
|
35
27
|
|
|
36
28
|
|
|
37
29
|
def __list_episodes(body):
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
|
|
2
|
+
function renderer(context) {
|
|
3
|
+
const step = context.step
|
|
4
|
+
const visList = context.environment.steps[0][0].visualize
|
|
5
|
+
const energyText = "CGRWLPFDM A"
|
|
6
|
+
|
|
7
|
+
const info = context.environment.info
|
|
8
|
+
const players = [
|
|
9
|
+
info?.TeamNames?.[0] || "Player 0",
|
|
10
|
+
info?.TeamNames?.[1] || "Player 1"
|
|
11
|
+
]
|
|
12
|
+
|
|
13
|
+
let canvas = context.parent.querySelector("canvas")
|
|
14
|
+
if (!canvas) {
|
|
15
|
+
container = document.createElement("div")
|
|
16
|
+
container.style.position = "relative"
|
|
17
|
+
context.parent.appendChild(container)
|
|
18
|
+
|
|
19
|
+
canvas = document.createElement("canvas")
|
|
20
|
+
canvas.width = 750
|
|
21
|
+
canvas.height = 700
|
|
22
|
+
container.appendChild(canvas)
|
|
23
|
+
|
|
24
|
+
if (visList) {
|
|
25
|
+
for (let k = 0; k < 2; k++) {
|
|
26
|
+
const button = document.createElement("button")
|
|
27
|
+
button.style.width = "120px"
|
|
28
|
+
button.style.height = "50px"
|
|
29
|
+
button.style.left = k == 0 ? "240px" : "380px"
|
|
30
|
+
button.style.top = "10px"
|
|
31
|
+
button.style.position = "absolute"
|
|
32
|
+
button.style.zIndex = 1
|
|
33
|
+
button.innerHTML = "Open Visualizer<br>" + players[k]
|
|
34
|
+
button.addEventListener("click", (e) => {
|
|
35
|
+
for (let i = 0; i < visList.length; i++) {
|
|
36
|
+
for (let j = 0; j < 2; j++) {
|
|
37
|
+
visList[i].current.players[j].ramainingTime = context.environment.steps[i][j].observation.remainingOverageTime
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
visList[0].ps = players
|
|
41
|
+
|
|
42
|
+
const input = document.createElement("input")
|
|
43
|
+
input.type = "hidden"
|
|
44
|
+
input.name = "json"
|
|
45
|
+
input.value = JSON.stringify(visList)
|
|
46
|
+
|
|
47
|
+
const form = document.createElement("form")
|
|
48
|
+
form.method = "POST"
|
|
49
|
+
form.action = "https://ptcgvis.heroz.jp/Visualizer/Replay/"
|
|
50
|
+
if (info.EpisodeId == null) {
|
|
51
|
+
form.action += k
|
|
52
|
+
} else {
|
|
53
|
+
form.action += info.EpisodeId + "/" + k
|
|
54
|
+
}
|
|
55
|
+
form.target = "_blank"
|
|
56
|
+
form.appendChild(input)
|
|
57
|
+
|
|
58
|
+
document.body.appendChild(form)
|
|
59
|
+
form.submit()
|
|
60
|
+
})
|
|
61
|
+
container.appendChild(button)
|
|
62
|
+
}
|
|
63
|
+
} else {
|
|
64
|
+
const ctx = canvas.getContext("2d")
|
|
65
|
+
ctx.strokeStyle = "#ccc"
|
|
66
|
+
ctx.fillStyle = "#fff"
|
|
67
|
+
ctx.font = "30px sans-serif"
|
|
68
|
+
ctx.fillText("No visualize data.", 10, 100)
|
|
69
|
+
const error = context.environment.steps[0][0].error
|
|
70
|
+
if (error) {
|
|
71
|
+
ctx.fillText(error, 10, 150)
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
if (visList.length <= step) {
|
|
77
|
+
return
|
|
78
|
+
}
|
|
79
|
+
const vis = visList[step]
|
|
80
|
+
const state = vis.current
|
|
81
|
+
|
|
82
|
+
const ctx = canvas.getContext("2d")
|
|
83
|
+
ctx.clearRect(0, 0, canvas.width, canvas.height)
|
|
84
|
+
|
|
85
|
+
ctx.strokeStyle = "#ccc"
|
|
86
|
+
ctx.fillStyle = "#fff"
|
|
87
|
+
ctx.lineWidth = 2
|
|
88
|
+
|
|
89
|
+
ctx.font = "20px sans-serif"
|
|
90
|
+
if (state.result >= 0) {
|
|
91
|
+
if (state.result == 2) {
|
|
92
|
+
ctx.fillText("Draw", 330, 70)
|
|
93
|
+
} else {
|
|
94
|
+
ctx.fillText(players[state.result] + " Win", 310, 120)
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
ctx.font = "12px sans-serif"
|
|
99
|
+
|
|
100
|
+
const drawCard = (x, y, card) => {
|
|
101
|
+
ctx.beginPath()
|
|
102
|
+
ctx.rect(x, y, 80, 60)
|
|
103
|
+
ctx.stroke()
|
|
104
|
+
nm = card.name
|
|
105
|
+
nm2 = null
|
|
106
|
+
if (nm.length >= 13) {
|
|
107
|
+
for (let i = 0; i < nm.length; i++) {
|
|
108
|
+
if (nm[i] == " ") {
|
|
109
|
+
nm2 = nm.substring(i + 1)
|
|
110
|
+
nm = nm.substring(0, i)
|
|
111
|
+
break
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
ctx.fillText(nm, x + 5, y + 13)
|
|
116
|
+
if (nm2 != null) {
|
|
117
|
+
ctx.fillText(nm2, x + 5, y + 27)
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
const drawField = (x, y, card) => {
|
|
121
|
+
drawCard(x, y, card)
|
|
122
|
+
ctx.fillText("HP " + card.hp, x + 5, y + 41)
|
|
123
|
+
energy = ""
|
|
124
|
+
for (let e of card.energies) {
|
|
125
|
+
energy = energy + energyText[e]
|
|
126
|
+
}
|
|
127
|
+
ctx.fillText(energy, x + 5, y + 55)
|
|
128
|
+
}
|
|
129
|
+
const posY = (index, len) => {
|
|
130
|
+
const center = 290
|
|
131
|
+
let height
|
|
132
|
+
if (len <= 8) {
|
|
133
|
+
height = 35 * len
|
|
134
|
+
} else {
|
|
135
|
+
height = 280
|
|
136
|
+
}
|
|
137
|
+
return center + height * (2 * index + 1 - len) / len
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
for (let j = 0; j < state.stadium.length; j++) {
|
|
141
|
+
drawCard(330, 420, state.stadium[j])
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
for (let i = 0; i < 2; i++) {
|
|
145
|
+
const ps = state.players[i]
|
|
146
|
+
|
|
147
|
+
ctx.fillText("Active", i == 0 ? 245 : 425, 270)
|
|
148
|
+
ctx.fillText("Bench", i == 0 ? 145 : 525, 10)
|
|
149
|
+
ctx.fillText("Hand", i == 0 ? 15 : 655, 10)
|
|
150
|
+
ctx.fillText("Deck " + ps.deckCount, i == 0 ? 258 : 438, 150)
|
|
151
|
+
ctx.fillText("Discard " + ps.discard.length, i == 0 ? 245 : 425, 170)
|
|
152
|
+
ctx.fillText("Prize " + ps.prize.length, i == 0 ? 258 : 438, 200)
|
|
153
|
+
|
|
154
|
+
for (let j = 0; j < ps.active.length; j++) {
|
|
155
|
+
drawField(i == 0 ? 240 : 420, posY(j, ps.active.length), ps.active[j])
|
|
156
|
+
}
|
|
157
|
+
for (let j = 0; j < ps.bench.length; j++) {
|
|
158
|
+
drawField(i == 0 ? 140 : 520, posY(j, ps.bench.length), ps.bench[j])
|
|
159
|
+
}
|
|
160
|
+
for (let j = 0; j < ps.hand.length; j++) {
|
|
161
|
+
drawCard(i == 0 ? 10 : 650, posY(j, ps.hand.length), ps.hand[j])
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "cabt",
|
|
3
|
+
"title": "Card Battle",
|
|
4
|
+
"description": "Limited Card Battle.",
|
|
5
|
+
"version": "1.0.0",
|
|
6
|
+
"agents": [2],
|
|
7
|
+
"configuration": {
|
|
8
|
+
"episodeSteps": 10000,
|
|
9
|
+
"actTimeout": 0,
|
|
10
|
+
"runTimeout": 3000
|
|
11
|
+
},
|
|
12
|
+
"reward": {
|
|
13
|
+
"description": "Lost:-1, Won:1, Draw:0",
|
|
14
|
+
"enum": [-1, 0, 1],
|
|
15
|
+
"default": 0
|
|
16
|
+
},
|
|
17
|
+
"observation": {
|
|
18
|
+
"remainingOverageTime": 600
|
|
19
|
+
},
|
|
20
|
+
"action": {
|
|
21
|
+
"description": "List of option index.",
|
|
22
|
+
"type": "array",
|
|
23
|
+
"default": []
|
|
24
|
+
},
|
|
25
|
+
"status": {
|
|
26
|
+
"defaults": ["INACTIVE", "INACTIVE"]
|
|
27
|
+
}
|
|
28
|
+
}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import random
|
|
2
|
+
import os
|
|
3
|
+
import json
|
|
4
|
+
|
|
5
|
+
from .cg.sim import Battle
|
|
6
|
+
from .cg.game import battle_start, battle_finish, battle_select, visualize_data
|
|
7
|
+
|
|
8
|
+
deck = [5,5,5,5,5,5,5,5,5,5,9,9,77,77,77,77,156,156,156,156,157,157,157,157,331,331,331,331,408,408,408,408,474,474,474,474,528,528,528,528,530,530,530,530,532,554,554,554,576,576,576,576,585,585,585,585,630,630,630,630]
|
|
9
|
+
|
|
10
|
+
def random_agent(obs: dict) -> list[int]:
|
|
11
|
+
if obs["select"] == None:
|
|
12
|
+
return deck
|
|
13
|
+
return random.sample(list(range(len(obs["select"]["option"]))), obs["select"]["maxCount"])
|
|
14
|
+
|
|
15
|
+
def first_agent(obs: dict) -> list[int]:
|
|
16
|
+
if obs["select"] == None:
|
|
17
|
+
return deck
|
|
18
|
+
return list(range(obs["select"]["maxCount"]))
|
|
19
|
+
|
|
20
|
+
agents = {"random": random_agent, "first": first_agent}
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def finish(env):
|
|
24
|
+
if len(env.steps) > 0:
|
|
25
|
+
env.steps[0][0]["visualize"] = json.loads(visualize_data())
|
|
26
|
+
battle_finish()
|
|
27
|
+
|
|
28
|
+
def interpreter(state, env):
|
|
29
|
+
if env.done:
|
|
30
|
+
Battle.battle_ptr = None
|
|
31
|
+
for i in range(2):
|
|
32
|
+
state[i].status = "ACTIVE"
|
|
33
|
+
o = state[i].observation
|
|
34
|
+
o["select"] = None
|
|
35
|
+
o["logs"] = []
|
|
36
|
+
o["current"] = None
|
|
37
|
+
o["search_begin_input"] = None
|
|
38
|
+
return state
|
|
39
|
+
elif Battle.battle_ptr == None:
|
|
40
|
+
decks = [state[0].action, state[1].action]
|
|
41
|
+
error = False
|
|
42
|
+
for i in range(2):
|
|
43
|
+
if state[i].status == "TIMEOUT" or state[i].status == "ERROR":
|
|
44
|
+
error = True
|
|
45
|
+
continue
|
|
46
|
+
if len(decks[i]) != 60:
|
|
47
|
+
state[i].status = "INVALID"
|
|
48
|
+
env.steps[0][0]["error"] = f"Player {i}'s deck does not have 60 cards."
|
|
49
|
+
error = True
|
|
50
|
+
if not error:
|
|
51
|
+
_, start_data = battle_start(state[0].action, state[1].action)
|
|
52
|
+
if start_data.errorPlayer >= 0:
|
|
53
|
+
state[start_data.errorPlayer].status = "INVALID"
|
|
54
|
+
env.steps[0][0]["error"] = f"Player {i}'s deck error."
|
|
55
|
+
error = True
|
|
56
|
+
if error:
|
|
57
|
+
for i in range(2):
|
|
58
|
+
if state[i].status == "ACTIVE":
|
|
59
|
+
state[i].status = "DONE"
|
|
60
|
+
return state
|
|
61
|
+
if Battle.battle_ptr == None:
|
|
62
|
+
raise ValueError("battle_ptr None.")
|
|
63
|
+
else:
|
|
64
|
+
error = False
|
|
65
|
+
select_player = Battle.obs["current"]["yourIndex"]
|
|
66
|
+
if state[select_player].status == "TIMEOUT" or state[select_player].status == "ERROR":
|
|
67
|
+
error = True
|
|
68
|
+
else:
|
|
69
|
+
try:
|
|
70
|
+
battle_select(state[select_player].action)
|
|
71
|
+
except:
|
|
72
|
+
state[select_player].status = "INVALID"
|
|
73
|
+
error = True
|
|
74
|
+
|
|
75
|
+
if error:
|
|
76
|
+
state[select_player].reward = -1
|
|
77
|
+
state[1 - select_player].status = "DONE"
|
|
78
|
+
state[1 - select_player].reward = 1
|
|
79
|
+
finish(env)
|
|
80
|
+
return state
|
|
81
|
+
|
|
82
|
+
obs = Battle.obs
|
|
83
|
+
s = obs["current"]
|
|
84
|
+
if s["result"] >= 0:
|
|
85
|
+
state[0].status = "DONE"
|
|
86
|
+
state[1].status = "DONE"
|
|
87
|
+
if s["result"] == 0:
|
|
88
|
+
state[0].reward = 1
|
|
89
|
+
state[1].reward = -1
|
|
90
|
+
elif s["result"] == 1:
|
|
91
|
+
state[0].reward = -1
|
|
92
|
+
state[1].reward = 1
|
|
93
|
+
else:
|
|
94
|
+
state[0].reward = 0
|
|
95
|
+
state[1].reward = 0
|
|
96
|
+
finish(env)
|
|
97
|
+
else:
|
|
98
|
+
index = s["yourIndex"]
|
|
99
|
+
state[index].status = "ACTIVE"
|
|
100
|
+
state[1 - index].status = "INACTIVE"
|
|
101
|
+
o = state[index].observation
|
|
102
|
+
o["select"] = obs["select"]
|
|
103
|
+
o["logs"] = obs["logs"]
|
|
104
|
+
o["current"] = obs["current"]
|
|
105
|
+
o["search_begin_input"] = obs["search_begin_input"]
|
|
106
|
+
return state
|
|
107
|
+
|
|
108
|
+
def renderer(state, env):
|
|
109
|
+
return json.dumps(Battle.obs)
|
|
110
|
+
|
|
111
|
+
def html_renderer():
|
|
112
|
+
jspath = os.path.abspath(os.path.join(os.path.dirname(__file__), "cabt.js"))
|
|
113
|
+
with open(jspath, encoding="utf-8") as f:
|
|
114
|
+
return f.read()
|
|
115
|
+
|
|
116
|
+
|
|
117
|
+
jsonpath = os.path.abspath(os.path.join(os.path.dirname(__file__), "cabt.json"))
|
|
118
|
+
with open(jsonpath) as f:
|
|
119
|
+
specification = json.load(f)
|
|
File without changes
|
|
Binary file
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import ctypes
|
|
2
|
+
import json
|
|
3
|
+
|
|
4
|
+
from .sim import StartData, lib, Battle
|
|
5
|
+
|
|
6
|
+
def _get_battle_data() -> dict:
|
|
7
|
+
"""Retrieve the current state.
|
|
8
|
+
|
|
9
|
+
Returns:
|
|
10
|
+
dict: Current observation.
|
|
11
|
+
"""
|
|
12
|
+
sd = lib.GetBattleData(Battle.battle_ptr)
|
|
13
|
+
Battle.obs = json.loads(sd.json.decode())
|
|
14
|
+
Battle.obs["search_begin_input"] = ctypes.string_at(sd.data, sd.count).decode('ascii')
|
|
15
|
+
return Battle.obs
|
|
16
|
+
|
|
17
|
+
def battle_start(deck0: list[int], deck1: list[int]) -> tuple[dict, StartData]:
|
|
18
|
+
"""Start the battle.
|
|
19
|
+
|
|
20
|
+
Args:
|
|
21
|
+
deck0: List of card IDs included in the first player’s deck.
|
|
22
|
+
deck1: List of card IDs included in the second player’s deck.
|
|
23
|
+
|
|
24
|
+
Returns:
|
|
25
|
+
tuple: A tuple containing:
|
|
26
|
+
- dict: First observation.
|
|
27
|
+
- StartData: Battle start data.
|
|
28
|
+
"""
|
|
29
|
+
if len(deck0) != 60 or len(deck1) != 60:
|
|
30
|
+
raise ValueError("The deck must contain 60 cards.")
|
|
31
|
+
cards = deck0 + deck1
|
|
32
|
+
arg = (ctypes.c_int*len(cards))(*cards)
|
|
33
|
+
start_data = lib.BattleStart(arg)
|
|
34
|
+
Battle.battle_ptr = start_data.battlePtr
|
|
35
|
+
if Battle.battle_ptr == None or Battle.battle_ptr == 0:
|
|
36
|
+
return (None, start_data)
|
|
37
|
+
else:
|
|
38
|
+
return (_get_battle_data(), start_data)
|
|
39
|
+
|
|
40
|
+
def battle_finish():
|
|
41
|
+
"""End the battle and free the memory used during it."""
|
|
42
|
+
lib.BattleFinish(Battle.battle_ptr)
|
|
43
|
+
|
|
44
|
+
def battle_select(select_list: list[int]) -> dict:
|
|
45
|
+
"""Select option.
|
|
46
|
+
|
|
47
|
+
Args:
|
|
48
|
+
select_list:
|
|
49
|
+
|
|
50
|
+
Returns:
|
|
51
|
+
dict: Next observation.
|
|
52
|
+
"""
|
|
53
|
+
if not isinstance(select_list, list) or not all(isinstance(i, int) for i in select_list):
|
|
54
|
+
raise ValueError("select_list is not list[int]")
|
|
55
|
+
arg = (ctypes.c_int*len(select_list))(*select_list)
|
|
56
|
+
err = lib.Select(Battle.battle_ptr, arg, len(select_list))
|
|
57
|
+
if err != 0:
|
|
58
|
+
if err == 30:
|
|
59
|
+
raise ValueError("battle_ptr broken.")
|
|
60
|
+
else:
|
|
61
|
+
raise IndexError()
|
|
62
|
+
return _get_battle_data()
|
|
63
|
+
|
|
64
|
+
def visualize_data() -> str:
|
|
65
|
+
"""Retrieve the data to be used by the visualizer.
|
|
66
|
+
|
|
67
|
+
Returns:
|
|
68
|
+
str: The data to be used by the visualizer.
|
|
69
|
+
"""
|
|
70
|
+
return lib.VisualizeData(Battle.battle_ptr).decode()
|
|
Binary file
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import ctypes
|
|
2
|
+
import os
|
|
3
|
+
|
|
4
|
+
class StartData(ctypes.Structure):
|
|
5
|
+
_fields_ = [
|
|
6
|
+
("battlePtr", ctypes.c_void_p),
|
|
7
|
+
("errorPlayer", ctypes.c_int),
|
|
8
|
+
("errorType", ctypes.c_int),
|
|
9
|
+
]
|
|
10
|
+
|
|
11
|
+
class SerialData(ctypes.Structure):
|
|
12
|
+
_fields_ = [
|
|
13
|
+
("json", ctypes.c_char_p),
|
|
14
|
+
("data", ctypes.POINTER(ctypes.c_ubyte)),
|
|
15
|
+
("count", ctypes.c_int),
|
|
16
|
+
("selectPlayer", ctypes.c_int)
|
|
17
|
+
]
|
|
18
|
+
|
|
19
|
+
if os.name == 'nt':
|
|
20
|
+
lib_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "cg.dll")
|
|
21
|
+
else:
|
|
22
|
+
lib_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "libcg.so")
|
|
23
|
+
lib = ctypes.cdll.LoadLibrary(lib_path)
|
|
24
|
+
|
|
25
|
+
lib.GameInitialize()
|
|
26
|
+
|
|
27
|
+
lib.BattleStart.restype = StartData
|
|
28
|
+
lib.BattleStart.argtypes = [ctypes.POINTER(ctypes.c_int)]
|
|
29
|
+
|
|
30
|
+
lib.BattleFinish.argtypes = [ctypes.c_void_p]
|
|
31
|
+
|
|
32
|
+
lib.GetBattleData.restype = SerialData
|
|
33
|
+
lib.GetBattleData.argtypes = [ctypes.c_void_p]
|
|
34
|
+
|
|
35
|
+
lib.Select.restype = ctypes.c_int
|
|
36
|
+
lib.Select.argtypes = [ctypes.c_void_p, ctypes.POINTER(ctypes.c_int), ctypes.c_int]
|
|
37
|
+
|
|
38
|
+
lib.VisualizeData.restype = ctypes.c_char_p
|
|
39
|
+
lib.VisualizeData.argtypes = [ctypes.c_void_p]
|
|
40
|
+
|
|
41
|
+
class Battle:
|
|
42
|
+
battle_ptr = None
|
|
43
|
+
obs = None
|
|
44
|
+
raminingTime = [[], []]
|
|
@@ -3,20 +3,21 @@ function renderer(options) {
|
|
|
3
3
|
|
|
4
4
|
const DEFAULT_NUM_ROWS = 8;
|
|
5
5
|
const DEFAULT_NUM_COLS = 8;
|
|
6
|
-
const
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
k: '
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
K: '
|
|
6
|
+
const PIECE_IMAGES_SRC = {
|
|
7
|
+
P: '',
|
|
8
|
+
R: '',
|
|
9
|
+
B: '',
|
|
10
|
+
Q: '',
|
|
11
|
+
n: '',
|
|
12
|
+
k: '',
|
|
13
|
+
p: '',
|
|
14
|
+
r: '',
|
|
15
|
+
b: '',
|
|
16
|
+
q: '',
|
|
17
|
+
N: '',
|
|
18
|
+
K: ''
|
|
19
19
|
};
|
|
20
|
+
|
|
20
21
|
const LIGHT_SQUARE_COLOR = '#f0d9b5';
|
|
21
22
|
const DARK_SQUARE_COLOR = '#b58863';
|
|
22
23
|
|
|
@@ -98,7 +99,7 @@ function renderer(options) {
|
|
|
98
99
|
alignItems: 'center'
|
|
99
100
|
});
|
|
100
101
|
const whitePawnImg = document.createElement('img');
|
|
101
|
-
whitePawnImg.src =
|
|
102
|
+
whitePawnImg.src = PIECE_IMAGES_SRC.P;
|
|
102
103
|
Object.assign(whitePawnImg.style, { height: '30px', marginRight: '8px' });
|
|
103
104
|
const whitePlayerName = document.createElement('span');
|
|
104
105
|
whitePlayerName.textContent = environment.info?.TeamNames?.[1] || 'Player 2';
|
|
@@ -134,7 +135,7 @@ function renderer(options) {
|
|
|
134
135
|
fontWeight: 'bold'
|
|
135
136
|
});
|
|
136
137
|
const blackPawnImg = document.createElement('img');
|
|
137
|
-
blackPawnImg.src =
|
|
138
|
+
blackPawnImg.src = PIECE_IMAGES_SRC.p;
|
|
138
139
|
Object.assign(blackPawnImg.style, { height: '30px', marginLeft: '8px' });
|
|
139
140
|
blackPlayerContainer.appendChild(blackPlayerName);
|
|
140
141
|
blackPlayerContainer.appendChild(blackPawnImg);
|
|
@@ -167,7 +168,6 @@ function renderer(options) {
|
|
|
167
168
|
justifyContent: 'center',
|
|
168
169
|
alignItems: 'center',
|
|
169
170
|
flexGrow: '1',
|
|
170
|
-
overflow: 'hidden',
|
|
171
171
|
width: '100%',
|
|
172
172
|
minHeight: '0'
|
|
173
173
|
});
|
|
@@ -197,7 +197,7 @@ function renderer(options) {
|
|
|
197
197
|
// Status Container
|
|
198
198
|
const statusContainer = document.createElement('div');
|
|
199
199
|
Object.assign(statusContainer.style, {
|
|
200
|
-
padding: '
|
|
200
|
+
padding: '5px',
|
|
201
201
|
backgroundColor: 'white',
|
|
202
202
|
borderRadius: '8px',
|
|
203
203
|
boxShadow: '0 4px 6px -1px rgba(0,0,0,0.1), 0 2px 4px -1px rgba(0,0,0,0.06)',
|
|
@@ -205,7 +205,7 @@ function renderer(options) {
|
|
|
205
205
|
width: 'auto',
|
|
206
206
|
minWidth: '200px',
|
|
207
207
|
maxWidth: '90vw',
|
|
208
|
-
marginTop: '
|
|
208
|
+
marginTop: '10px',
|
|
209
209
|
flexShrink: '0'
|
|
210
210
|
});
|
|
211
211
|
if (!environment.viewer) {
|
|
@@ -214,7 +214,7 @@ function renderer(options) {
|
|
|
214
214
|
|
|
215
215
|
currentStatusTextElement = document.createElement('p');
|
|
216
216
|
Object.assign(currentStatusTextElement.style, {
|
|
217
|
-
fontSize: isMobile ? '0.
|
|
217
|
+
fontSize: isMobile ? '0.8rem' : '1.1rem', // Responsive font size
|
|
218
218
|
fontWeight: '600',
|
|
219
219
|
margin: '0 0 5px 0'
|
|
220
220
|
});
|
|
@@ -222,7 +222,7 @@ function renderer(options) {
|
|
|
222
222
|
|
|
223
223
|
currentWinnerTextElement = document.createElement('p');
|
|
224
224
|
Object.assign(currentWinnerTextElement.style, {
|
|
225
|
-
fontSize: isMobile ? '
|
|
225
|
+
fontSize: isMobile ? '0.9rem' : '1.1rem', // Responsive font size
|
|
226
226
|
fontWeight: '700',
|
|
227
227
|
margin: '5px 0 0 0'
|
|
228
228
|
});
|
|
@@ -297,7 +297,10 @@ function renderer(options) {
|
|
|
297
297
|
// Calculate and apply board size
|
|
298
298
|
const containerWidth = currentBoardContainer?.clientWidth ?? width;
|
|
299
299
|
const containerHeight = currentBoardContainer?.clientHeight ?? height;
|
|
300
|
-
|
|
300
|
+
let smallestContainerEdge = Math.min(containerWidth, containerHeight);
|
|
301
|
+
// This is greedily trying to take as much space as possible, which can cause some conflict with flex box calculations for other elements
|
|
302
|
+
// we are going to take 24px off (arbitrary) to give the flex box renderer a bit of space to work with. Without it we will get some clipping.
|
|
303
|
+
smallestContainerEdge = smallestContainerEdge > 200 ? smallestContainerEdge - 24 : smallestContainerEdge;
|
|
301
304
|
const newSquareSize = Math.floor(smallestContainerEdge / displayCols);
|
|
302
305
|
|
|
303
306
|
if (newSquareSize !== squareSize) {
|
|
@@ -340,7 +343,7 @@ function renderer(options) {
|
|
|
340
343
|
const squareElement = currentBoardElement.querySelector(`#cell-${r_data}-${c_data}`);
|
|
341
344
|
if (squareElement && piece) {
|
|
342
345
|
const pieceImg = document.createElement('img');
|
|
343
|
-
pieceImg.src =
|
|
346
|
+
pieceImg.src = PIECE_IMAGES_SRC[piece];
|
|
344
347
|
pieceImg.style.width = `${pieceSize}px`;
|
|
345
348
|
pieceImg.style.height = `${pieceSize}px`;
|
|
346
349
|
squareElement.appendChild(pieceImg);
|