2019-04-26 04:29:35 +00:00
|
|
|
mod cache;
|
|
|
|
mod event;
|
2019-04-15 06:07:56 +00:00
|
|
|
mod graphic;
|
2019-04-26 04:29:35 +00:00
|
|
|
mod scale;
|
2019-04-29 20:37:19 +00:00
|
|
|
mod widgets;
|
2019-05-07 05:40:03 +00:00
|
|
|
#[macro_use]
|
2019-05-09 01:38:34 +00:00
|
|
|
pub mod img_ids;
|
2019-05-07 05:40:03 +00:00
|
|
|
#[macro_use]
|
|
|
|
mod font_ids;
|
2019-03-16 02:03:21 +00:00
|
|
|
|
2019-04-26 04:29:35 +00:00
|
|
|
pub use event::Event;
|
2019-05-07 06:25:26 +00:00
|
|
|
pub use graphic::Graphic;
|
2019-07-26 02:28:53 +00:00
|
|
|
pub use scale::{Scale, ScaleMode};
|
2019-05-19 14:15:50 +00:00
|
|
|
pub use widgets::{
|
|
|
|
image_slider::ImageSlider,
|
2019-05-20 06:57:44 +00:00
|
|
|
ingame::{Ingame, IngameAnchor, Ingameable},
|
2019-05-19 14:15:50 +00:00
|
|
|
toggle_button::ToggleButton,
|
2019-07-29 14:06:13 +00:00
|
|
|
tooltip::{Tooltip, Tooltipable},
|
2019-05-19 14:15:50 +00:00
|
|
|
};
|
2019-05-07 06:25:26 +00:00
|
|
|
|
2019-01-30 12:11:34 +00:00
|
|
|
use crate::{
|
|
|
|
render::{
|
2019-05-14 06:43:07 +00:00
|
|
|
create_ui_quad, create_ui_tri, Consts, DynamicModel, Globals, Mesh, RenderError, Renderer,
|
|
|
|
UiLocals, UiMode, UiPipeline,
|
2019-01-30 12:11:34 +00:00
|
|
|
},
|
2019-02-16 03:01:42 +00:00
|
|
|
window::Window,
|
2019-04-29 20:37:19 +00:00
|
|
|
Error,
|
|
|
|
};
|
2019-04-26 04:29:35 +00:00
|
|
|
use cache::Cache;
|
2019-08-04 19:54:08 +00:00
|
|
|
use common::{assets, util::srgba_to_linear};
|
2019-04-29 20:37:19 +00:00
|
|
|
use conrod_core::{
|
|
|
|
event::Input,
|
2019-05-07 05:40:03 +00:00
|
|
|
graph::Graph,
|
2019-05-14 06:43:07 +00:00
|
|
|
image::{self, Map},
|
2019-04-26 04:29:35 +00:00
|
|
|
input::{touch::Touch, Motion, Widget},
|
2019-05-14 06:43:07 +00:00
|
|
|
render::{Primitive, PrimitiveKind},
|
2019-04-26 04:29:35 +00:00
|
|
|
text::{self, font},
|
2019-05-14 06:43:07 +00:00
|
|
|
widget::{self, id::Generator},
|
2019-05-15 00:04:58 +00:00
|
|
|
Rect, UiBuilder, UiCell,
|
2019-01-30 12:11:34 +00:00
|
|
|
};
|
2019-04-26 04:29:35 +00:00
|
|
|
use graphic::Id as GraphicId;
|
2019-06-06 14:48:41 +00:00
|
|
|
use log::warn;
|
2019-06-23 20:42:17 +00:00
|
|
|
use std::{
|
2019-08-06 06:31:48 +00:00
|
|
|
fs::File,
|
2019-06-23 20:42:17 +00:00
|
|
|
io::{BufReader, Read},
|
|
|
|
ops::Range,
|
|
|
|
sync::Arc,
|
2019-07-29 14:06:13 +00:00
|
|
|
time::Duration,
|
2019-06-23 20:42:17 +00:00
|
|
|
};
|
2019-04-29 20:37:19 +00:00
|
|
|
use vek::*;
|
2019-07-29 14:06:13 +00:00
|
|
|
use widgets::tooltip::TooltipManager;
|
2019-01-30 12:11:34 +00:00
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum UiError {
|
|
|
|
RenderError(RenderError),
|
|
|
|
}
|
|
|
|
|
2019-03-20 05:13:42 +00:00
|
|
|
enum DrawKind {
|
2019-04-15 06:07:56 +00:00
|
|
|
Image,
|
2019-02-23 02:41:52 +00:00
|
|
|
// Text and non-textured geometry
|
2019-03-20 05:13:42 +00:00
|
|
|
Plain,
|
|
|
|
}
|
|
|
|
enum DrawCommand {
|
2019-05-04 14:28:21 +00:00
|
|
|
Draw { kind: DrawKind, verts: Range<usize> },
|
2019-03-20 05:13:42 +00:00
|
|
|
Scissor(Aabr<u16>),
|
2019-05-25 23:02:57 +00:00
|
|
|
WorldPos(Option<usize>),
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
|
|
|
impl DrawCommand {
|
2019-05-04 14:28:21 +00:00
|
|
|
fn image(verts: Range<usize>) -> DrawCommand {
|
2019-03-20 05:13:42 +00:00
|
|
|
DrawCommand::Draw {
|
2019-04-15 06:07:56 +00:00
|
|
|
kind: DrawKind::Image,
|
2019-05-04 14:28:21 +00:00
|
|
|
verts,
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-04 14:28:21 +00:00
|
|
|
fn plain(verts: Range<usize>) -> DrawCommand {
|
2019-03-20 05:13:42 +00:00
|
|
|
DrawCommand::Draw {
|
|
|
|
kind: DrawKind::Plain,
|
2019-05-04 14:28:21 +00:00
|
|
|
verts,
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
|
|
|
}
|
2019-02-16 03:01:42 +00:00
|
|
|
}
|
|
|
|
|
2019-05-07 06:25:26 +00:00
|
|
|
pub struct Font(text::Font);
|
|
|
|
impl assets::Asset for Font {
|
2019-08-06 06:31:48 +00:00
|
|
|
const ENDINGS: &'static [&'static str] = &["ttf"];
|
|
|
|
fn parse(mut buf_reader: BufReader<File>) -> Result<Self, assets::Error> {
|
2019-05-13 23:28:17 +00:00
|
|
|
let mut buf = Vec::new();
|
2019-06-23 20:42:17 +00:00
|
|
|
buf_reader.read_to_end(&mut buf)?;
|
2019-05-13 23:28:17 +00:00
|
|
|
Ok(Font(text::Font::from_bytes(buf.clone()).unwrap()))
|
2019-05-07 06:25:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-30 12:11:34 +00:00
|
|
|
pub struct Ui {
|
2019-05-15 00:04:58 +00:00
|
|
|
ui: conrod_core::Ui,
|
2019-04-15 06:07:56 +00:00
|
|
|
image_map: Map<GraphicId>,
|
2019-01-30 12:11:34 +00:00
|
|
|
cache: Cache,
|
2019-02-23 03:16:12 +00:00
|
|
|
// Draw commands for the next render
|
2019-02-23 02:41:52 +00:00
|
|
|
draw_commands: Vec<DrawCommand>,
|
2019-05-04 14:28:21 +00:00
|
|
|
// Model for drawing the ui
|
|
|
|
model: DynamicModel<UiPipeline>,
|
2019-05-14 06:43:07 +00:00
|
|
|
// Consts for default ui drawing position (ie the interface)
|
|
|
|
interface_locals: Consts<UiLocals>,
|
|
|
|
default_globals: Consts<Globals>,
|
2019-05-25 23:02:57 +00:00
|
|
|
// Consts to specify positions of ingame elements (e.g. Nametags)
|
|
|
|
ingame_locals: Vec<Consts<UiLocals>>,
|
2019-05-17 09:22:32 +00:00
|
|
|
// Window size for updating scaling
|
2019-03-03 23:55:07 +00:00
|
|
|
window_resized: Option<Vec2<f64>>,
|
2019-07-03 02:31:20 +00:00
|
|
|
// Used to delay cache resizing until after current frame is drawn
|
|
|
|
need_cache_resize: bool,
|
2019-03-03 23:55:07 +00:00
|
|
|
// Scaling of the ui
|
|
|
|
scale: Scale,
|
2019-07-29 14:06:13 +00:00
|
|
|
// Tooltips
|
|
|
|
tooltip_manager: TooltipManager,
|
2019-01-30 12:11:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Ui {
|
2019-02-16 03:01:42 +00:00
|
|
|
pub fn new(window: &mut Window) -> Result<Self, Error> {
|
2019-03-03 23:55:07 +00:00
|
|
|
let scale = Scale::new(window, ScaleMode::Absolute(1.0));
|
|
|
|
let win_dims = scale.scaled_window_size().into_array();
|
2019-04-25 12:20:35 +00:00
|
|
|
|
2019-06-06 14:48:41 +00:00
|
|
|
let renderer = window.renderer_mut();
|
2019-05-14 06:43:07 +00:00
|
|
|
|
2019-07-29 14:06:13 +00:00
|
|
|
let mut ui = UiBuilder::new(win_dims).build();
|
|
|
|
let tooltip_manager = TooltipManager::new(
|
|
|
|
ui.widget_id_generator(),
|
|
|
|
Duration::from_millis(1000),
|
|
|
|
Duration::from_millis(1000),
|
|
|
|
scale.scale_factor_logical(),
|
|
|
|
);
|
|
|
|
|
2019-01-30 12:11:34 +00:00
|
|
|
Ok(Self {
|
2019-07-29 14:06:13 +00:00
|
|
|
ui,
|
2019-02-12 04:14:55 +00:00
|
|
|
image_map: Map::new(),
|
2019-05-14 06:43:07 +00:00
|
|
|
cache: Cache::new(renderer)?,
|
2019-02-23 02:41:52 +00:00
|
|
|
draw_commands: vec![],
|
2019-05-14 06:43:07 +00:00
|
|
|
model: renderer.create_dynamic_model(100)?,
|
|
|
|
interface_locals: renderer.create_consts(&[UiLocals::default()])?,
|
|
|
|
default_globals: renderer.create_consts(&[Globals::default()])?,
|
2019-05-25 23:02:57 +00:00
|
|
|
ingame_locals: Vec::new(),
|
2019-05-04 14:28:21 +00:00
|
|
|
window_resized: None,
|
2019-07-03 02:31:20 +00:00
|
|
|
need_cache_resize: false,
|
2019-03-03 23:55:07 +00:00
|
|
|
scale,
|
2019-07-29 14:06:13 +00:00
|
|
|
tooltip_manager,
|
2019-01-30 12:11:34 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Set the scaling mode of the ui.
|
2019-07-26 02:28:53 +00:00
|
|
|
pub fn set_scaling_mode(&mut self, mode: ScaleMode) {
|
|
|
|
self.scale.set_scaling_mode(mode);
|
|
|
|
// To clear the cache (it won't be resized in this case)
|
|
|
|
self.need_cache_resize = true;
|
2019-05-17 09:22:32 +00:00
|
|
|
// Give conrod the new size.
|
2019-03-03 23:55:07 +00:00
|
|
|
let (w, h) = self.scale.scaled_window_size().into_tuple();
|
|
|
|
self.ui.handle_event(Input::Resize(w, h));
|
|
|
|
}
|
|
|
|
|
2019-07-26 02:28:53 +00:00
|
|
|
// Get a copy of Scale
|
|
|
|
pub fn scale(&self) -> Scale {
|
|
|
|
self.scale
|
|
|
|
}
|
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn add_graphic(&mut self, graphic: Graphic) -> image::Id {
|
2019-04-28 18:18:08 +00:00
|
|
|
self.image_map.insert(self.cache.add_graphic(graphic))
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 07:55:51 +00:00
|
|
|
pub fn new_font(&mut self, font: Arc<Font>) -> font::Id {
|
2019-05-07 06:25:26 +00:00
|
|
|
self.ui.fonts.insert(font.as_ref().0.clone())
|
2019-02-23 02:41:52 +00:00
|
|
|
}
|
|
|
|
|
2019-02-16 03:01:42 +00:00
|
|
|
pub fn id_generator(&mut self) -> Generator {
|
|
|
|
self.ui.widget_id_generator()
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
|
|
|
|
2019-07-29 14:06:13 +00:00
|
|
|
pub fn set_widgets(&mut self) -> (UiCell, &mut TooltipManager) {
|
|
|
|
(self.ui.set_widgets(), &mut self.tooltip_manager)
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Accepts Option so widget can be unfocused.
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn focus_widget(&mut self, id: Option<widget::Id>) {
|
2019-04-15 00:45:54 +00:00
|
|
|
self.ui.keyboard_capture(match id {
|
|
|
|
Some(id) => id,
|
|
|
|
None => self.ui.window,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Get id of current widget capturing keyboard.
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn widget_capturing_keyboard(&self) -> Option<widget::Id> {
|
2019-04-15 00:45:54 +00:00
|
|
|
self.ui.global_input().current.widget_capturing_keyboard
|
2019-03-30 02:15:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Get whether a widget besides the window is capturing the mouse.
|
2019-04-15 00:15:12 +00:00
|
|
|
pub fn no_widget_capturing_mouse(&self) -> bool {
|
2019-04-29 20:37:19 +00:00
|
|
|
self.ui
|
|
|
|
.global_input()
|
|
|
|
.current
|
|
|
|
.widget_capturing_mouse
|
|
|
|
.filter(|id| id != &self.ui.window)
|
|
|
|
.is_none()
|
2019-04-15 00:15:12 +00:00
|
|
|
}
|
2019-03-30 02:15:27 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Get the widget graph.
|
2019-05-07 03:25:25 +00:00
|
|
|
pub fn widget_graph(&self) -> &Graph {
|
|
|
|
self.ui.widget_graph()
|
|
|
|
}
|
2019-03-22 03:55:42 +00:00
|
|
|
pub fn handle_event(&mut self, event: Event) {
|
|
|
|
match event.0 {
|
2019-04-26 04:29:35 +00:00
|
|
|
Input::Resize(w, h) if w > 1.0 && h > 1.0 => {
|
|
|
|
self.window_resized = Some(Vec2::new(w, h))
|
|
|
|
}
|
2019-04-29 20:37:19 +00:00
|
|
|
Input::Touch(touch) => self.ui.handle_event(Input::Touch(Touch {
|
|
|
|
xy: self.scale.scale_point(touch.xy.into()).into_array(),
|
|
|
|
..touch
|
|
|
|
})),
|
|
|
|
Input::Motion(motion) => self.ui.handle_event(Input::Motion(match motion {
|
|
|
|
Motion::MouseCursor { x, y } => {
|
|
|
|
let (x, y) = self.scale.scale_point(Vec2::new(x, y)).into_tuple();
|
|
|
|
Motion::MouseCursor { x, y }
|
|
|
|
}
|
|
|
|
Motion::MouseRelative { x, y } => {
|
|
|
|
let (x, y) = self.scale.scale_point(Vec2::new(x, y)).into_tuple();
|
|
|
|
Motion::MouseRelative { x, y }
|
|
|
|
}
|
|
|
|
Motion::Scroll { x, y } => {
|
|
|
|
let (x, y) = self.scale.scale_point(Vec2::new(x, y)).into_tuple();
|
|
|
|
Motion::Scroll { x, y }
|
|
|
|
}
|
|
|
|
_ => motion,
|
|
|
|
})),
|
2019-03-22 03:55:42 +00:00
|
|
|
_ => self.ui.handle_event(event.0),
|
2019-03-03 23:55:07 +00:00
|
|
|
}
|
2019-02-16 03:01:42 +00:00
|
|
|
}
|
2019-02-12 04:14:55 +00:00
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn widget_input(&self, id: widget::Id) -> Widget {
|
2019-02-16 03:01:42 +00:00
|
|
|
self.ui.widget_input(id)
|
2019-01-30 12:11:34 +00:00
|
|
|
}
|
|
|
|
|
2019-05-21 03:43:24 +00:00
|
|
|
pub fn maintain(&mut self, renderer: &mut Renderer, cam_params: Option<(Mat4<f32>, f32)>) {
|
2019-07-29 14:06:13 +00:00
|
|
|
// Maintain tooltip manager
|
|
|
|
self.tooltip_manager
|
|
|
|
.maintain(self.ui.global_input(), self.scale.scale_factor_logical());
|
|
|
|
|
2019-03-03 23:55:07 +00:00
|
|
|
// Regenerate draw commands and associated models only if the ui changed
|
2019-05-04 14:28:21 +00:00
|
|
|
let mut primitives = match self.ui.draw_if_changed() {
|
|
|
|
Some(primitives) => primitives,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
2019-07-03 02:31:20 +00:00
|
|
|
if self.need_cache_resize {
|
|
|
|
// Resize graphic cache
|
|
|
|
self.cache.resize_graphic_cache(renderer).unwrap();
|
|
|
|
// Resize glyph cache
|
|
|
|
self.cache.resize_glyph_cache(renderer).unwrap();
|
|
|
|
|
|
|
|
self.need_cache_resize = false;
|
|
|
|
}
|
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
self.draw_commands.clear();
|
|
|
|
let mut mesh = Mesh::new();
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// TODO: this could be removed entirely if the draw call just used both textures,
|
|
|
|
// however this allows for flexibility if we want to interweave other draw calls later.
|
2019-05-04 14:28:21 +00:00
|
|
|
enum State {
|
|
|
|
Image,
|
|
|
|
Plain,
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut current_state = State::Plain;
|
|
|
|
let mut start = 0;
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
let window_scissor = default_scissor(renderer);
|
|
|
|
let mut current_scissor = window_scissor;
|
2019-05-04 14:28:21 +00:00
|
|
|
|
2019-05-25 23:02:57 +00:00
|
|
|
let mut ingame_local_index = 0;
|
|
|
|
|
2019-05-21 03:43:24 +00:00
|
|
|
enum Placement {
|
|
|
|
Interface,
|
2019-05-25 22:16:26 +00:00
|
|
|
// Number of primitives left to render ingame and relative scaling/resolution
|
2019-05-21 03:43:24 +00:00
|
|
|
InWorld(usize, Option<f32>),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut placement = Placement::Interface;
|
2019-05-20 06:09:20 +00:00
|
|
|
// TODO: maybe mutate an ingame scale factor instead of this, depends on if we want them to scale with other ui scaling or not
|
2019-05-14 06:43:07 +00:00
|
|
|
let mut p_scale_factor = self.scale.scale_factor_physical();
|
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
// Switches to the `Plain` state and completes the previous `Command` if not already in the
|
|
|
|
// `Plain` state.
|
|
|
|
macro_rules! switch_to_plain_state {
|
|
|
|
() => {
|
|
|
|
if let State::Image = current_state {
|
|
|
|
self.draw_commands
|
|
|
|
.push(DrawCommand::image(start..mesh.vertices().len()));
|
|
|
|
start = mesh.vertices().len();
|
2019-05-14 06:43:07 +00:00
|
|
|
current_state = State::Plain;
|
2019-05-04 14:28:21 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
while let Some(prim) = primitives.next() {
|
|
|
|
let Primitive {
|
|
|
|
kind,
|
|
|
|
scizzor,
|
|
|
|
rect,
|
|
|
|
..
|
|
|
|
} = prim;
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Check for a change in the scissor.
|
|
|
|
let new_scissor = {
|
2019-05-04 14:28:21 +00:00
|
|
|
let (l, b, w, h) = scizzor.l_b_w_h();
|
2019-05-20 06:57:44 +00:00
|
|
|
let scale_factor = self.scale.scale_factor_physical();
|
2019-05-04 14:28:21 +00:00
|
|
|
// Calculate minimum x and y coordinates while
|
2019-05-17 09:22:32 +00:00
|
|
|
// flipping y axis (from +up to +down) and
|
|
|
|
// moving origin to top-left corner (from middle).
|
2019-05-04 14:28:21 +00:00
|
|
|
let min_x = self.ui.win_w / 2.0 + l;
|
|
|
|
let min_y = self.ui.win_h / 2.0 - b - h;
|
|
|
|
Aabr {
|
|
|
|
min: Vec2 {
|
2019-05-20 06:57:44 +00:00
|
|
|
x: (min_x * scale_factor) as u16,
|
|
|
|
y: (min_y * scale_factor) as u16,
|
2019-05-04 14:28:21 +00:00
|
|
|
},
|
|
|
|
max: Vec2 {
|
2019-05-20 06:57:44 +00:00
|
|
|
x: ((min_x + w) * scale_factor) as u16,
|
|
|
|
y: ((min_y + h) * scale_factor) as u16,
|
2019-05-04 14:28:21 +00:00
|
|
|
},
|
|
|
|
}
|
2019-05-17 09:22:32 +00:00
|
|
|
.intersection(window_scissor)
|
2019-04-15 06:07:56 +00:00
|
|
|
};
|
2019-05-17 09:22:32 +00:00
|
|
|
if new_scissor != current_scissor {
|
|
|
|
// Finish the current command.
|
2019-05-04 14:28:21 +00:00
|
|
|
self.draw_commands.push(match current_state {
|
|
|
|
State::Plain => DrawCommand::plain(start..mesh.vertices().len()),
|
|
|
|
State::Image => DrawCommand::image(start..mesh.vertices().len()),
|
|
|
|
});
|
|
|
|
start = mesh.vertices().len();
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Update the scissor and produce a command.
|
|
|
|
current_scissor = new_scissor;
|
|
|
|
self.draw_commands.push(DrawCommand::Scissor(new_scissor));
|
2019-05-04 14:28:21 +00:00
|
|
|
}
|
2019-04-15 06:07:56 +00:00
|
|
|
|
2019-05-21 03:43:24 +00:00
|
|
|
match placement {
|
2019-05-25 22:16:26 +00:00
|
|
|
// No primitives left to place in the world at the current position, go back to drawing the interface
|
2019-05-21 03:43:24 +00:00
|
|
|
Placement::InWorld(0, _) => {
|
|
|
|
placement = Placement::Interface;
|
2019-05-14 06:43:07 +00:00
|
|
|
p_scale_factor = self.scale.scale_factor_physical();
|
|
|
|
// Finish current state
|
|
|
|
self.draw_commands.push(match current_state {
|
|
|
|
State::Plain => DrawCommand::plain(start..mesh.vertices().len()),
|
|
|
|
State::Image => DrawCommand::image(start..mesh.vertices().len()),
|
|
|
|
});
|
|
|
|
start = mesh.vertices().len();
|
|
|
|
// Push new position command
|
|
|
|
self.draw_commands.push(DrawCommand::WorldPos(None));
|
|
|
|
}
|
2019-05-25 22:16:26 +00:00
|
|
|
// Primitives still left to draw ingame
|
|
|
|
Placement::InWorld(num_prims, res) => match kind {
|
2019-06-01 04:52:20 +00:00
|
|
|
// Other types aren't drawn & shouldn't decrement the number of primitives left to draw ingame
|
2019-05-20 06:09:20 +00:00
|
|
|
PrimitiveKind::Other(_) => {}
|
2019-05-25 22:16:26 +00:00
|
|
|
// Decrement the number of primitives left
|
|
|
|
_ => placement = Placement::InWorld(num_prims - 1, res),
|
2019-05-19 14:15:50 +00:00
|
|
|
},
|
2019-05-21 03:43:24 +00:00
|
|
|
Placement::Interface => {}
|
2019-05-14 06:43:07 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Functions for converting for conrod scalar coords to GL vertex coords (-1.0 to 1.0).
|
2019-05-25 22:16:26 +00:00
|
|
|
let (ui_win_w, ui_win_h) = match placement {
|
|
|
|
Placement::InWorld(_, Some(res)) => (res as f64, res as f64),
|
|
|
|
// Behind the camera or far away
|
|
|
|
Placement::InWorld(_, None) => continue,
|
|
|
|
Placement::Interface => (self.ui.win_w, self.ui.win_h),
|
2019-05-14 06:43:07 +00:00
|
|
|
};
|
2019-05-04 14:28:21 +00:00
|
|
|
let vx = |x: f64| (x / ui_win_w * 2.0) as f32;
|
|
|
|
let vy = |y: f64| (y / ui_win_h * 2.0) as f32;
|
2019-05-15 00:04:58 +00:00
|
|
|
let gl_aabr = |rect: Rect| {
|
2019-05-04 14:28:21 +00:00
|
|
|
let (l, r, b, t) = rect.l_r_b_t();
|
|
|
|
Aabr {
|
|
|
|
min: Vec2::new(vx(l), vy(b)),
|
|
|
|
max: Vec2::new(vx(r), vy(t)),
|
|
|
|
}
|
|
|
|
};
|
2019-02-23 02:41:52 +00:00
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
match kind {
|
|
|
|
PrimitiveKind::Image {
|
|
|
|
image_id,
|
|
|
|
color,
|
2019-07-03 03:09:37 +00:00
|
|
|
source_rect: _, // TODO: <-- use this
|
2019-05-04 14:28:21 +00:00
|
|
|
} => {
|
|
|
|
let graphic_id = self
|
|
|
|
.image_map
|
|
|
|
.get(&image_id)
|
|
|
|
.expect("Image does not exist in image map");
|
|
|
|
let (graphic_cache, cache_tex) = self.cache.graphic_cache_mut_and_tex();
|
|
|
|
|
|
|
|
match graphic_cache.get_graphic(*graphic_id) {
|
|
|
|
Some(Graphic::Blank) | None => continue,
|
|
|
|
_ => {}
|
|
|
|
}
|
2019-03-20 05:13:42 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Switch to the image state if we are not in it already.
|
2019-05-04 14:28:21 +00:00
|
|
|
if let State::Plain = current_state {
|
2019-04-29 20:37:19 +00:00
|
|
|
self.draw_commands
|
2019-05-04 14:28:21 +00:00
|
|
|
.push(DrawCommand::plain(start..mesh.vertices().len()));
|
|
|
|
start = mesh.vertices().len();
|
|
|
|
current_state = State::Image;
|
2019-02-23 02:41:52 +00:00
|
|
|
}
|
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
let color =
|
2019-08-04 19:54:08 +00:00
|
|
|
srgba_to_linear(color.unwrap_or(conrod_core::color::WHITE).to_fsa().into());
|
2019-05-04 14:28:21 +00:00
|
|
|
|
|
|
|
let resolution = Vec2::new(
|
2019-05-09 01:38:34 +00:00
|
|
|
(rect.w() * p_scale_factor).round() as u16,
|
|
|
|
(rect.h() * p_scale_factor).round() as u16,
|
2019-05-04 14:28:21 +00:00
|
|
|
);
|
2019-05-17 09:22:32 +00:00
|
|
|
// Transform the source rectangle into uv coordinate.
|
|
|
|
// TODO: Make sure this is right.
|
2019-05-04 14:28:21 +00:00
|
|
|
let source_aabr = {
|
2019-05-14 06:43:07 +00:00
|
|
|
let (uv_l, uv_r, uv_b, uv_t) = (0.0, 1.0, 0.0, 1.0);
|
|
|
|
/*match source_rect {
|
|
|
|
Some(src_rect) => {
|
|
|
|
let (l, r, b, t) = src_rect.l_r_b_t();
|
|
|
|
((l / image_w) as f32,
|
|
|
|
(r / image_w) as f32,
|
|
|
|
(b / image_h) as f32,
|
|
|
|
(t / image_h) as f32)
|
|
|
|
}
|
|
|
|
None => (0.0, 1.0, 0.0, 1.0),
|
|
|
|
};*/
|
2019-05-04 14:28:21 +00:00
|
|
|
Aabr {
|
|
|
|
min: Vec2::new(uv_l, uv_b),
|
|
|
|
max: Vec2::new(uv_r, uv_t),
|
|
|
|
}
|
|
|
|
};
|
2019-07-03 01:21:08 +00:00
|
|
|
// TODO: get dims from graphic_cache (or have it return floats directly)
|
2019-05-04 14:28:21 +00:00
|
|
|
let (cache_w, cache_h) =
|
|
|
|
cache_tex.get_dimensions().map(|e| e as f32).into_tuple();
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Cache graphic at particular resolution.
|
2019-07-03 01:21:08 +00:00
|
|
|
let uv_aabr =
|
|
|
|
match graphic_cache.queue_res(*graphic_id, resolution, source_aabr) {
|
|
|
|
Some(aabr) => Aabr {
|
|
|
|
min: Vec2::new(
|
|
|
|
aabr.min.x as f32 / cache_w,
|
|
|
|
aabr.max.y as f32 / cache_h,
|
|
|
|
),
|
|
|
|
max: Vec2::new(
|
|
|
|
aabr.max.x as f32 / cache_w,
|
|
|
|
aabr.min.y as f32 / cache_h,
|
|
|
|
),
|
|
|
|
},
|
|
|
|
None => continue,
|
|
|
|
};
|
2019-03-20 05:13:42 +00:00
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
mesh.push_quad(create_ui_quad(gl_aabr(rect), uv_aabr, color, UiMode::Image));
|
|
|
|
}
|
|
|
|
PrimitiveKind::Text {
|
|
|
|
color,
|
|
|
|
text,
|
|
|
|
font_id,
|
|
|
|
} => {
|
|
|
|
switch_to_plain_state!();
|
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
let positioned_glyphs = text.positioned_glyphs(p_scale_factor as f32);
|
2019-05-04 14:28:21 +00:00
|
|
|
let (glyph_cache, cache_tex) = self.cache.glyph_cache_mut_and_tex();
|
2019-05-17 09:22:32 +00:00
|
|
|
// Queue the glyphs to be cached.
|
2019-05-04 14:28:21 +00:00
|
|
|
for glyph in positioned_glyphs {
|
|
|
|
glyph_cache.queue_glyph(font_id.index(), glyph.clone());
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
2019-02-23 03:16:12 +00:00
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
glyph_cache
|
|
|
|
.cache_queued(|rect, data| {
|
|
|
|
let offset = [rect.min.x as u16, rect.min.y as u16];
|
|
|
|
let size = [rect.width() as u16, rect.height() as u16];
|
|
|
|
|
|
|
|
let new_data = data
|
|
|
|
.iter()
|
|
|
|
.map(|x| [255, 255, 255, *x])
|
|
|
|
.collect::<Vec<[u8; 4]>>();
|
|
|
|
|
2019-06-06 14:48:41 +00:00
|
|
|
if let Err(err) =
|
|
|
|
renderer.update_texture(cache_tex, offset, size, &new_data)
|
|
|
|
{
|
|
|
|
warn!("Failed to update texture: {:?}", err);
|
|
|
|
}
|
2019-05-04 14:28:21 +00:00
|
|
|
})
|
|
|
|
.unwrap();
|
|
|
|
|
2019-08-04 19:54:08 +00:00
|
|
|
let color = srgba_to_linear(color.to_fsa().into());
|
2019-05-04 14:28:21 +00:00
|
|
|
|
|
|
|
for g in positioned_glyphs {
|
|
|
|
if let Ok(Some((uv_rect, screen_rect))) =
|
|
|
|
glyph_cache.rect_for(font_id.index(), g)
|
|
|
|
{
|
|
|
|
let uv = Aabr {
|
|
|
|
min: Vec2::new(uv_rect.min.x, uv_rect.max.y),
|
|
|
|
max: Vec2::new(uv_rect.max.x, uv_rect.min.y),
|
|
|
|
};
|
|
|
|
let rect = Aabr {
|
2019-04-29 20:37:19 +00:00
|
|
|
min: Vec2::new(
|
2019-05-15 00:04:58 +00:00
|
|
|
vx(screen_rect.min.x as f64 / p_scale_factor
|
|
|
|
- self.ui.win_w / 2.0),
|
|
|
|
vy(self.ui.win_h / 2.0
|
|
|
|
- screen_rect.max.y as f64 / p_scale_factor),
|
2019-04-29 20:37:19 +00:00
|
|
|
),
|
|
|
|
max: Vec2::new(
|
2019-05-15 00:04:58 +00:00
|
|
|
vx(screen_rect.max.x as f64 / p_scale_factor
|
|
|
|
- self.ui.win_w / 2.0),
|
|
|
|
vy(self.ui.win_h / 2.0
|
|
|
|
- screen_rect.min.y as f64 / p_scale_factor),
|
2019-04-29 20:37:19 +00:00
|
|
|
),
|
2019-05-04 14:28:21 +00:00
|
|
|
};
|
|
|
|
mesh.push_quad(create_ui_quad(rect, uv, color, UiMode::Text));
|
2019-02-23 02:41:52 +00:00
|
|
|
}
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
2019-05-04 14:28:21 +00:00
|
|
|
}
|
|
|
|
PrimitiveKind::Rectangle { color } => {
|
2019-08-04 19:54:08 +00:00
|
|
|
let color = srgba_to_linear(color.to_fsa().into());
|
2019-05-17 09:22:32 +00:00
|
|
|
// Don't draw a transparent rectangle.
|
2019-05-04 14:28:21 +00:00
|
|
|
if color[3] == 0.0 {
|
|
|
|
continue;
|
|
|
|
}
|
2019-03-04 07:28:16 +00:00
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
switch_to_plain_state!();
|
2019-03-04 07:28:16 +00:00
|
|
|
|
2019-05-04 14:28:21 +00:00
|
|
|
mesh.push_quad(create_ui_quad(
|
|
|
|
gl_aabr(rect),
|
|
|
|
Aabr {
|
|
|
|
min: Vec2::new(0.0, 0.0),
|
|
|
|
max: Vec2::new(0.0, 0.0),
|
|
|
|
},
|
|
|
|
color,
|
|
|
|
UiMode::Geometry,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
PrimitiveKind::TrianglesSingleColor { color, triangles } => {
|
2019-05-17 09:22:32 +00:00
|
|
|
// Don't draw transparent triangle or switch state if there are actually no triangles.
|
2019-08-04 19:54:08 +00:00
|
|
|
let color = srgba_to_linear(Rgba::from(Into::<[f32; 4]>::into(color)));
|
2019-05-04 14:28:21 +00:00
|
|
|
if triangles.is_empty() || color[3] == 0.0 {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_to_plain_state!();
|
|
|
|
|
|
|
|
for tri in triangles {
|
|
|
|
let p1 = Vec2::new(vx(tri[0][0]), vy(tri[0][1]));
|
|
|
|
let p2 = Vec2::new(vx(tri[1][0]), vy(tri[1][1]));
|
|
|
|
let p3 = Vec2::new(vx(tri[2][0]), vy(tri[2][1]));
|
2019-05-17 09:22:32 +00:00
|
|
|
// If triangle is clockwise, reverse it.
|
2019-05-04 14:28:21 +00:00
|
|
|
let (v1, v2): (Vec3<f32>, Vec3<f32>) = ((p2 - p1).into(), (p3 - p1).into());
|
|
|
|
let triangle = if v1.cross(v2).z > 0.0 {
|
|
|
|
[p1.into_array(), p2.into_array(), p3.into_array()]
|
|
|
|
} else {
|
|
|
|
[p2.into_array(), p1.into_array(), p3.into_array()]
|
|
|
|
};
|
|
|
|
mesh.push_tri(create_ui_tri(
|
|
|
|
triangle,
|
|
|
|
[[0.0; 2]; 3],
|
2019-03-04 07:28:16 +00:00
|
|
|
color,
|
|
|
|
UiMode::Geometry,
|
2019-04-04 14:45:57 +00:00
|
|
|
));
|
2019-03-04 07:28:16 +00:00
|
|
|
}
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
2019-05-14 06:43:07 +00:00
|
|
|
PrimitiveKind::Other(container) => {
|
|
|
|
if container.type_id == std::any::TypeId::of::<widgets::ingame::State>() {
|
2019-05-20 06:09:20 +00:00
|
|
|
// Calculate the scale factor to pixels at this 3d point using the camera.
|
2019-05-21 03:43:24 +00:00
|
|
|
if let Some((view_mat, fov)) = cam_params {
|
|
|
|
// Retrieve world position
|
|
|
|
let parameters = container
|
|
|
|
.state_and_style::<widgets::ingame::State, widgets::ingame::Style>()
|
|
|
|
.unwrap()
|
|
|
|
.state
|
|
|
|
.parameters;
|
|
|
|
|
|
|
|
let pos_in_view = view_mat * Vec4::from_point(parameters.pos);
|
|
|
|
let scale_factor = self.ui.win_w as f64
|
|
|
|
/ (-2.0
|
|
|
|
* pos_in_view.z as f64
|
|
|
|
* (0.5 * fov as f64).tan()
|
|
|
|
* parameters.res as f64);
|
|
|
|
// Don't process ingame elements behind the camera or very far away
|
2019-05-25 23:02:57 +00:00
|
|
|
placement = if scale_factor > 0.2 {
|
|
|
|
// Finish current state
|
|
|
|
self.draw_commands.push(match current_state {
|
|
|
|
State::Plain => {
|
|
|
|
DrawCommand::plain(start..mesh.vertices().len())
|
|
|
|
}
|
|
|
|
State::Image => {
|
|
|
|
DrawCommand::image(start..mesh.vertices().len())
|
|
|
|
}
|
|
|
|
});
|
|
|
|
start = mesh.vertices().len();
|
|
|
|
// Push new position command
|
|
|
|
if self.ingame_locals.len() > ingame_local_index {
|
|
|
|
renderer
|
|
|
|
.update_consts(
|
|
|
|
&mut self.ingame_locals[ingame_local_index],
|
|
|
|
&[parameters.pos.into()],
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
} else {
|
|
|
|
self.ingame_locals.push(
|
|
|
|
renderer.create_consts(&[parameters.pos.into()]).unwrap(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
self.draw_commands
|
|
|
|
.push(DrawCommand::WorldPos(Some(ingame_local_index)));
|
|
|
|
ingame_local_index += 1;
|
|
|
|
|
2019-05-21 03:43:24 +00:00
|
|
|
p_scale_factor = ((scale_factor * 10.0).log2().round().powi(2)
|
|
|
|
/ 10.0)
|
|
|
|
.min(1.6)
|
|
|
|
.max(0.2);
|
2019-05-27 19:20:02 +00:00
|
|
|
|
|
|
|
// Scale down ingame elements that are close to the camera
|
|
|
|
let res = if scale_factor > 3.2 {
|
|
|
|
parameters.res * scale_factor as f32 / 3.2
|
|
|
|
} else {
|
|
|
|
parameters.res
|
|
|
|
};
|
|
|
|
|
|
|
|
Placement::InWorld(parameters.num, Some(res))
|
2019-05-21 03:43:24 +00:00
|
|
|
} else {
|
|
|
|
Placement::InWorld(parameters.num, None)
|
|
|
|
};
|
2019-05-20 06:09:20 +00:00
|
|
|
}
|
2019-05-14 06:43:07 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-17 09:22:32 +00:00
|
|
|
_ => {} // TODO: Add this.
|
2019-05-04 14:28:21 +00:00
|
|
|
//PrimitiveKind::TrianglesMultiColor {..} => {println!("primitive kind multicolor with id {:?}", id);}
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
2019-05-04 14:28:21 +00:00
|
|
|
}
|
2019-05-17 09:22:32 +00:00
|
|
|
// Enter the final command.
|
2019-05-04 14:28:21 +00:00
|
|
|
self.draw_commands.push(match current_state {
|
|
|
|
State::Plain => DrawCommand::plain(start..mesh.vertices().len()),
|
|
|
|
State::Image => DrawCommand::image(start..mesh.vertices().len()),
|
|
|
|
});
|
|
|
|
|
2019-05-21 03:43:24 +00:00
|
|
|
// Draw glyph cache (use for debugging).
|
2019-05-20 06:57:44 +00:00
|
|
|
/*self.draw_commands
|
|
|
|
.push(DrawCommand::Scissor(default_scissor(renderer)));
|
|
|
|
start = mesh.vertices().len();
|
|
|
|
mesh.push_quad(create_ui_quad(
|
2019-05-21 03:43:24 +00:00
|
|
|
Aabr {
|
|
|
|
min: (-1.0, -1.0).into(),
|
|
|
|
max: (1.0, 1.0).into(),
|
|
|
|
},
|
|
|
|
Aabr {
|
|
|
|
min: (0.0, 1.0).into(),
|
|
|
|
max: (1.0, 0.0).into(),
|
|
|
|
},
|
2019-05-20 06:57:44 +00:00
|
|
|
Rgba::new(1.0, 1.0, 1.0, 0.8),
|
|
|
|
UiMode::Text,
|
|
|
|
));
|
|
|
|
self.draw_commands
|
|
|
|
.push(DrawCommand::plain(start..mesh.vertices().len()));*/
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Create a larger dynamic model if the mesh is larger than the current model size.
|
2019-05-04 14:28:21 +00:00
|
|
|
if self.model.vbuf.len() < mesh.vertices().len() {
|
|
|
|
self.model = renderer
|
|
|
|
.create_dynamic_model(mesh.vertices().len() * 4 / 3)
|
|
|
|
.unwrap();
|
|
|
|
}
|
2019-05-17 09:22:32 +00:00
|
|
|
// Update model with new mesh.
|
2019-07-03 01:21:08 +00:00
|
|
|
renderer.update_model(&self.model, &mesh, 0).unwrap();
|
|
|
|
|
|
|
|
// Move cached graphics to the gpu
|
|
|
|
let (graphic_cache, cache_tex) = self.cache.graphic_cache_mut_and_tex();
|
|
|
|
graphic_cache.cache_queued(|aabr, data| {
|
|
|
|
let offset = aabr.min.into_array();
|
|
|
|
let size = aabr.size().into_array();
|
|
|
|
if let Err(err) = renderer.update_texture(cache_tex, offset, size, data) {
|
|
|
|
warn!("Failed to update texture: {:?}", err);
|
|
|
|
}
|
|
|
|
});
|
2019-05-04 14:28:21 +00:00
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Handle window resizing.
|
2019-05-04 14:28:21 +00:00
|
|
|
if let Some(new_dims) = self.window_resized.take() {
|
2019-07-03 01:21:08 +00:00
|
|
|
let (old_w, old_h) = self.scale.scaled_window_size().into_tuple();
|
2019-05-04 14:28:21 +00:00
|
|
|
self.scale.window_resized(new_dims, renderer);
|
|
|
|
let (w, h) = self.scale.scaled_window_size().into_tuple();
|
|
|
|
self.ui.handle_event(Input::Resize(w, h));
|
|
|
|
|
2019-05-17 09:22:32 +00:00
|
|
|
// Avoid panic in graphic cache when minimizing.
|
2019-07-03 01:21:08 +00:00
|
|
|
// Avoid resetting cache if window size didn't change
|
2019-07-03 02:31:20 +00:00
|
|
|
// Somewhat inefficient for elements that won't change size after a window resize
|
|
|
|
let res = renderer.get_resolution();
|
|
|
|
self.need_cache_resize = res.x > 0 && res.y > 0 && !(old_w == w && old_h == h);
|
2019-02-12 04:14:55 +00:00
|
|
|
}
|
2019-01-30 12:11:34 +00:00
|
|
|
}
|
2019-02-16 03:01:42 +00:00
|
|
|
|
2019-05-14 06:43:07 +00:00
|
|
|
pub fn render(&self, renderer: &mut Renderer, maybe_globals: Option<&Consts<Globals>>) {
|
|
|
|
let mut scissor = default_scissor(renderer);
|
|
|
|
let globals = maybe_globals.unwrap_or(&self.default_globals);
|
|
|
|
let mut locals = &self.interface_locals;
|
2019-02-23 02:41:52 +00:00
|
|
|
for draw_command in self.draw_commands.iter() {
|
|
|
|
match draw_command {
|
2019-05-14 06:43:07 +00:00
|
|
|
DrawCommand::Scissor(new_scissor) => {
|
|
|
|
scissor = *new_scissor;
|
|
|
|
}
|
2019-05-25 23:02:57 +00:00
|
|
|
DrawCommand::WorldPos(index) => {
|
|
|
|
locals = index.map_or(&self.interface_locals, |i| &self.ingame_locals[i]);
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
2019-05-04 14:28:21 +00:00
|
|
|
DrawCommand::Draw { kind, verts } => {
|
2019-03-20 05:13:42 +00:00
|
|
|
let tex = match kind {
|
2019-04-29 20:37:19 +00:00
|
|
|
DrawKind::Image => self.cache.graphic_cache_tex(),
|
|
|
|
DrawKind::Plain => self.cache.glyph_cache_tex(),
|
2019-03-20 05:13:42 +00:00
|
|
|
};
|
2019-05-04 14:28:21 +00:00
|
|
|
let model = self.model.submodel(verts.clone());
|
2019-05-14 06:43:07 +00:00
|
|
|
renderer.render_ui_element(&model, &tex, scissor, globals, locals);
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
2019-02-16 03:01:42 +00:00
|
|
|
}
|
2019-02-23 02:41:52 +00:00
|
|
|
}
|
2019-02-16 03:01:42 +00:00
|
|
|
}
|
2019-01-30 12:11:34 +00:00
|
|
|
}
|
2019-03-20 05:13:42 +00:00
|
|
|
|
2019-05-20 06:57:44 +00:00
|
|
|
fn default_scissor(renderer: &Renderer) -> Aabr<u16> {
|
2019-03-20 05:13:42 +00:00
|
|
|
let (screen_w, screen_h) = renderer.get_resolution().map(|e| e as u16).into_tuple();
|
|
|
|
Aabr {
|
|
|
|
min: Vec2 { x: 0, y: 0 },
|
2019-04-29 20:37:19 +00:00
|
|
|
max: Vec2 {
|
|
|
|
x: screen_w,
|
|
|
|
y: screen_h,
|
|
|
|
},
|
2019-03-20 05:13:42 +00:00
|
|
|
}
|
2019-04-29 20:37:19 +00:00
|
|
|
}
|