2023-12-22 19:54:38 +00:00
|
|
|
/*
|
|
|
|
Copyright 2023, Savanni D'Gerinel <savanni@luminescent-dreams.com>
|
|
|
|
|
|
|
|
This file is part of FitnessTrax.
|
|
|
|
|
|
|
|
FitnessTrax is free software: you can redistribute it and/or modify it under the terms of the GNU
|
|
|
|
General Public License as published by the Free Software Foundation, either version 3 of the
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
FitnessTrax is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
|
|
|
|
even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with FitnessTrax. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
use crate::{
|
2023-12-22 20:08:34 +00:00
|
|
|
app::{AppInvocation, AppResponse},
|
2023-12-25 15:02:24 +00:00
|
|
|
components::DayDetail,
|
2023-12-22 19:54:38 +00:00
|
|
|
views::{HistoricalView, PlaceholderView, View, ViewName, WelcomeView},
|
|
|
|
};
|
|
|
|
use adw::prelude::*;
|
|
|
|
use async_channel::Sender;
|
2023-12-19 00:35:10 +00:00
|
|
|
use chrono::{FixedOffset, NaiveDate, TimeZone};
|
2023-12-25 00:13:49 +00:00
|
|
|
use dimensioned::si::{KG, M, S};
|
|
|
|
use ft_core::{Steps, TimeDistance, TraxRecord, Weight};
|
2023-12-22 19:54:38 +00:00
|
|
|
use gio::resources_lookup_data;
|
|
|
|
use gtk::STYLE_PROVIDER_PRIORITY_USER;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::{cell::RefCell, rc::Rc};
|
|
|
|
|
|
|
|
/// The application window, or the main window, is the main user interface for the app. Almost
|
|
|
|
/// everything occurs here.
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct AppWindow {
|
|
|
|
app_tx: Sender<AppInvocation>,
|
|
|
|
layout: gtk::Box,
|
|
|
|
current_view: Rc<RefCell<View>>,
|
|
|
|
settings: gio::Settings,
|
2023-12-25 06:09:31 +00:00
|
|
|
navigation: adw::NavigationView,
|
2023-12-22 19:54:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AppWindow {
|
|
|
|
/// Construct a new App Window.
|
|
|
|
///
|
|
|
|
/// adw_app is an Adwaita application. Application windows need to have access to this, but
|
|
|
|
/// otherwise we don't use this.
|
|
|
|
///
|
2023-12-24 17:00:12 +00:00
|
|
|
/// app is a core [crate::app::App] object which encapsulates all of the basic logic.
|
2023-12-22 19:54:38 +00:00
|
|
|
pub fn new(
|
|
|
|
app_id: &str,
|
|
|
|
resource_path: &str,
|
|
|
|
adw_app: &adw::Application,
|
|
|
|
app_tx: Sender<AppInvocation>,
|
|
|
|
) -> AppWindow {
|
|
|
|
let window = adw::ApplicationWindow::builder()
|
|
|
|
.application(adw_app)
|
|
|
|
.width_request(800)
|
|
|
|
.height_request(600)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let stylesheet = String::from_utf8(
|
|
|
|
resources_lookup_data(
|
|
|
|
&format!("{}style.css", resource_path),
|
|
|
|
gio::ResourceLookupFlags::NONE,
|
|
|
|
)
|
|
|
|
.expect("stylesheet must be available in the resources")
|
|
|
|
.to_vec(),
|
|
|
|
)
|
|
|
|
.expect("to parse stylesheet");
|
|
|
|
|
|
|
|
let provider = gtk::CssProvider::new();
|
|
|
|
provider.load_from_data(&stylesheet);
|
|
|
|
|
2023-12-22 20:08:34 +00:00
|
|
|
#[allow(deprecated)]
|
2023-12-22 19:54:38 +00:00
|
|
|
let context = window.style_context();
|
2023-12-22 20:08:34 +00:00
|
|
|
#[allow(deprecated)]
|
2023-12-22 19:54:38 +00:00
|
|
|
context.add_provider(&provider, STYLE_PROVIDER_PRIORITY_USER);
|
|
|
|
|
2023-12-25 06:09:31 +00:00
|
|
|
let navigation = adw::NavigationView::new();
|
2023-12-25 05:36:13 +00:00
|
|
|
|
2023-12-25 06:09:31 +00:00
|
|
|
/*
|
2023-12-22 19:54:38 +00:00
|
|
|
let header = adw::HeaderBar::builder()
|
|
|
|
.title_widget(>k::Label::new(Some("FitnessTrax")))
|
|
|
|
.build();
|
2023-12-25 06:09:31 +00:00
|
|
|
*/
|
2023-12-22 19:54:38 +00:00
|
|
|
|
|
|
|
let layout = gtk::Box::builder()
|
|
|
|
.orientation(gtk::Orientation::Vertical)
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let initial_view = View::Placeholder(PlaceholderView::new().upcast());
|
|
|
|
|
2023-12-25 06:09:31 +00:00
|
|
|
// layout.append(&header);
|
2023-12-22 19:54:38 +00:00
|
|
|
layout.append(initial_view.widget());
|
|
|
|
|
2023-12-25 06:09:31 +00:00
|
|
|
let nav_layout = gtk::Box::new(gtk::Orientation::Vertical, 0);
|
|
|
|
nav_layout.append(&adw::HeaderBar::new());
|
|
|
|
nav_layout.append(&layout);
|
|
|
|
navigation.push(
|
|
|
|
&adw::NavigationPage::builder()
|
|
|
|
.can_pop(false)
|
|
|
|
.title("FitnessTrax")
|
|
|
|
.child(&nav_layout)
|
|
|
|
.build(),
|
|
|
|
);
|
|
|
|
|
|
|
|
window.set_content(Some(&navigation));
|
2023-12-22 19:54:38 +00:00
|
|
|
window.present();
|
|
|
|
|
2023-12-25 05:36:13 +00:00
|
|
|
let gesture = gtk::GestureClick::new();
|
|
|
|
gesture.connect_released(|_, _, _, _| println!("detected gesture"));
|
|
|
|
layout.add_controller(gesture);
|
|
|
|
|
2023-12-22 19:54:38 +00:00
|
|
|
let s = Self {
|
|
|
|
app_tx,
|
|
|
|
layout,
|
|
|
|
current_view: Rc::new(RefCell::new(initial_view)),
|
|
|
|
settings: gio::Settings::new(app_id),
|
2023-12-25 06:09:31 +00:00
|
|
|
navigation,
|
2023-12-22 19:54:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
s
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn change_view(&self, view: ViewName) {
|
|
|
|
self.swap_main(self.construct_view(view));
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn process_response(&self, response: AppResponse) {
|
|
|
|
match response {
|
|
|
|
AppResponse::DatabaseChanged(db_path) => {
|
|
|
|
self.settings
|
|
|
|
.set_string("series-path", db_path.to_str().unwrap())
|
|
|
|
.unwrap();
|
|
|
|
self.change_view(ViewName::Historical);
|
|
|
|
}
|
|
|
|
AppResponse::NoDatabase => {
|
|
|
|
self.change_view(ViewName::Welcome);
|
|
|
|
}
|
|
|
|
AppResponse::Records => {
|
|
|
|
self.change_view(ViewName::Historical);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Switch views.
|
|
|
|
//
|
|
|
|
// This function only replaces the old view with the one which matches the current view state.
|
|
|
|
// It is responsible for ensuring that the new view goes into the layout in the correct
|
|
|
|
// position.
|
|
|
|
fn swap_main(&self, view: View) {
|
|
|
|
let mut current_widget = self.current_view.borrow_mut();
|
|
|
|
self.layout.remove(&*current_widget.widget());
|
|
|
|
*current_widget = view;
|
|
|
|
self.layout.append(&*current_widget.widget());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn construct_view(&self, view: ViewName) -> View {
|
|
|
|
match view {
|
|
|
|
ViewName::Welcome => View::Welcome(
|
|
|
|
WelcomeView::new({
|
|
|
|
let s = self.clone();
|
|
|
|
Box::new(move |path: PathBuf| {
|
|
|
|
s.app_tx
|
|
|
|
.send_blocking(AppInvocation::OpenDatabase(path))
|
|
|
|
.unwrap();
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.upcast(),
|
|
|
|
),
|
2023-12-25 00:13:49 +00:00
|
|
|
ViewName::Historical => View::Historical(
|
2023-12-25 05:36:13 +00:00
|
|
|
HistoricalView::new(
|
|
|
|
vec![
|
|
|
|
TraxRecord::Steps(Steps {
|
|
|
|
date: NaiveDate::from_ymd_opt(2023, 10, 13).unwrap(),
|
|
|
|
count: 1500,
|
|
|
|
}),
|
|
|
|
TraxRecord::Weight(Weight {
|
|
|
|
date: NaiveDate::from_ymd_opt(2023, 10, 13).unwrap(),
|
|
|
|
weight: 85. * KG,
|
|
|
|
}),
|
|
|
|
TraxRecord::Weight(Weight {
|
|
|
|
date: NaiveDate::from_ymd_opt(2023, 10, 14).unwrap(),
|
|
|
|
weight: 86. * KG,
|
|
|
|
}),
|
|
|
|
TraxRecord::BikeRide(TimeDistance {
|
|
|
|
datetime: FixedOffset::west_opt(10 * 60 * 60)
|
|
|
|
.unwrap()
|
|
|
|
.with_ymd_and_hms(2019, 6, 15, 12, 0, 0)
|
|
|
|
.unwrap(),
|
|
|
|
distance: Some(1000. * M),
|
|
|
|
duration: Some(150. * S),
|
|
|
|
comments: Some("Test Comments".to_owned()),
|
|
|
|
}),
|
|
|
|
TraxRecord::BikeRide(TimeDistance {
|
|
|
|
datetime: FixedOffset::west_opt(10 * 60 * 60)
|
|
|
|
.unwrap()
|
|
|
|
.with_ymd_and_hms(2019, 6, 15, 23, 0, 0)
|
|
|
|
.unwrap(),
|
|
|
|
distance: Some(1000. * M),
|
|
|
|
duration: Some(150. * S),
|
|
|
|
comments: Some("Test Comments".to_owned()),
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
{
|
|
|
|
let s = self.clone();
|
2023-12-25 06:09:31 +00:00
|
|
|
Rc::new(move |date: chrono::NaiveDate, records| {
|
|
|
|
let layout = gtk::Box::new(gtk::Orientation::Vertical, 0);
|
|
|
|
layout.append(&adw::HeaderBar::new());
|
2023-12-25 15:02:24 +00:00
|
|
|
layout.append(&DayDetail::new(date, records));
|
2023-12-25 06:09:31 +00:00
|
|
|
let page = &adw::NavigationPage::builder()
|
|
|
|
.title(date.format("%Y-%m-%d").to_string())
|
|
|
|
.child(&layout)
|
|
|
|
.build();
|
|
|
|
s.navigation.push(page);
|
|
|
|
})
|
2023-12-25 05:36:13 +00:00
|
|
|
},
|
|
|
|
)
|
2023-12-25 00:13:49 +00:00
|
|
|
.upcast(),
|
|
|
|
),
|
2023-12-22 19:54:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|