Skip to content

Commit

Permalink
feat(gui): add activity calendar to the homepage (#2160)
Browse files Browse the repository at this point in the history
* feat(gui): add activity calendar to the homepage

* localise week start
  • Loading branch information
ellie authored Jun 18, 2024
1 parent 7984f9e commit b8be23e
Show file tree
Hide file tree
Showing 6 changed files with 159 additions and 1 deletion.
18 changes: 18 additions & 0 deletions ui/backend/src/db.rs
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,24 @@ impl HistoryDB {
Ok(history)
}

pub async fn calendar(&self) -> Result<Vec<(String, u64)>, String> {
let query = "select count(1) as count, strftime('%F', datetime(timestamp / 1000000000, 'unixepoch')) as day from history where timestamp > ((unixepoch() - 31536000) * 1000000000) group by day;";

let calendar: Vec<(String, u64)> = sqlx::query(query)
// safe to cast, count(x) is never < 0
.map(|row: SqliteRow| {
(
row.get::<String, _>("day"),
row.get::<i64, _>("count") as u64,
)
})
.fetch_all(&self.0.pool)
.await
.map_err(|e| e.to_string())?;

Ok(calendar)
}

pub async fn global_stats(&self) -> Result<GlobalStats, String> {
let day_ago = time::OffsetDateTime::now_utc() - time::Duration::days(1);
let day_ago = day_ago.unix_timestamp_nanos();
Expand Down
49 changes: 49 additions & 0 deletions ui/backend/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -167,6 +167,54 @@ async fn home_info() -> Result<HomeInfo, String> {
Ok(info)
}

// Match the format that the frontend library we use expects
// All the processing in Rust, not JS.
// Faaaassssssst af ⚡️🦀
#[derive(Debug, serde::Serialize)]
pub struct HistoryCalendarDay {
pub date: String,
pub count: u64,
pub level: u8,
}

#[tauri::command]
async fn history_calendar() -> Result<Vec<HistoryCalendarDay>, String> {
let settings = Settings::new().map_err(|e| e.to_string())?;
let db_path = PathBuf::from(settings.db_path.as_str());
let db = HistoryDB::new(db_path, settings.local_timeout).await?;

let calendar = db.calendar().await?;

// probs don't want to iterate _this_ many times, but it's only the last year. so 365
// iterations at max. should be quick.

let max = calendar
.iter()
.max_by_key(|d| d.1)
.expect("Can't find max count");

let ret = calendar
.iter()
.map(|d| {
// calculate the "level". we have 5, so figure out which 5th it fits into
let percent: f64 = d.1 as f64 / max.1 as f64;
let level = if d.1 == 0 {
0.0
} else {
(percent / 0.2).round() + 1.0
};

HistoryCalendarDay {
date: d.0.clone(),
count: d.1,
level: std::cmp::min(4, level as u8),
}
})
.collect();

Ok(ret)
}

fn show_window(app: &AppHandle) {
let windows = app.webview_windows();

Expand All @@ -190,6 +238,7 @@ fn main() {
session,
login,
register,
history_calendar,
install::install_cli,
install::is_cli_installed,
install::setup_cli,
Expand Down
2 changes: 2 additions & 0 deletions ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,10 @@
"prism-react-renderer": "^2.3.1",
"prismjs": "^1.29.0",
"react": "^18.3.1",
"react-activity-calendar": "^2.2.10",
"react-dom": "^18.3.1",
"react-spinners": "^0.13.8",
"react-tooltip": "^5.27.0",
"react-window": "^1.8.10",
"react-window-infinite-loader": "^1.0.9",
"recharts": "^2.12.7",
Expand Down
43 changes: 43 additions & 0 deletions ui/pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

39 changes: 38 additions & 1 deletion ui/src/pages/Home.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
import { useEffect } from "react";
import React, { useEffect, useState } from "react";
import { formatRelative } from "date-fns";
import { Tooltip as ReactTooltip } from "react-tooltip";

import { useStore } from "@/state/store";
import { useToast } from "@/components/ui/use-toast";
import { invoke } from "@tauri-apps/api/core";

import ActivityCalendar from "react-activity-calendar";

function Stats({ stats }: any) {
return (
<div>
Expand Down Expand Up @@ -44,16 +47,32 @@ function Header({ name }: any) {
);
}

const explicitTheme: ThemeInput = {
light: ["#f0f0f0", "#c4edde", "#7ac7c4", "#f73859", "#384259"],
dark: ["#383838", "#4D455D", "#7DB9B6", "#F5E9CF", "#E96479"],
};

export default function Home() {
const [weekStart, setWeekStart] = useState(0);

const homeInfo = useStore((state) => state.homeInfo);
const user = useStore((state) => state.user);
const calendar = useStore((state) => state.calendar);

const refreshHomeInfo = useStore((state) => state.refreshHomeInfo);
const refreshUser = useStore((state) => state.refreshUser);
const refreshCalendar = useStore((state) => state.refreshCalendar);

const { toast } = useToast();

useEffect(() => {
let locale = new Intl.Locale(navigator.language);
let weekinfo = locale.getWeekInfo();
setWeekStart(weekinfo.firstDay);

refreshHomeInfo();
refreshUser();
refreshCalendar();

let setup = async () => {
let installed = await invoke("is_cli_installed");
Expand Down Expand Up @@ -112,6 +131,24 @@ export default function Home() {
]}
/>
</div>

<div className="pt-10">
<ActivityCalendar
theme={explicitTheme}
data={calendar}
weekStart={weekStart}
renderBlock={(block, activity) =>
React.cloneElement(block, {
"data-tooltip-id": "react-tooltip",
"data-tooltip-html": `${activity.count} commands on ${activity.date}`,
})
}
labels={{
totalCount: "{{count}} history records in the last year",
}}
/>
<ReactTooltip id="react-tooltip" />
</div>
</div>
</div>
);
Expand Down
9 changes: 9 additions & 0 deletions ui/src/state/store.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,10 @@ interface AtuinState {
aliases: Alias[];
vars: Var[];
shellHistory: ShellHistory[];
calendar: any[];

refreshHomeInfo: () => void;
refreshCalendar: () => void;
refreshAliases: () => void;
refreshVars: () => void;
refreshUser: () => void;
Expand All @@ -40,13 +42,20 @@ export const useStore = create<AtuinState>()((set, get) => ({
aliases: [],
vars: [],
shellHistory: [],
calendar: [],

refreshAliases: () => {
invoke("aliases").then((aliases: any) => {
set({ aliases: aliases });
});
},

refreshCalendar: () => {
invoke("history_calendar").then((calendar: any) => {
set({ calendar: calendar });
});
},

refreshVars: () => {
invoke("vars").then((vars: any) => {
set({ vars: vars });
Expand Down

0 comments on commit b8be23e

Please sign in to comment.