2023-03-31 02:25:11 +00:00
|
|
|
use gio::resources_lookup_data;
|
2023-03-24 14:14:01 +00:00
|
|
|
use glib::Object;
|
2023-03-31 02:25:11 +00:00
|
|
|
use gtk::{
|
|
|
|
gdk_pixbuf::{InterpType, Pixbuf},
|
|
|
|
prelude::*,
|
|
|
|
subclass::prelude::*,
|
|
|
|
};
|
|
|
|
use image::io::Reader as ImageReader;
|
2023-04-05 13:13:50 +00:00
|
|
|
use kifu_core::{
|
|
|
|
ui::{BoardElement, IntersectionElement},
|
|
|
|
Color,
|
|
|
|
};
|
2023-03-31 02:25:11 +00:00
|
|
|
use std::{cell::RefCell, io::Cursor, rc::Rc};
|
2023-03-25 13:00:56 +00:00
|
|
|
|
|
|
|
const WIDTH: i32 = 800;
|
|
|
|
const HEIGHT: i32 = 800;
|
2023-03-24 14:14:01 +00:00
|
|
|
|
2023-03-31 01:01:48 +00:00
|
|
|
#[derive(Clone, Default, PartialEq)]
|
|
|
|
struct Addr {
|
|
|
|
row: u8,
|
|
|
|
column: u8,
|
|
|
|
}
|
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
pub struct BoardPrivate {
|
2023-03-25 13:00:56 +00:00
|
|
|
drawing_area: gtk::DrawingArea,
|
|
|
|
|
2023-03-31 01:01:48 +00:00
|
|
|
current_player: Rc<RefCell<Color>>,
|
2023-04-05 13:13:50 +00:00
|
|
|
board: Rc<RefCell<BoardElement>>,
|
2023-03-31 01:01:48 +00:00
|
|
|
cursor_location: Rc<RefCell<Addr>>,
|
2023-03-25 13:00:56 +00:00
|
|
|
}
|
2023-03-24 14:14:01 +00:00
|
|
|
|
|
|
|
#[glib::object_subclass]
|
2023-04-05 13:13:50 +00:00
|
|
|
impl ObjectSubclass for BoardPrivate {
|
|
|
|
const NAME: &'static str = "Board";
|
|
|
|
type Type = Board;
|
2023-03-25 13:00:56 +00:00
|
|
|
type ParentType = gtk::Grid;
|
2023-03-31 01:01:48 +00:00
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
fn new() -> BoardPrivate {
|
|
|
|
BoardPrivate {
|
2023-03-31 01:01:48 +00:00
|
|
|
drawing_area: Default::default(),
|
|
|
|
current_player: Rc::new(RefCell::new(Color::Black)),
|
2023-04-05 13:13:50 +00:00
|
|
|
board: Default::default(),
|
2023-03-31 01:01:48 +00:00
|
|
|
cursor_location: Default::default(),
|
|
|
|
}
|
|
|
|
}
|
2023-03-24 14:14:01 +00:00
|
|
|
}
|
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
impl ObjectImpl for BoardPrivate {
|
2023-03-25 13:00:56 +00:00
|
|
|
fn constructed(&self) {
|
|
|
|
self.drawing_area.set_width_request(WIDTH);
|
|
|
|
self.drawing_area.set_height_request(HEIGHT);
|
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
let board = self.board.clone();
|
2023-03-31 01:01:48 +00:00
|
|
|
let cursor_location = self.cursor_location.clone();
|
|
|
|
let current_player = self.current_player.clone();
|
2023-03-31 02:25:11 +00:00
|
|
|
|
|
|
|
let wood_texture = resources_lookup_data(
|
|
|
|
"/com/luminescent-dreams/kifu-gtk/wood_texture.jpg",
|
|
|
|
gio::ResourceLookupFlags::NONE,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
let background = ImageReader::new(Cursor::new(wood_texture))
|
|
|
|
.with_guessed_format()
|
|
|
|
.unwrap()
|
|
|
|
.decode();
|
|
|
|
let background = background.map(|background| {
|
|
|
|
Pixbuf::from_bytes(
|
|
|
|
&glib::Bytes::from(background.as_bytes()),
|
|
|
|
gtk::gdk_pixbuf::Colorspace::Rgb,
|
|
|
|
false,
|
|
|
|
8,
|
|
|
|
background.width() as i32,
|
|
|
|
background.height() as i32,
|
|
|
|
background.to_rgb8().sample_layout().height_stride as i32,
|
|
|
|
)
|
|
|
|
.scale_simple(WIDTH, HEIGHT, InterpType::Nearest)
|
|
|
|
});
|
|
|
|
|
2023-03-25 13:00:56 +00:00
|
|
|
self.drawing_area
|
|
|
|
.set_draw_func(move |_, context, width, height| {
|
2023-04-05 13:13:50 +00:00
|
|
|
let board = board.borrow();
|
2023-03-31 02:25:11 +00:00
|
|
|
match background {
|
|
|
|
Ok(Some(ref background)) => {
|
|
|
|
context.set_source_pixbuf(&background, 0., 0.);
|
|
|
|
context.paint().expect("paint should succeed");
|
|
|
|
}
|
|
|
|
Ok(None) | Err(_) => context.set_source_rgb(0.7, 0.7, 0.7),
|
|
|
|
};
|
2023-03-25 13:00:56 +00:00
|
|
|
let _ = context.paint();
|
|
|
|
|
|
|
|
context.set_source_rgb(0.1, 0.1, 0.1);
|
|
|
|
context.set_line_width(2.);
|
2023-04-05 13:13:50 +00:00
|
|
|
let hspace_between = ((width - 40) as f64) / ((board.size.width - 1) as f64);
|
|
|
|
let vspace_between = ((height - 40) as f64) / ((board.size.height - 1) as f64);
|
2023-03-25 13:00:56 +00:00
|
|
|
|
|
|
|
let pen = Pen {
|
|
|
|
x_offset: 20.,
|
|
|
|
y_offset: 20.,
|
|
|
|
hspace_between,
|
|
|
|
vspace_between,
|
|
|
|
};
|
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
(0..board.size.width).for_each(|col| {
|
2023-03-25 13:00:56 +00:00
|
|
|
context.move_to(20.0 + (col as f64) * hspace_between, 20.0);
|
|
|
|
context.line_to(20.0 + (col as f64) * hspace_between, (height as f64) - 20.0);
|
|
|
|
let _ = context.stroke();
|
|
|
|
});
|
2023-04-05 13:13:50 +00:00
|
|
|
(0..board.size.height).for_each(|row| {
|
2023-03-25 13:00:56 +00:00
|
|
|
context.move_to(20.0, 20.0 + (row as f64) * vspace_between);
|
|
|
|
context.line_to((width - 20) as f64, 20.0 + (row as f64) * vspace_between);
|
|
|
|
let _ = context.stroke();
|
|
|
|
});
|
|
|
|
|
|
|
|
context.set_source_rgb(0.1, 0.1, 0.0);
|
2023-03-25 13:24:36 +00:00
|
|
|
vec![3, 9, 15].into_iter().for_each(|col| {
|
|
|
|
vec![3, 9, 15].into_iter().for_each(|row| {
|
2023-03-25 13:00:56 +00:00
|
|
|
pen.star_point(context, col, row);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
(0..19).for_each(|col| {
|
|
|
|
(0..19).for_each(|row| {
|
2023-04-05 13:13:50 +00:00
|
|
|
match board.stone(row, col) {
|
|
|
|
IntersectionElement::Unplayable => {}
|
|
|
|
IntersectionElement::Empty(request) => {}
|
|
|
|
IntersectionElement::Filled(stone) => {
|
|
|
|
pen.stone(&context, row, col, stone.color);
|
2023-03-25 13:00:56 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
})
|
|
|
|
});
|
2023-03-31 01:01:48 +00:00
|
|
|
|
|
|
|
let cursor = cursor_location.borrow();
|
2023-04-05 13:25:08 +00:00
|
|
|
match board.stone(cursor.row, cursor.column) {
|
|
|
|
IntersectionElement::Empty(_) => pen.ghost_stone(
|
|
|
|
context,
|
|
|
|
cursor.row,
|
|
|
|
cursor.column,
|
|
|
|
*current_player.borrow(),
|
|
|
|
),
|
|
|
|
_ => {}
|
|
|
|
}
|
2023-03-25 13:00:56 +00:00
|
|
|
});
|
2023-03-31 01:01:48 +00:00
|
|
|
|
|
|
|
let motion_controller = gtk::EventControllerMotion::new();
|
|
|
|
{
|
2023-04-05 13:13:50 +00:00
|
|
|
let board = self.board.clone();
|
2023-03-31 01:01:48 +00:00
|
|
|
let cursor = self.cursor_location.clone();
|
|
|
|
let drawing_area = self.drawing_area.clone();
|
|
|
|
motion_controller.connect_motion(move |_, x, y| {
|
2023-04-05 13:13:50 +00:00
|
|
|
let board = board.borrow();
|
2023-03-31 01:01:48 +00:00
|
|
|
let mut cursor = cursor.borrow_mut();
|
2023-04-05 13:13:50 +00:00
|
|
|
let hspace_between = ((WIDTH - 40) as f64) / ((board.size.width - 1) as f64);
|
|
|
|
let vspace_between = ((HEIGHT - 40) as f64) / ((board.size.height - 1) as f64);
|
2023-03-31 01:01:48 +00:00
|
|
|
|
|
|
|
let addr = Addr {
|
|
|
|
column: ((x.round() - 20.) / hspace_between).round() as u8,
|
|
|
|
row: ((y.round() - 20.) / vspace_between).round() as u8,
|
|
|
|
};
|
|
|
|
|
|
|
|
if *cursor != addr {
|
|
|
|
*cursor = addr;
|
|
|
|
drawing_area.queue_draw();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-04-05 13:48:33 +00:00
|
|
|
let gesture = gtk::GestureClick::new();
|
|
|
|
{
|
|
|
|
let board = self.board.clone();
|
|
|
|
let cursor = self.cursor_location.clone();
|
|
|
|
gesture.connect_released(move |_, _, _, _| {
|
|
|
|
let board = board.borrow();
|
|
|
|
let cursor = cursor.borrow();
|
|
|
|
match board.stone(cursor.row, cursor.column) {
|
|
|
|
IntersectionElement::Empty(request) => {
|
|
|
|
println!("need to send request: {:?}", request)
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-03-31 01:01:48 +00:00
|
|
|
self.drawing_area.add_controller(motion_controller);
|
2023-04-05 13:48:33 +00:00
|
|
|
self.drawing_area.add_controller(gesture);
|
2023-03-25 13:00:56 +00:00
|
|
|
}
|
|
|
|
}
|
2023-04-05 13:13:50 +00:00
|
|
|
impl WidgetImpl for BoardPrivate {}
|
|
|
|
impl GridImpl for BoardPrivate {}
|
2023-03-24 14:14:01 +00:00
|
|
|
|
|
|
|
glib::wrapper! {
|
2023-04-05 13:13:50 +00:00
|
|
|
pub struct Board(ObjectSubclass<BoardPrivate>) @extends gtk::Grid, gtk::Widget;
|
2023-03-24 14:14:01 +00:00
|
|
|
}
|
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
impl Board {
|
2023-03-24 14:14:01 +00:00
|
|
|
pub fn new() -> Self {
|
|
|
|
let s: Self = Object::builder().build();
|
2023-03-25 13:00:56 +00:00
|
|
|
s.attach(&s.imp().drawing_area, 1, 1, 1, 1);
|
2023-03-24 14:14:01 +00:00
|
|
|
s
|
|
|
|
}
|
2023-03-25 13:00:56 +00:00
|
|
|
|
2023-04-05 13:13:50 +00:00
|
|
|
pub fn set_board(&self, board: BoardElement) {
|
|
|
|
*self.imp().board.borrow_mut() = board;
|
2023-03-25 13:00:56 +00:00
|
|
|
self.imp().drawing_area.queue_draw();
|
|
|
|
}
|
2023-03-31 01:01:48 +00:00
|
|
|
|
|
|
|
pub fn set_current_player(&self, color: Color) {
|
|
|
|
*self.imp().current_player.borrow_mut() = color;
|
|
|
|
}
|
2023-03-25 13:00:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct Pen {
|
|
|
|
x_offset: f64,
|
|
|
|
y_offset: f64,
|
|
|
|
hspace_between: f64,
|
|
|
|
vspace_between: f64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Pen {
|
|
|
|
fn star_point(&self, context: &cairo::Context, row: u8, col: u8) {
|
|
|
|
context.arc(
|
|
|
|
self.x_offset + (col as f64) * self.hspace_between,
|
|
|
|
self.y_offset + (row as f64) * self.vspace_between,
|
|
|
|
10.,
|
|
|
|
0.,
|
|
|
|
2. * std::f64::consts::PI,
|
|
|
|
);
|
|
|
|
let _ = context.fill();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn stone(&self, context: &cairo::Context, row: u8, col: u8, color: Color) {
|
|
|
|
match color {
|
|
|
|
Color::White => context.set_source_rgb(0.9, 0.9, 0.9),
|
|
|
|
Color::Black => context.set_source_rgb(0.0, 0.0, 0.0),
|
|
|
|
};
|
|
|
|
context.arc(
|
|
|
|
20.0 + (col as f64) * self.hspace_between,
|
|
|
|
20.0 + (row as f64) * self.vspace_between,
|
|
|
|
25.0,
|
|
|
|
0.0,
|
|
|
|
2.0 * std::f64::consts::PI,
|
|
|
|
);
|
|
|
|
let _ = context.fill();
|
|
|
|
}
|
2023-03-31 01:01:48 +00:00
|
|
|
|
|
|
|
fn ghost_stone(&self, context: &cairo::Context, row: u8, col: u8, color: Color) {
|
|
|
|
match color {
|
|
|
|
Color::White => context.set_source_rgba(0.9, 0.9, 0.9, 0.5),
|
|
|
|
Color::Black => context.set_source_rgba(0.0, 0.0, 0.0, 0.5),
|
|
|
|
};
|
|
|
|
context.arc(
|
|
|
|
20.0 + (col as f64) * self.hspace_between,
|
|
|
|
20.0 + (row as f64) * self.vspace_between,
|
|
|
|
25.0,
|
|
|
|
0.0,
|
|
|
|
2.0 * std::f64::consts::PI,
|
|
|
|
);
|
|
|
|
let _ = context.fill();
|
|
|
|
}
|
2023-03-24 14:14:01 +00:00
|
|
|
}
|