Compare commits
	
		
			20 Commits
		
	
	
		
			add-fronte
			...
			9573c3b8ff
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						
						
							
						
						9573c3b8ff
	
				 | 
					
					
						|||
| 
						
						
							
						
						c4354a1380
	
				 | 
					
					
						|||
| 
						
						
							
						
						a245b6ff0f
	
				 | 
					
					
						|||
| 
						
						
							
						
						6329d380b7
	
				 | 
					
					
						|||
| 
						
						
							
						
						76fbc39fed
	
				 | 
					
					
						|||
| 
						
						
							
						
						4b6734c173
	
				 | 
					
					
						|||
| 
						
						
							
						
						b505b5b430
	
				 | 
					
					
						|||
| 
						
						
							
						
						87553ebdc5
	
				 | 
					
					
						|||
| 
						
						
							
						
						ba4fc0cac5
	
				 | 
					
					
						|||
| 
						
						
							
						
						8cb0276215
	
				 | 
					
					
						|||
| 
						
						
							
						
						f9a51ee83d
	
				 | 
					
					
						|||
| 
						
						
							
						
						c9deba7d65
	
				 | 
					
					
						|||
| 
						
						
							
						
						c55fbe86b5
	
				 | 
					
					
						|||
| 
						
						
							
						
						0e93993498
	
				 | 
					
					
						|||
| 
						
						
							
						
						9fccdfbff0
	
				 | 
					
					
						|||
| 
						
						
							
						
						d78139a5b3
	
				 | 
					
					
						|||
| 
						
						
							
						
						7dc43fbf77
	
				 | 
					
					
						|||
| 
						
						
							
						
						5442926457
	
				 | 
					
					
						|||
| 
						
						
							
						
						db4c635260
	
				 | 
					
					
						|||
| 
						
						
							
						
						4a1d08d4df
	
				 | 
					
					
						
@ -1,27 +0,0 @@
 | 
			
		||||
name: Django CI/CD
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
    branches: [ main ]
 | 
			
		||||
  pull_request:
 | 
			
		||||
    branches: [ main ]
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  build-and-push:
 | 
			
		||||
    needs: test
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: Checkout
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
    - name: Set up Docker Buildx
 | 
			
		||||
      uses: docker/setup-buildx-action@v3
 | 
			
		||||
    - name: Build and push
 | 
			
		||||
      uses: docker/build-push-action@v5
 | 
			
		||||
      with:
 | 
			
		||||
        context: .
 | 
			
		||||
        push: true
 | 
			
		||||
        tags: |
 | 
			
		||||
          registry.kucharczyk.xyz/timetracker:latest
 | 
			
		||||
          registry.kucharczyk.xyz/timetracker:${{ env.VERSION_NUMBER }}          
 | 
			
		||||
    env:
 | 
			
		||||
      VERSION_NUMBER: 1.5.1
 | 
			
		||||
							
								
								
									
										36
									
								
								.github/workflows/build-docker.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								.github/workflows/build-docker.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@ -0,0 +1,36 @@
 | 
			
		||||
name: Django CI/CD
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  push:
 | 
			
		||||
    paths-ignore: [ 'README.md' ]
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4
 | 
			
		||||
      - uses: actions/setup-python@v4
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: 3.12
 | 
			
		||||
      - run: |
 | 
			
		||||
          python -m pip install poetry
 | 
			
		||||
          poetry install
 | 
			
		||||
          poetry env info
 | 
			
		||||
          poetry run python manage.py migrate
 | 
			
		||||
          poetry run pytest
 | 
			
		||||
  build-and-push:
 | 
			
		||||
    needs: test
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    if: github.ref == 'refs/heads/main'
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v4
 | 
			
		||||
      - uses: docker/setup-buildx-action@v3
 | 
			
		||||
      - uses: docker/build-push-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          context: .
 | 
			
		||||
          push: true
 | 
			
		||||
          tags: |
 | 
			
		||||
            registry.kucharczyk.xyz/timetracker:latest
 | 
			
		||||
            registry.kucharczyk.xyz/timetracker:${{ env.VERSION_NUMBER }}          
 | 
			
		||||
    env:
 | 
			
		||||
      VERSION_NUMBER: 1.5.1
 | 
			
		||||
@ -8,3 +8,8 @@ repos:
 | 
			
		||||
  hooks:
 | 
			
		||||
    - id: isort
 | 
			
		||||
      name: isort (python)
 | 
			
		||||
- repo: https://github.com/Riverside-Healthcare/djLint
 | 
			
		||||
  rev: v1.34.0
 | 
			
		||||
  hooks:
 | 
			
		||||
    - id: djlint-reformat-django
 | 
			
		||||
    - id: djlint-django
 | 
			
		||||
 | 
			
		||||
@ -23,6 +23,12 @@
 | 
			
		||||
  font-style: normal;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
