Compare commits

16 Commits

Author SHA1 Message Date
4dc74c0011 fix: corriger l'utilisation de usePrecision pour totalHours dans le composant Stats 2026-01-04 20:04:38 +01:00
f95a417b37 fix: mettre à jour le statut du projet de 'En cours' à 'Terminé' dans data-visualisation.md 2026-01-04 20:01:00 +01:00
055c16e198 Refactor code structure for improved readability and maintainability 2026-01-04 20:00:43 +01:00
669eec60a1 fix: refactor le calcul de totalHours dans le composant Stats 2026-01-04 19:57:42 +01:00
4cda11a6a3 fix: ajouter des conditions d'affichage pour les cartes dans le composant Stats 2026-01-04 19:54:18 +01:00
df33efc731 fix: corriger la déclaration de totalHours dans le composant Stats 2026-01-04 19:53:15 +01:00
305c91199a fix: corriger la récupération de la date de début dans le composant Stats 2026-01-04 19:46:58 +01:00
ed3019d1b9 fix: corriger l'utilisation des valeurs retournées pour yearsCollected et formattedDate dans le composant Stats 2026-01-04 19:43:10 +01:00
988e18b2f7 fix: gérer les valeurs nulles pour les statistiques dans le composant Stats 2026-01-04 19:41:22 +01:00
5d4f6ee9a4 fix: uniformiser le nom de cache pour les fonctions Wakatime dans le composant Stats 2026-01-04 19:40:10 +01:00
d13594e5d1 fix: uniformiser le nom de cache pour les données Wakatime dans le composant Stats 2026-01-04 19:34:43 +01:00
9aabe422b3 fix: supprimer le chargement paresseux des données dans le composant Stats et améliorer la gestion des requêtes API 2026-01-04 19:30:48 +01:00
e30d956f58 fix: activer le chargement paresseux des données dans le composant Stats 2026-01-04 19:23:08 +01:00
56c63d8db7 fix: mettre à jour les couleurs des icônes et des barres de progression dans le composant Stats 2026-01-04 19:07:47 +01:00
c8116e10b8 fix: supprimer l'affichage des langues, éditeurs et systèmes d'exploitation dans le composant Stats et ajuster les couleurs 2026-01-04 19:03:32 +01:00
9dc21f4287 fix: afficher les langues, éditeurs et systèmes d'exploitation dans le composant Stats 2026-01-04 18:59:57 +01:00
5 changed files with 72 additions and 36 deletions

View File

@@ -4,15 +4,19 @@ import { usePrecision } from '@vueuse/math'
const { data: stats } = await useAsyncData<Stats>('stats', () => $fetch('/api/stats'))
const startDate = computed(() => new Date(stats.value!.coding.range.start))
const yearsCollected = useTimeAgo(startDate).value
const formattedDate = useDateFormat(startDate, 'MMM DD, YYYY').value
const startDate = computed(() => new Date(stats.value?.coding?.range?.start ?? new Date()))
const rawHours = computed(() => {
const seconds = stats.value?.coding?.grand_total?.total_seconds_including_other_language ?? 0
return seconds / 3600
})
const totalHours = usePrecision((stats.value!.coding.grand_total.total_seconds_including_other_language ?? 0) / 3600, 0)
const totalHours = usePrecision(rawHours, 0)
const yearsCollected = useTimeAgo(startDate)
const formattedDate = useDateFormat(startDate, 'MMM DD, YYYY')
const topLanguages = computed(() => stats.value!.languages.slice(0, 4))
const topEditors = computed(() => stats.value!.editors.slice(0, 3))
const topOS = computed(() => stats.value!.os.slice(0, 2))
const topLanguages = computed(() => stats.value?.languages.slice(0, 4) ?? [])
const topEditors = computed(() => stats.value?.editors.slice(0, 3) ?? [])
const topOS = computed(() => stats.value?.os.slice(0, 2) ?? [])
</script>
<template>
@@ -21,8 +25,12 @@ const topOS = computed(() => stats.value!.os.slice(0, 2))
v-if="stats"
class="space-y-6"
>
{{ rawHours }}
{{ totalHours }}
{{ usePrecision(rawHours, 0) }}
{{ usePrecision(rawHours, 0) }}
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
<UCard>
<UCard v-if="totalHours">
<div class="flex items-center gap-4">
<div class="p-3 bg-primary-200 dark:bg-primary-900 rounded-lg text-primary-500 flex items-center justify-center">
<UIcon
@@ -41,7 +49,7 @@ const topOS = computed(() => stats.value!.os.slice(0, 2))
</div>
</UCard>
<UCard>
<UCard v-if="formattedDate && yearsCollected">
<div class="flex items-center gap-4">
<div class="p-3 bg-emerald-100 dark:bg-emerald-900/20 rounded-lg text-emerald-500 flex items-center justify-center">
<UIcon
@@ -71,11 +79,14 @@ const topOS = computed(() => stats.value!.os.slice(0, 2))
</div>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
<div class="col-span-1 lg:col-span-1 space-y-4">
<div
v-if="topLanguages.length"
class="col-span-1 lg:col-span-1 space-y-4"
>
<h4 class="text-sm font-semibold text-neutral-900 dark:text-white flex items-center gap-2">
<UIcon
name="i-ph-code-block-duotone"
class="text-primary-500 w-5 h-5"
class="text-red-500 w-5 h-5"
/>
Top Languages
</h4>
@@ -91,18 +102,21 @@ const topOS = computed(() => stats.value!.os.slice(0, 2))
</div>
<UProgress
v-model="lang.percent"
color="primary"
color="red"
size="sm"
/>
</div>
</div>
</div>
<div class="col-span-1 lg:col-span-1 space-y-4">
<div
v-if="topEditors.length"
class="col-span-1 lg:col-span-1 space-y-4"
>
<h4 class="text-sm font-semibold text-neutral-900 dark:text-white flex items-center gap-2">
<UIcon
name="i-ph-terminal-window-duotone"
class="text-orange-500 w-5 h-5"
class="text-green-500 w-5 h-5"
/>
Preferred Editors
</h4>
@@ -118,14 +132,17 @@ const topOS = computed(() => stats.value!.os.slice(0, 2))
</div>
<UProgress
v-model="editor.percent"
color="orange"
color="green"
size="sm"
/>
</div>
</div>
</div>
<div class="col-span-1 lg:col-span-1 space-y-4">
<div
v-if="topOS.length"
class="col-span-1 lg:col-span-1 space-y-4"
>
<h4 class="text-sm font-semibold text-neutral-900 dark:text-white flex items-center gap-2">
<UIcon
name="i-ph-desktop-duotone"

