Compare commits
	
		
			14 Commits
		
	
	
		
			1.1.1
			...
			2bd07e5f2d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 2bd07e5f2d | |||
| 058b83522c | |||
| f13ed8a078 | |||
| 02d5adcb3c | |||
| d6fb16bb74 | |||
| 71b90b8202 | |||
| 3ee36932c3 | |||
| 391fcc79a8 | |||
| 57d4fd7212 | |||
| a5b2854bf6 | |||
| 518c0ecd56 | |||
| a6cd7a3430 | |||
| dba8414fd9 | |||
| 0e2113eefd | 
							
								
								
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @ -1,3 +1,31 @@ | |||||||
|  | ## Unreleased | ||||||
|  |  | ||||||
|  | ### New | ||||||
|  | * Add Stats to the main navigation | ||||||
|  | * Allow selecting year on the Stats page | ||||||
|  |  | ||||||
|  | ### Improved | ||||||
|  | * Make navigation more compact | ||||||
|  |  | ||||||
|  | ### Fixed | ||||||
|  | * Correctly limit sessions to a single year for stats | ||||||
|  |  | ||||||
|  | ## 1.2.0 / 2023-11-01 20:18+01:00 | ||||||
|  |  | ||||||
|  | ### New | ||||||
|  | * Add yearly stats page (https://git.kucharczyk.xyz/lukas/timetracker/issues/15) | ||||||
|  |  | ||||||
|  | ### Enhancements | ||||||
|  | * Add a button to start session from game overview | ||||||
|  |  | ||||||
|  | ## 1.1.2 / 2023-10-13 16:30+02:00 | ||||||
|  |  | ||||||
|  | ### Enhancements | ||||||
|  | * Durations are formatted in a consisent manner across all pages | ||||||
|  |  | ||||||
|  | ### Fixes | ||||||
|  | * Game Overview: display duration when >1 hour instead of displaying 0 | ||||||
|  |  | ||||||
| ## 1.1.1 / 2023-10-09 20:52+02:00 | ## 1.1.1 / 2023-10-09 20:52+02:00 | ||||||
|  |  | ||||||
| ### New | ### New | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ RUN npm install && \ | |||||||
|  |  | ||||||
| FROM python:3.10.9-slim-bullseye | FROM python:3.10.9-slim-bullseye | ||||||
|  |  | ||||||
| ENV VERSION_NUMBER 1.1.1 | ENV VERSION_NUMBER 1.2.0 | ||||||
| ENV PROD 1 | ENV PROD 1 | ||||||
| ENV PYTHONUNBUFFERED=1 | ENV PYTHONUNBUFFERED=1 | ||||||
|  |  | ||||||
|  | |||||||
| @ -32,6 +32,8 @@ def format_duration( | |||||||
|     from the formatting string. For example: |     from the formatting string. For example: | ||||||
|     - 61 seconds as "%s" = 61 seconds |     - 61 seconds as "%s" = 61 seconds | ||||||
|     - 61 seconds as "%m %s" = 1 minutes 1 seconds" |     - 61 seconds as "%m %s" = 1 minutes 1 seconds" | ||||||
|  |     Format specifiers can include width and precision options: | ||||||
|  |     - %5.2H: hours formatted with width 5 and 2 decimal places (padded with zeros) | ||||||
|     """ |     """ | ||||||
|     minute_seconds = 60 |     minute_seconds = 60 | ||||||
|     hour_seconds = 60 * minute_seconds |     hour_seconds = 60 * minute_seconds | ||||||
| @ -46,18 +48,32 @@ def format_duration( | |||||||
|     remainder = seconds = seconds_total |     remainder = seconds = seconds_total | ||||||
|     if "%d" in format_string: |     if "%d" in format_string: | ||||||
|         days, remainder = divmod(seconds_total, day_seconds) |         days, remainder = divmod(seconds_total, day_seconds) | ||||||
|     if "%H" in format_string: |     if re.search(r"%\d*\.?\d*H", format_string): | ||||||
|         hours, remainder = divmod(remainder, hour_seconds) |         hours_float, remainder = divmod(remainder, hour_seconds) | ||||||
|     if "%m" in format_string: |         hours = float(hours_float) + remainder / hour_seconds | ||||||
|  |     if re.search(r"%\d*\.?\d*m", format_string): | ||||||
|         minutes, seconds = divmod(remainder, minute_seconds) |         minutes, seconds = divmod(remainder, minute_seconds) | ||||||
|     literals = { |     literals = { | ||||||
|         "%d": str(days), |         "d": str(days), | ||||||
|         "%H": str(hours), |         "H": str(hours), | ||||||
|         "%m": str(minutes), |         "m": str(minutes), | ||||||
|         "%s": str(seconds), |         "s": str(seconds), | ||||||
|         "%r": str(seconds_total), |         "r": str(seconds_total), | ||||||
|     } |     } | ||||||
|     formatted_string = format_string |     formatted_string = format_string | ||||||
|     for pattern, replacement in literals.items(): |     for pattern, replacement in literals.items(): | ||||||
|         formatted_string = re.sub(pattern, replacement, formatted_string) |         # Match format specifiers with optional width and precision | ||||||
|  |         match = re.search(rf"%(\d*\.?\d*){pattern}", formatted_string) | ||||||
|  |         if match: | ||||||
|  |             format_spec = match.group(1) | ||||||
|  |             if "." in format_spec: | ||||||
|  |                 # Format the number as float if precision is specified | ||||||
|  |                 replacement = f"{float(replacement):{format_spec}f}" | ||||||
|  |             else: | ||||||
|  |                 # Format the number as integer if no precision is specified | ||||||
|  |                 replacement = f"{int(float(replacement)):>{format_spec}}" | ||||||
|  |             # Replace the format specifier with the formatted number | ||||||
|  |             formatted_string = re.sub( | ||||||
|  |                 rf"%\d*\.?\d*{pattern}", replacement, formatted_string | ||||||
|  |             ) | ||||||
|     return formatted_string |     return formatted_string | ||||||
|  | |||||||
| @ -114,7 +114,7 @@ class Session(models.Model): | |||||||
|         return timedelta(seconds=(manual + calculated).total_seconds()) |         return timedelta(seconds=(manual + calculated).total_seconds()) | ||||||
|  |  | ||||||
|     def duration_formatted(self) -> str: |     def duration_formatted(self) -> str: | ||||||
|         result = format_duration(self.duration_seconds(), "%H:%m") |         result = format_duration(self.duration_seconds(), "%02.0H:%02.0m") | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|  | |||||||
| @ -755,6 +755,10 @@ select { | |||||||
|   position: absolute; |   position: absolute; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .relative { | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  |  | ||||||
| .bottom-2 { | .bottom-2 { | ||||||
|   bottom: 0.5rem; |   bottom: 0.5rem; | ||||||
| } | } | ||||||
| @ -791,14 +795,27 @@ select { | |||||||
|   margin-bottom: 1rem; |   margin-bottom: 1rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .my-6 { | ||||||
|  |   margin-top: 1.5rem; | ||||||
|  |   margin-bottom: 1.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .mb-1 { | .mb-1 { | ||||||
|   margin-bottom: 0.25rem; |   margin-bottom: 0.25rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .mb-10 { | ||||||
|  |   margin-bottom: 2.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .mb-4 { | .mb-4 { | ||||||
|   margin-bottom: 1rem; |   margin-bottom: 1rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .ml-1 { | ||||||
|  |   margin-left: 0.25rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .ml-2 { | .ml-2 { | ||||||
|   margin-left: 0.5rem; |   margin-left: 0.5rem; | ||||||
| } | } | ||||||
| @ -811,18 +828,14 @@ select { | |||||||
|   margin-top: 1rem; |   margin-top: 1rem; | ||||||
| } | } | ||||||
|  |  | ||||||
| .ml-8 { |  | ||||||
|   margin-left: 2rem; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .ml-1 { |  | ||||||
|   margin-left: 0.25rem; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .block { | .block { | ||||||
|   display: block; |   display: block; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .inline-block { | ||||||
|  |   display: inline-block; | ||||||
|  | } | ||||||
|  |  | ||||||
| .inline { | .inline { | ||||||
|   display: inline; |   display: inline; | ||||||
| } | } | ||||||
| @ -839,10 +852,6 @@ select { | |||||||
|   display: none; |   display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .h-6 { |  | ||||||
|   height: 1.5rem; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .h-4 { | .h-4 { | ||||||
|   height: 1rem; |   height: 1rem; | ||||||
| } | } | ||||||
| @ -851,26 +860,34 @@ select { | |||||||
|   height: 1.25rem; |   height: 1.25rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .h-6 { | ||||||
|  |   height: 1.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .min-h-screen { | .min-h-screen { | ||||||
|   min-height: 100vh; |   min-height: 100vh; | ||||||
| } | } | ||||||
|  |  | ||||||
| .w-6 { |  | ||||||
|   width: 1.5rem; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .w-full { |  | ||||||
|   width: 100%; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .w-5 { | .w-5 { | ||||||
|   width: 1.25rem; |   width: 1.25rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .w-6 { | ||||||
|  |   width: 1.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .w-7 { | .w-7 { | ||||||
|   width: 1.75rem; |   width: 1.75rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .w-auto { | ||||||
|  |   width: auto; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .w-full { | ||||||
|  |   width: 100%; | ||||||
|  | } | ||||||
|  |  | ||||||
| .max-w-screen-lg { | .max-w-screen-lg { | ||||||
|   max-width: 1024px; |   max-width: 1024px; | ||||||
| } | } | ||||||
| @ -913,6 +930,10 @@ select { | |||||||
|   justify-content: space-between; |   justify-content: space-between; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .gap-2 { | ||||||
|  |   gap: 0.5rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .self-center { | .self-center { | ||||||
|   align-self: center; |   align-self: center; | ||||||
| } | } | ||||||
| @ -935,6 +956,10 @@ select { | |||||||
|   border-radius: 0.5rem; |   border-radius: 0.5rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .rounded-sm { | ||||||
|  |   border-radius: 0.125rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .border-gray-200 { | .border-gray-200 { | ||||||
|   --tw-border-opacity: 1; |   --tw-border-opacity: 1; | ||||||
|   border-color: rgb(229 231 235 / var(--tw-border-opacity)); |   border-color: rgb(229 231 235 / var(--tw-border-opacity)); | ||||||
| @ -945,26 +970,26 @@ select { | |||||||
|   border-color: rgb(100 116 139 / var(--tw-border-opacity)); |   border-color: rgb(100 116 139 / var(--tw-border-opacity)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .bg-gray-200 { | ||||||
|  |   --tw-bg-opacity: 1; | ||||||
|  |   background-color: rgb(229 231 235 / var(--tw-bg-opacity)); | ||||||
|  | } | ||||||
|  |  | ||||||
| .bg-green-600 { | .bg-green-600 { | ||||||
|   --tw-bg-opacity: 1; |   --tw-bg-opacity: 1; | ||||||
|   background-color: rgb(22 163 74 / var(--tw-bg-opacity)); |   background-color: rgb(22 163 74 / var(--tw-bg-opacity)); | ||||||
| } | } | ||||||
|  |  | ||||||
| .bg-white { |  | ||||||
|   --tw-bg-opacity: 1; |  | ||||||
|   background-color: rgb(255 255 255 / var(--tw-bg-opacity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .bg-blue-600 { |  | ||||||
|   --tw-bg-opacity: 1; |  | ||||||
|   background-color: rgb(37 99 235 / var(--tw-bg-opacity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .bg-violet-600 { | .bg-violet-600 { | ||||||
|   --tw-bg-opacity: 1; |   --tw-bg-opacity: 1; | ||||||
|   background-color: rgb(124 58 237 / var(--tw-bg-opacity)); |   background-color: rgb(124 58 237 / var(--tw-bg-opacity)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .bg-white { | ||||||
|  |   --tw-bg-opacity: 1; | ||||||
|  |   background-color: rgb(255 255 255 / var(--tw-bg-opacity)); | ||||||
|  | } | ||||||
|  |  | ||||||
| .p-4 { | .p-4 { | ||||||
|   padding: 1rem; |   padding: 1rem; | ||||||
| } | } | ||||||
| @ -974,6 +999,11 @@ select { | |||||||
|   padding-right: 0.5rem; |   padding-right: 0.5rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .px-4 { | ||||||
|  |   padding-left: 1rem; | ||||||
|  |   padding-right: 1rem; | ||||||
|  | } | ||||||
|  |  | ||||||
| .py-1 { | .py-1 { | ||||||
|   padding-top: 0.25rem; |   padding-top: 0.25rem; | ||||||
|   padding-bottom: 0.25rem; |   padding-bottom: 0.25rem; | ||||||
| @ -992,8 +1022,8 @@ select { | |||||||
|   padding-right: 1rem; |   padding-right: 1rem; | ||||||
| } | } | ||||||
|  |  | ||||||
| .pl-8 { | .pt-1 { | ||||||
|   padding-left: 2rem; |   padding-top: 0.25rem; | ||||||
| } | } | ||||||
|  |  | ||||||
| .text-center { | .text-center { | ||||||
| @ -1014,6 +1044,11 @@ select { | |||||||
|   line-height: 2.5rem; |   line-height: 2.5rem; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .text-5xl { | ||||||
|  |   font-size: 3rem; | ||||||
|  |   line-height: 1; | ||||||
|  | } | ||||||
|  |  | ||||||
| .text-base { | .text-base { | ||||||
|   font-size: 1rem; |   font-size: 1rem; | ||||||
|   line-height: 1.5rem; |   line-height: 1.5rem; | ||||||
| @ -1042,6 +1077,11 @@ select { | |||||||
|   font-style: italic; |   font-style: italic; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .text-gray-700 { | ||||||
|  |   --tw-text-opacity: 1; | ||||||
|  |   color: rgb(55 65 81 / var(--tw-text-opacity)); | ||||||
|  | } | ||||||
|  |  | ||||||
| .text-slate-300 { | .text-slate-300 { | ||||||
|   --tw-text-opacity: 1; |   --tw-text-opacity: 1; | ||||||
|   color: rgb(203 213 225 / var(--tw-text-opacity)); |   color: rgb(203 213 225 / var(--tw-text-opacity)); | ||||||
| @ -1277,16 +1317,16 @@ th label { | |||||||
|   box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow); |   box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hover\:bg-gray-400:hover { | ||||||
|  |   --tw-bg-opacity: 1; | ||||||
|  |   background-color: rgb(156 163 175 / var(--tw-bg-opacity)); | ||||||
|  | } | ||||||
|  |  | ||||||
| .hover\:bg-green-700:hover { | .hover\:bg-green-700:hover { | ||||||
|   --tw-bg-opacity: 1; |   --tw-bg-opacity: 1; | ||||||
|   background-color: rgb(21 128 61 / var(--tw-bg-opacity)); |   background-color: rgb(21 128 61 / var(--tw-bg-opacity)); | ||||||
| } | } | ||||||
|  |  | ||||||
| .hover\:bg-blue-700:hover { |  | ||||||
|   --tw-bg-opacity: 1; |  | ||||||
|   background-color: rgb(29 78 216 / var(--tw-bg-opacity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .hover\:bg-violet-700:hover { | .hover\:bg-violet-700:hover { | ||||||
|   --tw-bg-opacity: 1; |   --tw-bg-opacity: 1; | ||||||
|   background-color: rgb(109 40 217 / var(--tw-bg-opacity)); |   background-color: rgb(109 40 217 / var(--tw-bg-opacity)); | ||||||
| @ -1312,11 +1352,6 @@ th label { | |||||||
|   --tw-ring-color: rgb(34 197 94 / var(--tw-ring-opacity)); |   --tw-ring-color: rgb(34 197 94 / var(--tw-ring-opacity)); | ||||||
| } | } | ||||||
|  |  | ||||||
| .focus\:ring-blue-500:focus { |  | ||||||
|   --tw-ring-opacity: 1; |  | ||||||
|   --tw-ring-color: rgb(59 130 246 / var(--tw-ring-opacity)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .focus\:ring-violet-500:focus { | .focus\:ring-violet-500:focus { | ||||||
|   --tw-ring-opacity: 1; |   --tw-ring-opacity: 1; | ||||||
|   --tw-ring-color: rgb(139 92 246 / var(--tw-ring-opacity)); |   --tw-ring-color: rgb(139 92 246 / var(--tw-ring-opacity)); | ||||||
| @ -1334,6 +1369,10 @@ th label { | |||||||
|   --tw-ring-offset-color: #ddd6fe; |   --tw-ring-offset-color: #ddd6fe; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .group:hover .group-hover\:block { | ||||||
|  |   display: block; | ||||||
|  | } | ||||||
|  |  | ||||||
| :is(.dark .dark\:bg-gray-800) { | :is(.dark .dark\:bg-gray-800) { | ||||||
|   --tw-bg-opacity: 1; |   --tw-bg-opacity: 1; | ||||||
|   background-color: rgb(31 41 55 / var(--tw-bg-opacity)); |   background-color: rgb(31 41 55 / var(--tw-bg-opacity)); | ||||||
|  | |||||||
| @ -25,19 +25,37 @@ | |||||||
|                     <div class="w-full md:block md:w-auto"> |                     <div class="w-full md:block md:w-auto"> | ||||||
|                         <ul |                         <ul | ||||||
|                             class="flex flex-col md:flex-row p-4 mt-4 dark:text-white"> |                             class="flex flex-col md:flex-row p-4 mt-4 dark:text-white"> | ||||||
|                             <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_game' %}">New Game</a></li> |                             <li class="relative group"> | ||||||
|                             <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_platform' %}">New Platform</a></li> |                                 <a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_game' %}">New</a> | ||||||
|                             {% if game_available and platform_available %} |                                 <ul class="absolute hidden text-gray-700 pt-1 group-hover:block  w-auto whitespace-nowrap"> | ||||||
|                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_edition' %}">New Edition</a></li> |                                     {% if purchase_available %} | ||||||
|                             {% endif %} |                                         <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_device' %}">Device</a></li> | ||||||
|                             {% if edition_available %} |                                     {% endif %} | ||||||
|                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_purchase' %}">New Purchase</a></li> |                                     <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_game' %}">Game</a></li> | ||||||
|                             {% endif %} |                                     {% if game_available and platform_available %} | ||||||
|                             {% if purchase_available %} |                                         <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_edition' %}">Edition</a></li> | ||||||
|                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_session' %}">New Session</a></li> |                                     {% endif %} | ||||||
|                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'add_device' %}">New Device</a></li> |                                     <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_platform' %}">Platform</a></li> | ||||||
|                             {% endif %} |                                     {% if edition_available %} | ||||||
|  |                                         <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_purchase' %}">Purchase</a></li> | ||||||
|  |                                     {% endif %} | ||||||
|  |                                     {% if purchase_available %} | ||||||
|  |                                         <li><a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'add_session' %}">Session</a></li> | ||||||
|  |                                     {% endif %} | ||||||
|  |                                      | ||||||
|  |                                 </ul> | ||||||
|  |                             </li> | ||||||
|                             {% if session_count > 0 %} |                             {% if session_count > 0 %} | ||||||
|  |                                     <li class="relative group"> | ||||||
|  |                                         <a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'stats_current_year' %}">Stats</a> | ||||||
|  |                                         <ul class="absolute hidden text-gray-700 pt-1 group-hover:block"> | ||||||
|  |                                             {% for year in stats_dropdown_year_range %} | ||||||
|  |                                                 <li> | ||||||
|  |                                                     <a class="bg-gray-200 hover:bg-gray-400 py-2 px-4 block whitespace-no-wrap" href="{% url 'stats_by_year' year %}">{{ year }}</a> | ||||||
|  |                                                 </li> | ||||||
|  |                                             {% endfor %} | ||||||
|  |                                         </ul> | ||||||
|  |                                     </li> | ||||||
|                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'list_sessions' %}">All Sessions</a></li> |                                 <li><a class="block py-2 pl-3 pr-4 hover:underline" href="{% url 'list_sessions' %}">All Sessions</a></li> | ||||||
|                             {% endif %} |                             {% endif %} | ||||||
|                         </ul> |                         </ul> | ||||||
|  | |||||||
| @ -1,10 +1,13 @@ | |||||||
| <button | {% comment %}  | ||||||
|   type="button" | title | ||||||
|  | text | ||||||
|  | {% endcomment %} | ||||||
|  | <a | ||||||
|  |   href="{{ link }}" | ||||||
|   title="{{ title }}" |   title="{{ title }}" | ||||||
|   autofocus |   class="truncate max-w-xs py-1 px-2 text-xs bg-green-600 hover:bg-green-700 focus:ring-green-500 focus:ring-offset-blue-200 text-white transition ease-in duration-200 text-center font-semibold shadow-md focus:outline-none focus:ring-2 focus:ring-offset-2 rounded-sm" | ||||||
|   class="truncate max-w-xs sm:max-w-md lg:max-w-lg py-1 px-2 bg-green-600 hover:bg-green-700 focus:ring-green-500 focus:ring-offset-blue-200 text-white transition ease-in duration-200 text-center text-base font-semibold shadow-md focus:outline-none focus:ring-2 focus:ring-offset-2 rounded-lg" |  | ||||||
| > | > | ||||||
|   <svg |   {% comment %} <svg | ||||||
|     xmlns="http://www.w3.org/2000/svg" |     xmlns="http://www.w3.org/2000/svg" | ||||||
|     fill="none" |     fill="none" | ||||||
|     viewBox="0 0 24 24" |     viewBox="0 0 24 24" | ||||||
| @ -16,7 +19,8 @@ | |||||||
|       stroke-linecap="round" |       stroke-linecap="round" | ||||||
|       stroke-linejoin="round" |       stroke-linejoin="round" | ||||||
|       d="M5.25 5.653c0-.856.917-1.398 1.667-.986l11.54 6.348a1.125 1.125 0 010 1.971l-11.54 6.347a1.125 1.125 0 01-1.667-.985V5.653z" |       d="M5.25 5.653c0-.856.917-1.398 1.667-.986l11.54 6.348a1.125 1.125 0 010 1.971l-11.54 6.347a1.125 1.125 0 01-1.667-.985V5.653z" | ||||||
|     /> |       />  | ||||||
|   </svg> |     </svg> | ||||||
|  |     {% endcomment %} | ||||||
|   {{ text }} |   {{ text }} | ||||||
| </button> | </a> | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								games/templates/components/button_start.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								games/templates/components/button_start.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | {% comment %}  | ||||||
|  | title | ||||||
|  | text | ||||||
|  | {% endcomment %} | ||||||
|  | <button | ||||||
|  |   type="button" | ||||||
|  |   title="{{ title }}" | ||||||
|  |   autofocus | ||||||
|  |   class="truncate max-w-xs sm:max-w-md lg:max-w-lg py-1 px-2 bg-green-600 hover:bg-green-700 focus:ring-green-500 focus:ring-offset-blue-200 text-white transition ease-in duration-200 text-center text-base font-semibold shadow-md focus:outline-none focus:ring-2 focus:ring-offset-2 rounded-lg" | ||||||
|  | > | ||||||
|  |   <svg | ||||||
|  |     xmlns="http://www.w3.org/2000/svg" | ||||||
|  |     fill="none" | ||||||
|  |     viewBox="0 0 24 24" | ||||||
|  |     stroke-width="1.5" | ||||||
|  |     stroke="currentColor" | ||||||
|  |     class="self-center w-6 h-6 inline" | ||||||
|  |   > | ||||||
|  |     <path | ||||||
|  |       stroke-linecap="round" | ||||||
|  |       stroke-linejoin="round" | ||||||
|  |       d="M5.25 5.653c0-.856.917-1.398 1.667-.986l11.54 6.348a1.125 1.125 0 010 1.971l-11.54 6.347a1.125 1.125 0 01-1.667-.985V5.653z" | ||||||
|  |     /> | ||||||
|  |   </svg> | ||||||
|  |   {{ text }} | ||||||
|  | </button> | ||||||
| @ -10,8 +10,8 @@ | |||||||
| <div class="mx-auto text-center my-4"> | <div class="mx-auto text-center my-4"> | ||||||
|   <a |   <a | ||||||
|     id="last-session-start" |     id="last-session-start" | ||||||
|     href="{% url 'start_session' last.id %}" |     href="{% url 'start_session_same_as_last' last.id %}" | ||||||
|     hx-get="{% url 'start_session' last.id %}" |     hx-get="{% url 'start_session_same_as_last' last.id %}" | ||||||
|     hx-indicator="#indicator" |     hx-indicator="#indicator" | ||||||
|     hx-swap="afterbegin" |     hx-swap="afterbegin" | ||||||
|     hx-target=".responsive-table tbody" |     hx-target=".responsive-table tbody" | ||||||
| @ -19,7 +19,7 @@ | |||||||
|     onClick="document.querySelector('#last-session-start').classList.add('invisible')" |     onClick="document.querySelector('#last-session-start').classList.add('invisible')" | ||||||
|     class="{% if last.timestamp_end == null %}invisible{% endif %}" |     class="{% if last.timestamp_end == null %}invisible{% endif %}" | ||||||
|   > |   > | ||||||
|     {% include 'components/button.html' with text=last.purchase title="Start session of last played game" only %} |     {% include 'components/button_start.html' with text=last.purchase title="Start session of last played game" only %} | ||||||
|   </a> |   </a> | ||||||
| </div> | </div> | ||||||
| {% endif %} | {% endif %} | ||||||
|  | |||||||
							
								
								
									
										72
									
								
								games/templates/stats.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								games/templates/stats.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | {% extends "base.html" %} | ||||||
|  |  | ||||||
|  | {% block title %}{{ title }}{% endblock title %} | ||||||
|  |  | ||||||
|  | {% load static %} | ||||||
|  |  | ||||||
|  | {% block content %} | ||||||
|  |     <div class="dark:text-white max-w-sm sm:max-w-xl lg:max-w-3xl mx-auto"> | ||||||
|  |         <div class="flex justify-center items-center"> | ||||||
|  |             <form method="get" class="text-center"> | ||||||
|  |                 <label class="text-5xl text-center inline-block mb-10" for="yearSelect">Stats for:</label> | ||||||
|  |                 <select name="year" id="yearSelect" onchange="this.form.submit();" class="mx-2"> | ||||||
|  |                     <option value="2022" {% if year == 2022 %}selected{% endif %}>2022</option> | ||||||
|  |                     <option value="2023" {% if year == 2023 %}selected{% endif %}>2023</option> | ||||||
|  |                 </select> | ||||||
|  |             </form> | ||||||
|  |         </div> | ||||||
|  |         <table class="responsive-table"> | ||||||
|  |             <thead> | ||||||
|  |                 <tr> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Total hours</th> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Total games</th> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Released that year</th> | ||||||
|  |                 </tr> | ||||||
|  |             <tbody> | ||||||
|  |                 <tr> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_hours }}</td> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_games }}</td> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_2023_games }}</td> | ||||||
|  |                 </tr> | ||||||
|  |             </tbody> | ||||||
|  |         </table> | ||||||
|  |         <h1 class="text-5xl text-center my-6">Top games by playtime</h1> | ||||||
|  |         <table class="responsive-table"> | ||||||
|  |             <thead> | ||||||
|  |                 <tr> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Name</th> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Playtime (hours)</th> | ||||||
|  |                 </tr> | ||||||
|  |             </thead> | ||||||
|  |             <tbody> | ||||||
|  |                 {% for game in top_10_games_by_playtime %} | ||||||
|  |                 <tr> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono"> | ||||||
|  |                         <a href="{% url 'view_game' game.id  %}">{{ game.name }} | ||||||
|  |  | ||||||
|  |                         </a> | ||||||
|  |                     </td> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ game.formatted_playtime }}</td> | ||||||
|  |                 </tr> | ||||||
|  |                 {% endfor %} | ||||||
|  |             </tbody> | ||||||
|  |         </table> | ||||||
|  |         <h1 class="text-5xl text-center my-6">Platforms by playtime</h1> | ||||||
|  |         <table class="responsive-table"> | ||||||
|  |             <thead> | ||||||
|  |                 <tr> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Platform</th> | ||||||
|  |                     <th class="px-2 sm:px-4 md:px-6 md:py-2">Playtime (hours)</th> | ||||||
|  |                 </tr> | ||||||
|  |             </thead> | ||||||
|  |             <tbody> | ||||||
|  |                 {% for item in total_playtime_per_platform %} | ||||||
|  |                 <tr> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ item.platform_name }}                    </td> | ||||||
|  |                     <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ item.formatted_playtime }}</td> | ||||||
|  |                 </tr> | ||||||
|  |                 {% endfor %} | ||||||
|  |             </tbody> | ||||||
|  |         </table> | ||||||
|  |     </div> | ||||||
|  | {% endblock content %} | ||||||
| @ -47,7 +47,14 @@ | |||||||
|             </li> |             </li> | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </ul> |         </ul> | ||||||
|         <h1 class="text-3xl mt-4 mb-1">Sessions <span class="dark:text-slate-500">({{ sessions.count }})</span></h1> |         <h1 class="text-3xl mt-4 mb-1 flex gap-2 items-center"> | ||||||
|  |             Sessions | ||||||
|  |             <span class="dark:text-slate-500"> | ||||||
|  |                 ({{ sessions.count }}) | ||||||
|  |             </span> | ||||||
|  |             {% url 'start_game_session' game.id as add_session_link %} | ||||||
|  |             {% include 'components/button.html' with title="Start new session" text="New" link=add_session_link %} | ||||||
|  |         </h1> | ||||||
|         <ul> |         <ul> | ||||||
|             {% for session in sessions %} |             {% for session in sessions %} | ||||||
|             <li class="sm:pl-2 flex items-center"> |             <li class="sm:pl-2 flex items-center"> | ||||||
|  | |||||||
| @ -19,9 +19,14 @@ urlpatterns = [ | |||||||
|         name="update_session", |         name="update_session", | ||||||
|     ), |     ), | ||||||
|     path( |     path( | ||||||
|         "start-session/<int:last_session_id>", |         "start-session-same-as-last/<int:last_session_id>", | ||||||
|         views.start_session, |         views.start_session_same_as_last, | ||||||
|         name="start_session", |         name="start_session_same_as_last", | ||||||
|  |     ), | ||||||
|  |     path( | ||||||
|  |         "start-session/<int:game_id>", | ||||||
|  |         views.start_game_session, | ||||||
|  |         name="start_game_session", | ||||||
|     ), |     ), | ||||||
|     # path( |     # path( | ||||||
|     #     "delete_session/by-id/<int:session_id>", |     #     "delete_session/by-id/<int:session_id>", | ||||||
| @ -68,4 +73,10 @@ urlpatterns = [ | |||||||
|         {"filter": "ownership_type"}, |         {"filter": "ownership_type"}, | ||||||
|         name="list_sessions_by_ownership_type", |         name="list_sessions_by_ownership_type", | ||||||
|     ), |     ), | ||||||
|  |     path("stats/", views.stats, name="stats_current_year"), | ||||||
|  |     path( | ||||||
|  |         "stats/<int:year>", | ||||||
|  |         views.stats, | ||||||
|  |         name="stats_by_year", | ||||||
|  |     ), | ||||||
| ] | ] | ||||||
|  | |||||||
| @ -1,10 +1,12 @@ | |||||||
| from datetime import datetime, timedelta |  | ||||||
| from zoneinfo import ZoneInfo |  | ||||||
|  |  | ||||||
| from common.time import now as now_with_tz |  | ||||||
| from common.time import format_duration | from common.time import format_duration | ||||||
|  | from common.time import now as now_with_tz | ||||||
|  | from datetime import datetime, timedelta | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.db.models import Sum, F | ||||||
|  | from django.http import HttpResponseRedirect | ||||||
| from django.shortcuts import redirect, render | from django.shortcuts import redirect, render | ||||||
|  | from django.urls import reverse | ||||||
|  | from zoneinfo import ZoneInfo | ||||||
|  |  | ||||||
| from .forms import ( | from .forms import ( | ||||||
|     GameForm, |     GameForm, | ||||||
| @ -27,6 +29,10 @@ def model_counts(request): | |||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def stats_dropdown_year_range(request): | ||||||
|  |     return {"stats_dropdown_year_range": range(2022, 2024)} | ||||||
|  |  | ||||||
|  |  | ||||||
| def add_session(request): | def add_session(request): | ||||||
|     context = {} |     context = {} | ||||||
|     initial = {} |     initial = {} | ||||||
| @ -101,8 +107,8 @@ def view_game(request, game_id=None): | |||||||
|     context["sessions"] = Session.objects.filter( |     context["sessions"] = Session.objects.filter( | ||||||
|         purchase__edition__game_id=game_id |         purchase__edition__game_id=game_id | ||||||
|     ).order_by("-timestamp_start") |     ).order_by("-timestamp_start") | ||||||
|     context["total_hours"] = int( |     context["total_hours"] = float( | ||||||
|         format_duration(context["sessions"].total_duration_unformatted(), "%H") |         format_duration(context["sessions"].total_duration_unformatted(), "%2.1H") | ||||||
|     ) |     ) | ||||||
|     context["session_average"] = round( |     context["session_average"] = round( | ||||||
|         (context["total_hours"]) / int(context["sessions"].count()), 1 |         (context["total_hours"]) / int(context["sessions"].count()), 1 | ||||||
| @ -140,7 +146,24 @@ def edit_edition(request, edition_id=None): | |||||||
|     return render(request, "add.html", context) |     return render(request, "add.html", context) | ||||||
|  |  | ||||||
|  |  | ||||||
| def start_session(request, last_session_id: int): | def start_game_session(request, game_id: int): | ||||||
|  |     last_session = ( | ||||||
|  |         Session.objects.filter(purchase__edition__game_id=game_id) | ||||||
|  |         .order_by("-timestamp_start") | ||||||
|  |         .first() | ||||||
|  |     ) | ||||||
|  |     session = SessionForm( | ||||||
|  |         { | ||||||
|  |             "purchase": last_session.purchase.id, | ||||||
|  |             "timestamp_start": now_with_tz(), | ||||||
|  |             "device": last_session.device, | ||||||
|  |         } | ||||||
|  |     ) | ||||||
|  |     session.save() | ||||||
|  |     return redirect("list_sessions") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def start_session_same_as_last(request, last_session_id: int): | ||||||
|     last_session = Session.objects.get(id=last_session_id) |     last_session = Session.objects.get(id=last_session_id) | ||||||
|     session = SessionForm( |     session = SessionForm( | ||||||
|         { |         { | ||||||
| @ -212,6 +235,57 @@ def list_sessions( | |||||||
|     return render(request, "list_sessions.html", context) |     return render(request, "list_sessions.html", context) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def stats(request, year: int = 0): | ||||||
|  |     selected_year = request.GET.get("year") | ||||||
|  |     if selected_year: | ||||||
|  |         return HttpResponseRedirect(reverse("stats_by_year", args=[selected_year])) | ||||||
|  |     if year == 0: | ||||||
|  |         year = now_with_tz().year | ||||||
|  |     first_day_of_year = datetime(year, 1, 1) | ||||||
|  |     last_day_of_year = datetime(year + 1, 1, 1) | ||||||
|  |     year_sessions = Session.objects.filter( | ||||||
|  |         timestamp_start__gte=first_day_of_year | ||||||
|  |     ).filter(timestamp_start__lt=last_day_of_year) | ||||||
|  |     year_purchases = Purchase.objects.filter(session__in=year_sessions).distinct() | ||||||
|  |  | ||||||
|  |     games_with_playtime = ( | ||||||
|  |         Game.objects.filter(edition__purchase__session__in=year_sessions) | ||||||
|  |         .annotate( | ||||||
|  |             total_playtime=Sum( | ||||||
|  |                 F("edition__purchase__session__duration_calculated") | ||||||
|  |                 + F("edition__purchase__session__duration_manual") | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         .values("id", "name", "total_playtime") | ||||||
|  |     ) | ||||||
|  |     top_10_games_by_playtime = games_with_playtime.order_by("-total_playtime")[:10] | ||||||
|  |     for game in top_10_games_by_playtime: | ||||||
|  |         game["formatted_playtime"] = format_duration(game["total_playtime"], "%2.0H") | ||||||
|  |  | ||||||
|  |     total_playtime_per_platform = ( | ||||||
|  |         year_sessions.values("purchase__platform__name") | ||||||
|  |         .annotate(total_playtime=Sum(F("duration_calculated") + F("duration_manual"))) | ||||||
|  |         .annotate(platform_name=F("purchase__platform__name")) | ||||||
|  |         .values("platform_name", "total_playtime") | ||||||
|  |         .order_by("-total_playtime") | ||||||
|  |     ) | ||||||
|  |     for item in total_playtime_per_platform: | ||||||
|  |         item["formatted_playtime"] = format_duration(item["total_playtime"], "%2.0H") | ||||||
|  |  | ||||||
|  |     context = { | ||||||
|  |         "total_hours": format_duration( | ||||||
|  |             year_sessions.total_duration_unformatted(), "%2.0H" | ||||||
|  |         ), | ||||||
|  |         "total_games": year_purchases.count(), | ||||||
|  |         "total_2023_games": year_purchases.filter(edition__year_released=year).count(), | ||||||
|  |         "top_10_games_by_playtime": top_10_games_by_playtime, | ||||||
|  |         "year": year, | ||||||
|  |         "total_playtime_per_platform": total_playtime_per_platform, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return render(request, "stats.html", context) | ||||||
|  |  | ||||||
|  |  | ||||||
| def add_purchase(request): | def add_purchase(request): | ||||||
|     context = {} |     context = {} | ||||||
|     now = datetime.now() |     now = datetime.now() | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| [tool.poetry] | [tool.poetry] | ||||||
| name = "timetracker" | name = "timetracker" | ||||||
| version = "1.1.1" | version = "1.2.0" | ||||||
| description = "A simple time tracker." | description = "A simple time tracker." | ||||||
| authors = ["Lukáš Kucharczyk <lukas@kucharczyk.xyz>"] | authors = ["Lukáš Kucharczyk <lukas@kucharczyk.xyz>"] | ||||||
| license = "GPL" | license = "GPL" | ||||||
|  | |||||||
| @ -6,7 +6,6 @@ from common.time import format_duration | |||||||
|  |  | ||||||
| class FormatDurationTest(unittest.TestCase): | class FormatDurationTest(unittest.TestCase): | ||||||
|     def setUp(self) -> None: |     def setUp(self) -> None: | ||||||
|  |  | ||||||
|         return super().setUp() |         return super().setUp() | ||||||
|  |  | ||||||
|     def test_only_days(self): |     def test_only_days(self): | ||||||
| @ -19,6 +18,21 @@ class FormatDurationTest(unittest.TestCase): | |||||||
|         result = format_duration(delta, "%H hours") |         result = format_duration(delta, "%H hours") | ||||||
|         self.assertEqual(result, "1 hours") |         self.assertEqual(result, "1 hours") | ||||||
|  |  | ||||||
|  |     def test_only_hours_fractional(self): | ||||||
|  |         delta = timedelta(hours=1) | ||||||
|  |         result = format_duration(delta, "%.1H hours") | ||||||
|  |         self.assertEqual(result, "1.0 hours") | ||||||
|  |  | ||||||
|  |     def test_less_than_hour_with_precision(self): | ||||||
|  |         delta = timedelta(hours=0.5) | ||||||
|  |         result = format_duration(delta, "%.1H hours") | ||||||
|  |         self.assertEqual(result, "0.5 hours") | ||||||
|  |  | ||||||
|  |     def test_less_than_hour_without_precision(self): | ||||||
|  |         delta = timedelta(hours=0.5) | ||||||
|  |         result = format_duration(delta, "%H hours") | ||||||
|  |         self.assertEqual(result, "0 hours") | ||||||
|  |  | ||||||
|     def test_overflow_hours(self): |     def test_overflow_hours(self): | ||||||
|         delta = timedelta(hours=25) |         delta = timedelta(hours=25) | ||||||
|         result = format_duration(delta, "%H hours") |         result = format_duration(delta, "%H hours") | ||||||
|  | |||||||
| @ -68,6 +68,7 @@ TEMPLATES = [ | |||||||
|                 "django.contrib.auth.context_processors.auth", |                 "django.contrib.auth.context_processors.auth", | ||||||
|                 "django.contrib.messages.context_processors.messages", |                 "django.contrib.messages.context_processors.messages", | ||||||
|                 "games.views.model_counts", |                 "games.views.model_counts", | ||||||
|  |                 "games.views.stats_dropdown_year_range", | ||||||
|             ], |             ], | ||||||
|         }, |         }, | ||||||
|     }, |     }, | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user