a:hover {
 | 
			
		||||
  text-decoration-color: #ff4400;
 | 
			
		||||
  color: rgb(254, 185, 160);
 | 
			
		||||
  transition: all 0.2s ease-out;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
form label {
 | 
			
		||||
  @apply dark:text-slate-400;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										17
									
								
								games/migrations/0033_alter_edition_unique_together.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								games/migrations/0033_alter_edition_unique_together.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,17 @@
 | 
			
		||||
# Generated by Django 4.2.7 on 2023-11-28 13:43
 | 
			
		||||
 | 
			
		||||
from django.db import migrations
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class Migration(migrations.Migration):
 | 
			
		||||
 | 
			
		||||
    dependencies = [
 | 
			
		||||
        ("games", "0032_alter_session_options_session_modified_at_and_more"),
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    operations = [
 | 
			
		||||
        migrations.AlterUniqueTogether(
 | 
			
		||||
            name="edition",
 | 
			
		||||
            unique_together={("name", "platform", "year_released")},
 | 
			
		||||
        ),
 | 
			
		||||
    ]
 | 
			
		||||
@ -34,7 +34,7 @@ class Game(models.Model):
 | 
			
		||||
 | 
			
		||||
class Edition(models.Model):
 | 
			
		||||
    class Meta:
 | 
			
		||||
        unique_together = [["name", "platform"]]
 | 
			
		||||
        unique_together = [["name", "platform", "year_released"]]
 | 
			
		||||
 | 
			
		||||
    game = models.ForeignKey("Game", on_delete=models.CASCADE)
 | 
			
		||||
    name = models.CharField(max_length=255)
 | 
			
		||||
 | 
			
		||||
@ -1173,6 +1173,12 @@ select {
 | 
			
		||||
  font-style: normal;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
a:hover {
 | 
			
		||||
  text-decoration-color: #ff4400;
 | 
			
		||||
  color: rgb(254, 185, 160);
 | 
			
		||||
  transition: all 0.2s ease-out;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
:is(.dark form label) {
 | 
			
		||||
  --tw-text-opacity: 1;
 | 
			
		||||
  color: rgb(148 163 184 / var(--tw-text-opacity));
 | 
			
		||||
@ -1477,10 +1483,6 @@ th label {
 | 
			
		||||
    display: block;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .md\:w-1\/2 {
 | 
			
		||||
    width: 50%;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .md\:w-auto {
 | 
			
		||||
    width: auto;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@ -1,24 +1,24 @@
 | 
			
		||||
import { syncSelectInputUntilChanged } from './utils.js';
 | 
			
		||||
import { syncSelectInputUntilChanged } from "./utils.js";
 | 
			
		||||
 | 
			
		||||
let syncData = [
 | 
			
		||||
  {
 | 
			
		||||
    "source": "#id_game",
 | 
			
		||||
    "source_value": "dataset.name",
 | 
			
		||||
    "target": "#id_name",
 | 
			
		||||
    "target_value": "value"
 | 
			
		||||
    source: "#id_game",
 | 
			
		||||
    source_value: "dataset.name",
 | 
			
		||||
    target: "#id_name",
 | 
			
		||||
    target_value: "value",
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    "source": "#id_game",
 | 
			
		||||
    "source_value": "textContent",
 | 
			
		||||
    "target": "#id_sort_name",
 | 
			
		||||
    "target_value": "value"
 | 
			
		||||
    source: "#id_game",
 | 
			
		||||
    source_value: "textContent",
 | 
			
		||||
    target: "#id_sort_name",
 | 
			
		||||
    target_value: "value",
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    "source": "#id_game",
 | 
			
		||||
    "source_value": "dataset.year",
 | 
			
		||||
    "target": "#id_year_released",
 | 
			
		||||
    "target_value": "value"
 | 
			
		||||
  },  
 | 
			
		||||
]
 | 
			
		||||
    source: "#id_game",
 | 
			
		||||
    source_value: "dataset.year",
 | 
			
		||||
    target: "#id_year_released",
 | 
			
		||||
    target_value: "value",
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
syncSelectInputUntilChanged(syncData, "form");
 | 
			
		||||
 | 
			
		||||
@ -1,12 +1,12 @@
 | 
			
		||||
import { syncSelectInputUntilChanged } from './utils.js'
 | 
			
		||||
import { syncSelectInputUntilChanged } from "./utils.js";
 | 
			
		||||
 | 
			
		||||
let syncData = [
 | 
			
		||||
  {
 | 
			
		||||
    "source": "#id_name",
 | 
			
		||||
    "source_value": "value",
 | 
			
		||||
    "target": "#id_sort_name",
 | 
			
		||||
    "target_value": "value"
 | 
			
		||||
  }
 | 
			
		||||
]
 | 
			
		||||
    source: "#id_name",
 | 
			
		||||
    source_value: "value",
 | 
			
		||||
    target: "#id_sort_name",
 | 
			
		||||
    target_value: "value",
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
syncSelectInputUntilChanged(syncData, "form")
 | 
			
		||||
syncSelectInputUntilChanged(syncData, "form");
 | 
			
		||||
 | 
			
		||||
@ -2,7 +2,7 @@ import {
 | 
			
		||||
  syncSelectInputUntilChanged,
 | 
			
		||||
  getEl,
 | 
			
		||||
  disableElementsWhenTrue,
 | 
			
		||||
  disableElementsWhenFalse,
 | 
			
		||||
  disableElementsWhenValueNotEqual,
 | 
			
		||||
} from "./utils.js";
 | 
			
		||||
 | 
			
		||||
let syncData = [
 | 
			
		||||
@ -21,7 +21,11 @@ function setupElementHandlers() {
 | 
			
		||||
    "#id_name",
 | 
			
		||||
    "#id_related_purchase",
 | 
			
		||||
  ]);
 | 
			
		||||
  disableElementsWhenFalse("#id_type", "game", ["#id_date_finished"]);
 | 
			
		||||
  disableElementsWhenValueNotEqual(
 | 
			
		||||
    "#id_type",
 | 
			
		||||
    ["game", "dlc"],
 | 
			
		||||
    ["#id_date_finished"]
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
document.addEventListener("DOMContentLoaded", setupElementHandlers);
 | 
			
		||||
@ -30,14 +34,14 @@ getEl("#id_type").onchange = () => {
 | 
			
		||||
  setupElementHandlers();
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
document.body.addEventListener('htmx:beforeRequest', function(event) {
 | 
			
		||||
document.body.addEventListener("htmx:beforeRequest", function (event) {
 | 
			
		||||
  // Assuming 'Purchase1' is the element that triggers the HTMX request
 | 
			
		||||
  if (event.target.id === 'id_edition') {
 | 
			
		||||
      var idEditionValue = document.getElementById('id_edition').value;
 | 
			
		||||
  if (event.target.id === "id_edition") {
 | 
			
		||||
    var idEditionValue = document.getElementById("id_edition").value;
 | 
			
		||||
 | 
			
		||||
      // Condition to check - replace this with your actual logic
 | 
			
		||||
      if (idEditionValue != '') {
 | 
			
		||||
          event.preventDefault(); // This cancels the HTMX request
 | 
			
		||||
      }
 | 
			
		||||
    // Condition to check - replace this with your actual logic
 | 
			
		||||
    if (idEditionValue != "") {
 | 
			
		||||
      event.preventDefault(); // This cancels the HTMX request
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
@ -7,10 +7,14 @@ for (let button of document.querySelectorAll("[data-target]")) {
 | 
			
		||||
  button.addEventListener("click", (event) => {
 | 
			
		||||
    event.preventDefault();
 | 
			
		||||
    if (type == "now") {
 | 
			
		||||
      targetElement.value = toISOUTCString(new Date);
 | 
			
		||||
      targetElement.value = toISOUTCString(new Date());
 | 
			
		||||
    } else if (type == "copy") {
 | 
			
		||||
      const oppositeName = targetElement.name == "timestamp_start" ? "timestamp_end" : "timestamp_start";
 | 
			
		||||
      document.querySelector(`[name='${oppositeName}']`).value = targetElement.value;
 | 
			
		||||
      const oppositeName =
 | 
			
		||||
        targetElement.name == "timestamp_start"
 | 
			
		||||
          ? "timestamp_end"
 | 
			
		||||
          : "timestamp_start";
 | 
			
		||||
      document.querySelector(`[name='${oppositeName}']`).value =
 | 
			
		||||
        targetElement.value;
 | 
			
		||||
    } else if (type == "toggle") {
 | 
			
		||||
      if (targetElement.type == "datetime-local") targetElement.type = "text";
 | 
			
		||||
      else targetElement.type = "datetime-local";
 | 
			
		||||
 | 
			
		||||
@ -75,7 +75,14 @@ function syncSelectInputUntilChanged(syncData, parentSelector = document) {
 | 
			
		||||
 * @param {string} property - The property to retrieve the value from.
 | 
			
		||||
 */
 | 
			
		||||
function getValueFromProperty(sourceElement, property) {
 | 
			
		||||
  let source = (sourceElement instanceof HTMLSelectElement) ? sourceElement.selectedOptions[0] : sourceElement
 | 
			
		||||
  let source =
 | 
			
		||||
    sourceElement instanceof HTMLSelectElement
 | 
			
		||||
      ? sourceElement.selectedOptions[0]
 | 
			
		||||
      : sourceElement;
 | 
			
		||||
  let source =
 | 
			
		||||
    sourceElement instanceof HTMLSelectElement
 | 
			
		||||
      ? sourceElement.selectedOptions[0]
 | 
			
		||||
      : sourceElement;
 | 
			
		||||
  if (property.startsWith("dataset.")) {
 | 
			
		||||
    let datasetKey = property.slice(8); // Remove 'dataset.' part
 | 
			
		||||
    return source.dataset[datasetKey];
 | 
			
		||||
@ -93,13 +100,11 @@ function getValueFromProperty(sourceElement, property) {
 | 
			
		||||
 */
 | 
			
		||||
function getEl(selector) {
 | 
			
		||||
  if (selector.startsWith("#")) {
 | 
			
		||||
    return document.getElementById(selector.slice(1))
 | 
			
		||||
  }
 | 
			
		||||
  else if (selector.startsWith(".")) {
 | 
			
		||||
    return document.getElementsByClassName(selector)
 | 
			
		||||
  }
 | 
			
		||||
  else {
 | 
			
		||||
    return document.getElementsByTagName(selector)
 | 
			
		||||
    return document.getElementById(selector.slice(1));
 | 
			
		||||
  } else if (selector.startsWith(".")) {
 | 
			
		||||
    return document.getElementsByClassName(selector);
 | 
			
		||||
  } else {
 | 
			
		||||
    return document.getElementsByTagName(selector);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -116,7 +121,7 @@ function getEl(selector) {
 | 
			
		||||
function conditionalElementHandler(...configs) {
 | 
			
		||||
  configs.forEach(([condition, targetElements, callbackfn1, callbackfn2]) => {
 | 
			
		||||
    if (condition()) {
 | 
			
		||||
      targetElements.forEach(elementName => {
 | 
			
		||||
      targetElements.forEach((elementName) => {
 | 
			
		||||
        let el = getEl(elementName);
 | 
			
		||||
        if (el === null) {
 | 
			
		||||
          console.error(`Element ${elementName} doesn't exist.`);
 | 
			
		||||
@ -125,7 +130,7 @@ function conditionalElementHandler(...configs) {
 | 
			
		||||
        }
 | 
			
		||||
      });
 | 
			
		||||
    } else {
 | 
			
		||||
      targetElements.forEach(elementName => {
 | 
			
		||||
      targetElements.forEach((elementName) => {
 | 
			
		||||
        let el = getEl(elementName);
 | 
			
		||||
        if (el === null) {
 | 
			
		||||
          console.error(`Element ${elementName} doesn't exist.`);
 | 
			
		||||
@ -137,16 +142,44 @@ function conditionalElementHandler(...configs) {
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function disableElementsWhenFalse(targetSelect, targetValue, elementList) {
 | 
			
		||||
function disableElementsWhenValueNotEqual(
 | 
			
		||||
  targetSelect,
 | 
			
		||||
  targetValue,
 | 
			
		||||
  elementList
 | 
			
		||||
) {
 | 
			
		||||
  return conditionalElementHandler([
 | 
			
		||||
    () => {
 | 
			
		||||
      return getEl(targetSelect).value != targetValue;
 | 
			
		||||
      let target = getEl(targetSelect);
 | 
			
		||||
      console.debug(
 | 
			
		||||
        `${disableElementsWhenTrue.name}: triggered on ${target.id}`
 | 
			
		||||
      );
 | 
			
		||||
      console.debug(`
 | 
			
		||||
      ${disableElementsWhenTrue.name}: matching against value(s): ${targetValue}`);
 | 
			
		||||
      if (targetValue instanceof Array) {
 | 
			
		||||
        if (targetValue.every((value) => target.value != value)) {
 | 
			
		||||
          console.debug(
 | 
			
		||||
            `${disableElementsWhenTrue.name}: none of the values is equal to ${target.value}, returning true.`
 | 
			
		||||
          );
 | 
			
		||||
          return true;
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        console.debug(
 | 
			
		||||
          `${disableElementsWhenTrue.name}: none of the values is equal to ${target.value}, returning true.`
 | 
			
		||||
        );
 | 
			
		||||
        return target.value != targetValue;
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
    elementList,
 | 
			
		||||
    (el) => {
 | 
			
		||||
      console.debug(
 | 
			
		||||
        `${disableElementsWhenTrue.name}: evaluated true, disabling ${el.id}.`
 | 
			
		||||
      );
 | 
			
		||||
      el.disabled = "disabled";
 | 
			
		||||
    },
 | 
			
		||||
    (el) => {
 | 
			
		||||
      console.debug(
 | 
			
		||||
        `${disableElementsWhenTrue.name}: evaluated false, NOT disabling ${el.id}.`
 | 
			
		||||
      );
 | 
			
		||||
      el.disabled = "";
 | 
			
		||||
    },
 | 
			
		||||
  ]);
 | 
			
		||||
@ -167,4 +200,12 @@ function disableElementsWhenTrue(targetSelect, targetValue, elementList) {
 | 
			
		||||
  ]);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export { toISOUTCString, syncSelectInputUntilChanged, getEl, conditionalElementHandler, disableElementsWhenFalse, disableElementsWhenTrue, getValueFromProperty };
 | 
			
		||||
export {
 | 
			
		||||
  toISOUTCString,
 | 
			
		||||
  syncSelectInputUntilChanged,
 | 
			
		||||
  getEl,
 | 
			
		||||
  conditionalElementHandler,
 | 
			
		||||
  disableElementsWhenValueNotEqual,
 | 
			
		||||
  disableElementsWhenTrue,
 | 
			
		||||
  getValueFromProperty,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@ -16,7 +16,7 @@
 | 
			
		||||
                    {% endif %}
 | 
			
		||||
                    {% if field.name == "timestamp_start" or field.name == "timestamp_end" %}
 | 
			
		||||
                        <td>
 | 
			
		||||
                            <div class="basic-button-container">
 | 
			
		||||
                            <div class="basic-button-container" hx-boost="false">
 | 
			
		||||
                                <button class="basic-button" data-target="{{ field.name }}" data-type="now">Set to now</button>
 | 
			
		||||
                                <button class="basic-button"
 | 
			
		||||
                                        data-target="{{ field.name }}"
 | 
			
		||||
 | 
			
		||||
@ -14,16 +14,23 @@
 | 
			
		||||
        <script src="{% static 'js/htmx.min.js' %}"></script>
 | 
			
		||||
        <link rel="stylesheet" href="{% static 'base.css' %}" />
 | 
			
		||||
    </head>
 | 
			
		||||
    <body class="dark" hx-indicator="#indicator" hx-boost="true">
 | 
			
		||||
    <body class="dark" hx-indicator="#indicator">
 | 
			
		||||
        <img id="indicator"
 | 
			
		||||
             src="{% static 'icons/loading.png' %}"
 | 
			
		||||
             class="absolute right-3 top-3 animate-spin htmx-indicator" />
 | 
			
		||||
             class="absolute right-3 top-3 animate-spin htmx-indicator"
 | 
			
		||||
             height="24"
 | 
			
		||||
             width="24"
 | 
			
		||||
             alt="loading indicator" />
 | 
			
		||||
        <div class="dark:bg-gray-800 min-h-screen">
 | 
			
		||||
            <nav class="mb-4 bg-white dark:bg-gray-900 border-gray-200 rounded">
 | 
			
		||||
                <div class="container flex flex-wrap items-center justify-between mx-auto">
 | 
			
		||||
                    <a href="{% url 'list_sessions_recent' %}" class="flex items-center">
 | 
			
		||||
                        <span class="text-4xl">
 | 
			
		||||
                            <img src="{% static 'icons/schedule.png' %}" width="48" class="mr-4" />
 | 
			
		||||
                            <img src="{% static 'icons/schedule.png' %}"
 | 
			
		||||
                                 height="48"
 | 
			
		||||
                                 width="48"
 | 
			
		||||
                                 alt="Timetracker Logo"
 | 
			
		||||
                                 class="mr-4" />
 | 
			
		||||
                        </span>
 | 
			
		||||
                        <span class="self-center text-xl font-semibold whitespace-nowrap text-white">Timetracker</span>
 | 
			
		||||
                    </a>
 | 
			
		||||
 | 
			
		||||
@ -18,74 +18,82 @@
 | 
			
		||||
                </select>
 | 
			
		||||
            </form>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="flex flex-column flex-wrap justify-center">
 | 
			
		||||
            <div class="md:w-1/2">
 | 
			
		||||
                <h1 class="text-5xl text-center my-6">Playtime</h1>
 | 
			
		||||
                <table class="responsive-table">
 | 
			
		||||
                    <tbody>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Hours</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_hours }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Sessions</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_sessions }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Days</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ unique_days }} ({{ unique_days_percent }}%)</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Games</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_games }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Games ({{ year }})</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_2023_games }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Finished</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ all_finished_this_year.count }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Finished ({{ year }})</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ this_year_finished_this_year.count }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                    </tbody>
 | 
			
		||||
                </table>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="md:w-1/2">
 | 
			
		||||
                <h1 class="text-5xl text-center my-6">Purchases</h1>
 | 
			
		||||
                <table class="responsive-table">
 | 
			
		||||
                    <tbody>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Total</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ all_purchased_this_year.count }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Refunded</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                                {{ all_purchased_refunded_this_year.count }} ({{ refunded_percent }}%)
 | 
			
		||||
                            </td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Unfinished</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                                {{ purchased_unfinished.count }} ({{ unfinished_purchases_percent }}%)
 | 
			
		||||
                            </td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Backlog Decrease</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ backlog_decrease_count }}</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                        <tr>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2">Spendings ({{ total_spent_currency }})</td>
 | 
			
		||||
                            <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_spent }} ({{ spent_per_game }}/game)</td>
 | 
			
		||||
                        </tr>
 | 
			
		||||
                    </tbody>
 | 
			
		||||
                </table>
 | 
			
		||||
            </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <h1 class="text-5xl text-center my-6">Playtime</h1>
 | 
			
		||||
        <table class="responsive-table">
 | 
			
		||||
            <tbody>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Hours</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_hours }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Sessions</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_sessions }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Days</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ unique_days }} ({{ unique_days_percent }}%)</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Games</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_games }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Games ({{ year }})</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_2023_games }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Finished</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ all_finished_this_year.count }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Finished ({{ year }})</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ this_year_finished_this_year.count }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Longest session</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ longest_session_time }} ({{ longest_session_game }})</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Most sessions</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ highest_session_count }} ({{ highest_session_count_game }})</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Highest session average</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                        {{ highest_session_average }} ({{ highest_session_average_game }})
 | 
			
		||||
                    </td>
 | 
			
		||||
                </tr>
 | 
			
		||||
            </tbody>
 | 
			
		||||
        </table>
 | 
			
		||||
        <h1 class="text-5xl text-center my-6">Purchases</h1>
 | 
			
		||||
        <table class="responsive-table">
 | 
			
		||||
            <tbody>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Total</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ all_purchased_this_year.count }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Refunded</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                        {{ all_purchased_refunded_this_year.count }} ({{ refunded_percent }}%)
 | 
			
		||||
                    </td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Unfinished</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                        {{ purchased_unfinished.count }} ({{ unfinished_purchases_percent }}%)
 | 
			
		||||
                    </td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Backlog Decrease</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ backlog_decrease_count }}</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
                <tr>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2">Spendings ({{ total_spent_currency }})</td>
 | 
			
		||||
                    <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ total_spent }} ({{ spent_per_game }}/game)</td>
 | 
			
		||||
                </tr>
 | 
			
		||||
            </tbody>
 | 
			
		||||
        </table>
 | 
			
		||||
        <h1 class="text-5xl text-center my-6">Top games by playtime</h1>
 | 
			
		||||
        <table class="responsive-table">
 | 
			
		||||
            <thead>
 | 
			
		||||
@ -136,7 +144,13 @@
 | 
			
		||||
                    <tr>
 | 
			
		||||
                        <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">
 | 
			
		||||
                            <a class="underline decoration-slate-500 sm:decoration-2"
 | 
			
		||||
                               href="{% url 'edit_purchase' purchase.id %}">{{ purchase.edition.name }}</a>
 | 
			
		||||
                               href="{% url 'edit_purchase' purchase.id %}">
 | 
			
		||||
                                {% if purchase.type == 'dlc' %}
 | 
			
		||||
                                    {{ purchase.name }} ({{ purchase.edition.name }} DLC)
 | 
			
		||||
                                {% else %}
 | 
			
		||||
                                    {{ purchase.edition.name }}
 | 
			
		||||
                                {% endif %}
 | 
			
		||||
                            </a>
 | 
			
		||||
                        </td>
 | 
			
		||||
                        <td class="px-2 sm:px-4 md:px-6 md:py-2 font-mono">{{ purchase.date_finished | date:"d/m/Y" }}</td>
 | 
			
		||||
                    </tr>
 | 
			
		||||
 | 
			
		||||
@ -61,7 +61,6 @@
 | 
			
		||||
            {% 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 %}
 | 
			
		||||
            and Notes <span class="dark:text-slate-500">({{ sessions_with_notes_count }})</span>
 | 
			
		||||
            
 | 
			
		||||
        </h1>
 | 
			
		||||
        <ul>
 | 
			
		||||
            {% for session in sessions %}
 | 
			
		||||
 | 
			
		||||
@ -2,8 +2,8 @@ from datetime import datetime, timedelta
 | 
			
		||||
from typing import Any, Callable
 | 
			
		||||
 | 
			
		||||
from django.core.exceptions import ObjectDoesNotExist
 | 
			
		||||
from django.db.models import Count, F, Prefetch, Sum
 | 
			
		||||
from django.db.models.functions import TruncDate
 | 
			
		||||
from django.db.models import Avg, Count, ExpressionWrapper, F, Prefetch, Q, Sum, fields
 | 
			
		||||
from django.db.models.functions import Extract, TruncDate
 | 
			
		||||
from django.http import (
 | 
			
		||||
    HttpRequest,
 | 
			
		||||
    HttpResponse,
 | 
			
		||||
@ -30,11 +30,11 @@ from .models import Edition, Game, Platform, Purchase, Session
 | 
			
		||||
 | 
			
		||||
def model_counts(request):
 | 
			
		||||
    return {
 | 
			
		||||
        "game_available": Game.objects.count() != 0,
 | 
			
		||||
        "edition_available": Edition.objects.count() != 0,
 | 
			
		||||
        "platform_available": Platform.objects.count() != 0,
 | 
			
		||||
        "purchase_available": Purchase.objects.count() != 0,
 | 
			
		||||
        "session_count": Session.objects.count(),
 | 
			
		||||
        "game_available": Game.objects.exists(),
 | 
			
		||||
        "edition_available": Edition.objects.exists(),
 | 
			
		||||
        "platform_available": Platform.objects.exists(),
 | 
			
		||||
        "purchase_available": Purchase.objects.exists(),
 | 
			
		||||
        "session_count": Session.objects.exists(),
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -321,6 +321,22 @@ def stats(request, year: int = 0):
 | 
			
		||||
    if year == 0:
 | 
			
		||||
        year = timezone.now().year
 | 
			
		||||
    this_year_sessions = Session.objects.filter(timestamp_start__year=year)
 | 
			
		||||
    this_year_sessions_with_durations = this_year_sessions.annotate(
 | 
			
		||||
        duration=ExpressionWrapper(
 | 
			
		||||
            F("timestamp_end") - F("timestamp_start"),
 | 
			
		||||
            output_field=fields.DurationField(),
 | 
			
		||||
        )
 | 
			
		||||
    )
 | 
			
		||||
    longest_session = this_year_sessions_with_durations.order_by("-duration").first()
 | 
			
		||||
    this_year_games_with_session_counts = Game.objects.annotate(
 | 
			
		||||
        session_count=Count(
 | 
			
		||||
            "edition__purchase__session",
 | 
			
		||||
            filter=Q(edition__purchase__session__timestamp_start__year=year),
 | 
			
		||||
        )
 | 
			
		||||
    )
 | 
			
		||||
    game_highest_session_count = this_year_games_with_session_counts.order_by(
 | 
			
		||||
        "-session_count"
 | 
			
		||||
    ).first()
 | 
			
		||||
    selected_currency = "CZK"
 | 
			
		||||
    unique_days = (
 | 
			
		||||
        this_year_sessions.annotate(date=TruncDate("timestamp_start"))
 | 
			
		||||
@ -344,8 +360,8 @@ def stats(request, year: int = 0):
 | 
			
		||||
    this_year_purchases_unfinished = this_year_purchases_without_refunded.filter(
 | 
			
		||||
        date_finished__isnull=True
 | 
			
		||||
    ).filter(
 | 
			
		||||
        type=Purchase.GAME
 | 
			
		||||
    )  # do not count DLC etc.
 | 
			
		||||
        Q(type=Purchase.GAME) | Q(type=Purchase.DLC)
 | 
			
		||||
    )  # do not count battle passes etc.
 | 
			
		||||
 | 
			
		||||
    this_year_purchases_unfinished_percent = int(
 | 
			
		||||
        safe_division(
 | 
			
		||||
@ -381,6 +397,14 @@ def stats(request, year: int = 0):
 | 
			
		||||
        )
 | 
			
		||||
        .values("id", "name", "total_playtime")
 | 
			
		||||
    )
 | 
			
		||||
    highest_session_average_game = (
 | 
			
		||||
        Game.objects.filter(edition__purchase__session__in=this_year_sessions)
 | 
			
		||||
        .annotate(
 | 
			
		||||
            session_average=Avg("edition__purchase__session__duration_calculated")
 | 
			
		||||
        )
 | 
			
		||||
        .order_by("-session_average")
 | 
			
		||||
        .first()
 | 
			
		||||
    )
 | 
			
		||||
    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")
 | 
			
		||||
@ -444,6 +468,17 @@ def stats(request, year: int = 0):
 | 
			
		||||
            "date_purchased"
 | 
			
		||||
        ),
 | 
			
		||||
        "backlog_decrease_count": backlog_decrease_count,
 | 
			
		||||
        "longest_session_time": format_duration(
 | 
			
		||||
            longest_session.duration if longest_session else timedelta(0),
 | 
			
		||||
            "%2.0Hh %2.0mm",
 | 
			
		||||
        ),
 | 
			
		||||
        "longest_session_game": longest_session.purchase.edition.name,
 | 
			
		||||
        "highest_session_count": game_highest_session_count.session_count,
 | 
			
		||||
        "highest_session_count_game": game_highest_session_count.name,
 | 
			
		||||
        "highest_session_average": format_duration(
 | 
			
		||||
            highest_session_average_game.session_average, "%2.0Hh %2.0mm"
 | 
			
		||||
        ),
 | 
			
		||||
        "highest_session_average_game": highest_session_average_game,
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    request.session["return_path"] = request.path
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										47
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										47
									
								
								poetry.lock
									
									
									
										generated
									
									
									
								
							@ -1,4 +1,4 @@
 | 
			
		||||
# This file is automatically @generated by Poetry 1.7.0 and should not be changed by hand.
 | 
			
		||||
# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand.
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "asgiref"
 | 
			
		||||
@ -129,6 +129,21 @@ tzdata = {version = "*", markers = "sys_platform == \"win32\""}
 | 
			
		||||
argon2 = ["argon2-cffi (>=19.1.0)"]
 | 
			
		||||
bcrypt = ["bcrypt"]
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "django-debug-toolbar"
 | 
			
		||||
version = "4.2.0"
 | 
			
		||||
description = "A configurable set of panels that display various debug information about the current request/response."
 | 
			
		||||
optional = false
 | 
			
		||||
python-versions = ">=3.8"
 | 
			
		||||
files = [
 | 
			
		||||
    {file = "django_debug_toolbar-4.2.0-py3-none-any.whl", hash = "sha256:af99128c06e8e794479e65ab62cc6c7d1e74e1c19beb44dcbf9bad7a9c017327"},
 | 
			
		||||
    {file = "django_debug_toolbar-4.2.0.tar.gz", hash = "sha256:bc7fdaafafcdedefcc67a4a5ad9dac96efd6e41db15bc74d402a54a2ba4854dc"},
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[package.dependencies]
 | 
			
		||||
django = ">=3.2.4"
 | 
			
		||||
sqlparse = ">=0.2"
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "django-extensions"
 | 
			
		||||
version = "3.2.3"
 | 
			
		||||
@ -262,13 +277,13 @@ files = [
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "identify"
 | 
			
		||||
version = "2.5.31"
 | 
			
		||||
version = "2.5.32"
 | 
			
		||||
description = "File identification library for Python"
 | 
			
		||||
optional = false
 | 
			
		||||
python-versions = ">=3.8"
 | 
			
		||||
files = [
 | 
			
		||||
    {file = "identify-2.5.31-py2.py3-none-any.whl", hash = "sha256:90199cb9e7bd3c5407a9b7e81b4abec4bb9d249991c79439ec8af740afc6293d"},
 | 
			
		||||
    {file = "identify-2.5.31.tar.gz", hash = "sha256:7736b3c7a28233637e3c36550646fc6389bedd74ae84cb788200cc8e2dd60b75"},
 | 
			
		||||
    {file = "identify-2.5.32-py2.py3-none-any.whl", hash = "sha256:0b7656ef6cba81664b783352c73f8c24b39cf82f926f78f4550eda928e5e0545"},
 | 
			
		||||
    {file = "identify-2.5.32.tar.gz", hash = "sha256:5d9979348ec1a21c768ae07e0a652924538e8bce67313a73cb0f681cf08ba407"},
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[package.extras]
 | 
			
		||||
@ -497,13 +512,13 @@ files = [
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "platformdirs"
 | 
			
		||||
version = "3.11.0"
 | 
			
		||||
version = "4.0.0"
 | 
			
		||||
description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."
 | 
			
		||||
optional = false
 | 
			
		||||
python-versions = ">=3.7"
 | 
			
		||||
files = [
 | 
			
		||||
    {file = "platformdirs-3.11.0-py3-none-any.whl", hash = "sha256:e9d171d00af68be50e9202731309c4e658fd8bc76f55c11c7dd760d023bda68e"},
 | 
			
		||||
    {file = "platformdirs-3.11.0.tar.gz", hash = "sha256:cf8ee52a3afdb965072dcc652433e0c7e3e40cf5ea1477cd4b3b1d2eb75495b3"},
 | 
			
		||||
    {file = "platformdirs-4.0.0-py3-none-any.whl", hash = "sha256:118c954d7e949b35437270383a3f2531e99dd93cf7ce4dc8340d3356d30f173b"},
 | 
			
		||||
    {file = "platformdirs-4.0.0.tar.gz", hash = "sha256:cb633b2bcf10c51af60beb0ab06d2f1d69064b43abf4c185ca6b28865f3f9731"},
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[package.extras]
 | 
			
		||||
@ -721,17 +736,17 @@ files = [
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "setuptools"
 | 
			
		||||
version = "68.2.2"
 | 
			
		||||
version = "69.0.2"
 | 
			
		||||
description = "Easily download, build, install, upgrade, and uninstall Python packages"
 | 
			
		||||
optional = false
 | 
			
		||||
python-versions = ">=3.8"
 | 
			
		||||
files = [
 | 
			
		||||
    {file = "setuptools-68.2.2-py3-none-any.whl", hash = "sha256:b454a35605876da60632df1a60f736524eb73cc47bbc9f3f1ef1b644de74fd2a"},
 | 
			
		||||
    {file = "setuptools-68.2.2.tar.gz", hash = "sha256:4ac1475276d2f1c48684874089fefcd83bd7162ddaafb81fac866ba0db282a87"},
 | 
			
		||||
    {file = "setuptools-69.0.2-py3-none-any.whl", hash = "sha256:1e8fdff6797d3865f37397be788a4e3cba233608e9b509382a2777d25ebde7f2"},
 | 
			
		||||
    {file = "setuptools-69.0.2.tar.gz", hash = "sha256:735896e78a4742605974de002ac60562d286fa8051a7e2299445e8e8fbb01aa6"},
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[package.extras]
 | 
			
		||||
docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-hoverxref (<2)", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"]
 | 
			
		||||
docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"]
 | 
			
		||||
testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"]
 | 
			
		||||
testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.1)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"]
 | 
			
		||||
 | 
			
		||||
@ -824,19 +839,19 @@ standard = ["colorama (>=0.4)", "httptools (>=0.5.0)", "python-dotenv (>=0.13)",
 | 
			
		||||
 | 
			
		||||
[[package]]
 | 
			
		||||
name = "virtualenv"
 | 
			
		||||
version = "20.24.6"
 | 
			
		||||
version = "20.24.7"
 | 
			
		||||
description = "Virtual Python Environment builder"
 | 
			
		||||
optional = false
 | 
			
		||||
python-versions = ">=3.7"
 | 
			
		||||
files = [
 | 
			
		||||
    {file = "virtualenv-20.24.6-py3-none-any.whl", hash = "sha256:520d056652454c5098a00c0f073611ccbea4c79089331f60bf9d7ba247bb7381"},
 | 
			
		||||
    {file = "virtualenv-20.24.6.tar.gz", hash = "sha256:02ece4f56fbf939dbbc33c0715159951d6bf14aaf5457b092e4548e1382455af"},
 | 
			
		||||
    {file = "virtualenv-20.24.7-py3-none-any.whl", hash = "sha256:a18b3fd0314ca59a2e9f4b556819ed07183b3e9a3702ecfe213f593d44f7b3fd"},
 | 
			
		||||
    {file = "virtualenv-20.24.7.tar.gz", hash = "sha256:69050ffb42419c91f6c1284a7b24e0475d793447e35929b488bf6a0aade39353"},
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[package.dependencies]
 | 
			
		||||
distlib = ">=0.3.7,<1"
 | 
			
		||||
filelock = ">=3.12.2,<4"
 | 
			
		||||
platformdirs = ">=3.9.1,<4"
 | 
			
		||||
platformdirs = ">=3.9.1,<5"
 | 
			
		||||
 | 
			
		||||
[package.extras]
 | 
			
		||||
docs = ["furo (>=2023.7.26)", "proselint (>=0.13)", "sphinx (>=7.1.2)", "sphinx-argparse (>=0.4)", "sphinxcontrib-towncrier (>=0.2.1a0)", "towncrier (>=23.6)"]
 | 
			
		||||
@ -862,4 +877,4 @@ watchdog = ["watchdog (>=2.3)"]
 | 
			
		||||
[metadata]
 | 
			
		||||
lock-version = "2.0"
 | 
			
		||||
python-versions = "^3.12"
 | 
			
		||||
content-hash = "32e7c40e7148530effb10ebd5d67a4f1c8fe30794a4d3b5d213d4f30048c79ea"
 | 
			
		||||
content-hash = "498b3358998a9f3bbfb74fae7d6a90de7b55b9bdc76845bce52f65785afd0c1e"
 | 
			
		||||
 | 
			
		||||
@ -24,6 +24,7 @@ djhtml = "^1.5.2"
 | 
			
		||||
djlint = "^1.19.11"
 | 
			
		||||
isort = "^5.11.4"
 | 
			
		||||
pre-commit = "^3.5.0"
 | 
			
		||||
django-debug-toolbar = "^4.2.0"
 | 
			
		||||
 | 
			
		||||
[tool.isort]
 | 
			
		||||
profile = "black"
 | 
			
		||||
 | 
			
		||||
@ -43,6 +43,7 @@ INSTALLED_APPS = [
 | 
			
		||||
if DEBUG:
 | 
			
		||||
    INSTALLED_APPS.append("django_extensions")
 | 
			
		||||
    INSTALLED_APPS.append("django.contrib.admin")
 | 
			
		||||
    INSTALLED_APPS.append("debug_toolbar")
 | 
			
		||||
 | 
			
		||||
MIDDLEWARE = [
 | 
			
		||||
    "django.middleware.security.SecurityMiddleware",
 | 
			
		||||
@ -54,6 +55,11 @@ MIDDLEWARE = [
 | 
			
		||||
    "django.middleware.clickjacking.XFrameOptionsMiddleware",
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
if DEBUG:
 | 
			
		||||
    MIDDLEWARE.append("debug_toolbar.middleware.DebugToolbarMiddleware")
 | 
			
		||||
    INTERNAL_IPS = ["127.0.0.1"]
 | 
			
		||||
    DEBUG_TOOLBAR_CONFIG = {"ROOT_TAG_EXTRA_ATTRS": "hx-preserve"}
 | 
			
		||||
 | 
			
		||||
ROOT_URLCONF = "timetracker.urls"
 | 
			
		||||
 | 
			
		||||
TEMPLATES = [
 | 
			
		||||
 | 
			
		||||
@ -25,3 +25,4 @@ urlpatterns = [
 | 
			
		||||
 | 
			
		||||
if settings.DEBUG:
 | 
			
		||||
    urlpatterns.append(path("admin/", admin.site.urls))
 | 
			
		||||
    urlpatterns.append(path("__debug__/", include("debug_toolbar.urls")))
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user