add tab window to KL (#280)

* KL: init draft interface

* KL: fix images

* KL: put icons

* added score in local

* KL: divisors

* KL: add lateral buttons

* KL: background

* KL: init hero portraits

* KL: add lateral header buttons

* KL: minor styles fixes

* KL: add real keys in card user

* KL: fix negative margin

* KL: ready button

* KL: interface functionality

* KL: fix draft height problems

* KL: port the remaing parts from cardplayer

* KL: remove order of card users in interface

* KL: add bottom buttons on draft interface

* KL: add a little animation

* KL: draw filter interface

* KL: add filters prop to game

* KL: fix height and overflow

* KL: init filter functionality

* KL: fix filter reversing

* added role and b64 images to heroes

* BoardBuilder: change mouse cursor do 32x32

* KL: fix name bug

* KL: allow game to init

* KL: fix board position in html

* KL: set initial player pos in canvas board

* KL: undo scrollbar

* KL: add functional role filter

* KL: fix initial positons in canvas board

* KL: players now choose initial coordinates ingame

* show bases

* fix allies choosing same coord

* draw indicators to player know where clicked in base

* fix indicatos for users know base location

* add delay before game start

* refactor code

* remove actions when not in preparation moment

* fix arena board

* fix automatic coord choose

* fix lava board

* show skill description

* change cursor in buttons which does not work yet

* draw top of interface

* draw user hud interface

* add skills hud interface

* add cast list interface

* clean draw play code

* improve cross browser

* fix ap bars positioning

* fix ap bars positioning 2snd part

* animate ready button

* KL: delete draft for draft interface file

* KL: shows turns left to player come back to life

* KL: do ready button animation

* KL: add light indicator in card player if player is ready

* KL: add hover in header buttons

* KL: let skill lits all have same width

* KL: init tab window creation

* KL: add pressed tab buttons in tab window

* KL: finnish add skill tab in tab window

* KL: put table in stats tab

* KL: start organize play's draw files

* KL: fix some identation

* KL: end stats tab tables

* Kind: remove + sign in word show

* KL: change row qtt to 10

* KL: implement tab shadow

* KL: init recap tab

* KL: put order in draft skills tab on user card

* Merge branch 'master' into rheidner/interface

* KL: fix life_bar in recap graph

* KL: end recap tab

* KL: fix recap heals

Co-authored-by: Derenash <eyer.derenash@gmail.com>
This commit is contained in:
Rheidner Achiles 2021-08-19 10:50:50 -03:00 committed by GitHub
parent eedb46ba88
commit 57887f5520
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
27 changed files with 2103 additions and 1011 deletions

View File

@ -9,13 +9,13 @@ type App.KL.Game.State.Local {
room: String
tab: Maybe<String>
page: String // page in draft -> heroes or users for now
score: Maybe<String>
info: Hexagonal.Axial
preview: App.KL.Game.Cast.Preview
mouse: Pair<U32, U32>
screen_size: Pair<U32, U32>
filter: App.KL.Game.Phase.Draft.Filter
hud: App.KL.Game.State.Local.hud // HUD controls
hud: App.KL.Game.Controls.Hud // HUD controls
draft: App.KL.Game.Controls.Draft // draft page controls
)
}
@ -26,8 +26,9 @@ App.KL.Game.State.Local.init(user: String, room: String, screen_size: Pair<U32,
let info = Hexagonal.Axial.new(0,0)
let page = "heroes"
let filter = App.KL.Game.Phase.Draft.Filter.default
let hud = App.KL.Game.State.Local.hud.new(none)
App.KL.Game.State.Local.new(user, room, tab, page, none, info, preview, mouse, screen_size, filter, hud)
let hud = App.KL.Game.Controls.Hud.initial
let draft = App.KL.Game.Controls.Draft.initial
App.KL.Game.State.Local.new(user, room, tab, page, info, preview, mouse, screen_size, filter, hud, draft)
// Global State
// ============

View File

@ -0,0 +1,56 @@
// BUTTON ANIMATION
// ==========================
type App.KL.Game.Assets.ButtonAnimation {
new(frames: List<String>, init_loop: Nat, seconds_per_frame: Nat)
}
type App.KL.Game.Assets.ButtonAnimation.Button { // animation and actual frame
new(animation: App.KL.Game.Assets.ButtonAnimation, frame: Nat)
}
App.KL.Game.Assets.ButtonAnimation.Button.get_img(
button: App.KL.Game.Assets.ButtonAnimation.Button
): Maybe<String>
App.KL.Game.Assets.ButtonAnimation.get(button@animation, button@frame)
App.KL.Game.Assets.ButtonAnimation.get(
animation: App.KL.Game.Assets.ButtonAnimation,
frame: Nat
): Maybe<String>
let index = Nat.div(frame, animation@seconds_per_frame)
List.get!(index, animation@frames)
App.KL.Game.Assets.ButtonAnimation.get_limit(
animation: App.KL.Game.Assets.ButtonAnimation
): Nat
(List.length!(animation@frames) * animation@seconds_per_frame) - 1
// BUTTON CLICK
// ==========================
type App.KL.Game.Assets.ButtonClick.Image {
new(
normal: String,
hover: String,
click: String
)
}
type App.KL.Game.Assets.ButtonClick {
new(image: App.KL.Game.Assets.ButtonClick.Image, hovered: Bool, clicked: Bool)
}
App.KL.Game.Assets.ButtonClick.generate(
image: App.KL.Game.Assets.ButtonClick.Image
): App.KL.Game.Assets.ButtonClick
App.KL.Game.Assets.ButtonClick.new(image, false, false)
App.KL.Game.Assets.ButtonClick.get_img(
button: App.KL.Game.Assets.ButtonClick
): String
if button@clicked then
button@image@click
else
if button@hovered then
button@image@hover
else
button@image@normal

View File

@ -0,0 +1,100 @@
// Draft controls
type App.KL.Game.Controls.Draft {
new(
// ready button controls
main_button: App.KL.Game.Assets.ButtonAnimation.Button
main_button_hovered: Bool
// header button controls
header: Map<App.KL.Game.Assets.ButtonClick> // id -> (img, state, action)
)
}
App.KL.Game.Controls.Draft.initial: App.KL.Game.Controls.Draft
App.KL.Game.Controls.Draft.new(
// ready button
App.KL.Game.Assets.ButtonAnimation.Button.new(App.KL.Game.Phase.Draft.Assets.main_button.ready, 0)
false
// header buttons
App.KL.Game.Controls.Draft.header.buttons
)
// default when mouse_move/mouse_down is not active
App.KL.Game.Controls.Draft.default(
ready: Bool,
controls: App.KL.Game.Controls.Draft
): App.KL.Game.Controls.Draft
let main_button =
if ready then
controls@main_button@animation <- App.KL.Game.Phase.Draft.Assets.main_button.waiting
else
controls@main_button@animation <- App.KL.Game.Phase.Draft.Assets.main_button.ready
let main_button_hovered = false
let header.buttons = App.KL.Game.Controls.Draft.header.buttons
App.KL.Game.Controls.Draft.new(main_button, main_button_hovered, header.buttons)
// HEADER BUTTONS
App.KL.Game.Controls.Draft.header.buttons: Map<App.KL.Game.Assets.ButtonClick>
{
"exit": App.KL.Game.Assets.ButtonClick.generate(exit_button),
"help": App.KL.Game.Assets.ButtonClick.generate(help_button),
"friend": App.KL.Game.Assets.ButtonClick.generate(friend_button),
"collection": App.KL.Game.Assets.ButtonClick.generate(collection_button),
"store": App.KL.Game.Assets.ButtonClick.generate(store_button),
"config": App.KL.Game.Assets.ButtonClick.generate(config_button),
}
App.KL.Game.Controls.Draft.header.update(
event_id: String
controls: App.KL.Game.Controls.Draft
): App.KL.Game.Controls.Draft
let buttons = controls@header
let button = buttons{event_id}
case button {
none: controls
some:
let buttons = buttons{event_id} <- button.value@hovered <- true
controls@header <- buttons
}
// assets
exit_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)
help_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)
friend_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)
config_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)
collection_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)
store_button: App.KL.Game.Assets.ButtonClick.Image
App.KL.Game.Assets.ButtonClick.Image.new(
""
""
""
)

View File

@ -0,0 +1,50 @@
// HUD controls
type App.KL.Game.Controls.Hud {
new(
skill: Maybe<App.KL.Game.Skill>
// tab
tab_active: Bool
tab_which: App.KL.Game.Controls.Hud.Tab
)
}
type App.KL.Game.Controls.Hud.Tab.Label {
new(image: String, content: DOM)
}
App.KL.Game.Controls.Hud.initial: App.KL.Game.Controls.Hud
let skill = none
let tab_active = false
let tab_which = App.KL.Game.Controls.Hud.Tab.stats
App.KL.Game.Controls.Hud.new(
skill,
tab_active,
tab_which,
)
type App.KL.Game.Controls.Hud.Tab {
stats
skills
recap
}
App.KL.Game.Controls.Hud.Tab.show(
tab: App.KL.Game.Controls.Hud.Tab
): String
case tab {
stats: "stats"
skills: "skills"
recap: "recap"
}
App.KL.Game.Controls.Hud.Tab.eql(
a: App.KL.Game.Controls.Hud.Tab
b: App.KL.Game.Controls.Hud.Tab
): Bool
case a b {
stats stats: true
recap recap: true
skills skills: true
} default false

View File

