puzzle/common/GameCommon.js

667 lines
17 KiB
JavaScript
Raw Normal View History

2020-11-17 22:34:15 +01:00
import Geometry from './Geometry.js'
2020-12-23 01:19:30 +01:00
import Protocol from './Protocol.js'
2020-11-25 22:03:35 +01:00
import Util from './Util.js'
2020-11-17 22:34:15 +01:00
// Map<gameId, GameObject>
2020-11-17 22:34:15 +01:00
const GAMES = {}
function exists(gameId) {
return (!!GAMES[gameId]) || false
}
function __createGameObject(id, rng, puzzle, players, evtInfos) {
2020-12-05 19:45:34 +01:00
return {
id: id,
2020-12-21 18:34:57 +01:00
rng: rng,
2020-12-05 19:45:34 +01:00
puzzle: puzzle,
players: players,
evtInfos: evtInfos,
}
}
2020-12-22 22:35:09 +01:00
function __createPlayerObject(id, ts) {
2020-12-05 19:45:34 +01:00
return {
id: id,
x: 0,
y: 0,
d: 0, // mouse down
2020-12-07 12:20:09 +01:00
name: null, // 'anon'
color: null, // '#ffffff'
bgcolor: null, // '#222222'
2020-12-05 19:45:34 +01:00
points: 0,
ts: ts,
}
}
function newGame({id, rng, puzzle, players, evtInfos}) {
const game = __createGameObject(id, rng, puzzle, players, evtInfos)
2020-12-05 19:45:34 +01:00
setGame(id, game)
return game
}
2020-11-17 22:34:15 +01:00
function setGame(gameId, game) {
GAMES[gameId] = game
}
2020-12-23 01:19:30 +01:00
function getPlayerIndexById(gameId, playerId) {
let i = 0;
for (let player of GAMES[gameId].players) {
if (Util.decodePlayer(player).id === playerId) {
return i
}
i++
}
return -1
}
function getPlayerIdByIndex(gameId, playerIndex) {
if (GAMES[gameId].players.length > playerIndex) {
return Util.decodePlayer(GAMES[gameId].players[playerIndex]).id
}
return null
}
2020-12-05 19:45:34 +01:00
function getPlayer(gameId, playerId) {
2020-12-23 01:19:30 +01:00
let idx = getPlayerIndexById(gameId, playerId)
return Util.decodePlayer(GAMES[gameId].players[idx])
2020-12-05 19:45:34 +01:00
}
function setPlayer(gameId, playerId, player) {
2020-12-23 01:19:30 +01:00
let idx = getPlayerIndexById(gameId, playerId)
if (idx === -1) {
GAMES[gameId].players.push(Util.encodePlayer(player))
} else {
GAMES[gameId].players[idx] = Util.encodePlayer(player)
}
2020-12-05 19:45:34 +01:00
}
function setTile(gameId, tileIdx, tile) {
GAMES[gameId].puzzle.tiles[tileIdx] = Util.encodeTile(tile)
}
function setPuzzleData(gameId, data) {
GAMES[gameId].puzzle.data = data
}
2020-12-03 21:11:52 +01:00
function playerExists(gameId, playerId) {
2020-12-23 01:19:30 +01:00
const idx = getPlayerIndexById(gameId, playerId)
return idx !== -1
2020-12-03 21:11:52 +01:00
}
2020-12-22 22:35:09 +01:00
function getRelevantPlayers(gameId, ts) {
2020-12-21 02:29:14 +01:00
const minTs = ts - 30000
return getAllPlayers(gameId).filter(player => {
return player.ts >= minTs || player.points > 0
})
}
2020-12-22 22:35:09 +01:00
function getActivePlayers(gameId, ts) {
2020-12-05 19:45:34 +01:00
const minTs = ts - 30000
2020-12-21 02:29:14 +01:00
return getAllPlayers(gameId).filter(player => {
return player.ts >= minTs
})
2020-12-05 19:45:34 +01:00
}
2020-12-22 22:35:09 +01:00
function addPlayer(gameId, playerId, ts) {
2020-12-23 01:19:30 +01:00
if (!playerExists(gameId, playerId)) {
2020-12-22 22:35:09 +01:00
setPlayer(gameId, playerId, __createPlayerObject(playerId, ts))
2020-11-25 22:03:35 +01:00
} else {
changePlayer(gameId, playerId, { ts })
2020-11-17 22:34:15 +01:00
}
2020-12-03 21:11:52 +01:00
if (!GAMES[gameId].evtInfos[playerId]) {
GAMES[gameId].evtInfos[playerId] = {
_last_mouse: null,
_last_mouse_down: null,
}
2020-11-17 22:34:15 +01:00
}
}
2020-11-25 22:03:35 +01:00
function getAllGames() {
return Object.values(GAMES).sort((a, b) => {
// when both have same finished state, sort by started
if (isFinished(a.id) === isFinished(b.id)) {
return b.puzzle.data.started - a.puzzle.data.started
}
// otherwise, sort: unfinished, finished
return isFinished(a.id) ? 1 : -1
})
2020-12-05 19:45:34 +01:00
}
function getAllPlayers(gameId) {
return GAMES[gameId]
2020-12-23 01:19:30 +01:00
? GAMES[gameId].players.map(Util.decodePlayer)
2020-12-05 19:45:34 +01:00
: []
2020-11-25 22:03:35 +01:00
}
2020-11-17 22:34:15 +01:00
function get(gameId) {
return GAMES[gameId]
}
2020-12-05 19:45:34 +01:00
function getTileCount(gameId) {
return GAMES[gameId].puzzle.tiles.length
}
function getImageUrl(gameId) {
return GAMES[gameId].puzzle.info.imageUrl
}
function isFinished(gameId) {
return getFinishedTileCount(gameId) === getTileCount(gameId)
}
2020-12-05 19:45:34 +01:00
function getFinishedTileCount(gameId) {
let count = 0
for (let t of GAMES[gameId].puzzle.tiles) {
if (Util.decodeTile(t).owner === -1) {
count++
}
}
return count
}
function getTilesSortedByZIndex(gameId) {
const tiles = GAMES[gameId].puzzle.tiles.map(Util.decodeTile)
return tiles.sort((t1, t2) => t1.z - t2.z)
}
2020-11-17 22:34:15 +01:00
function changePlayer(gameId, playerId, change) {
2020-12-05 19:45:34 +01:00
const player = getPlayer(gameId, playerId)
2020-11-17 22:34:15 +01:00
for (let k of Object.keys(change)) {
2020-12-05 19:45:34 +01:00
player[k] = change[k]
2020-11-17 22:34:15 +01:00
}
2020-12-05 19:45:34 +01:00
setPlayer(gameId, playerId, player)
2020-11-17 22:34:15 +01:00
}
function changeData(gameId, change) {
for (let k of Object.keys(change)) {
GAMES[gameId].puzzle.data[k] = change[k]
}
}
function changeTile(gameId, tileIdx, change) {
for (let k of Object.keys(change)) {
2020-12-05 19:45:34 +01:00
const tile = Util.decodeTile(GAMES[gameId].puzzle.tiles[tileIdx])
tile[k] = change[k]
GAMES[gameId].puzzle.tiles[tileIdx] = Util.encodeTile(tile)
2020-11-17 22:34:15 +01:00
}
}
const getTile = (gameId, tileIdx) => {
2020-12-05 19:45:34 +01:00
return Util.decodeTile(GAMES[gameId].puzzle.tiles[tileIdx])
2020-11-17 22:34:15 +01:00
}
const getTileGroup = (gameId, tileIdx) => {
const tile = getTile(gameId, tileIdx)
return tile.group
}
const getFinalTilePos = (gameId, tileIdx) => {
const info = GAMES[gameId].puzzle.info
const boardPos = {
x: (info.table.width - info.width) / 2,
y: (info.table.height - info.height) / 2
}
const srcPos = srcPosByTileIdx(gameId, tileIdx)
return Geometry.pointAdd(boardPos, srcPos)
}
const getTilePos = (gameId, tileIdx) => {
const tile = getTile(gameId, tileIdx)
return tile.pos
}
const getTileZIndex = (gameId, tileIdx) => {
const tile = getTile(gameId, tileIdx)
return tile.z
}
const getFirstOwnedTileIdx = (gameId, userId) => {
for (let t of GAMES[gameId].puzzle.tiles) {
2020-12-05 19:45:34 +01:00
const tile = Util.decodeTile(t)
if (tile.owner === userId) {
return tile.idx
2020-11-17 22:34:15 +01:00
}
}
return -1
}
2020-12-05 19:45:34 +01:00
const getFirstOwnedTile = (gameId, userId) => {
const idx = getFirstOwnedTileIdx(gameId, userId)
return idx < 0 ? null : GAMES[gameId].puzzle.tiles[idx]
}
const getTileDrawOffset = (gameId) => {
return GAMES[gameId].puzzle.info.tileDrawOffset
}
const getTileDrawSize = (gameId) => {
return GAMES[gameId].puzzle.info.tileDrawSize
}
2020-12-07 02:38:07 +01:00
const getStartTs = (gameId) => {
return GAMES[gameId].puzzle.data.started
}
const getFinishTs = (gameId) => {
return GAMES[gameId].puzzle.data.finished
}
2020-11-17 22:34:15 +01:00
const getMaxGroup = (gameId) => {
return GAMES[gameId].puzzle.data.maxGroup
}
const getMaxZIndex = (gameId) => {
return GAMES[gameId].puzzle.data.maxZ
}
const getMaxZIndexByTileIdxs = (gameId, tileIdxs) => {
let maxZ = 0
for (let tileIdx of tileIdxs) {
let tileZIndex = getTileZIndex(gameId, tileIdx)
if (tileZIndex > maxZ) {
maxZ = tileZIndex
}
}
return maxZ
}
function srcPosByTileIdx(gameId, tileIdx) {
const info = GAMES[gameId].puzzle.info
2020-12-05 19:45:34 +01:00
const c = Util.coordByTileIdx(info, tileIdx)
2020-11-17 22:34:15 +01:00
const cx = c.x * info.tileSize
const cy = c.y * info.tileSize
return { x: cx, y: cy }
}
function getSurroundingTilesByIdx(gameId, tileIdx) {
const info = GAMES[gameId].puzzle.info
2020-12-05 19:45:34 +01:00
const c = Util.coordByTileIdx(info, tileIdx)
2020-11-17 22:34:15 +01:00
return [
// top
2020-12-05 19:45:34 +01:00
(c.y > 0) ? (tileIdx - info.tilesX) : -1,
2020-11-17 22:34:15 +01:00
// right
2020-12-05 19:45:34 +01:00
(c.x < info.tilesX - 1) ? (tileIdx + 1) : -1,
2020-11-17 22:34:15 +01:00
// bottom
2020-12-05 19:45:34 +01:00
(c.y < info.tilesY - 1) ? (tileIdx + info.tilesX) : -1,
2020-11-17 22:34:15 +01:00
// left
2020-12-05 19:45:34 +01:00
(c.x > 0) ? (tileIdx - 1) : -1,
2020-11-17 22:34:15 +01:00
]
}
const setTilesZIndex = (gameId, tileIdxs, zIndex) => {
for (let tilesIdx of tileIdxs) {
changeTile(gameId, tilesIdx, { z: zIndex })
}
}
const moveTileDiff = (gameId, tileIdx, diff) => {
const oldPos = getTilePos(gameId, tileIdx)
const pos = Geometry.pointAdd(oldPos, diff)
changeTile(gameId, tileIdx, { pos })
}
const moveTilesDiff = (gameId, tileIdxs, diff) => {
for (let tileIdx of tileIdxs) {
moveTileDiff(gameId, tileIdx, diff)
}
}
const finishTiles = (gameId, tileIdxs) => {
for (let tileIdx of tileIdxs) {
changeTile(gameId, tileIdx, { owner: -1, z: 1 })
}
}
const setTilesOwner = (gameId, tileIdxs, owner) => {
for (let tileIdx of tileIdxs) {
changeTile(gameId, tileIdx, { owner })
}
}
// get all grouped tiles for a tile
function getGroupedTileIdxs(gameId, tileIdx) {
const tiles = GAMES[gameId].puzzle.tiles
2020-12-05 19:45:34 +01:00
const tile = Util.decodeTile(tiles[tileIdx])
2020-11-17 22:34:15 +01:00
const grouped = []
if (tile.group) {
for (let other of tiles) {
2020-12-05 19:45:34 +01:00
const otherTile = Util.decodeTile(other)
if (otherTile.group === tile.group) {
grouped.push(otherTile.idx)
2020-11-17 22:34:15 +01:00
}
}
} else {
grouped.push(tile.idx)
}
return grouped
}
// Returns the index of the puzzle tile with the highest z index
// that is not finished yet and that matches the position
const freeTileIdxByPos = (gameId, pos) => {
let info = GAMES[gameId].puzzle.info
let tiles = GAMES[gameId].puzzle.tiles
let maxZ = -1
let tileIdx = -1
for (let idx = 0; idx < tiles.length; idx++) {
2020-12-05 19:45:34 +01:00
const tile = Util.decodeTile(tiles[idx])
2020-11-17 22:34:15 +01:00
if (tile.owner !== 0) {
continue
}
const collisionRect = {
x: tile.pos.x,
y: tile.pos.y,
w: info.tileSize,
h: info.tileSize,
}
if (Geometry.pointInBounds(pos, collisionRect)) {
if (maxZ === -1 || tile.z > maxZ) {
maxZ = tile.z
tileIdx = idx
}
}
}
return tileIdx
}
2020-12-05 19:45:34 +01:00
const getPlayerBgColor = (gameId, playerId) => {
2020-12-22 22:35:09 +01:00
const p = getPlayer(gameId, playerId)
return p ? p.bgcolor : null
2020-12-05 19:45:34 +01:00
}
const getPlayerColor = (gameId, playerId) => {
2020-12-22 22:35:09 +01:00
const p = getPlayer(gameId, playerId)
return p ? p.color : null
2020-12-05 19:45:34 +01:00
}
const getPlayerName = (gameId, playerId) => {
2020-12-22 22:35:09 +01:00
const p = getPlayer(gameId, playerId)
return p ? p.name : null
2020-12-05 19:45:34 +01:00
}
const getPlayerPoints = (gameId, playerId) => {
2020-12-22 22:35:09 +01:00
const p = getPlayer(gameId, playerId)
return p ? p.points : null
2020-12-05 19:45:34 +01:00
}
2020-11-17 22:34:15 +01:00
// determine if two tiles are grouped together
const areGrouped = (gameId, tileIdx1, tileIdx2) => {
const g1 = getTileGroup(gameId, tileIdx1)
const g2 = getTileGroup(gameId, tileIdx2)
return g1 && g1 === g2
}
2020-12-05 19:45:34 +01:00
const getTableWidth = (gameId) => {
return GAMES[gameId].puzzle.info.table.width
}
const getTableHeight = (gameId) => {
return GAMES[gameId].puzzle.info.table.height
}
2020-12-22 22:35:09 +01:00
const getPuzzle = (gameId) => {
return GAMES[gameId].puzzle
}
const getRng = (gameId) => {
return GAMES[gameId].rng.obj
}
2020-12-05 19:45:34 +01:00
const getPuzzleWidth = (gameId) => {
return GAMES[gameId].puzzle.info.width
}
const getPuzzleHeight = (gameId) => {
return GAMES[gameId].puzzle.info.height
}
2020-12-22 22:35:09 +01:00
function handleInput(gameId, playerId, input, ts) {
2020-12-05 19:45:34 +01:00
const puzzle = GAMES[gameId].puzzle
2020-11-17 22:34:15 +01:00
let evtInfo = GAMES[gameId].evtInfos[playerId]
let changes = []
const _dataChange = () => {
changes.push(['data', puzzle.data])
}
const _tileChange = (tileIdx) => {
changes.push(['tile', getTile(gameId, tileIdx)])
}
const _tileChanges = (tileIdxs) => {
for (let tileIdx of tileIdxs) {
_tileChange(tileIdx)
}
}
const _playerChange = () => {
2020-12-23 01:19:30 +01:00
changes.push(['player', Util.encodePlayer(getPlayer(gameId, playerId))])
2020-11-17 22:34:15 +01:00
}
// put both tiles (and their grouped tiles) in the same group
const groupTiles = (gameId, tileIdx1, tileIdx2) => {
let tiles = GAMES[gameId].puzzle.tiles
let group1 = getTileGroup(gameId, tileIdx1)
let group2 = getTileGroup(gameId, tileIdx2)
let group
let searchGroups = []
if (group1) {
searchGroups.push(group1)
}
if (group2) {
searchGroups.push(group2)
}
if (group1) {
group = group1
} else if (group2) {
group = group2
} else {
let maxGroup = getMaxGroup(gameId) + 1
changeData(gameId, { maxGroup })
_dataChange()
group = getMaxGroup(gameId)
}
changeTile(gameId, tileIdx1, { group })
_tileChange(tileIdx1)
changeTile(gameId, tileIdx2, { group })
_tileChange(tileIdx2)
// TODO: strange
if (searchGroups.length > 0) {
2020-12-05 19:45:34 +01:00
for (let t of tiles) {
const tile = Util.decodeTile(t)
2020-11-17 22:34:15 +01:00
if (searchGroups.includes(tile.group)) {
changeTile(gameId, tile.idx, { group })
_tileChange(tile.idx)
}
}
}
}
2020-11-25 22:03:35 +01:00
const type = input[0]
2020-12-23 01:19:30 +01:00
if (type === Protocol.INPUT_EV_BG_COLOR) {
2020-12-03 21:24:59 +01:00
const bgcolor = input[1]
changePlayer(gameId, playerId, { bgcolor, ts })
_playerChange()
2020-12-23 01:19:30 +01:00
} else if (type === Protocol.INPUT_EV_PLAYER_COLOR) {
2020-11-25 22:03:35 +01:00
const color = input[1]
changePlayer(gameId, playerId, { color, ts })
_playerChange()
2020-12-23 01:19:30 +01:00
} else if (type === Protocol.INPUT_EV_PLAYER_NAME) {
2020-11-25 22:03:35 +01:00
const name = `${input[1]}`.substr(0, 16)
changePlayer(gameId, playerId, { name, ts })
_playerChange()
2020-12-23 01:19:30 +01:00
} else if (type === Protocol.INPUT_EV_MOUSE_DOWN) {
2020-11-25 22:03:35 +01:00
const x = input[1]
const y = input[2]
const pos = {x, y}
changePlayer(gameId, playerId, { d: 1, ts })
2020-11-17 22:34:15 +01:00
_playerChange()
evtInfo._last_mouse_down = pos
const tileIdxAtPos = freeTileIdxByPos(gameId, pos)
if (tileIdxAtPos >= 0) {
let maxZ = getMaxZIndex(gameId) + 1
changeData(gameId, { maxZ })
_dataChange()
const tileIdxs = getGroupedTileIdxs(gameId, tileIdxAtPos)
setTilesZIndex(gameId, tileIdxs, getMaxZIndex(gameId))
setTilesOwner(gameId, tileIdxs, playerId)
_tileChanges(tileIdxs)
}
2020-11-25 22:03:35 +01:00
evtInfo._last_mouse = pos
2020-11-17 22:34:15 +01:00
2020-12-23 01:19:30 +01:00
} else if (type === Protocol.INPUT_EV_MOUSE_MOVE) {
2020-11-25 22:03:35 +01:00
const x = input[1]
const y = input[2]
const pos = {x, y}
changePlayer(gameId, playerId, {x, y, ts})
2020-11-17 22:34:15 +01:00
_playerChange()
if (evtInfo._last_mouse_down !== null) {
let tileIdx = getFirstOwnedTileIdx(gameId, playerId)
if (tileIdx >= 0) {
const diffX = x - evtInfo._last_mouse_down.x
const diffY = y - evtInfo._last_mouse_down.y
const diff = { x: diffX, y: diffY }
const tileIdxs = getGroupedTileIdxs(gameId, tileIdx)
moveTilesDiff(gameId, tileIdxs, diff)
_tileChanges(tileIdxs)
}
evtInfo._last_mouse_down = pos
}
2020-11-25 22:03:35 +01:00
evtInfo._last_mouse = pos
2020-12-23 01:19:30 +01:00
} else if (type === Protocol.INPUT_EV_MOUSE_UP) {
2020-11-25 22:03:35 +01:00
const x = input[1]
const y = input[2]
const pos = {x, y}
changePlayer(gameId, playerId, { d: 0, ts })
2020-11-17 22:34:15 +01:00
_playerChange()
evtInfo._last_mouse_down = null
let tileIdx = getFirstOwnedTileIdx(gameId, playerId)
if (tileIdx >= 0) {
// drop the tile(s)
let tileIdxs = getGroupedTileIdxs(gameId, tileIdx)
setTilesOwner(gameId, tileIdxs, 0)
_tileChanges(tileIdxs)
// Check if the tile was dropped near the final location
let tilePos = getTilePos(gameId, tileIdx)
let finalPos = getFinalTilePos(gameId, tileIdx)
if (Geometry.pointDistance(finalPos, tilePos) < puzzle.info.snapDistance) {
let diff = Geometry.pointSub(finalPos, tilePos)
// Snap the tile to the final destination
moveTilesDiff(gameId, tileIdxs, diff)
finishTiles(gameId, tileIdxs)
2020-12-05 19:45:34 +01:00
changePlayer(gameId, playerId, { points: getPlayerPoints(gameId, playerId) + tileIdxs.length })
2020-11-17 22:34:15 +01:00
_tileChanges(tileIdxs)
// check if the puzzle is finished
if (getFinishedTileCount(gameId) === getTileCount(gameId)) {
2020-12-22 22:35:09 +01:00
changeData(gameId, { finished: ts })
_dataChange()
}
2020-11-17 22:34:15 +01:00
} else {
// Snap to other tiles
const check = (gameId, tileIdx, otherTileIdx, off) => {
let info = GAMES[gameId].puzzle.info
if (otherTileIdx < 0) {
return false
}
if (areGrouped(gameId, tileIdx, otherTileIdx)) {
return false
}
const tilePos = getTilePos(gameId, tileIdx)
const dstPos = Geometry.pointAdd(
getTilePos(gameId, otherTileIdx),
{x: off[0] * info.tileSize, y: off[1] * info.tileSize}
)
if (Geometry.pointDistance(tilePos, dstPos) < info.snapDistance) {
let diff = Geometry.pointSub(dstPos, tilePos)
let tileIdxs = getGroupedTileIdxs(gameId, tileIdx)
moveTilesDiff(gameId, tileIdxs, diff)
groupTiles(gameId, tileIdx, otherTileIdx)
tileIdxs = getGroupedTileIdxs(gameId, tileIdx)
const zIndex = getMaxZIndexByTileIdxs(gameId, tileIdxs)
setTilesZIndex(gameId, tileIdxs, zIndex)
_tileChanges(tileIdxs)
return true
}
return false
}
for (let tileIdxTmp of getGroupedTileIdxs(gameId, tileIdx)) {
let othersIdxs = getSurroundingTilesByIdx(gameId, tileIdxTmp)
if (
check(gameId, tileIdxTmp, othersIdxs[0], [0, 1]) // top
|| check(gameId, tileIdxTmp, othersIdxs[1], [-1, 0]) // right
|| check(gameId, tileIdxTmp, othersIdxs[2], [0, -1]) // bottom
|| check(gameId, tileIdxTmp, othersIdxs[3], [1, 0]) // left
) {
break
}
}
}
}
2020-11-25 22:03:35 +01:00
evtInfo._last_mouse = pos
} else {
changePlayer(gameId, playerId, { ts })
_playerChange()
2020-11-17 22:34:15 +01:00
}
return changes
}
export default {
2020-12-22 22:35:09 +01:00
__createGameObject,
__createPlayerObject,
2020-12-05 19:45:34 +01:00
newGame,
2020-11-17 22:34:15 +01:00
exists,
2020-12-03 21:11:52 +01:00
playerExists,
2020-12-21 02:29:14 +01:00
getRelevantPlayers,
2020-12-05 19:45:34 +01:00
getActivePlayers,
2020-11-17 22:34:15 +01:00
addPlayer,
2020-12-05 19:45:34 +01:00
getFinishedTileCount,
getTileCount,
getImageUrl,
2020-11-17 22:34:15 +01:00
get,
2020-11-25 22:03:35 +01:00
getAllGames,
2020-12-05 19:45:34 +01:00
getPlayerBgColor,
getPlayerColor,
getPlayerName,
2020-12-23 01:19:30 +01:00
getPlayerIndexById,
getPlayerIdByIndex,
2020-12-05 19:45:34 +01:00
changePlayer,
setPlayer,
setTile,
setPuzzleData,
getTableWidth,
getTableHeight,
2020-12-22 22:35:09 +01:00
getPuzzle,
getRng,
2020-12-05 19:45:34 +01:00
getPuzzleWidth,
getPuzzleHeight,
getTilesSortedByZIndex,
getFirstOwnedTile,
getTileDrawOffset,
getTileDrawSize,
2020-12-07 02:38:07 +01:00
getStartTs,
getFinishTs,
2020-11-17 22:34:15 +01:00
handleInput,
}