View File

@@ -5,7 +5,7 @@ type: Academic Project
description: An interactive data visualization project built with R, R Shiny, and ggplot2 for creating dynamic, explorable visualizations.
publishedAt: 2026-01-05
readingTime: 1
status: In progress
status: Completed
tags:
- R
- R Shiny
@@ -44,9 +44,11 @@ This project involves creating an interactive data visualization application usi
## 📚 Resources
You can find the code here: [Data Visualisation](https://go.arthurdanjou.fr/dataviz)
You can find the code here: [Data Visualisation Code](https://go.arthurdanjou.fr/datavis-code)
And the online application here: [Data Visualisation App](https://go.arthurdanjou.fr/datavis-app)
## 📄 Detailed Report
<iframe src="/projects/dataviz.pdf" width="100%" height="1000px">
<iframe src="/projects/datavis.pdf" width="100%" height="1000px">
</iframe>

BIN
public/projects/datavis.pdf Normal file

Binary file not shown.

View File

@@ -1,42 +1,56 @@
import type { H3Event } from 'h3'
const WAKATIME_HEADERS = {
'User-Agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36',
'Accept': 'application/json'
}
const fetchWakatime = async (url: string) => {
try {
return await $fetch<{ data: unknown[] }>(url, {
headers: WAKATIME_HEADERS,
timeout: 5000
})
}
catch (err) {
console.error(`[Wakatime Error] Failed to fetch ${url}`, err)
return { data: [] }
}
}
const cachedWakatimeCoding = defineCachedFunction(async (event: H3Event) => {
const config = useRuntimeConfig(event)
return await $fetch<{ data: unknown[] }>(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.coding}.json`)
return await fetchWakatime(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.coding}.json`)
}, {
maxAge: 60,
name: 'wakatime',
maxAge: 60 * 50,
name: 'stats',
getKey: () => 'coding'
})
const cachedWakatimeEditors = defineCachedFunction(async (event: H3Event) => {
const config = useRuntimeConfig(event)
return await $fetch<{ data: unknown[] }>(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.editors}.json`)
return await fetchWakatime(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.editors}.json`)
}, {
maxAge: 60,
name: 'wakatime',
maxAge: 60 * 60,
name: 'stats',
getKey: () => 'editors'
})
const cachedWakatimeOs = defineCachedFunction(async (event: H3Event) => {
const config = useRuntimeConfig(event)
return await $fetch<{ data: unknown[] }>(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.os}.json`)
return await fetchWakatime(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.os}.json`)
}, {
maxAge: 60,
name: 'wakatime',
maxAge: 60 * 60,
name: 'stats',
getKey: () => 'os'
})
const cachedWakatimeLanguages = defineCachedFunction(async (event: H3Event) => {
const config = useRuntimeConfig(event)
return await $fetch<{ data: unknown[] }>(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.languages}.json`)
return await fetchWakatime(`https://wakatime.com/share/${config.wakatime.userId}/${config.wakatime.languages}.json`)
}, {
maxAge: 60,
name: 'wakatime',
maxAge: 60 * 60,
name: 'stats',
getKey: () => 'languages'
})

View File

@@ -1,7 +1,10 @@
/* eslint-disable */
// Generated by Wrangler by running `wrangler types` (hash: 373e9a05bf207b93549ab53665d07e4b)
// Generated by Wrangler by running `wrangler types` (hash: 8c48032b4b2801cdbac6e8dbc9d26203)
// Runtime types generated with workerd@1.20251210.0 2025-12-13 nodejs_compat
declare namespace Cloudflare {
interface GlobalProps {
mainModule: typeof import("./.output/server/index");
}
interface Env {
CACHE: KVNamespace;
STUDIO_GITHUB_CLIENT_ID: string;