@ -11,7 +11,7 @@ type App.KL.Game.Hero {
role: Maybe<App.KL.Game.Hero.Role>
description: String
)
}
}
type App.KL.Game.Hero.Draw.Pose {
idle

View File

@ -0,0 +1,13 @@
// background
// =====================
App.KL.Game.Phase.Draft.Assets.background.hexagons.1: String
""
App.KL.Game.Phase.Draft.Assets.background.hexagons.2: String
""
App.KL.Game.Phase.Draft.Assets.background.hexagons.3: String
""
App.KL.Game.Phase.Draft.Assets.background.hexagons.4: String
""

View File

@ -0,0 +1,7 @@
// bottom buttons
// =====================
App.KL.Game.Phase.Draft.Assets.bottom.list_button: String
""
App.KL.Game.Phase.Draft.Assets.bottom.chat_buttom: String
""

View File

@ -0,0 +1,37 @@
// ASSETS
// user and allies card
// =====================
App.KL.Game.Phase.Draft.Assets.card.button: String
""
App.KL.Game.Phase.Draft.Assets.card.divisor: String
""
App.KL.Game.Phase.Draft.Assets.card.name_divisor: String
""
App.KL.Game.Phase.Draft.Assets.card.attr_bar(i: Nat, j: Nat): String
case Nat.cmp(j, i) {
ltn: ""
eql: ""
gtn: ""
}
App.KL.Game.Phase.Draft.Assets.card.attr_icons: Map<String>
{
"damage": "" // offensive
"resistance": "" // defensive
"mobility": "" // mobility
"range": "" // range
"utility": ""// utility
}
App.KL.Game.Phase.Draft.Assets.card.active: String
""
App.KL.Game.Phase.Draft.Assets.card.main: String
""
App.KL.Game.Phase.Draft.Assets.card.allies: String
""

View File

@ -0,0 +1,16 @@
// filter
// =====================
App.KL.Game.Phase.Draft.Assets.filter.background: String
""
App.KL.Game.Phase.Draft.Assets.filter.icon: String
""
App.KL.Game.Phase.Draft.Assets.filter.input: String
""
App.KL.Game.Phase.Draft.Assets.filter.check_false: String
""
App.KL.Game.Phase.Draft.Assets.filter.check_true: String
""

View File

@ -0,0 +1,16 @@
// footer
// =====================
App.KL.Game.Phase.Draft.Assets.footer.stencil.main: String
""
App.KL.Game.Phase.Draft.Assets.footer.stencil.left: String
""
App.KL.Game.Phase.Draft.Assets.footer.stencil.diagonal_left: String
""
App.KL.Game.Phase.Draft.Assets.footer.stencil.diagonal_right: String
""
App.KL.Game.Phase.Draft.Assets.footer.stencil.right: String
""

View File

@ -0,0 +1,4 @@
// header
// =====================
App.KL.Game.Phase.Draft.Assets.header: String
""

View File

@ -0,0 +1,48 @@
Velocity: Nat // move to constants TODO?
6
// ready button
// =====================
App.KL.Game.Phase.Draft.Assets.main_button.ready: App.KL.Game.Assets.ButtonAnimation
{
[
""
""
""
""
],
0,
Velocity
}
App.KL.Game.Phase.Draft.Assets.main_button.waiting: App.KL.Game.Assets.ButtonAnimation
{
[
""
""
""
""
""
""
""
""
],
0,
Velocity
}
App.KL.Game.Phase.Draft.Assets.main_button.cancel: App.KL.Game.Assets.ButtonAnimation
{
[
""
""
""
""
""
""
""
""
],
4 * Velocity,
Velocity
}

View File

@ -0,0 +1,7 @@
// hero portraits
// =====================
App.KL.Game.Phase.Draft.Assets.portrait.hero.background: String
""
App.KL.Game.Phase.Draft.Assets.portrait.hero.border: String
""

File diff suppressed because one or more lines are too long

View File

@ -5,15 +5,51 @@ App.KL.Game.Phase.Draft.when(
event : App.Event,
): IO<Maybe<App.State.local<App.KL.State>>>
let players = global@players
let room = String.take(16, Crypto.Keccak.hash(local@room))
let room = String.take(16, Crypto.Keccak.hash(local@room))
let ready = Maybe.if!((x) x@ready, players{local@user}) // verify if local user is ready
let controls = local@draft // buttons controls (hover, click, active, etc)
case event {
frame:
let main_button = controls@main_button
let init = main_button@animation@init_loop
let end = App.KL.Game.Assets.ButtonAnimation.get_limit(main_button@animation)
let actual = main_button@frame
let main_button =
if actual >=? end then
main_button@frame <- init
else
main_button@frame <- main_button@frame + 1
let controls = controls@main_button <- main_button
let local = local@draft <- controls
App.set_local!(App.KL.State.Local.game(local))
input:
switch String.starts_with(event.id) {
"HF": // HF = hero-filter = name of hero to filter
let local = local@filter <- filter@hero_name <- event.text
App.set_local!(App.KL.State.Local.game(local))
} default App.pass!
mouse_move:
let default_controls = App.KL.Game.Controls.Draft.default(ready, controls)
let controls = switch String.starts_with(event.id) {
"R": // ready
if ready then
if controls@main_button_hovered then
controls
else
let new_button = App.KL.Game.Assets.ButtonAnimation.Button.new(App.KL.Game.Phase.Draft.Assets.main_button.cancel, 0)
let controls = controls@main_button <- new_button
let controls = controls@main_button_hovered <- true
controls
else
controls
"btn-header-": // hover in header buttons
let event_id = String.drop(11, event.id)
App.KL.Game.Controls.Draft.header.update(event_id, default_controls)
} default default_controls
let local = local@draft <- controls
App.set_local!(App.KL.State.Local.game(local))
mouse_click:
switch String.starts_with(event.id) {
"H" : // Set Player Hero
@ -33,12 +69,17 @@ App.KL.Game.Phase.Draft.when(
case player {
none: App.pass!
some:
let can_ready = App.KL.Game.Player.has_hero(player.value)
let ready_u8 =
if player.value@ready then 0#8
else if can_ready then 1#8
else 0#8
App.new_post!(room, App.KL.Game.Phase.Draft.Event.set_ready.serial(ready_u8))
let can_ready = App.KL.Game.Player.has_hero(player.value)
let ready_info =
if player.value@ready then {0#8, App.KL.Game.Phase.Draft.Assets.main_button.ready}
else
if can_ready then { 1#8, App.KL.Game.Phase.Draft.Assets.main_button.waiting}
else {0#8, App.KL.Game.Phase.Draft.Assets.main_button.ready}
let local = local@draft <- controls@main_button <- controls@main_button@animation <- ready_info@snd
IO {
App.new_post!(room, App.KL.Game.Phase.Draft.Event.set_ready.serial(ready_info@fst))
App.set_local!(App.KL.State.Local.game(local))
}
}
"T" : // Set a team
let player_count = String.drop(1, event.id)

View File

@ -0,0 +1,32 @@
App.KL.Game.Phase.Play.Assets.interface.top: String
""
App.KL.Game.Phase.Play.Assets.interface.top.dead: String
""
App.KL.Game.Phase.Play.Assets.interface.top.life_placeholder: String
""
App.KL.Game.Phase.Play.Assets.interface.top.life_green: String
""
App.KL.Game.Phase.Play.Assets.interface.top.life_red: String
""
App.KL.Game.Phase.Play.Assets.interface.top.ap: String
""
App.KL.Game.Phase.Play.Assets.interface.top.portrait_placeholder: String
""
App.KL.Game.Phase.Play.Assets.interface.bottom: String
""
App.KL.Game.Phase.Play.Assets.interface.user: String
""
App.KL.Game.Phase.Play.Assets.interface.user.life: String
""
App.KL.Game.Phase.Play.Assets.interface.user.ap: String
""

File diff suppressed because one or more lines are too long

View File

@ -4,19 +4,18 @@ App.KL.Game.Phase.Play.draw.fator.number: F64
App.KL.Game.Phase.Play.draw.fator.string: String
F64.show(App.KL.Game.Phase.Play.draw.fator.number)
// BOARD DRAWING
// =============
App.KL.Game.Phase.Play.draw.theme.secondary: String
"#eb6bff"
type App.KL.Game.Phase.Play.draw.SkillInfo {
new(
hero: String
name: String
delay: U64
)
}
PlayersInfo: Type
Pair<Maybe<App.KL.Game.Creature>, App.KL.Game.Player>
SkillInfo: Type
Pair<String,Pair<App.KL.Game.Hero, App.KL.Game.Skill>>
// draw all the screen (round, seconds, canvas, list of skills)
App.KL.Game.Phase.Play.draw(img: VoxBox, local: App.KL.Game.State.Local, game: App.KL.Game): DOM
let players_info = App.KL.Game.Phase.Play.draw.get_players_info(game@players, game@board)
<div style={
"width": "max(100vw, 900px)",
"height": "max(100vh, 650px)",
@ -25,470 +24,135 @@ App.KL.Game.Phase.Play.draw(img: VoxBox, local: App.KL.Game.State.Local, game: A
"background-color":" #071840"
}>
// interface
{ App.KL.Game.Phase.Play.draw.interface(local, game) }
{ App.KL.Game.Phase.Play.draw.board_canvas(local, game, img) }
{ App.KL.Game.Phase.Play.draw.interface(local, game, players_info) }
// canvas
{ App.KL.Game.Phase.Play.draw.canvas.elem(local, game, img) }
// tab
{ App.KL.Game.Phase.Play.draw.tab(local, game, players_info) }
</div>
App.KL.Game.Phase.Play.draw.interface(local: App.KL.Game.State.Local, game: App.KL.Game): DOM
let players_info = App.KL.Game.Phase.Play.draw.interface.top.get_players(game@players, game@board)
<div>
{App.KL.Game.Phase.Play.draw.interface.top(players_info, local, game)}
{App.KL.Game.Phase.Play.draw.interface.bottom(local@user, players_info, local@hud@skill)}
{App.KL.Game.Phase.Play.draw.interface.user(local@user, players_info)}
{App.KL.Game.Phase.Play.draw.interface.casts(local@user, game@moment, game@players)}
</div>
// AUX FUNCTIONS
// =================================
App.KL.Game.Phase.Play.draw.interface.casts(user: String, moment: App.KL.Game.Moment, players: Map<App.KL.Game.Player>): DOM
case moment {
execution:
casts = moment.casts
p_cst = moment.previous_casts
<div style={
"position": "absolute"
"bottom": "0"
"right": "0"
"width": "300px"
"margin-right": "5px"
}>
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, p_cst[1], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, p_cst[0], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(true, casts[0], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, casts[1], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, casts[2], players)}
</div>
} default <span style={"display": "none"}></span>
App.KL.Game.Phase.Play.draw.interface.casts.item(main: Bool, cast: Maybe<App.KL.Game.Cast>, players: Map<App.KL.Game.Player>): DOM
let name = Maybe {
get cast = cast
get player = players{cast@player}
App.KL.Game.Phase.Play.draw.get_player_hero(player: App.KL.Game.Player): Maybe<App.KL.Game.Hero>
Maybe {
get hero_id = player@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
get skill = hero@skills{Char.to_string(cast@letter)}
return hero@name | " - " | skill@name
} <> ""
<div style={
"color": "white"
"background-color": if main && Bool.not(name =? "") then "#2a1956" else "transparent"
"width": if main then "100%" else "98%"
"padding": "0 20px"
"margin-left": if main then "0" else "2%"
"height": if main then "48px" else "36px"
"flex": if main then "2" else "1"
"display": "flex"
"border-radius": "5px"
"align-items": "center"
}>name</div>
App.KL.Game.Phase.Play.draw.interface.top(players_info: Map<PlayersInfo>, local: App.KL.Game.State.Local, game: App.KL.Game): DOM
let inner =
<div>
{App.KL.Game.Phase.Play.draw.interface.top.timer(game@moment)}
{App.KL.Game.Phase.Play.draw.interface.top.round(game@turn)}
{App.KL.Game.Phase.Play.draw.interface.top.scores(game@score)}
{App.KL.Game.Phase.Play.draw.interface.top.players(local@user, players_info, game@cemetery, game@board)}
</div>
<header style={
"position": "absolute",
"top": "0"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(624, 91, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.draw.assets.interface.top, inner)}
</header>
App.KL.Game.Phase.Play.draw.interface.top.timer(moment: App.KL.Game.Moment): DOM
let bar = case moment {
initial:
if moment.countdown >? App.KL.Constants.initital_delay then
U64.to_f64(moment.countdown - App.KL.Constants.initital_delay) / U64.to_f64(App.KL.Constants.initial_time)
else
0.0
preparation:
U64.to_f64(moment.countdown) / U64.to_f64(App.KL.Constants.round_time)
execution: 0.0
return hero
}
<div style={
"position": "absolute"
"top": "4%"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"height": "8.5%"
"background-color": "#eb6bff"
"border-radius": "2px"
"width": String.show_clean(F64.show(bar * 44.5)) | "%"
}>
</div>
App.KL.Game.Phase.Play.draw.get_players_heroes(players: Map<App.KL.Game.Player>): List<App.KL.Game.Hero>
let list = Map.values!(players)
let list = List.mapped!(list)!((player) App.KL.Game.Hero.get_by_id(player@hero_id <> 65536))
let list = List.somes!(list)
list
App.KL.Game.Phase.Play.draw.interface.top.round(round: U64): DOM
<div style={
"position": "absolute"
"bottom": "8%"
"font-size": "1.2rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}>
<span>Nat.show(U64.to_nat(round))</span>
</div>
App.KL.Game.Phase.Play.draw.get_players_skills(players: Map<App.KL.Game.Player>): List<SkillInfo>
let game_heroes = App.KL.Game.Phase.Play.draw.get_players_heroes(players)
let game_skills = List.flatten!(List.mapped!(game_heroes)!((hero) List.mapped!(Map.to_list!(hero@skills))!((x) {Pair.fst!!(x), {hero,Pair.snd!!(x)}})))
let game_skills = List.sort!((a,b) let as = Pair.snd!!(Pair.snd!!(a)); bs = Pair.snd!!(Pair.snd!!(b)); U64.cmp(as@delay, bs@delay), game_skills)
game_skills
App.KL.Game.Phase.Play.draw.interface.top.scores(scores: App.KL.Game.Score): DOM
let blue_hp = I32.sub(App.KL.Constants.max_score, scores@points@snd)
let red_hp = I32.sub(App.KL.Constants.max_score, scores@points@fst)
// return a list with skill info of all players
// the either serves to tell where the walk skills are
App.KL.Game.Phase.Play.draw.get_skills_to_tab(
players: Map<App.KL.Game.Player>
): List<Either<Unit, SkillInfo>>
let skills = App.KL.Game.Phase.Play.draw.get_players_skills(players)
App.KL.Game.Phase.Play.draw.get_skills_to_tab.go(skills, false)
App.KL.Game.Phase.Play.draw.get_skills_to_tab.go(
skills: List<SkillInfo>
found_walk: Bool
): List<Either<Unit, SkillInfo>>
case skills {
nil: []
cons:
let skill = skills.head@snd@snd
if (skill@name =? "Walk") || (skill@delay =? 500) then
if found_walk then
App.KL.Game.Phase.Play.draw.get_skills_to_tab.go(skills.tail, found_walk)
else
Either.left!!(unit) & App.KL.Game.Phase.Play.draw.get_skills_to_tab.go(skills.tail, true)
else
Either.right!!(skills.head) & App.KL.Game.Phase.Play.draw.get_skills_to_tab.go(skills.tail, found_walk)
}
<div style={
"position": "absolute"
"top": "35%"
"font-size": "1.5rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}>
<span style={"margin-right": "2.2%"}>Nat.show(I32.to_nat(blue_hp))</span>
<span>Nat.show(I32.to_nat(red_hp))</span>
</div>
App.KL.Game.Phase.Play.draw.apply_condition(
condition: App.KL.Game.Skill -> Bool
skill_info: SkillInfo
): Bool
let skill = skill_info@snd@snd
condition(skill)
type PlayersInfo {
new(
team: App.KL.Game.Team
creature: Maybe<App.KL.Game.Creature>
hero_id: Maybe<Nat>
)
}
App.KL.Game.Phase.Play.draw.get_allies_enemies(
user: String
players_info: Map<PlayersInfo>
): Pair<Map<PlayersInfo>, Map<PlayersInfo>>
let {blues, reds} = App.KL.Game.Phase.Play.draw.sep_blue_red(players_info)
let player_info = players_info{user}
case player_info {
none: {blues, reds} // when local user isnt a player, blues come first
some:
let player = player_info.value@snd
let player_team = player@team // if local user is a player, then his team come first
if App.KL.Game.Team.blue =? player_team then
{blues, reds}
else
{reds, blues}
}
App.KL.Game.Phase.Play.draw.interface.top.get_players(
// separate blue and red teams
App.KL.Game.Phase.Play.draw.sep_blue_red(
players: Map<PlayersInfo>
): Pair<Map<PlayersInfo>, Map<PlayersInfo>>
let result = {{} :: Map<PlayersInfo>, {} :: Map<PlayersInfo>}
for id:player_info in players with result:
let {allies, enemies} = result
let player = player_info@snd
if App.KL.Game.Team.eql(player@team, App.KL.Game.Team.blue) then
let allies = allies{id}<-player_info
{allies, enemies}
else
let enemies = enemies{id}<-player_info
{allies, enemies}
result
App.KL.Game.Phase.Play.draw.get_players_info(
players: Map<App.KL.Game.Player>,
board: App.KL.Game.Board
): Map<PlayersInfo>
let result = {}
for id:player in players with result:
result{id} <- PlayersInfo.new(player@team, none, player@hero_id)
result{id} <- {none, player}
for coord:tile in board with result:
Maybe {
get creature = tile@creature
get id = creature@player
get info = result{id}
return result{id} <- info@creature <- some(creature)
return result{id} <- info@fst <- some(creature)
} <> result
result
App.KL.Game.Phase.Play.draw.interface.top.players(user: String, players_info: Map<PlayersInfo>, cemetery: App.KL.Game.Cemetery, board: App.KL.Game.Board): DOM
// get team to show in left and team to show in right
let {user_team, enemy_team} =
Maybe {
get info = players_info{user}
let team = info@team
case team {
blue: some({App.KL.Game.Team.blue, App.KL.Game.Team.red})
red: some({App.KL.Game.Team.red, App.KL.Game.Team.blue})
neutral: none
}
} <> {App.KL.Game.Team.blue, App.KL.Game.Team.red} // set as default team
App.KL.Game.Phase.Play.draw.get_life(creature: Maybe<App.KL.Game.Creature>): I32
case creature {
none: 0
some: creature.value@hp
}
// TODO change when formap for DOM published
let players_info = Map.to_list!(players_info)
<div style={"display": "contents"}>
<div style={
"position": "absolute"
"top": "18%"
"left": "3.2%"
"width": "39.7%"
"height": "51%"
"display": "grid"
"grid-template-columns": "repeat(5, 1fr)"
"grid-template-rows": "1fr"
"grid-column-gap": "3px"
}>
for info in players_info:
let {id, snd_info} = info
let team = snd_info@team
let creature = snd_info@creature
let dead = List.find!((x) id =? x@fst, cemetery)
if team =? user_team then
App.KL.Game.Phase.Play.draw.interface.top.creature(creature, dead, false, id =? user)
else
<span style={"display": "none"}></span>
</div>
<div style={
"position": "absolute"
"top": "18%"
"left": "57%"
"width": "39.7%"
"height": "51%"
"display": "grid"
"grid-template-columns": "repeat(5, 1fr)"
"grid-template-rows": "1fr"
"grid-column-gap": "3px"
}>
for info in players_info:
let {id, snd_info} = info
let team = snd_info@team
let creature = snd_info@creature
let dead = List.find!((x) id =? x@fst, cemetery)
if team =? enemy_team then
App.KL.Game.Phase.Play.draw.interface.top.creature(creature, dead, true, false)
else
<span style={"display": "none"}></span>
</div>
</div>
App.KL.Game.Phase.Play.draw.interface.get_life_percentage(creature: Maybe<App.KL.Game.Creature>): F64
App.KL.Game.Phase.Play.draw.get_life_percentage(creature: Maybe<App.KL.Game.Creature>): F64
case creature {
none: 0
some: F64.mul(100.0, I32.to_f64(creature.value@hp) / I32.to_f64(creature.value@hero@max_hp))
}
App.KL.Game.Phase.Play.draw.interface.get_ap(creature: Maybe<App.KL.Game.Creature>): I32
App.KL.Game.Phase.Play.draw.get_ap(creature: Maybe<App.KL.Game.Creature>): I32
case creature {
none: 0
some: creature.value@ap
}
App.KL.Game.Phase.Play.draw.interface.top.creature(creature: Maybe<App.KL.Game.Creature>, dead: Maybe<App.KL.Game.Cemetery.Dead>, is_enemy: Bool, is_user: Bool): DOM
let life_percentage = String.show_clean(F64.show(App.KL.Game.Phase.Play.draw.interface.get_life_percentage(creature)))
let ap = App.KL.Game.Phase.Play.draw.interface.get_ap(creature)
<div style={"position": "relative", "width": "100%", "height": "100%", "order": if is_user then "1" else "2"}>
// portrait
<img src=App.KL.Game.Phase.Play.draw.assets.interface.top.portrait_placeholder style={
"top": "2%"
"left": "4.5%"
"transform-origin": "top left"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"position": "absolute"
}></img>
<div style={"position": "absolute", "top": "46%", "left": "63%", "transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")", "transform-origin": "top left"}>
{ App.KL.Game.Phase.Play.draw.interface.top.creature.life_ap(life_percentage, ap, is_enemy) }
</div>
// death portrait
{
case dead {
none: <span style={"display":"none"}></span>
some:
<div style={"display": "contents"}>
<img src=App.KL.Game.Phase.Play.draw.assets.interface.top.dead style={
"left": "0"
"transform-origin": "top left"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"position": "absolute"
}></img>
<span style={
"position": "absolute"
"top": "38%"
"font-size": ".9rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}> String.show_clean(I32.show(dead.value@snd)) | " TURNS"</span>
</div>
}
}
</div>
App.KL.Game.Phase.Play.draw.interface.top.creature.life_ap(life_percentage: String, ap: I32, is_enemy: Bool): DOM
<div style={"position": "relative"}>
// life bar
<img src=App.KL.Game.Phase.Play.draw.assets.interface.top.life_placeholder></img>
<img
src= if is_enemy then App.KL.Game.Phase.Play.draw.assets.interface.top.life_red else App.KL.Game.Phase.Play.draw.assets.interface.top.life_green
style={
"left": "2px"
"top": "1px"
"position": "absolute"
"clip-path": "polygon(0 0%, "|life_percentage|"% 0%, "|life_percentage|"% 100%, 0% 100%)"
}>
</img>
// ap bar
for img in App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap, [{2.0,21.0}, {8.0,15.0}, {14.0,9.0}]):
img
</div>
App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap: I32, positions: List<Pair<F64, F64>>): List<DOM>
if ap <=? 0 then []
else
case positions {
nil: []
cons:
<img src=App.KL.Game.Phase.Play.draw.assets.interface.top.ap style={
"position": "absolute"
"left": String.show_clean(F64.show(positions.head@fst)) | "px"
"top": String.show_clean(F64.show(positions.head@snd)) | "px"
}></img> & App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap - 1, positions.tail)
}
App.KL.game.Phase.Play.Draw.interface.bottom.get_skills(skills: Map<App.KL.Game.Skill>, max: Nat): List<Pair<String, App.KL.Game.Skill>>
result = {[] :: List<Pair<String, App.KL.Game.Skill>>, max}
for key: skill in skills with result:
if (max <=? 0) || (skill@name =? "Walk") then
result
else
let result = result@fst <- {key, skill} & result@fst
result@snd <- result@snd - 1
result@fst
App.KL.Game.Phase.Play.Draw.interface.bottom.get_walk_key(skills: Map<App.KL.Game.Skill>): String
case skills {
tip: ""
bin:
if skills.val@name =? "Walk" then
skills.key
else
App.KL.Game.Phase.Play.Draw.interface.bottom.get_walk_key(skills.left) | App.KL.Game.Phase.Play.Draw.interface.bottom.get_walk_key(skills.right)
}
App.KL.Game.Phase.Play.draw.interface.bottom(user: String, players_info: Map<PlayersInfo>, skill_hovered: Maybe<App.KL.Game.Skill>): DOM
// get user in players map
// if user isnt a player this interface isnt shown
let user = players_info{user}
without user: <span style={"display": "none"}></span>
// get hero of user, if none this interface isnt shown
let hero =
Maybe {
get hero_id = user@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
return hero
}
without hero: <span style={"display": "none"}></span>
// draw content TODO change when formap implemented
let skills = hero@skills
let skills_list = App.KL.game.Phase.Play.Draw.interface.bottom.get_skills(skills, 4)
let content =
<div style={
"position": "relative"
"width": "100%"
"height": "100%"
}>
{
let walk_key = App.KL.Game.Phase.Play.Draw.interface.bottom.get_walk_key(skills)
<span style={"position": "absolute", "left": "1.25%", "top": "25%"}>walk_key</span>
}
<div style={
"position": "absolute"
"display": "grid"
"grid-template-columns": "repeat(4, 1fr)"
"grid-template-rows": "1fr"
"gap": "2%"
"width": "73%"
"height": "73%"
"left": "13.5%"
}>
for info in skills_list:
let {key, skill} = info
let id = "SK"|key
let show = Maybe.if!((x) String.eql(skill@name, x@name), skill_hovered)
<div style={
"position": "relative"
}>
<div id=id style={
"background-color": "#190d1f"
"border-radius": "5px"
"padding": "5px"
"animation": "fade-in 0.1s"
"position": "absolute"
"transform-origin": "bottom center"
"transform": "translate(-15%, -101%)"
"width": "150%"
"color": "white"
"display": if show then "block" else "none
"
}>skill@description</div>
<span id=id style={"position": "absolute", "bottom": "1%", "left": "7%"}>key</span>
<span id=id style={"position": "absolute", "width": "100%", "height": "100%", "display": "flex", "justify-content": "center", "align-items": "center", "color": "white"}>skill@name</span>
</div>
</div>
</div>
<footer style={
"position": "absolute",
"bottom": "0"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(280, 65, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.draw.assets.interface.bottom, content)}
</footer>
App.KL.Game.Phase.Play.draw.interface.user(user: String, players_info: Map<PlayersInfo>): DOM
let user = players_info{user}
without user: <span style={"display": "none"}></span>
let creature = user@creature
let life_percentage = App.KL.Game.Phase.Play.draw.interface.get_life_percentage(creature)
let ap = App.KL.Game.Phase.Play.draw.interface.get_ap(creature)
let content =
<div style={
"position": "relative"
"width": "100%"
"height": "100%"
}>
<img src=App.KL.Game.Phase.Play.draw.assets.interface.top.portrait_placeholder style={
"position": "absolute"
"transform": "scale("|F64.show(App.KL.Game.Phase.Play.draw.fator.number * 2)|")"
"transform-origin": "top left"
"top": "4%"
"left": "4%"
}></img>
<img src= App.KL.Game.Phase.Play.draw.assets.interface.user.life style={
"position": "absolute"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"transform-origin": "top left"
"top": "61%"
"left": "34%"
"clip-path": "polygon(0 0%, "|String.show_clean(F64.show(life_percentage + 8))|"% 0%, "|String.show_clean(F64.show(life_percentage))|"% 100%, 0% 100%)"
}></img>
<div style={
"position": "absolute"
"top": "74%"
"left": "31%"
"display": "flex"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"transform-origin": "top left"
}>
for i in [0 to ap : I32]:
<img src=App.KL.Game.Phase.Play.draw.assets.interface.user.ap style={
"margin-right": "-3px"
}></img>
</div>
</div>
<footer style={
"position": "absolute",
"bottom": "0"
"left": "0"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(249, 112, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.draw.assets.interface.user, content)}
</footer>
App.KL.Game.Phase.Play.draw.wrapper(width: Nat, height: Nat, bt: Nat, id: String, fator: F64, background: String, inner: DOM): DOM
<div class="pixel-art" id=id style={
"height": Nat.show(F64.to_nat(fator * Nat.to_f64(height))) | "px",
@ -502,339 +166,6 @@ App.KL.Game.Phase.Play.draw.wrapper(width: Nat, height: Nat, bt: Nat, id: String
{inner}
</div>
App.KL.Game.Phase.Play.draw.assets.interface.top: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.dead: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.life_placeholder: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.life_green: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.life_red: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.ap: String
""
App.KL.Game.Phase.Play.draw.assets.interface.top.portrait_placeholder: String
""
App.KL.Game.Phase.Play.draw.assets.interface.bottom: String
""
App.KL.Game.Phase.Play.draw.assets.interface.user: String
""
App.KL.Game.Phase.Play.draw.assets.interface.user.life: String
""
App.KL.Game.Phase.Play.draw.assets.interface.user.ap: String
""
App.KL.Game.Phase.Play.draw.board_canvas(local: App.KL.Game.State.Local, game: App.KL.Game, img: VoxBox): DOM
<div style={
"cursor": "url("| App.KL.Game.Phase.Play.draw.get_cursor(local, game) | "), auto"
"max-width": "max(100vw, 900px)",
"max-height": "max(100vh, 650px)",
"overflow": "scroll",
"display": "flex"
"justify-content": "center"
"align-items": "center"
}>{
let width = Nat.show(U32.to_nat(local@screen_size@fst / 2))
let height = Nat.show(U32.to_nat(local@screen_size@snd / 2))
DOM.vbox({
"class": "pixel-art"
"id": "game_screen",
"width": width,
"height": height,
"scale": "2"
},
{},
App.KL.Game.Phase.Play.draw.canvas(img, local, game))
}</div>
// CANVAS DRAWING
// ================
// draw canvas
App.KL.Game.Phase.Play.draw.canvas(
img: VoxBox
local: App.KL.Game.State.Local
game: App.KL.Game
): VoxBox
open game
let hits = case game.moment {
initial: none
preparation: none
execution: some(game.moment.hits)
}
let img = App.KL.Game.Phase.Play.draw.board(game, local@preview, hits, local@user, local@mouse, local@screen_size, img)
let img = App.KL.Game.Phase.Play.draw.bases(game, local@user, local@screen_size, img)
let img = App.KL.Game.Phase.Play.draw.cursor(local@mouse, game.board, local@screen_size, img)
img
// draw canvas board
App.KL.Game.Phase.Play.draw.landscape(blueprint: App.KL.Game.Board.Blueprint, screen: Pair<U32, U32> img: VoxBox): VoxBox
let landscape = blueprint@landscape
for coord:tile in landscape with img:
App.KL.Game.Phase.Play.draw.tile.terrain(tile, Hexagonal.Axial.Map.new!, none, coord, screen, img)
img
App.KL.Game.Phase.Play.draw.board(
game: App.KL.Game
preview: App.KL.Game.Cast.Preview
hits: Maybe<App.KL.Game.Indicators>
user: String
mouse: Pair<U32, U32>
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
open game
let map = game@board
let mouse_coord = Hexagonal.Axial.from_screen_xy(mouse, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
let indicators = App.KL.Game.Phase.Play.Draw.get_indicators(preview, mouse_coord, user, game)
let ap_used = App.KL.Game.Moment.get_player_used_ap(game) // If there is a skill being cast, returns the player address and used ap
let img = App.KL.Game.Phase.Play.draw.landscape(game@blueprint, screen, img)
for coord:tile in map with img:
let indicator = Hexagonal.Axial.Map.get!(coord, hits <> indicators)
let img = App.KL.Game.Phase.Play.draw.grid(coord, game@moment, screen, img)
let img = App.KL.Game.Phase.Play.draw.tile.terrain(tile@terrain, preview@picks, indicator, coord, screen, img)
let img = App.KL.Game.Phase.Play.draw.tile.indicator(indicator, coord, screen, img)
let img = App.KL.Game.Phase.Play.draw.tile.creature(game, tile@creature, coord, ap_used, screen, img)
let img = App.KL.Game.Phase.Play.draw.tile.token(game, tile@token, coord, screen, img)
img
img
// TODO optimize?
App.KL.Game.Phase.Play.draw.grid(
coord: Hexagonal.Axial
moment: App.KL.Game.Moment
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
case moment {
initial: img
preparation:
let indicator = App.KL.Game.Indicator.gray_line
let indicator_img = App.KL.Game.Indicator.get_img(indicator)
let {x, y} = App.KL.Game.Phase.Play.draw.centralize.new(coord, screen)
VoxBox.Draw.image(x, y, App.KL.Constants.z_index.grid, indicator_img, img)
execution: img
}
App.KL.Game.Phase.Play.draw.tile.token(
game: App.KL.Game
token: Maybe<App.KL.Game.Token>,
token_coord: Hexagonal.Axial,
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
without token: img
let col = case App.KL.Game.Token.get_dominance(token) as team {
none: Col32.new(100,100,100,255)
some:
case team.value {
blue: Col32.new(0, 0, 255, 255)
red: Col32.new(255, 0, 0, 255)
} default Col32.new(100,100,100,255)
}
let coords = Hexagonal.Axial.range(token_coord, token@range)
for coord in coords with img:
let {cx, cy} =
Hexagonal.Axial.to_screen_xy(coord, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
if Hexagonal.Axial.eql(coord, token_coord) then
VoxBox.Draw.square(cx,cy,App.KL.Constants.z_index.token,10,10,col,img)
else
VoxBox.Draw.square(cx,cy,App.KL.Constants.z_index.token,5,5,col,img)
img
App.KL.Game.Phase.Play.draw.bases(
game: App.KL.Game
user: String
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
case game@moment {
initial:
let player = game@players{user}
without player: img
let allies = App.KL.Game.Player.get_allies(player, game@players)
let bases = game@blueprint@bases
for coord:team in bases with img:
let indicator = App.KL.Game.Team.get_indicator(team)
App.KL.Game.Phase.Play.draw.tile.indicator(some(indicator), coord, screen, img)
for l_user:player in allies with img:
let init_pos = player@init_pos
without init_pos: img
let indicator =
if String.eql(user, l_user) then
App.KL.Game.Indicator.green_line
else
App.KL.Game.Indicator.gray_line
App.KL.Game.Phase.Play.draw.tile.indicator(some(indicator), init_pos, screen, img)
img
preparation: img
execution: img
}
App.KL.Game.Phase.Play.Draw.get_indicators(
preview: App.KL.Game.Cast.Preview
target: Hexagonal.Axial,
user: String,
game: App.KL.Game
): App.KL.Game.Indicators
Maybe {
get pair = preview@skill
let skill_key = pair@fst
get skill = App.KL.Game.Skill.get(user, skill_key, game)
let targets = App.KL.Game.Picks.get_skill_coords(skill_key, preview@picks)
let targets = target & targets
let game = App.KL.Game.Cast.simulate(skill@delay, (x) String.eql(x@player, user), game)
get center = App.KL.Game.Board.find_player_coord(user, game@board)
get creature = App.KL.Game.Board.Creature.get(center, game@board)
let indicator = App.KL.Game.Indicator.blue_line
let indicators = Hexagonal.Axial.Map.new!
let distance = App.KL.Game.Creature.Status.movement_range(Nat.to_i32(U64.to_nat(skill@range)), creature)
let coords = Hexagonal.Axial.range(center, distance)
// set range indicator
for coord in coords with indicators:
Hexagonal.Axial.Map.set!(coord, indicator, indicators)
let areas = App.KL.Game.Effect.indicators.get_indicators(center, skill, {skill_key, user} targets, game@board)
return Hexagonal.Axial.Map.union!(indicators, areas)
} <> Hexagonal.Axial.Map.new<App.KL.Game.Indicator>
// draw mouse indicator
App.KL.Game.Phase.Play.draw.cursor(
mouse: Pair<U32, U32>
board: App.KL.Game.Board
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
coord = Hexagonal.Axial.from_screen_xy(mouse, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
{x,y} = App.KL.Game.Phase.Play.draw.centralize.old(coord, screen)
let has_tile = App.KL.Game.Tile.get(coord, board)
case has_tile {
none:
img
some:
VoxBox.Draw.image(x, y, App.KL.Constants.z_index.cursor, App.KL.Game.Field.Mouse.mouse_ui, img)
}
// draws tile terrain
App.KL.Game.Phase.Play.draw.tile.terrain(
terrain: App.KL.Game.Terrain
picks: App.KL.Game.Picks
indicator: Maybe<App.KL.Game.Indicator>
tile_coord: Hexagonal.Axial
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
let indicator = indicator <> App.KL.Game.Indicator.background
let {i, j} = App.KL.Game.Phase.Play.draw.centralize.new(tile_coord, screen)
let field = App.KL.Game.Field.get_by_id.default(terrain@field_id)
let field_drawing = field@draw(terrain, indicator)
let tile_drawing = VoxBox.Draw.image(i, j, I32.to_u32(tile_coord@j + App.KL.Constants.z_index.terrain), field_drawing, img)
let tile_drawing = App.KL.Game.Phase.Play.draw.letter(tile_coord, picks, screen, tile_drawing)
tile_drawing
App.KL.Game.Phase.Play.draw.tile.indicator(
indicator: Maybe<App.KL.Game.Indicator>
tile_coord: Hexagonal.Axial
screen: Pair<U32,U32>
img: VoxBox
): VoxBox
without indicator: img
let {i, j} = App.KL.Game.Phase.Play.draw.centralize.new(tile_coord, screen)
let ind_img = App.KL.Game.Indicator.get_img(indicator)
let img = VoxBox.Draw.image(i, j, App.KL.Constants.z_index.indicator, ind_img, img)
img
App.KL.Game.Phase.Play.draw.tile.creature.get_draw_pose(game: App.KL.Game, coord: Hexagonal.Axial): App.KL.Game.Hero.Draw.Pose
open game
case game.moment {
initial: App.KL.Game.Hero.Draw.Pose.idle
preparation:
App.KL.Game.Hero.Draw.Pose.idle
execution: case game.moment.casts {
nil:
App.KL.Game.Hero.Draw.Pose.idle
cons: Maybe {
//get success = if game.moment.success then some(unit) else none
get success = case game.moment.success {
left: none
right: some(unit)
}
get creature = App.KL.Game.Board.Creature.get(coord, game.board)
let cast = game.moment.casts.head
get player = creature@player
open cast
if String.eql(player, cast.player) then Maybe {
let frame = game.moment.frame
let center = game.moment.coord <> coord
let target = cast.target
let letter = cast.letter
get skill = App.KL.Game.Cast.get_skill(cast, game)
get creature = App.KL.Game.Cast.get_creature(cast, game)
return App.KL.Game.Hero.Draw.Pose.cast(frame, center, target, letter, skill, creature)
} else none
} <> App.KL.Game.Hero.Draw.Pose.idle
}
}
// draws tile creature if any
App.KL.Game.Phase.Play.draw.tile.creature(
game: App.KL.Game
creature: Maybe<App.KL.Game.Creature>,
coord: Hexagonal.Axial,
ap_used: Maybe<Pair<String, I32>>
screen: Pair<U32,U32>
img: VoxBox
): VoxBox
case creature {
none: img
some:
let hero = creature.value@hero
let draw_pose = App.KL.Game.Phase.Play.draw.tile.creature.get_draw_pose(game, coord)
let {draw_coord, draw_voxbox} = hero@draw@vbox_img(draw_pose)
let draw_coord = draw_coord <> coord
//let aux = I32.to_u32(App.KL.Constants.hexagon_radius) //U32
//let cy = cy - (aux * 2) //U32
//let cx = cx - aux //U32
//log(U32.show( cy + 100))
{cx, cy} = Hexagonal.Axial.to_screen_xy(draw_coord, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
let img_x = cx - (128::U32)
let img_y = cy - (128::U32)
let height = I32.to_u32(hero@draw@height)
let img = VoxBox.Draw.image(img_x, img_y, App.KL.Constants.z_index.hero, draw_voxbox, img)
open creature.value as creature
case creature.player {
none:
img
some:
let img = App.KL.Game.Phase.Play.Bar.show_hp(cx - 17, (cy - height), creature.value, img)
let img = App.KL.Game.Phase.Play.draw.ap(img_x - 12, (img_y - height) - 3, creature.value, ap_used, img)
img
}
}
// AUX FUNCTIONS
// =============
App.KL.Game.Phase.Play.draw.get_cursor(
local: App.KL.Game.State.Local
game : App.KL.Game
@ -949,7 +280,7 @@ App.KL.Game.Phase.Play.draw.shield(
let shield = Nat.show(Int.to_nat(shield))
VoxBox.Draw.text(shield, PixelFont.small_black, Pos32.new(cx, cy, 0), img)
App.KL.Game.Phase.Play.Draw.gray_overlay(game: App.KL.Game): DOM
App.KL.Game.Phase.Play.draw.gray_overlay(game: App.KL.Game): DOM
open game
case game.moment {
initial: <div></div>

View File

@ -0,0 +1,299 @@
App.KL.Game.Phase.Play.draw.canvas.elem(
local: App.KL.Game.State.Local,
game: App.KL.Game, img: VoxBox
): DOM
<div style={
"cursor": "url("| App.KL.Game.Phase.Play.draw.get_cursor(local, game) | "), auto"
"max-width": "max(100vw, 900px)",
"max-height": "max(100vh, 650px)",
"overflow": "scroll",
"display": "flex"
"justify-content": "center"
"align-items": "center"
}>{
let width = Nat.show(U32.to_nat(local@screen_size@fst / 2))
let height = Nat.show(U32.to_nat(local@screen_size@snd / 2))
DOM.vbox({
"class": "pixel-art"
"id": "game_screen",
"width": width,
"height": height,
"scale": "2"
},
{},
App.KL.Game.Phase.Play.draw.canvas(img, local, game))
}</div>
// CANVAS DRAWING
// ================
// draw canvas
App.KL.Game.Phase.Play.draw.canvas(
img: VoxBox
local: App.KL.Game.State.Local
game: App.KL.Game
): VoxBox
open game
let hits = case game.moment {
initial: none
preparation: none
execution: some(game.moment.hits)
}
let img = App.KL.Game.Phase.Play.draw.canvas.board(game, local@preview, hits, local@user, local@mouse, local@screen_size, img)
let img = App.KL.Game.Phase.Play.draw.canvas.bases(game, local@user, local@screen_size, img)
let img = App.KL.Game.Phase.Play.draw.canvas.cursor(local@mouse, game.board, local@screen_size, img)
img
// draw canvas board
App.KL.Game.Phase.Play.draw.canvas.landscape(blueprint: App.KL.Game.Board.Blueprint, screen: Pair<U32, U32> img: VoxBox): VoxBox
let landscape = blueprint@landscape
for coord:tile in landscape with img:
App.KL.Game.Phase.Play.draw.canvas.tile.terrain(tile, Hexagonal.Axial.Map.new!, none, coord, screen, img)
img
App.KL.Game.Phase.Play.draw.canvas.board(
game: App.KL.Game
preview: App.KL.Game.Cast.Preview
hits: Maybe<App.KL.Game.Indicators>
user: String
mouse: Pair<U32, U32>
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
open game
let map = game@board
let mouse_coord = Hexagonal.Axial.from_screen_xy(mouse, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
let indicators = App.KL.Game.Phase.Play.Draw.canvas.get_indicators(preview, mouse_coord, user, game)
let ap_used = App.KL.Game.Moment.get_player_used_ap(game) // If there is a skill being cast, returns the player address and used ap
let img = App.KL.Game.Phase.Play.draw.canvas.landscape(game@blueprint, screen, img)
for coord:tile in map with img:
let indicator = Hexagonal.Axial.Map.get!(coord, hits <> indicators)
let img = App.KL.Game.Phase.Play.draw.canvas.grid(coord, game@moment, screen, img)
let img = App.KL.Game.Phase.Play.draw.canvas.tile.terrain(tile@terrain, preview@picks, indicator, coord, screen, img)
let img = App.KL.Game.Phase.Play.draw.canvas.tile.indicator(indicator, coord, screen, img)
let img = App.KL.Game.Phase.Play.draw.canvas.tile.creature(game, tile@creature, coord, ap_used, screen, img)
let img = App.KL.Game.Phase.Play.draw.canvas.tile.token(game, tile@token, coord, screen, img)
img
img
// TODO optimize?
App.KL.Game.Phase.Play.draw.canvas.grid(
coord: Hexagonal.Axial
moment: App.KL.Game.Moment
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
case moment {
initial: img
preparation:
let indicator = App.KL.Game.Indicator.gray_line
let indicator_img = App.KL.Game.Indicator.get_img(indicator)
let {x, y} = App.KL.Game.Phase.Play.draw.centralize.new(coord, screen)
VoxBox.Draw.image(x, y, App.KL.Constants.z_index.grid, indicator_img, img)
execution: img
}
App.KL.Game.Phase.Play.draw.canvas.tile.token(
game: App.KL.Game
token: Maybe<App.KL.Game.Token>,
token_coord: Hexagonal.Axial,
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
without token: img
let col = case App.KL.Game.Token.get_dominance(token) as team {
none: Col32.new(100,100,100,255)
some:
case team.value {
blue: Col32.new(0, 0, 255, 255)
red: Col32.new(255, 0, 0, 255)
} default Col32.new(100,100,100,255)
}
let coords = Hexagonal.Axial.range(token_coord, token@range)
for coord in coords with img:
let {cx, cy} =
Hexagonal.Axial.to_screen_xy(coord, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
if Hexagonal.Axial.eql(coord, token_coord) then
VoxBox.Draw.square(cx,cy,App.KL.Constants.z_index.token,10,10,col,img)
else
VoxBox.Draw.square(cx,cy,App.KL.Constants.z_index.token,5,5,col,img)
img
App.KL.Game.Phase.Play.draw.canvas.bases(
game: App.KL.Game
user: String
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
case game@moment {
initial:
let player = game@players{user}
without player: img
let allies = App.KL.Game.Player.get_allies(player, game@players)
let bases = game@blueprint@bases
for coord:team in bases with img:
let indicator = App.KL.Game.Team.get_indicator(team)
App.KL.Game.Phase.Play.draw.canvas.tile.indicator(some(indicator), coord, screen, img)
for l_user:player in allies with img:
let init_pos = player@init_pos
without init_pos: img
let indicator =
if String.eql(user, l_user) then
App.KL.Game.Indicator.green_line
else
App.KL.Game.Indicator.gray_line
App.KL.Game.Phase.Play.draw.canvas.tile.indicator(some(indicator), init_pos, screen, img)
img
preparation: img
execution: img
}
App.KL.Game.Phase.Play.Draw.canvas.get_indicators(
preview: App.KL.Game.Cast.Preview
target: Hexagonal.Axial,
user: String,
game: App.KL.Game
): App.KL.Game.Indicators
Maybe {
get pair = preview@skill
let skill_key = pair@fst
get skill = App.KL.Game.Skill.get(user, skill_key, game)
let targets = App.KL.Game.Picks.get_skill_coords(skill_key, preview@picks)
let targets = target & targets
let game = App.KL.Game.Cast.simulate(skill@delay, (x) String.eql(x@player, user), game)
get center = App.KL.Game.Board.find_player_coord(user, game@board)
get creature = App.KL.Game.Board.Creature.get(center, game@board)
let indicator = App.KL.Game.Indicator.blue_line
let indicators = Hexagonal.Axial.Map.new!
let distance = App.KL.Game.Creature.Status.movement_range(Nat.to_i32(U64.to_nat(skill@range)), creature)
let coords = Hexagonal.Axial.range(center, distance)
// set range indicator
for coord in coords with indicators:
Hexagonal.Axial.Map.set!(coord, indicator, indicators)
let areas = App.KL.Game.Effect.indicators.get_indicators(center, skill, {skill_key, user} targets, game@board)
return Hexagonal.Axial.Map.union!(indicators, areas)
} <> Hexagonal.Axial.Map.new<App.KL.Game.Indicator>
// draw mouse indicator
App.KL.Game.Phase.Play.draw.canvas.cursor(
mouse: Pair<U32, U32>
board: App.KL.Game.Board
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
coord = Hexagonal.Axial.from_screen_xy(mouse, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
{x,y} = App.KL.Game.Phase.Play.draw.centralize.old(coord, screen)
let has_tile = App.KL.Game.Tile.get(coord, board)
case has_tile {
none:
img
some:
VoxBox.Draw.image(x, y, App.KL.Constants.z_index.cursor, App.KL.Game.Field.Mouse.mouse_ui, img)
}
// draws tile terrain
App.KL.Game.Phase.Play.draw.canvas.tile.terrain(
terrain: App.KL.Game.Terrain
picks: App.KL.Game.Picks
indicator: Maybe<App.KL.Game.Indicator>
tile_coord: Hexagonal.Axial
screen: Pair<U32, U32>
img: VoxBox
): VoxBox
let indicator = indicator <> App.KL.Game.Indicator.background
let {i, j} = App.KL.Game.Phase.Play.draw.centralize.new(tile_coord, screen)
let field = App.KL.Game.Field.get_by_id.default(terrain@field_id)
let field_drawing = field@draw(terrain, indicator)
let tile_drawing = VoxBox.Draw.image(i, j, I32.to_u32(tile_coord@j + App.KL.Constants.z_index.terrain), field_drawing, img)
let tile_drawing = App.KL.Game.Phase.Play.draw.letter(tile_coord, picks, screen, tile_drawing)
tile_drawing
App.KL.Game.Phase.Play.draw.canvas.tile.indicator(
indicator: Maybe<App.KL.Game.Indicator>
tile_coord: Hexagonal.Axial
screen: Pair<U32,U32>
img: VoxBox
): VoxBox
without indicator: img
let {i, j} = App.KL.Game.Phase.Play.draw.centralize.new(tile_coord, screen)
let ind_img = App.KL.Game.Indicator.get_img(indicator)
let img = VoxBox.Draw.image(i, j, App.KL.Constants.z_index.indicator, ind_img, img)
img
App.KL.Game.Phase.Play.draw.canvas.tile.creature.get_draw_pose(game: App.KL.Game, coord: Hexagonal.Axial): App.KL.Game.Hero.Draw.Pose
open game
case game.moment {
initial: App.KL.Game.Hero.Draw.Pose.idle
preparation:
App.KL.Game.Hero.Draw.Pose.idle
execution: case game.moment.casts {
nil:
App.KL.Game.Hero.Draw.Pose.idle
cons: Maybe {
//get success = if game.moment.success then some(unit) else none
get success = case game.moment.success {
left: none
right: some(unit)
}
get creature = App.KL.Game.Board.Creature.get(coord, game.board)
let cast = game.moment.casts.head
get player = creature@player
open cast
if String.eql(player, cast.player) then Maybe {
let frame = game.moment.frame
let center = game.moment.coord <> coord
let target = cast.target
let letter = cast.letter
get skill = App.KL.Game.Cast.get_skill(cast, game)
get creature = App.KL.Game.Cast.get_creature(cast, game)
return App.KL.Game.Hero.Draw.Pose.cast(frame, center, target, letter, skill, creature)
} else none
} <> App.KL.Game.Hero.Draw.Pose.idle
}
}
// draws tile creature if any
App.KL.Game.Phase.Play.draw.canvas.tile.creature(
game: App.KL.Game
creature: Maybe<App.KL.Game.Creature>,
coord: Hexagonal.Axial,
ap_used: Maybe<Pair<String, I32>>
screen: Pair<U32,U32>
img: VoxBox
): VoxBox
case creature {
none: img
some:
let hero = creature.value@hero
let draw_pose = App.KL.Game.Phase.Play.draw.canvas.tile.creature.get_draw_pose(game, coord)
let {draw_coord, draw_voxbox} = hero@draw@vbox_img(draw_pose)
let draw_coord = draw_coord <> coord
//let aux = I32.to_u32(App.KL.Constants.hexagon_radius) //U32
//let cy = cy - (aux * 2) //U32
//let cx = cx - aux //U32
//log(U32.show( cy + 100))
{cx, cy} = Hexagonal.Axial.to_screen_xy(draw_coord, App.KL.Constants.hexagon_radius, U32.to_i32(screen@fst) / 4, U32.to_i32(screen@snd) / 4)
let img_x = cx - (128::U32)
let img_y = cy - (128::U32)
let height = I32.to_u32(hero@draw@height)
let img = VoxBox.Draw.image(img_x, img_y, App.KL.Constants.z_index.hero, draw_voxbox, img)
open creature.value as creature
case creature.player {
none:
img
some:
let img = App.KL.Game.Phase.Play.Bar.show_hp(cx - 17, (cy - height), creature.value, img)
let img = App.KL.Game.Phase.Play.draw.ap(img_x - 12, (img_y - height) - 3, creature.value, ap_used, img)
img
}
}

View File

@ -0,0 +1,429 @@
App.KL.Game.Phase.Play.draw.interface(
local: App.KL.Game.State.Local,
game: App.KL.Game
players_info: Map<PlayersInfo>
): DOM
<div>
{App.KL.Game.Phase.Play.draw.interface.top(players_info, local, game)}
{App.KL.Game.Phase.Play.draw.interface.bottom(local@user, players_info, local@hud@skill)}
{App.KL.Game.Phase.Play.draw.interface.user(local@user, players_info)}
{App.KL.Game.Phase.Play.draw.interface.casts(local@user, game@moment, game@players)}
</div>
App.KL.Game.Phase.Play.draw.interface.casts(user: String, moment: App.KL.Game.Moment, players: Map<App.KL.Game.Player>): DOM
case moment {
execution:
casts = moment.casts
p_cst = moment.previous_casts
<div style={
"position": "absolute"
"bottom": "0"
"right": "0"
"width": "300px"
"margin-right": "5px"
}>
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, p_cst[1], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, p_cst[0], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(true, casts[0], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, casts[1], players)}
{App.KL.Game.Phase.Play.draw.interface.casts.item(false, casts[2], players)}
</div>
} default <span style={"display": "none"}></span>
App.KL.Game.Phase.Play.draw.interface.casts.item(main: Bool, cast: Maybe<App.KL.Game.Cast>, players: Map<App.KL.Game.Player>): DOM
let name = Maybe {
get cast = cast
get player = players{cast@player}
get hero_id = player@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
get skill = hero@skills{Char.to_string(cast@letter)}
return hero@name | " - " | skill@name
} <> ""
<div style={
"color": "white"
"background-color": if main && Bool.not(name =? "") then "#2a1956" else "transparent"
"padding": "0 20px"
"margin-left": if main then "0" else "2%"
"height": if main then "48px" else "36px"
"flex": if main then "2" else "1"
"display": "flex"
"border-radius": "5px"
"align-items": "center"
}>name</div>
App.KL.Game.Phase.Play.draw.interface.top(players_info: Map<PlayersInfo>, local: App.KL.Game.State.Local, game: App.KL.Game): DOM
let inner =
<div>
{App.KL.Game.Phase.Play.draw.interface.top.timer(game@moment)}
{App.KL.Game.Phase.Play.draw.interface.top.round(game@turn)}
{App.KL.Game.Phase.Play.draw.interface.top.scores(game@score)}
{App.KL.Game.Phase.Play.draw.interface.top.players(local@user, players_info, game@cemetery, game@board)}
</div>
<header style={
"position": "absolute",
"top": "0"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(624, 91, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.Assets.interface.top, inner)}
</header>
App.KL.Game.Phase.Play.draw.interface.top.timer(moment: App.KL.Game.Moment): DOM
let bar = case moment {
initial:
if moment.countdown >? App.KL.Constants.initital_delay then
U64.to_f64(moment.countdown - App.KL.Constants.initital_delay) / U64.to_f64(App.KL.Constants.initial_time)
else
0.0
preparation:
U64.to_f64(moment.countdown) / U64.to_f64(App.KL.Constants.round_time)
execution: 0.0
}
<div style={
"position": "absolute"
"top": "4%"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"height": "8.5%"
"background-color": App.KL.Game.Phase.Play.draw.theme.secondary
"border-radius": "2px"
"width": String.show_clean(F64.show(bar * 44.5)) | "%"
}>
</div>
App.KL.Game.Phase.Play.draw.interface.top.round(round: U64): DOM
<div style={
"position": "absolute"
"bottom": "8%"
"font-size": "1.2rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}>
<span>Nat.show(U64.to_nat(round))</span>
</div>
App.KL.Game.Phase.Play.draw.interface.top.scores(scores: App.KL.Game.Score): DOM
let blue_hp = I32.sub(App.KL.Constants.max_score, scores@points@snd)
let red_hp = I32.sub(App.KL.Constants.max_score, scores@points@fst)
<div style={
"position": "absolute"
"top": "35%"
"font-size": "1.5rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}>
<span style={"margin-right": "2.2%"}>Nat.show(I32.to_nat(blue_hp))</span>
<span>Nat.show(I32.to_nat(red_hp))</span>
</div>
App.KL.Game.Phase.Play.draw.interface.top.players(user: String, players_info: Map<PlayersInfo>, cemetery: App.KL.Game.Cemetery, board: App.KL.Game.Board): DOM
// get team to show in left and team to show in right
let {user_team, enemy_team} =
Maybe {
get info = players_info{user}
let player = info@snd
let team = player@team
case team {
blue: some({App.KL.Game.Team.blue, App.KL.Game.Team.red})
red: some({App.KL.Game.Team.red, App.KL.Game.Team.blue})
neutral: none
}
} <> {App.KL.Game.Team.blue, App.KL.Game.Team.red} // set as default team
// TODO change when formap for DOM published
let players_info = Map.to_list!(players_info)
<div style={"display": "contents"}>
<div style={
"position": "absolute"
"top": "18%"
"left": "3.2%"
"width": "39.7%"
"height": "51%"
"display": "grid"
"grid-template-columns": "repeat(5, 1fr)"
"grid-template-rows": "1fr"
"grid-column-gap": "3px"
}>
for info in players_info:
let {id, player_info} = info
let {creature, player} = player_info
let team = player@team
let dead = List.find!((x) id =? x@fst, cemetery)
if team =? user_team then
App.KL.Game.Phase.Play.draw.interface.top.creature(creature, dead, false, id =? user)
else
<span style={"display": "none"}></span>
</div>
<div style={
"position": "absolute"
"top": "18%"
"left": "57%"
"width": "39.7%"
"height": "51%"
"display": "grid"
"grid-template-columns": "repeat(5, 1fr)"
"grid-template-rows": "1fr"
"grid-column-gap": "3px"
}>
for info in players_info:
let {id, player_info} = info
let {creature, player} = player_info
let team = player@team
let dead = List.find!((x) id =? x@fst, cemetery)
if team =? enemy_team then
App.KL.Game.Phase.Play.draw.interface.top.creature(creature, dead, true, false)
else
<span style={"display": "none"}></span>
</div>
</div>
App.KL.Game.Phase.Play.draw.interface.top.creature(creature: Maybe<App.KL.Game.Creature>, dead: Maybe<App.KL.Game.Cemetery.Dead>, is_enemy: Bool, is_user: Bool): DOM
let life_percentage = String.show_clean(F64.show(App.KL.Game.Phase.Play.draw.get_life_percentage(creature)))
let ap = App.KL.Game.Phase.Play.draw.get_ap(creature)
<div style={"position": "relative", "width": "100%", "height": "100%", "order": if is_user then "1" else "2"}>
// portrait
<img src=App.KL.Game.Phase.Play.Assets.interface.top.portrait_placeholder style={
"top": "2%"
"left": "4.5%"
"transform-origin": "top left"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"position": "absolute"
}></img>
<div style={"position": "absolute", "top": "46%", "left": "63%", "transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")", "transform-origin": "top left"}>
{ App.KL.Game.Phase.Play.draw.interface.top.creature.life_ap(life_percentage, ap, is_enemy) }
</div>
// death portrait
{
case dead {
none: <span style={"display":"none"}></span>
some:
<div style={"display": "contents"}>
<img src=App.KL.Game.Phase.Play.Assets.interface.top.dead style={
"left": "0"
"transform-origin": "top left"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"position": "absolute"
}></img>
<span style={
"position": "absolute"
"top": "38%"
"font-size": ".9rem"
"margin-left":" auto"
"margin-right":" auto"
"left":" 0"
"right":" 0"
"text-align":" center"
"color": "white"
}> String.show_clean(I32.show(dead.value@snd)) | " TURNS"</span>
</div>
}
}
</div>
App.KL.Game.Phase.Play.draw.interface.top.creature.life_ap(life_percentage: String, ap: I32, is_enemy: Bool): DOM
<div style={"position": "relative"}>
// life bar
<img src=App.KL.Game.Phase.Play.Assets.interface.top.life_placeholder></img>
<img
src= if is_enemy then App.KL.Game.Phase.Play.Assets.interface.top.life_red else App.KL.Game.Phase.Play.Assets.interface.top.life_green
style={
"left": "2px"
"top": "1px"
"position": "absolute"
"clip-path": "polygon(0 0%, "|life_percentage|"% 0%, "|life_percentage|"% 100%, 0% 100%)"
}>
</img>
// ap bar
for img in App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap, [{2.0,21.0}, {8.0,15.0}, {14.0,9.0}]):
img
</div>
App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap: I32, positions: List<Pair<F64, F64>>): List<DOM>
if ap <=? 0 then []
else
case positions {
nil: []
cons:
<img src=App.KL.Game.Phase.Play.Assets.interface.top.ap style={
"position": "absolute"
"left": String.show_clean(F64.show(positions.head@fst)) | "px"
"top": String.show_clean(F64.show(positions.head@snd)) | "px"
}></img> & App.KL.Game.Phase.Play.draw.interface.top.creature.ap(ap - 1, positions.tail)
}
App.KL.Game.Phase.Play.draw.interface.bottom(user: String, players_info: Map<PlayersInfo>, skill_hovered: Maybe<App.KL.Game.Skill>): DOM
// get user in players map
// if user isnt a player this interface isnt shown
let info = players_info{user}
without info: <span style={"display": "none"}></span>
// get hero of user, if none this interface isnt shown
let hero =
Maybe {
let player = info@snd
get hero_id = player@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
return hero
}
without hero: <span style={"display": "none"}></span>
// draw content TODO change when formap implemented
let skills = hero@skills
let skills_list = App.KL.Game.Phase.Play.draw.interface.bottom.get_skills(skills, 4)
let content =
<div style={
"position": "relative"
"width": "100%"
"height": "100%"
}>
{
let walk_key = App.KL.Game.Phase.Play.draw.interface.bottom.get_walk_key(skills)
<span style={"position": "absolute", "left": "1.25%", "top": "25%"}>walk_key</span>
}
<div style={
"position": "absolute"
"display": "grid"
"grid-template-columns": "repeat(4, 1fr)"
"grid-template-rows": "1fr"
"gap": "2%"
"width": "73%"
"height": "73%"
"left": "13.5%"
}>
for info in skills_list:
let {key, skill} = info
let id = "SK"|key
let show = Maybe.if!((x) String.eql(skill@name, x@name), skill_hovered)
<div style={
"position": "relative"
}>
<div id=id style={
"background-color": "#190d1f"
"border-radius": "5px"
"padding": "5px"
"animation": "fade-in 0.1s"
"position": "absolute"
"transform-origin": "bottom center"
"transform": "translate(-15%, -101%)"
"width": "150%"
"color": "white"
"display": if show then "block" else "none
"
}>skill@description</div>
<span id=id style={"position": "absolute", "bottom": "1%", "left": "7%"}>key</span>
<span id=id style={"position": "absolute", "width": "100%", "height": "100%", "display": "flex", "justify-content": "center", "align-items": "center", "color": "white"}>skill@name</span>
</div>
</div>
</div>
<footer style={
"position": "absolute",
"bottom": "0"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(280, 65, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.Assets.interface.bottom, content)}
</footer>
App.KL.Game.Phase.Play.draw.interface.user(user: String, players_info: Map<PlayersInfo>): DOM
let info = players_info{user}
without info: <span style={"display": "none"}></span>
let creature = info@fst
let life_percentage = App.KL.Game.Phase.Play.draw.get_life_percentage(creature)
let ap = App.KL.Game.Phase.Play.draw.get_ap(creature)
let content =
<div style={
"position": "relative"
"width": "100%"
"height": "100%"
}>
<img src=App.KL.Game.Phase.Play.Assets.interface.top.portrait_placeholder style={
"position": "absolute"
"transform": "scale("|F64.show(App.KL.Game.Phase.Play.draw.fator.number * 2)|")"
"transform-origin": "top left"
"top": "4%"
"left": "4%"
}></img>
<img src= App.KL.Game.Phase.Play.Assets.interface.user.life style={
"position": "absolute"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"transform-origin": "top left"
"top": "61%"
"left": "34%"
"clip-path": "polygon(0 0%, "|String.show_clean(F64.show(life_percentage + 8))|"% 0%, "|String.show_clean(F64.show(life_percentage))|"% 100%, 0% 100%)"
}></img>
<div style={
"position": "absolute"
"top": "74%"
"left": "31%"
"display": "flex"
"transform": "scale("|App.KL.Game.Phase.Play.draw.fator.string|")"
"transform-origin": "top left"
}>
for i in [0 to ap : I32]:
<img src=App.KL.Game.Phase.Play.Assets.interface.user.ap style={
"margin-right": "-3px"
}></img>
</div>
</div>
<footer style={
"position": "absolute",
"bottom": "0"
"left": "0"
"width": "min-content"
}>
{App.KL.Game.Phase.Play.draw.wrapper(249, 112, 0, "", App.KL.Game.Phase.Play.draw.fator.number, App.KL.Game.Phase.Play.Assets.interface.user, content)}
</footer>
// AUX FUNCTION
// =========================================
App.KL.Game.Phase.Play.draw.interface.bottom.get_skills(
skills: Map<App.KL.Game.Skill>,
max: Nat
): List<Pair<String, App.KL.Game.Skill>>
result = {[] :: List<Pair<String, App.KL.Game.Skill>>, max}
for key: skill in skills with result:
if (max <=? 0) || (skill@name =? "Walk") then
result
else
let result = result@fst <- {key, skill} & result@fst
result@snd <- result@snd - 1
result@fst
App.KL.Game.Phase.Play.draw.interface.bottom.get_walk_key(skills: Map<App.KL.Game.Skill>): String
case skills {
tip: ""
bin:
if skills.val@name =? "Walk" then
skills.key
else
App.KL.Game.Phase.Play.draw.interface.bottom.get_walk_key(skills.left) | App.KL.Game.Phase.Play.draw.interface.bottom.get_walk_key(skills.right)
}

View File

@ -0,0 +1,520 @@
App.KL.Game.Phase.Play.draw.tab(
local: App.KL.Game.State.Local,
game: App.KL.Game,
players_info: Map<PlayersInfo>
): DOM
let tab_active = local@hud@tab_active
let tab_which = local@hud@tab_which
if tab_active then
<div style={
"position": "fixed"
"top": "0"
"left": "0"
"width": "100vw"
"height": "100vh"
"background-color": "rgba(0,0,0,0.35)"
"animation": "fade-in 100ms"
}>
<div class="pixel-art" style={
"position": "fixed"
"top": "50%"
"left": "50%"
"transform": "scale("| App.KL.Game.Phase.Play.draw.fator.string |") translate(-50%, -50%)"
"transform-origin": "top left"
"box-shadow": "100% 100% 30px black"
}>
<div style={
"position": "relative"
}>
<img src=App.KL.Game.Phase.Play.Assets.tab.background></img>
{ App.KL.Game.Phase.Play.draw.tab.close }
{ App.KL.Game.Phase.Play.draw.tab.score(game@score) }
{ App.KL.Game.Phase.Play.draw.tab.timer }
{ App.KL.Game.Phase.Play.draw.tab.content(tab_which, local@user, players_info) }
{ App.KL.Game.Phase.Play.draw.tab.controls(tab_which) }
</div>
</div>
</div>
else
<span style={"display": "none"}></span>
App.KL.Game.Phase.Play.draw.tab.score(score: App.KL.Game.Score): DOM
let blue_hp = I32.sub(App.KL.Constants.max_score, score@points@snd)
let red_hp = I32.sub(App.KL.Constants.max_score, score@points@fst)
<div style={
"position": "absolute"
"width": "87%"
"top": "5%"
"display": "flex"
"justify-content": "space-between"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"color": "white"
}>
<span> String.show_clean(I32.show(blue_hp)) </span>
<span> String.show_clean(I32.show(red_hp)) </span>
</div>
App.KL.Game.Phase.Play.draw.tab.close: DOM
<div style={
"position": "absolute"
"right": "-12px"
"top": "-8px"
}>
<img src=App.KL.Game.Phase.Play.Assets.tab.button.close></img>
</div>
App.KL.Game.Phase.Play.draw.tab.timer: DOM
<div style={
"position": "absolute"
"top": "8.3%"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
"color": App.KL.Game.Phase.Play.draw.theme.secondary
}>
<span>"13:58"</span>
</div>
App.KL.Game.Phase.Play.draw.tab.content(
tab_which: App.KL.Game.Controls.Hud.Tab
user: String
players: Map<PlayersInfo>
): DOM
<div style={
"left": "1%"
"top": "15%"
"position": "absolute"
"height": "72%"
"width": "98%"
}>
{
case tab_which {
stats: App.KL.Game.Phase.Play.draw.tab.content.stats(user, players)
recap: App.KL.Game.Phase.Play.draw.tab.content.recap(user, players)
skills:
let players = BBT.map!!!((x) x@snd, players)
App.KL.Game.Phase.Play.draw.tab.content.skills(players)
}
}
</div>
App.KL.Game.Phase.Play.draw.tab.content.skills(players: Map<App.KL.Game.Player>): DOM
let rows_quantity = 10 // 7 is the number of rows
let skills = App.KL.Game.Phase.Play.draw.get_skills_to_tab(players)
let blocks = List.chunks_of!(rows_quantity, skills) // 7 is the number of rows
let blocks = List.indices!(blocks)
let last = List.length!(blocks) - 1
<div style={
"height": "100%"
"display": "flex"
"margin": "0 5%"
"overflow-x": "auto"
"animation": "fade-in 100ms"
}>
for block in blocks:
let { block_number, skills_info } = block
<div style={"display": "contents"}>
<div style={"height": "100%"}>
<div style={
"height": "8%"
"display": "flex"
"align-items": "center"
"margin-bottom": "5px"
}>
{
if block_number =? 0 then
<div style={"display": "contents"}>
<img
style={"margin-right": "5px"}
src=App.KL.Game.Phase.Play.Assets.tab.arrow.1
></img>
<span style={"color": "#861cef"}>"Beginning"</span>
</div>
else
<img src=App.KL.Game.Phase.Play.Assets.tab.arrow.2></img>
}
</div>
for skill_info in skills_info:
<div style={
"height": "7%"
"display": "flex"
"align-items": "center"
"text-transform": "uppercase"
"color": "white"
"margin-bottom": "5px"
}>
{
case skill_info {
left:
<img
style={"width": "100%"}
src=App.KL.Game.Phase.Play.Assets.tab.divisor.walk
> </img>
right:
let {key, info} = skill_info.value
let {hero, skill} = info
<div style={"display": "contents"}>
<img
style={"margin-right": "5px", "height": "100%"}
src=App.KL.Game.Phase.Play.Assets.tab.portrait.neutral
></img>
<img
style={"margin-right": "5px", "height": "100%"}
src=App.KL.Game.Phase.Play.Assets.tab.portrait.neutral
></img>
<span style={"font-size": ".55rem"}>skill@name</span>
</div>
}
}
</div>
</div>
<div style={"margin": "0 10px", "display": "flex", "align-items": "flex-end"}>
{
if block_number =? last then
<span style={"display": "none"}></span>
else
<img
src=App.KL.Game.Phase.Play.Assets.tab.divisor.vertical
style={"height": "91.5%"}
></img>
}
</div>
</div>
</div>
App.KL.Game.Phase.Play.draw.tab.content.stats(
user: String,
players: Map<PlayersInfo>
): DOM
let {allies, enemies} = App.KL.Game.Phase.Play.draw.get_allies_enemies(user, players)
let allies = List.indices!(Map.values!(allies))
let enemies = List.indices!(Map.values!(enemies))
<div style={"display": "flex", "justify-content": "space-between", "width": "100%", "height": "100%", "animation": "fade-in 100ms"}>
{ App.KL.Game.Phase.Play.draw.tab.content.stats.table(allies, false) }
<img
style={"height": "100%"}
src=App.KL.Game.Phase.Play.Assets.tab.divisor.vertical
></img>
{ App.KL.Game.Phase.Play.draw.tab.content.stats.table(enemies, true) }
</div>
App.KL.Game.Phase.Play.draw.tab.content.stats.table(
players: List<Pair<Nat, PlayersInfo>>
reversed: Bool
): DOM
let last = List.length!(players) - 1
let flex_direction =
if reversed then "row-reverse" else "row"
<div style={
"width": "45%"
"height": "100%"
"text-align": "center"
"color": "white"
}>
<div> // header
<div style={
"display": "flex"
"align-items": "center"
"color": App.KL.Game.Phase.Play.draw.theme.secondary
"font-size": "0.7rem"
"flex-direction": flex_direction
}>
<div style={"flex": "2"}></div>
<div style={"flex": "1"}>"Hp"</div>
<div style={"flex": "1"}>"Kills"</div>
<div style={"flex": "1"}>"Deaths"</div>
<div style={"flex": "1"}>"Assists"</div>
</div>
</div>
for info in players: // body
let {indice, player_info} = info
let creature = player_info@fst
let player = player_info@snd
let hero = App.KL.Game.Phase.Play.draw.get_player_hero(player)
let last = indice =? last
case hero {
none: <span style={"display": "none"}></span>
some: App.KL.Game.Phase.Play.draw.tab.stats.table.row(reversed, last, hero.value, player, creature)
}
</div>
App.KL.Game.Phase.Play.draw.tab.stats.table.row(
reversed: Bool
last: Bool
hero: App.KL.Game.Hero
player: App.KL.Game.Player
creature: Maybe<App.KL.Game.Creature>
): DOM
let inside_margin = // apply margin left if in right and vice-versa
if reversed then "0 0 0 5px" else "0 5px 0 0"
let outside_margin =
if reversed then "0 12px 0 0" else "0 0 0 12px"
let text_align =
if reversed then "right" else "left"
let flex_direction =
if reversed then "row-reverse" else "row"
let portrait =
if reversed then
App.KL.Game.Phase.Play.Assets.tab.portrait.enemy
else
App.KL.Game.Phase.Play.Assets.tab.portrait.ally
<div style={"height": "17%", "font-size": "0.8rem"}>
<div style={
"display": "flex"
"align-items": "center"
"flex-direction": flex_direction
"height": "calc(100% - 11px)" // 10px from 3px of bottom image + 8px of margin
}>
<div style={
"flex": "2"
"height": "100%"
}>
<div style={
"display": "flex"
"align-items": "center"
"flex-direction": flex_direction
"height": "100%"
"margin": outside_margin
}>
<img
style={"margin": inside_margin, "height": "100%"}
src=portrait
></img>
<div style={"text-transform": "uppercase", "text-align": text_align}>
<p style={"font-size": "0.8rem"}>hero@name</p>
<p style={"font-size": "0.7rem", "color": App.KL.Game.Phase.Play.draw.theme.secondary}>"User"</p>
</div>
</div>
</div>
<div style={"flex": "1"}><p>I32.show(App.KL.Game.Phase.Play.draw.get_life(creature))</p></div>
<div style={"flex": "1"}><p>I32.show(player@scoreboard@kills)</p></div>
<div style={"flex": "1"}><p>I32.show(player@scoreboard@deaths)</p></div>
<div style={"flex": "1"}><p>"100"</p></div>
</div>
{
if last then
<span style={"display": "none"}></span>
else
<img
style={"display": "flex", "margin": "4px 0", "width": "100%"}
src=App.KL.Game.Phase.Play.Assets.tab.divisor.horizontal
></img>
}
</div>
App.KL.Game.Phase.Play.draw.tab.content.recap(
user: String
players_info: Map<PlayersInfo>
): DOM
let player_info = players_info{user}
without player_info: App.KL.Game.Phase.Play.draw.tab.content.recap.placeholder // case user not in players map (espectator?)
let creature = player_info@fst
let player = player_info@snd
let recaps = List.take!(10, player@recap)
let recap_len = List.length!(recaps)
if recap_len =? 0 then
App.KL.Game.Phase.Play.draw.tab.content.recap.placeholder
else
<div style={
"width": "80%"
"height": "100%"
"animation": "fade-in 100ms"
"display": "flex"
"align-items": "flex-end"
"margin": "0 auto"
}>
<div style={"width": "100%"}>
<div style={"position": "relative", "display": "flex"}>
<img
src=App.KL.Game.Phase.Play.Assets.tab.recap.graph
style={"width": "100%"}
></img>
<div style={
"position": "absolute",
"top": "0",
"left": "0"
"width": "100%"
"height": "100%"
"display": "flex"
"flex-direction": "row-reverse"
"align-items": "flex-end"
"justify-content": "space-around"
"padding": "1px 0" // border of image
}>
for recap in recaps:
let hp = recap@new_hp
let diff = recap@hp_diff
let diff_percentage = App.KL.Game.Phase.Play.draw.tab.content.recap.get_percentage(diff, creature)
let life_percentage = App.KL.Game.Phase.Play.draw.tab.content.recap.get_percentage(hp, creature)
let life_percentage =
if diff >? 0 then life_percentage - diff_percentage
else life_percentage
let life_percentage = F64.show(life_percentage)
let diff_percentage = F64.show(diff_percentage)
<div style={
"width": "8%",
"height": "100%",
"background-color": "#2d2b86",
"border": "2px solid #6b1fcc"
"display": "flex",
"flex-direction": "column"
"justify-content": "flex-end"
}>
{
case diff >? 0 {
true: <div style={"width": "100%", "height": diff_percentage|"%", "background": "#b5e61c"}></div>
false:<div style={"width": "100%", "height": diff_percentage|"%", "background": "#ed1c24"}></div>
}
}
<div style={"width": "100%", "height": life_percentage|"%", "background": "#eb6aff"}></div>
</div>
</div>
</div>
<div style={"width": "100%", "display": "flex", "justify-content": "space-around", "flex-direction": "row-reverse", "margin-top": "15px"}>
for recap in recaps:
let skill = App.KL.Game.Phase.Play.draw.tab.content.recap.get_skill(recap, players_info)
<div style={"width": "8%"}>
<div style={
"position": "relative"
"width": "100%"
}>
<img
src=App.KL.Game.Phase.Play.Assets.tab.portrait.neutral
style={"width": "100%"}
></img>
<div style={
"display": "flex",
"justify-content": "center",
"align-items": "center",
"position": "absolute"
"width": "100%", "height": "100%"
"top": "0", "left": "0"
"color": "white"
"font-size": "0.55rem"
}>
{
case skill {
none: <span></span>
some: <span>skill.value@name</span>
}
}
</div>
</div>
<img
src=App.KL.Game.Phase.Play.Assets.tab.portrait.neutral
style={"width": "50%", "margin": "-12px auto 0 auto", "display": "block", "position": "relative"}
></img>
</div>
</div>
</div>
</div>
App.KL.Game.Phase.Play.draw.tab.content.recap.get_skill(
recap: App.KL.Game.Recap.Episode
players_info: Map<PlayersInfo>
): Maybe<App.KL.Game.Skill>
Maybe {
let address = recap@address
get player_info = players_info{address}
let player = player_info@snd
get hero_id = player@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
let skill_key = Char.to_string(recap@skill_key)
get skill = hero@skills{skill_key}
return skill
}
App.KL.Game.Phase.Play.draw.tab.content.recap.placeholder: DOM
<div style={
"width": "100%",
"height": "100%",
"display": "flex"
"justify-content": "center"
"align-items": "center"
"color": "white"
}><span>"No recap to show... :("</span></div>
App.KL.Game.Phase.Play.draw.tab.content.recap.get_percentage(
number: I32,
creature: Maybe<App.KL.Game.Creature>
): F64
case creature {
none: 0
some:
let number = I32.abs(number)
F64.mul(100.0, I32.to_f64(number) / I32.to_f64(creature.value@hero@max_hp))
}
App.KL.Game.Phase.Play.draw.tab.controls(
active_tab: App.KL.Game.Controls.Hud.Tab
): DOM
<div style={
"position": "absolute"
"top": "89%"
"margin-left": "auto"
"margin-right": "auto"
"left": "0"
"right": "0"
"text-align": "center"
}>
{
App.KL.Game.Phase.Play.draw.tab.controls.button(
App.KL.Game.Phase.Play.Assets.tab.button.left,
App.KL.Game.Controls.Hud.Tab.stats,
active_tab
)
}
{
App.KL.Game.Phase.Play.draw.tab.controls.button(
App.KL.Game.Phase.Play.Assets.tab.button.middle,
App.KL.Game.Controls.Hud.Tab.skills,
active_tab
)
}
{
App.KL.Game.Phase.Play.draw.tab.controls.button(
App.KL.Game.Phase.Play.Assets.tab.button.right,
App.KL.Game.Controls.Hud.Tab.recap,
active_tab
)
}
</div>
App.KL.Game.Phase.Play.draw.tab.controls.button(
button: App.KL.Game.Phase.Play.Assets.tab.button,
button_tab: App.KL.Game.Controls.Hud.Tab
active_tab: App.KL.Game.Controls.Hud.Tab
): DOM
let tab_text = String.to_lower(App.KL.Game.Controls.Hud.Tab.show(button_tab))
let is_active = App.KL.Game.Controls.Hud.Tab.eql(button_tab, active_tab)
<div id=tab_text style={"position": "relative", "display": "inline-flex", "margin-right": "1px"}>
<img id=tab_text src=button(is_active)></img>
<div id=tab_text style={
"position": "absolute"
"top": "0"
"width": "100%"
"height": "100%"
"display": "flex"
"justify-content": "center"
"align-items": "center"
"color": "white"
"text-transform": "capitalize"
}>
tab_text
</div>
</div>

View File

@ -4,6 +4,8 @@ App.KL.Game.Phase.Play.when(
event: App.Event,
): IO<Maybe<App.State.local<App.KL.State>>>
let room = String.take(16, Crypto.Keccak.hash(local@room))
let controls = local@hud
open game
case event {
frame:
@ -13,63 +15,78 @@ App.KL.Game.Phase.Play.when(
let new_local = new_local@screen_size <- event.info@screen_size
App.set_local!(App.KL.State.Local.game(new_local))
mouse_move:
// mouse move on canvas
if String.eql("game_screen", event.id) then
let {x,y} = event.mouse_pos
let local = local@hud <- App.KL.Game.State.Local.hud.new(none)
let local = local@hud <- local@hud@skill <- none
let local = local@mouse <- {x / 2, y / 2}
App.set_local!(App.KL.State.Local.game(local))
// mouse move on interface
else
switch String.starts_with(event.id) {
"SK":
let key = String.drop(2, event.id)
Maybe {
get user = game@players{local@user}
get hero_id = user@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
get skill = hero@skills{key}
let local = local@hud <- local@hud@skill <- some(skill)
return App.set_local!(App.KL.State.Local.game(local))
} <> App.set_local!(App.KL.State.Local.game(local@hud <- local@hud@skill <- none))
} default App.set_local!(App.KL.State.Local.game(local@hud <- App.KL.Game.State.Local.hud.new(none)))
let default_controls = App.KL.Game.Controls.Hud.initial
let controls =
switch String.starts_with(event.id) {
// hover on skills
"SK":
let key = String.drop(2, event.id)
Maybe {
get user = game@players{local@user}
get hero_id = user@hero_id
get hero = App.KL.Game.Hero.get_by_id(hero_id)
get skill = hero@skills{key}
let controls = local@hud@skill <- some(skill)
return controls
} <> controls
} default controls
let local = local@hud <- controls
App.set_local!(App.KL.State.Local.game(local))
key_down:
case game@moment {
preparation:
let preview = local@preview
switch U16.eql(event.code) {
9#16:
let score = case local@score {
none: some("score")
some: none
}
let local = local@score <- score
App.set_local!(App.KL.State.Local.game(local))
27#16:
let preview = preview@skill <- none
IO {
App.new_post!(room, App.KL.Game.Phase.Play.Event.cast.cancel_casts)
App.set_local!(App.KL.State.Local.game(local@preview <- preview))
}
}default
local =
Maybe {
get casted_skill = App.KL.Game.Skill.get(local@user, event.code, game)
let skill =
case preview@skill as skill {
none:
some({event.code, 1})
some:
if skill.value@fst =? event.code then
none
else
// if tab (esc) pressed
if event.code =? 32#16 then
log("ENTROU")
let local = App.KL.Game.Phase.Play.when.invert_tab(true, controls, local)
App.set_local!(App.KL.State.Local.game(local))
else
case game@moment {
preparation:
let preview = local@preview
switch U16.eql(event.code) {
27#16:
let preview = preview@skill <- none
IO {
App.new_post!(room, App.KL.Game.Phase.Play.Event.cast.cancel_casts)
App.set_local!(App.KL.State.Local.game(local@preview <- preview))
}
}default
local =
Maybe {
get casted_skill = App.KL.Game.Skill.get(local@user, event.code, game)
let skill =
case preview@skill as skill {
none:
some({event.code, 1})
}
return local@preview <- preview@skill <- skill
} <> local
IO {
App.new_post!(room, App.KL.Game.Phase.Play.Event.cast.cancel_skill(event.code))
App.set_local!(App.KL.State.Local.game(local))
}
} default App.pass!
some:
if skill.value@fst =? event.code then
none
else
some({event.code, 1})
}
return local@preview <- preview@skill <- skill
} <> local
IO {
App.new_post!(room, App.KL.Game.Phase.Play.Event.cast.cancel_skill(event.code))
App.set_local!(App.KL.State.Local.game(local))
}
} default App.pass!
key_up:
// if tab (esc) released
if event.code =? 32#16 then
log("SAIU")
let local = App.KL.Game.Phase.Play.when.invert_tab(false, controls, local)
App.set_local!(App.KL.State.Local.game(local))
else
App.pass!
mouse_down:
case game.moment {
initial:
@ -81,8 +98,41 @@ App.KL.Game.Phase.Play.when(
let preview = preview@skill <- none
App.set_local!(App.KL.State.Local.game(local@preview <- preview))
}
mouse_click:
// switch tab
let local = App.KL.Game.Phase.Play.when.switch_tab(event.id, local)
App.set_local!(App.KL.State.Local.game(local))
} default App.pass!
App.KL.Game.Phase.Play.when.switch_tab(
event_id: String
local: App.KL.Game.State.Local
): App.KL.Game.State.Local
let controls = local@hud
let actual_tab = controls@tab_which
let to_string = App.KL.Game.Controls.Hud.Tab.show
let stats = App.KL.Game.Controls.Hud.Tab.stats
let skills = App.KL.Game.Controls.Hud.Tab.skills
let recap = App.KL.Game.Controls.Hud.Tab.recap
let tab = switch String.eql(event_id) {
to_string(stats): stats
to_string(recap): recap
to_string(skills): skills
} default actual_tab
let controls = controls@tab_which <- tab
let local = local@hud <- controls
local
App.KL.Game.Phase.Play.when.invert_tab(
active: Bool
controls: App.KL.Game.Controls.Hud
local: App.KL.Game.State.Local
): App.KL.Game.State.Local
let controls = controls@tab_active <- active
let local = local@hud <- controls
local
App.KL.Game.Phase.Play.when.initial.mouse_down(room: String, local: App.KL.Game.State.Local, game: App.KL.Game): IO<Maybe<App.State.local<App.KL.State>>>
let coord = Hexagonal.Axial.from_screen_xy(local@mouse, App.KL.Constants.hexagon_radius, U32.to_i32(local@screen_size@fst) / 4, U32.to_i32(local@screen_size@snd) / 4)

View File

@ -3,12 +3,4 @@ App.KL.Game.Player.get_allies(
players: Map<App.KL.Game.Player>
): Map<App.KL.Game.Player>
let team = player@team
let result = {}
for user:player in players with result:
if App.KL.Game.Team.eql(player@team, team) then
result{user} <- player
else
result
result
App.KL.Game.Player.get_from_team(team,players)

View File

@ -0,0 +1,14 @@
App.KL.Game.Player.get_enemies(
player: App.KL.Game.Player
players: Map<App.KL.Game.Player>
): Map<App.KL.Game.Player>
let team = player@team
let result = {}
for user:player in players with result:
if Bool.not(App.KL.Game.Team.eql(player@team, team)) then
result{user} <- player
else
result
result

View File

@ -0,0 +1,13 @@
App.KL.Game.Player.get_from_team(
team: App.KL.Game.Team
players: Map<App.KL.Game.Player>
): Map<App.KL.Game.Player>
let result = {}
for user:player in players with result:
if App.KL.Game.Team.eql(player@team, team) then
result{user} <- player
else
result
result

View File

@ -1,14 +1,14 @@
List.find_index<A: Type>(xs: List<A>, f: A -> Bool): Maybe<Pair<A,Nat>>
List.find_index<A: Type>(xs: List<A>, f: A -> Bool): Maybe<Nat>
List.find_index.go<A>(xs,f,Nat.zero)
List.find_index.go<A: Type>(xs: List<A>, f: A -> Bool, i: Nat): Maybe<Pair<A,Nat>>
List.find_index.go<A: Type>(xs: List<A>, f: A -> Bool, i: Nat): Maybe<Nat>
case xs {
nil : Maybe.none!,
cons: case f(xs.head) {
true : Maybe.some!(Pair.new!!(xs.head)(i)),
true : Maybe.some!(i),
false: List.find_index.go!(xs.tail,f,Nat.succ(i))
}
}

View File

@ -1,5 +1,5 @@
Word.s_show(size: Nat)(a: Word(size)): String
neg = Word.is_neg(size, a)
n = Word.to_nat!(Word.abs!(a))
sgn = if neg then "-" else "+"
sgn = if neg then "-" else ""
sgn | Nat.show(n)