give visualizer a facelift

This commit is contained in:
ajuvercr 2020-04-01 22:16:55 +02:00
parent b6fa959f1e
commit e0a634464e
4 changed files with 156 additions and 46 deletions

View file

@ -74,7 +74,7 @@ async fn debug_get() -> Result<Template, String> {
#[get("/visualizer")] #[get("/visualizer")]
async fn visualizer_get() -> Template { async fn visualizer_get() -> Template {
let game_options = get_games().await; let game_options = get_games().await;
let context = Context::new_with("Visualizer", ContextT::Games(game_options)); let context = Context::new_with("Visualizer", json!({"games": game_options, "colours": COLOURS}));
Template::render("visualizer", &context) Template::render("visualizer", &context)
} }

View file

@ -9,6 +9,10 @@ static NAV: [(&'static str, &'static str); 5] = [
("/debug", "Debug Station"), ("/debug", "Debug Station"),
]; ];
pub static COLOURS: [&'static str; 9] = [
"gray", "blue", "cyan", "green", "yellow", "orange", "red", "pink", "purple",
];
#[derive(Serialize)] #[derive(Serialize)]
pub struct Map { pub struct Map {
name: String, name: String,
@ -91,7 +95,9 @@ impl Ord for GameState {
} }
impl From<FinishedState> for GameState { impl From<FinishedState> for GameState {
fn from(state: FinishedState) -> Self { fn from(mut state: FinishedState) -> Self {
state.players.sort_by_key(|x| x.0);
GameState::Finished { GameState::Finished {
map: String::new(), map: String::new(),
players: state players: state
@ -136,11 +142,11 @@ pub struct Lobby {
pub maps: Vec<Map>, pub maps: Vec<Map>,
} }
#[derive(Serialize)] // #[derive(Serialize)]
#[serde(rename_all = "camelCase")] // #[serde(rename_all = "camelCase")]
pub enum ContextT { // pub enum ContextT {
Games(Vec<GameState>), // Games(Vec<GameState>),
} // }
#[derive(Serialize)] #[derive(Serialize)]
pub struct Context<T> { pub struct Context<T> {
@ -213,11 +219,15 @@ pub async fn get_games() -> Vec<GameState> {
match fs::File::open("games.ini").await { match fs::File::open("games.ini").await {
Ok(file) => { Ok(file) => {
let mut file = BufReader::new(file); let mut file = BufReader::new(file);
file.lines().filter_map(move |maybe| async { file.lines()
maybe .filter_map(move |maybe| async {
.ok() maybe
.and_then(|line| serde_json::from_str::<FinishedState>(&line).ok()) .ok()
}).map(|state| state.into()).collect().await .and_then(|line| serde_json::from_str::<FinishedState>(&line).ok())
})
.map(|state| state.into())
.collect()
.await
} }
Err(_) => Vec::new(), Err(_) => Vec::new(),
} }

View file

@ -1,6 +1,7 @@
{% extends "base" %} {% extends "base" %}
{% block content %} {% block content %}
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">
<link rel="stylesheet" type="text/css" href="/style/visualizer.css"> <link rel="stylesheet" type="text/css" href="/style/visualizer.css">
<link rel="stylesheet" type="text/css" href="/style/state.css"> <link rel="stylesheet" type="text/css" href="/style/state.css">
@ -24,20 +25,21 @@
<input type="range" min="0" max="1" value="1" class="slider" id="turnSlider"> <input type="range" min="0" max="1" value="1" class="slider" id="turnSlider">
</div> </div>
</div> </div>
<div class="lds-roller"><div></div><div></div><div></div><div></div><div></div><div></div><div></div><div></div></div>
</div> </div>
<div class="options"> <div class="options">
{% for state in games %} {% for state in games %}
<div class="option" onclick="visualizer.handle('/games/{{ state.file }}', '{{ state.name }}')"> <div class="option" onclick="visualizer.handle('/games/{{ state.file }}', '{{ state.name }}')">
<p>{{state.name}} ({{ state.map }}) <span style="float: right;">{{state.turns}} turns</span></p> <p>{{state.name}} ({{ state.map }}) <span style="float: right;">{{state.turns}} turns</span></p>
<div class="content"> <div>
<div class="players"> <div class="players">
{% for player in state.players %} {% for player in state.players %}
<p class="{% if player[1] %}winner{% endif %}">{{ player[0] }}</p> <p style="color: {{colours[loop.index]}}" class="{% if player[1] %}winner{% endif %}">{{ player[0] }}</p>
{% endfor %} {% endfor %}
</div>
</div> </div>
</div> </div>
</div>
{% endfor %} {% endfor %}
</div> </div>
</div> </div>

View file

@ -16,11 +16,12 @@ p {
min-height: 100%; min-height: 100%;
width: 100%; width: 100%;
height: 100%; height: 100%;
display: flex;
} }
#main { #main {
height: calc(100% - 200px); height: 100%;
width: 100%; flex-grow: 1;
position: relative; position: relative;
} }
@ -40,46 +41,143 @@ p {
} }
.options { .options {
background-color: #444; background-color: black;
overflow-y: hidden; max-width: 300px;
overflow-x: scroll; width: 20%;
white-space: nowrap; min-width: 150px;
height: 200px; height: 100%;
min-height: 200px !important; display: flex;
max-height: 200px !important; flex-direction: column;
overflow-y: auto;
} }
.option { .option {
display: inline-block;
color: white; color: white;
text-align: center; margin: 0 0 20px 0;
width: 200px; padding: 10px;
height: 100%; background-color: grey;
vertical-align: top }
.option:last-child {
margin: 0;
} }
.option:hover { .option:hover {
background-color: #777; background-color: #777;
} }
.loading { .lds-roller {
position: relative; display: none;
}
.loading>.lds-roller {
display: inline-block; display: inline-block;
} }
.loading::before { .lds-roller {
content: "";
position: absolute; position: absolute;
width: 100px;
height: 100px;
background-color: #1c5ba2;
border-radius: 100%;
z-index: 5;
left: 50%; left: 50%;
top: 50%; top: 50%;
transform: translate(-50%, -50%); transform: translate(-50%, -50%);
-webkit-animation: slide-top 0.5s ease-out infinite alternate; width: 80px;
animation: slide-top 0.5s ease-out infinite alternate; height: 80px;
}
.lds-roller div {
animation: lds-roller 1.2s cubic-bezier(0.5, 0, 0.5, 1) infinite;
transform-origin: 40px 40px;
}
.lds-roller div:after {
content: " ";
display: block;
position: absolute;
width: 7px;
height: 7px;
border-radius: 50%;
background: #fff;
margin: -4px 0 0 -4px;
}
.lds-roller div:nth-child(1) {
animation-delay: -0.036s;
}
.lds-roller div:nth-child(1):after {
top: 63px;
left: 63px;
}
.lds-roller div:nth-child(2) {
animation-delay: -0.072s;
}
.lds-roller div:nth-child(2):after {
top: 68px;
left: 56px;
}
.lds-roller div:nth-child(3) {
animation-delay: -0.108s;
}
.lds-roller div:nth-child(3):after {
top: 71px;
left: 48px;
}
.lds-roller div:nth-child(4) {
animation-delay: -0.144s;
}
.lds-roller div:nth-child(4):after {
top: 72px;
left: 40px;
}
.lds-roller div:nth-child(5) {
animation-delay: -0.18s;
}
.lds-roller div:nth-child(5):after {
top: 71px;
left: 32px;
}
.lds-roller div:nth-child(6) {
animation-delay: -0.216s;
}
.lds-roller div:nth-child(6):after {
top: 68px;
left: 24px;
}
.lds-roller div:nth-child(7) {
animation-delay: -0.252s;
}
.lds-roller div:nth-child(7):after {
top: 63px;
left: 17px;
}
.lds-roller div:nth-child(8) {
animation-delay: -0.288s;
}
.lds-roller div:nth-child(8):after {
top: 56px;
left: 12px;
}
@keyframes lds-roller {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
} }
#speed { #speed {