Compare commits
255 Commits
d78139a5b3
...
main
Author | SHA1 | Date | |
---|---|---|---|
5cc1652002
|
|||
7cf2180192
|
|||
ad0641f95b
|
|||
abdcfdfe64
|
|||
31daf2efe0
|
|||
6d53fca910
|
|||
f7e426e030
|
|||
b29e4edd72
|
|||
3c58851b88
|
|||
99f3540825
|
|||
5e778bec30
|
|||
fea9d9784d
|
|||
23b4a7a069
|
|||
89de85c00d
|
|||
d892659132
|
|||
341e62283b
|
|||
61b6c1c55f
|
|||
eeaa02bada
|
|||
9d16bc2546
|
|||
7a52b59b3d
|
|||
0ce59a8cc6
|
|||
e0dfc0fc3e
|
|||
8cb67ca002
|
|||
be2a01840c
|
|||
612c42ebb7
|
|||
e2255a1c85
|
|||
0b274b4403
|
|||
ddd75f22b0
|
|||
843eed64d6
|
|||
50e7efcfae
|
|||
3e713a7637
|
|||
2d7342c0d5
|
|||
aba9bc994d
|
|||
967ff7df07
|
|||
2ab497fd54
|
|||
34148466c7
|
|||
b22e185d47
|
|||
b2b69339b3
|
|||
89d1bbdd9e
|
|||
637e3e6493
|
|||
d213a3d35d
|
|||
2f4e16dd54
|
|||
6f62889e92
|
|||
4ec808eeec
|
|||
69d27958f3
|
|||
4ec1cf5f28
|
|||
d936fdc60d
|
|||
2116cfc219
|
|||
6bd8271291
|
|||
e571feadef
|
|||
23c1ce1f96
|
|||
33103daebc
|
|||
ba6028e43d
|
|||
c2853a3ecc
|
|||
cd90d60475
|
|||
11cea2142a
|
|||
24578b64fe
|
|||
13e607f9a7
|
|||
fc0d8db8e8
|
|||
8acc4f9c5b
|
|||
6b7a96dc06
|
|||
5c5fd5f26a
|
|||
7181b6472c
|
|||
af06d07ee3
|
|||
315e22a8ac
|
|||
19676f8441 | |||
f61cde180f | |||
a53818257c | |||
2d3ea714c4 | |||
832bb48983 | |||
c6b1badf39 | |||
a3ed93c154 | |||
cf503a7b7d | |||
d81df6452a | |||
d9290373b0
|
|||
f8d621e710
|
|||
9992d9c9bd
|
|||
2ae81bb00f
|
|||
993abb4710
|
|||
23502eab85
|
|||
c517d735c7
|
|||
19056f846e
|
|||
0759ad0804
|
|||
228fc2bf5f
|
|||
a5a7041920
|
|||
fbd829f70e
|
|||
4873f25248
|
|||
3578f1707f
|
|||
b74ccb6eaa
|
|||
b0b1bb2d42
|
|||
c40764a02f
|
|||
649351efde
|
|||
698c8966c0
|
|||
7f6584ecf7
|
|||
540f5ee42c
|
|||
1c73268258
|
|||
3063a3d143
|
|||
b589199ca6
|
|||
2fc661dade
|
|||
1f535a6e84
|
|||
a9c1135639
|
|||
58cfaca1a9
|
|||
c1b3493c80
|
|||
a1df8720f5
|
|||
5a852bc2b9
|
|||
8ab9bfeeeb
|
|||
5eee7176d4
|
|||
98c9c1faee
|
|||
645ffa0dad
|
|||
4358708262
|
|||
c738245783
|
|||
57184ceea0
|
|||
c2b9409562
|
|||
e067e65bce
|
|||
b8258e2937
|
|||
9af4c79947
|
|||
d8b8182b91
|
|||
2fd44c1f53
|
|||
c3f99d124c
|
|||
51f5b9fceb
|
|||
973f4416de
|
|||
a84209eb81
|
|||
498cd69328
|
|||
b28c42d945
|
|||
3099f02145
|
|||
74b9d0421c
|
|||
c61adad180
|
|||
298ecb4092
|
|||
020e12e20b
|
|||
6ef56bfed5
|
|||
fda4913c97
|
|||
e85b32e22f
|
|||
2d6d6d24a4
|
|||
00993a85db
|
|||
4f7e708255
|
|||
238e4839e0
|
|||
b0ad806a93
|
|||
453b4fd922
|
|||
bb0d24809e
|
|||
3abd4c4af9
|
|||
2e5e77b4e5
|
|||
e79cf5de7a
|
|||
c15eaca205
|
|||
496c99ccf1
|
|||
992622e8d1
|
|||
cabe36c822
|
|||
d84b67c460
|
|||
1c28950b53
|
|||
b54bcdd9e9
|
|||
9ec6c958c8
|
|||
25deac6ea9
|
|||
a5ac10b20d
|
|||
3de40ccad3
|
|||
6a5dc9b62c
|
|||
b6014a72e0
|
|||
245b47b8b3
|
|||
e33f23c18f
|
|||
33012bc328
|
|||
447bd4820c
|
|||
72e89dae77
|
|||
1cd0a8c0fb
|
|||
a9a430f856
|
|||
0ee4c50a24
|
|||
714f0d97a9
|
|||
d622ddfbf3
|
|||
86fd40cc4a
|
|||
e174850262
|
|||
6328d835ee
|
|||
34d42e2af5
|
|||
e19caf47bf
|
|||
72998ffc02
|
|||
ba44814474
|
|||
86f8fde8fa
|
|||
811fec4b11
|
|||
fe6cf2758c
|
|||
1e1372ca56
|
|||
d91c0bc255
|
|||
a14f5d3ae5
|
|||
4ac13053d5
|
|||
e9311225e7
|
|||
44c70a5ee7
|
|||
cd804f2c77
|
|||
15997bd5af
|
|||
880ea93424
|
|||
dc1a9d5c4f
|
|||
51c25659a9
|
|||
973dda59d2
|
|||
64edca9ffa
|
|||
86e25b84ab
|
|||
edc1d062bc
|
|||
12a517c9fa
|
|||
c1882f66e3
|
|||
1e87e67eb1
|
|||
84552e088b
|
|||
79dc8ae25c
|
|||
cee06e4f64
|
|||
d9b5f0eab2
|
|||
ff28600710
|
|||
7517bf5f37
|
|||
780a04d13f
|
|||
fd04e9fa77
|
|||
18902aedac
|
|||
f9e37e9b1e
|
|||
c747cd1fd8
|
|||
6a5457191a
|
|||
76f6d0c377
|
|||
ae93703c08
|
|||
c55176090c
|
|||
081b8a92de
|
|||
d02a60675f
|
|||
4670568acb
|
|||
4b75a1dea9
|
|||
e2b7ff2e15
|
|||
b94aa49fc3
|
|||
73a92e5636
|
|||
42b28665e1
|
|||
6ba187f8e4
|
|||
a765fd8d00
|
|||
854e3cc54a
|
|||
2d8eb32e90
|
|||
1f1ed79ee5
|
|||
01fd7bad69
|
|||
44f49e5974
|
|||
0cf3411f63
|
|||
aa669710e1
|
|||
242833f886
|
|||
0cdfd3c298
|
|||
a98b4839dd
|
|||
1999f13cf2
|
|||
8466f67c86
|
|||
d9fbb4b896
|
|||
4ff3692606
|
|||
8289c48896
|
|||
d1b9202337
|
|||
fde93cb875
|
|||
d1c3ac6079
|
|||
d921c2d8a6
|
|||
52513e1ed8
|
|||
cb380814a7
|
|||
5ef8c07f30
|
|||
9573c3b8ff
|
|||
c4354a1380
|
|||
a245b6ff0f
|
|||
6329d380b7
|
|||
76fbc39fed
|
|||
4b6734c173
|
|||
b505b5b430
|
|||
87553ebdc5
|
|||
ba4fc0cac5
|
|||
8cb0276215
|
|||
f9a51ee83d
|
|||
c9deba7d65
|
|||
c55fbe86b5
|
|||
0e93993498
|
|||
9fccdfbff0
|
25
.devcontainer/devcontainer.json
Normal file
25
.devcontainer/devcontainer.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"name": "Django Time Tracker",
|
||||||
|
"dockerFile": "../devcontainer.Dockerfile",
|
||||||
|
"customizations": {
|
||||||
|
"vscode": {
|
||||||
|
"settings": {
|
||||||
|
"python.pythonPath": "/usr/local/bin/python",
|
||||||
|
"python.defaultInterpreterPath": "/usr/local/bin/python",
|
||||||
|
"terminal.integrated.defaultProfile.linux": "bash"
|
||||||
|
},
|
||||||
|
"extensions": [
|
||||||
|
"ms-python.python",
|
||||||
|
"ms-python.debugpy",
|
||||||
|
"ms-python.vscode-pylance",
|
||||||
|
"ms-azuretools.vscode-docker",
|
||||||
|
"batisteo.vscode-django",
|
||||||
|
"charliermarsh.ruff",
|
||||||
|
"bradlc.vscode-tailwindcss",
|
||||||
|
"EditorConfig.EditorConfig"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"forwardPorts": [8000],
|
||||||
|
"postCreateCommand": "poetry install && poetry run python manage.py migrate && npm install && make dev",
|
||||||
|
}
|
@ -15,3 +15,6 @@ indent_size = 4
|
|||||||
[**/*.js]
|
[**/*.js]
|
||||||
indent_style = space
|
indent_style = space
|
||||||
indent_size = 2
|
indent_size = 2
|
||||||
|
|
||||||
|
[*.html]
|
||||||
|
insert_final_newline = false
|
||||||
|
2
.github/workflows/build-docker.yml
vendored
2
.github/workflows/build-docker.yml
vendored
@ -17,7 +17,7 @@ jobs:
|
|||||||
poetry install
|
poetry install
|
||||||
poetry env info
|
poetry env info
|
||||||
poetry run python manage.py migrate
|
poetry run python manage.py migrate
|
||||||
poetry run pytest
|
# PROD=1 poetry run pytest
|
||||||
build-and-push:
|
build-and-push:
|
||||||
needs: test
|
needs: test
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
5
.gitignore
vendored
5
.gitignore
vendored
@ -1,9 +1,12 @@
|
|||||||
__pycache__
|
__pycache__
|
||||||
.mypy_cache
|
.mypy_cache
|
||||||
.pytest_cache
|
.pytest_cache
|
||||||
.venv
|
.venv/
|
||||||
node_modules
|
node_modules
|
||||||
package-lock.json
|
package-lock.json
|
||||||
db.sqlite3
|
db.sqlite3
|
||||||
/static/
|
/static/
|
||||||
dist/
|
dist/
|
||||||
|
.DS_Store
|
||||||
|
.python-version
|
||||||
|
.direnv
|
||||||
|
@ -1,10 +0,0 @@
|
|||||||
repos:
|
|
||||||
- repo: https://github.com/psf/black
|
|
||||||
rev: 22.12.0
|
|
||||||
hooks:
|
|
||||||
- id: black
|
|
||||||
- repo: https://github.com/pycqa/isort
|
|
||||||
rev: 5.12.0
|
|
||||||
hooks:
|
|
||||||
- id: isort
|
|
||||||
name: isort (python)
|
|
11
.vscode/extensions.json
vendored
Normal file
11
.vscode/extensions.json
vendored
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"recommendations": [
|
||||||
|
"charliermarsh.ruff",
|
||||||
|
"ms-python.python",
|
||||||
|
"ms-python.vscode-pylance",
|
||||||
|
"ms-python.debugpy",
|
||||||
|
"batisteo.vscode-django",
|
||||||
|
"bradlc.vscode-tailwindcss",
|
||||||
|
"EditorConfig.EditorConfig"
|
||||||
|
]
|
||||||
|
}
|
26
.vscode/launch.json
vendored
Normal file
26
.vscode/launch.json
vendored
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
// Use IntelliSense to learn about possible attributes.
|
||||||
|
// Hover to view descriptions of existing attributes.
|
||||||
|
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
|
||||||
|
"version": "0.2.0",
|
||||||
|
"configurations": [
|
||||||
|
{
|
||||||
|
"name": "Python Debugger: Current File",
|
||||||
|
"type": "debugpy",
|
||||||
|
"request": "launch",
|
||||||
|
"program": "${file}",
|
||||||
|
"console": "integratedTerminal"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Python Debugger: Django",
|
||||||
|
"type": "debugpy",
|
||||||
|
"request": "launch",
|
||||||
|
"args": [
|
||||||
|
"runserver"
|
||||||
|
],
|
||||||
|
"django": true,
|
||||||
|
"autoStartBrowser": false,
|
||||||
|
"program": "${workspaceFolder}/manage.py"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
26
.vscode/settings.json
vendored
26
.vscode/settings.json
vendored
@ -4,8 +4,30 @@
|
|||||||
],
|
],
|
||||||
"python.testing.unittestEnabled": false,
|
"python.testing.unittestEnabled": false,
|
||||||
"python.testing.pytestEnabled": true,
|
"python.testing.pytestEnabled": true,
|
||||||
"python.analysis.typeCheckingMode": "basic",
|
"python.analysis.typeCheckingMode": "strict",
|
||||||
"[python]": {
|
"[python]": {
|
||||||
"editor.defaultFormatter": "ms-python.black-formatter"
|
"editor.defaultFormatter": "charliermarsh.ruff",
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.codeActionsOnSave": {
|
||||||
|
"source.fixAll": "explicit",
|
||||||
|
"source.organizeImports": "explicit"
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
"ruff.path": ["/nix/store/jaibb3v0rrnlw5ib54qqq3452yhp1xcb-ruff-0.5.7/bin/ruff"],
|
||||||
|
"tailwind-fold.supportedLanguages": [
|
||||||
|
"html",
|
||||||
|
"typescriptreact",
|
||||||
|
"javascriptreact",
|
||||||
|
"typescript",
|
||||||
|
"javascript",
|
||||||
|
"vue-html",
|
||||||
|
"vue",
|
||||||
|
"php",
|
||||||
|
"markdown",
|
||||||
|
"coffeescript",
|
||||||
|
"svelte",
|
||||||
|
"astro",
|
||||||
|
"erb",
|
||||||
|
"django-html"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
35
CHANGELOG.md
35
CHANGELOG.md
@ -1,8 +1,41 @@
|
|||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
|
## New
|
||||||
|
* Render notes as Markdown
|
||||||
|
* Require login by default
|
||||||
|
* Add stats for dropped purchases, monthly playtimes
|
||||||
|
* Allow deleting purchases
|
||||||
|
* Add all-time stats
|
||||||
|
* Manage purchases
|
||||||
|
* Automatically convert purchase prices
|
||||||
|
* Add emulated property to sessions
|
||||||
|
* Add today's and last 7 days playtime stats to navbar
|
||||||
|
|
||||||
## Improved
|
## Improved
|
||||||
* game overview: improve how editions and purchases are displayed
|
* mark refunded purchases red on game overview
|
||||||
|
* increase session count on game overview when starting a new session
|
||||||
|
* game overview:
|
||||||
|
* sort purchases also by date purchased (on top of date released)
|
||||||
|
* improve header format, make it more appealing
|
||||||
|
* ignore manual sessions when calculating session average
|
||||||
|
* stats: improve purchase name consistency
|
||||||
|
* session list: use display name instead of sort name
|
||||||
|
* unify the appearance of game links, and make them expand to full size on hover
|
||||||
|
|
||||||
|
## Fixed
|
||||||
|
* Fix title not being displayed on the Recent sessions page
|
||||||
|
* Avoid errors when displaying game overview with zero sessions
|
||||||
|
|
||||||
|
## 1.5.2 / 2024-01-14 21:27+01:00
|
||||||
|
|
||||||
|
## Improved
|
||||||
|
* game overview:
|
||||||
|
* improve how editions and purchases are displayed
|
||||||
|
* make it possible to end session from overview
|
||||||
* add purchase: only allow choosing purchases of selected edition
|
* add purchase: only allow choosing purchases of selected edition
|
||||||
|
* session list:
|
||||||
|
* starting and ending sessions is much faster/doest not reload the page
|
||||||
|
* listing sessions is much faster
|
||||||
|
|
||||||
## 1.5.1 / 2023-11-14 21:10+01:00
|
## 1.5.1 / 2023-11-14 21:10+01:00
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
FROM python:3.12.0-slim-bullseye
|
FROM python:3.12.0-slim-bullseye
|
||||||
|
|
||||||
ENV VERSION_NUMBER=1.5.1 \
|
ENV VERSION_NUMBER=1.5.2 \
|
||||||
PROD=1 \
|
PROD=1 \
|
||||||
PYTHONUNBUFFERED=1 \
|
PYTHONUNBUFFERED=1 \
|
||||||
PYTHONFAULTHANDLER=1 \
|
PYTHONFAULTHANDLER=1 \
|
||||||
|
20
Makefile
20
Makefile
@ -3,6 +3,7 @@ all: css migrate
|
|||||||
initialize: npm css migrate sethookdir loadplatforms
|
initialize: npm css migrate sethookdir loadplatforms
|
||||||
|
|
||||||
HTMLFILES := $(shell find games/templates -type f)
|
HTMLFILES := $(shell find games/templates -type f)
|
||||||
|
PYTHON_VERSION = 3.12
|
||||||
|
|
||||||
npm:
|
npm:
|
||||||
npm install
|
npm install
|
||||||
@ -10,17 +11,26 @@ npm:
|
|||||||
css: common/input.css
|
css: common/input.css
|
||||||
npx tailwindcss -i ./common/input.css -o ./games/static/base.css
|
npx tailwindcss -i ./common/input.css -o ./games/static/base.css
|
||||||
|
|
||||||
css-dev: css
|
|
||||||
npx tailwindcss -i ./common/input.css -o ./games/static/base.css --watch
|
|
||||||
|
|
||||||
makemigrations:
|
makemigrations:
|
||||||
poetry run python manage.py makemigrations
|
poetry run python manage.py makemigrations
|
||||||
|
|
||||||
migrate: makemigrations
|
migrate: makemigrations
|
||||||
poetry run python manage.py migrate
|
poetry run python manage.py migrate
|
||||||
|
|
||||||
dev: migrate
|
init:
|
||||||
poetry run python manage.py runserver
|
pyenv install -s $(PYTHON_VERSION)
|
||||||
|
pyenv local $(PYTHON_VERSION)
|
||||||
|
pip install poetry
|
||||||
|
poetry install
|
||||||
|
npm install
|
||||||
|
|
||||||
|
dev:
|
||||||
|
@npx concurrently \
|
||||||
|
--names "Django,Tailwind" \
|
||||||
|
--prefix-colors "blue,green" \
|
||||||
|
"poetry run python -Wa manage.py runserver" \
|
||||||
|
"npx tailwindcss -i ./common/input.css -o ./games/static/base.css --watch"
|
||||||
|
|
||||||
|
|
||||||
caddy:
|
caddy:
|
||||||
caddy run --watch
|
caddy run --watch
|
||||||
|
12
README.md
12
README.md
@ -1,3 +1,15 @@
|
|||||||
# Timetracker
|
# Timetracker
|
||||||
|
|
||||||
A simple game catalogue and play session tracker.
|
A simple game catalogue and play session tracker.
|
||||||
|
|
||||||
|
# Development
|
||||||
|
|
||||||
|
The project uses `pyenv` to manage installed Python versions.
|
||||||
|
If you have `pyenv` installed, you can simply run:
|
||||||
|
|
||||||
|
```
|
||||||
|
make init
|
||||||
|
```
|
||||||
|
|
||||||
|
This will make sure the correct Python version is installed, and it will install all dependencies using `poetry`.
|
||||||
|
Afterwards, you can start the development server using `make dev`.
|
287
common/components.py
Normal file
287
common/components.py
Normal file
@ -0,0 +1,287 @@
|
|||||||
|
from random import choices as random_choices
|
||||||
|
from string import ascii_lowercase
|
||||||
|
from typing import Any, Callable
|
||||||
|
|
||||||
|
from django.template import TemplateDoesNotExist
|
||||||
|
from django.template.defaultfilters import floatformat
|
||||||
|
from django.template.loader import render_to_string
|
||||||
|
from django.urls import NoReverseMatch, reverse
|
||||||
|
from django.utils.safestring import SafeText, mark_safe
|
||||||
|
|
||||||
|
from common.utils import truncate
|
||||||
|
from games.models import Game, Purchase, Session
|
||||||
|
|
||||||
|
HTMLAttribute = tuple[str, str | int | bool]
|
||||||
|
HTMLTag = str
|
||||||
|
|
||||||
|
|
||||||
|
def Component(
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
template: str = "",
|
||||||
|
tag_name: str = "",
|
||||||
|
) -> HTMLTag:
|
||||||
|
if not tag_name and not template:
|
||||||
|
raise ValueError("One of template or tag_name is required.")
|
||||||
|
if isinstance(children, str):
|
||||||
|
children = [children]
|
||||||
|
childrenBlob = "\n".join(children)
|
||||||
|
if len(attributes) == 0:
|
||||||
|
attributesBlob = ""
|
||||||
|
else:
|
||||||
|
attributesList = [f'{name}="{value}"' for name, value in attributes]
|
||||||
|
# make attribute list into a string
|
||||||
|
# and insert space between tag and attribute list
|
||||||
|
attributesBlob = f" {' '.join(attributesList)}"
|
||||||
|
tag: str = ""
|
||||||
|
if tag_name != "":
|
||||||
|
tag = f"<{tag_name}{attributesBlob}>{childrenBlob}</{tag_name}>"
|
||||||
|
elif template != "":
|
||||||
|
tag = render_to_string(
|
||||||
|
template,
|
||||||
|
{name: value for name, value in attributes}
|
||||||
|
| {"slot": mark_safe("\n".join(children))},
|
||||||
|
)
|
||||||
|
return mark_safe(tag)
|
||||||
|
|
||||||
|
|
||||||
|
def randomid(seed: str = "", length: int = 10) -> str:
|
||||||
|
return seed + "".join(random_choices(ascii_lowercase, k=length))
|
||||||
|
|
||||||
|
|
||||||
|
def Popover(
|
||||||
|
popover_content: str,
|
||||||
|
wrapped_content: str = "",
|
||||||
|
wrapped_classes: str = "",
|
||||||
|
children: list[HTMLTag] = [],
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
) -> str:
|
||||||
|
if not wrapped_content and not children:
|
||||||
|
raise ValueError("One of wrapped_content or children is required.")
|
||||||
|
id = randomid()
|
||||||
|
return Component(
|
||||||
|
attributes=attributes
|
||||||
|
+ [
|
||||||
|
("id", id),
|
||||||
|
("wrapped_content", wrapped_content),
|
||||||
|
("popover_content", popover_content),
|
||||||
|
("wrapped_classes", wrapped_classes),
|
||||||
|
],
|
||||||
|
children=children,
|
||||||
|
template="cotton/popover.html",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def PopoverTruncated(
|
||||||
|
input_string: str,
|
||||||
|
popover_content: str = "",
|
||||||
|
popover_if_not_truncated: bool = False,
|
||||||
|
length: int = 30,
|
||||||
|
ellipsis: str = "…",
|
||||||
|
endpart: str = "",
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Returns `input_string` truncated after `length` of characters
|
||||||
|
and displays the untruncated text in a popover HTML element.
|
||||||
|
The truncated text ends in `ellipsis`, and optionally
|
||||||
|
an always-visible `endpart` can be specified.
|
||||||
|
`popover_content` can be specified if:
|
||||||
|
1. It needs to be always displayed regardless if text is truncated.
|
||||||
|
2. It needs to differ from `input_string`.
|
||||||
|
"""
|
||||||
|
if (truncated := truncate(input_string, length, ellipsis, endpart)) != input_string:
|
||||||
|
return Popover(
|
||||||
|
wrapped_content=truncated,
|
||||||
|
popover_content=popover_content if popover_content else input_string,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if popover_content and popover_if_not_truncated:
|
||||||
|
return Popover(
|
||||||
|
wrapped_content=input_string,
|
||||||
|
popover_content=popover_content if popover_content else "",
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
return input_string
|
||||||
|
|
||||||
|
|
||||||
|
def A(
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
url: str | Callable[..., Any] = "",
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Returns the HTML tag "a".
|
||||||
|
"url" can either be:
|
||||||
|
- URL (string)
|
||||||
|
- path name passed to reverse() (string)
|
||||||
|
- function
|
||||||
|
"""
|
||||||
|
additional_attributes = []
|
||||||
|
if url:
|
||||||
|
if type(url) is str:
|
||||||
|
try:
|
||||||
|
url_result = reverse(url)
|
||||||
|
except NoReverseMatch:
|
||||||
|
url_result = url
|
||||||
|
elif callable(url):
|
||||||
|
url_result = url()
|
||||||
|
else:
|
||||||
|
raise TypeError("'url' is neither str nor function.")
|
||||||
|
additional_attributes = [("href", url_result)]
|
||||||
|
return Component(
|
||||||
|
tag_name="a", attributes=attributes + additional_attributes, children=children
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def Button(
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
size: str = "base",
|
||||||
|
icon: bool = False,
|
||||||
|
color: str = "blue",
|
||||||
|
):
|
||||||
|
return Component(
|
||||||
|
template="cotton/button.html",
|
||||||
|
attributes=attributes + [("size", size), ("icon", icon), ("color", color)],
|
||||||
|
children=children,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def Div(
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
):
|
||||||
|
return Component(tag_name="div", attributes=attributes, children=children)
|
||||||
|
|
||||||
|
|
||||||
|
def Input(
|
||||||
|
type: str = "text",
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
):
|
||||||
|
return Component(
|
||||||
|
tag_name="input", attributes=attributes + [("type", type)], children=children
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def Form(
|
||||||
|
action="",
|
||||||
|
method="get",
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
children: list[HTMLTag] | HTMLTag = [],
|
||||||
|
):
|
||||||
|
return Component(
|
||||||
|
tag_name="form",
|
||||||
|
attributes=attributes + [("action", action), ("method", method)],
|
||||||
|
children=children,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def Icon(
|
||||||
|
name: str,
|
||||||
|
attributes: list[HTMLAttribute] = [],
|
||||||
|
):
|
||||||
|
try:
|
||||||
|
result = Component(template=f"cotton/icon/{name}.html", attributes=attributes)
|
||||||
|
except TemplateDoesNotExist:
|
||||||
|
result = Icon(name="unspecified", attributes=attributes)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def LinkedPurchase(purchase: Purchase) -> SafeText:
|
||||||
|
link = reverse("view_purchase", args=[int(purchase.id)])
|
||||||
|
link_content = ""
|
||||||
|
popover_content = ""
|
||||||
|
game_count = purchase.games.count()
|
||||||
|
popover_if_not_truncated = False
|
||||||
|
if game_count == 1:
|
||||||
|
link_content += purchase.games.first().name
|
||||||
|
popover_content = link_content
|
||||||
|
if game_count > 1:
|
||||||
|
if purchase.name:
|
||||||
|
link_content += f"{purchase.name}"
|
||||||
|
popover_content += f"<h1>{purchase.name}</h1><br>"
|
||||||
|
else:
|
||||||
|
link_content += f"{game_count} games"
|
||||||
|
popover_if_not_truncated = True
|
||||||
|
popover_content += f"""
|
||||||
|
<ul class="list-disc list-inside">
|
||||||
|
{"".join(f"<li>{game.name}</li>" for game in purchase.games.all())}
|
||||||
|
</ul>
|
||||||
|
"""
|
||||||
|
icon = purchase.platform.icon if game_count == 1 else "unspecified"
|
||||||
|
if link_content == "":
|
||||||
|
raise ValueError("link_content is empty!!")
|
||||||
|
a_content = Div(
|
||||||
|
[("class", "inline-flex gap-2 items-center")],
|
||||||
|
[
|
||||||
|
Icon(
|
||||||
|
icon,
|
||||||
|
[("title", "Multiple")],
|
||||||
|
),
|
||||||
|
PopoverTruncated(
|
||||||
|
input_string=link_content,
|
||||||
|
popover_content=mark_safe(popover_content),
|
||||||
|
popover_if_not_truncated=popover_if_not_truncated,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
return mark_safe(A(url=link, children=[a_content]))
|
||||||
|
|
||||||
|
|
||||||
|
def NameWithIcon(
|
||||||
|
name: str = "",
|
||||||
|
platform: str = "",
|
||||||
|
game_id: int = 0,
|
||||||
|
session_id: int = 0,
|
||||||
|
purchase_id: int = 0,
|
||||||
|
linkify: bool = True,
|
||||||
|
emulated: bool = False,
|
||||||
|
) -> SafeText:
|
||||||
|
create_link = False
|
||||||
|
link = ""
|
||||||
|
platform = None
|
||||||
|
if (game_id != 0 or session_id != 0 or purchase_id != 0) and linkify:
|
||||||
|
create_link = True
|
||||||
|
if session_id:
|
||||||
|
session = Session.objects.get(pk=session_id)
|
||||||
|
emulated = session.emulated
|
||||||
|
game_id = session.game.pk
|
||||||
|
if purchase_id:
|
||||||
|
purchase = Purchase.objects.get(pk=purchase_id)
|
||||||
|
game_id = purchase.games.first().pk
|
||||||
|
if game_id:
|
||||||
|
game = Game.objects.get(pk=game_id)
|
||||||
|
name = name or game.name
|
||||||
|
platform = game.platform
|
||||||
|
link = reverse("view_game", args=[int(game_id)])
|
||||||
|
content = Div(
|
||||||
|
[("class", "inline-flex gap-2 items-center")],
|
||||||
|
[
|
||||||
|
Icon(
|
||||||
|
platform.icon,
|
||||||
|
[("title", platform.name)],
|
||||||
|
)
|
||||||
|
if platform
|
||||||
|
else "",
|
||||||
|
Icon("emulated", [("title", "Emulated")]) if emulated else "",
|
||||||
|
PopoverTruncated(name),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
return mark_safe(
|
||||||
|
A(
|
||||||
|
url=link,
|
||||||
|
children=[content],
|
||||||
|
)
|
||||||
|
if create_link
|
||||||
|
else content,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def PurchasePrice(purchase) -> str:
|
||||||
|
return Popover(
|
||||||
|
popover_content=f"{floatformat(purchase.price)} {purchase.price_currency}",
|
||||||
|
wrapped_content=f"{floatformat(purchase.converted_price)} {purchase.converted_currency}",
|
||||||
|
wrapped_classes="underline decoration-dotted",
|
||||||
|
)
|
122
common/input.css
122
common/input.css
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "IBM Plex Mono";
|
font-family: "IBM Plex Mono";
|
||||||
src: url("fonts/IBMPlexMono-regular.woff2") format("woff2");
|
src: url("fonts/IBMPlexMono-Regular.woff2") format("woff2");
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
}
|
}
|
||||||
@ -23,12 +23,33 @@
|
|||||||
font-style: normal;
|
font-style: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
form label {
|
@font-face {
|
||||||
@apply dark:text-slate-400;
|
font-family: "IBM Plex Serif";
|
||||||
|
src: url("fonts/IBMPlexSerif-Bold.woff2") format("woff2");
|
||||||
|
font-weight: 700;
|
||||||
|
font-style: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: "IBM Plex Sans Condensed";
|
||||||
|
src: url("fonts/IBMPlexSansCondensed-Regular.woff2") format("woff2");
|
||||||
|
font-weight: 400;
|
||||||
|
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;
|
||||||
|
} */
|
||||||
|
|
||||||
.responsive-table {
|
.responsive-table {
|
||||||
@apply dark:text-white mx-auto;
|
@apply dark:text-white mx-auto table-fixed;
|
||||||
}
|
}
|
||||||
|
|
||||||
.responsive-table tr:nth-child(even) {
|
.responsive-table tr:nth-child(even) {
|
||||||
@ -49,61 +70,62 @@ form label {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@layer utilities {
|
@layer utilities {
|
||||||
|
.min-w-20char {
|
||||||
|
min-width: 20ch;
|
||||||
|
}
|
||||||
.max-w-20char {
|
.max-w-20char {
|
||||||
max-width: 20ch;
|
max-width: 20ch;
|
||||||
}
|
}
|
||||||
|
.min-w-30char {
|
||||||
|
min-width: 30ch;
|
||||||
|
}
|
||||||
|
.max-w-30char {
|
||||||
|
max-width: 30ch;
|
||||||
|
}
|
||||||
.max-w-35char {
|
.max-w-35char {
|
||||||
max-width: 40ch;
|
max-width: 35ch;
|
||||||
}
|
}
|
||||||
.max-w-40char {
|
.max-w-40char {
|
||||||
max-width: 40ch;
|
max-width: 40ch;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
form input,
|
/* form input,
|
||||||
select,
|
select,
|
||||||
textarea {
|
textarea {
|
||||||
@apply dark:border dark:border-slate-900 dark:bg-slate-500 dark:text-slate-100;
|
@apply dark:border dark:border-slate-900 dark:bg-slate-500 dark:text-slate-100;
|
||||||
}
|
} */
|
||||||
|
|
||||||
form input:disabled,
|
form input:disabled,
|
||||||
select:disabled,
|
select:disabled,
|
||||||
textarea:disabled {
|
textarea:disabled {
|
||||||
@apply dark:bg-slate-700 dark:text-slate-400;
|
@apply dark:bg-slate-800 dark:text-slate-500 cursor-not-allowed;
|
||||||
}
|
}
|
||||||
|
|
||||||
.errorlist {
|
.errorlist {
|
||||||
@apply mt-4 mb-1 pl-3 py-2 bg-red-600 text-slate-200 w-[300px];
|
@apply mt-4 mb-1 pl-3 py-2 bg-red-600 text-slate-200 w-[300px];
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (min-width: 768px) {
|
/* @media screen and (min-width: 768px) {
|
||||||
form input,
|
form input,
|
||||||
select,
|
select,
|
||||||
textarea {
|
textarea {
|
||||||
width: 300px;
|
width: 300px;
|
||||||
}
|
}
|
||||||
}
|
} */
|
||||||
|
|
||||||
@media screen and (max-width: 768px) {
|
/* @media screen and (max-width: 768px) {
|
||||||
form input,
|
form input,
|
||||||
select,
|
select,
|
||||||
textarea {
|
textarea {
|
||||||
width: 150px;
|
width: 150px;
|
||||||
}
|
}
|
||||||
}
|
} */
|
||||||
|
|
||||||
#button-container button {
|
#button-container button {
|
||||||
@apply mx-1;
|
@apply mx-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
th {
|
|
||||||
@apply text-right;
|
|
||||||
}
|
|
||||||
|
|
||||||
th label {
|
|
||||||
@apply mr-4;
|
|
||||||
}
|
|
||||||
|
|
||||||
.basic-button-container {
|
.basic-button-container {
|
||||||
@apply flex space-x-2 justify-center;
|
@apply flex space-x-2 justify-center;
|
||||||
}
|
}
|
||||||
@ -111,3 +133,63 @@ th label {
|
|||||||
.basic-button {
|
.basic-button {
|
||||||
@apply inline-block px-6 py-2.5 bg-blue-600 text-white font-medium text-xs leading-tight uppercase rounded shadow-md hover:bg-blue-700 hover:shadow-lg focus:bg-blue-700 focus:shadow-lg focus:outline-none focus:ring-0 active:bg-blue-800 active:shadow-lg transition duration-150 ease-in-out;
|
@apply inline-block px-6 py-2.5 bg-blue-600 text-white font-medium text-xs leading-tight uppercase rounded shadow-md hover:bg-blue-700 hover:shadow-lg focus:bg-blue-700 focus:shadow-lg focus:outline-none focus:ring-0 active:bg-blue-800 active:shadow-lg transition duration-150 ease-in-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.markdown-content ul {
|
||||||
|
list-style-type: disc;
|
||||||
|
list-style-position: inside;
|
||||||
|
padding-left: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown-content ol {
|
||||||
|
list-style-type: decimal;
|
||||||
|
list-style-position: inside;
|
||||||
|
padding-left: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown-content ul,
|
||||||
|
.markdown-content ol {
|
||||||
|
list-style-position: outside;
|
||||||
|
padding-left: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown-content ul ul,
|
||||||
|
.markdown-content ul ol,
|
||||||
|
.markdown-content ol ul,
|
||||||
|
.markdown-content ol ol {
|
||||||
|
list-style-type: circle;
|
||||||
|
margin-top: 0.5em;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
padding-left: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* .truncate-container {
|
||||||
|
@apply inline-block relative;
|
||||||
|
a {
|
||||||
|
@apply inline-block truncate max-w-20char transition-all group-hover:absolute group-hover:max-w-none group-hover:-top-8 group-hover:-left-6 group-hover:min-w-60 group-hover:px-6 group-hover:py-3.5 group-hover:bg-purple-600 group-hover:rounded-sm group-hover:outline-dashed group-hover:outline-purple-400 group-hover:outline-4;
|
||||||
|
|
||||||
|
}
|
||||||
|
} */
|
||||||
|
|
||||||
|
label {
|
||||||
|
@apply dark:text-slate-500;
|
||||||
|
}
|
||||||
|
|
||||||
|
[type="text"], [type="password"], [type="datetime-local"], [type="datetime"], [type="date"], [type="number"], select, textarea {
|
||||||
|
@apply dark:bg-slate-600 dark:text-slate-300;
|
||||||
|
}
|
||||||
|
|
||||||
|
[type="submit"] {
|
||||||
|
@apply dark:text-white font-bold dark:bg-blue-600 px-4 py-2;
|
||||||
|
}
|
||||||
|
|
||||||
|
form div label {
|
||||||
|
@apply dark:text-white;
|
||||||
|
}
|
||||||
|
|
||||||
|
form div {
|
||||||
|
@apply flex flex-col;
|
||||||
|
}
|
||||||
|
|
||||||
|
div [type="submit"] {
|
||||||
|
@apply mt-3;
|
||||||
|
}
|
||||||
|
103
common/time.py
103
common/time.py
@ -1,9 +1,19 @@
|
|||||||
import re
|
import re
|
||||||
from datetime import timedelta
|
from datetime import date, datetime, timedelta
|
||||||
|
|
||||||
|
from django.utils import timezone
|
||||||
|
|
||||||
|
from common.utils import generate_split_ranges
|
||||||
|
|
||||||
|
dateformat: str = "%d/%m/%Y"
|
||||||
|
datetimeformat: str = "%d/%m/%Y %H:%M"
|
||||||
|
timeformat: str = "%H:%M"
|
||||||
|
durationformat: str = "%2.1H hours"
|
||||||
|
durationformat_manual: str = "%H hours"
|
||||||
|
|
||||||
|
|
||||||
def _safe_timedelta(duration: timedelta | int | None):
|
def _safe_timedelta(duration: timedelta | int | None):
|
||||||
if duration == None:
|
if duration is None:
|
||||||
return timedelta(0)
|
return timedelta(0)
|
||||||
elif isinstance(duration, int):
|
elif isinstance(duration, int):
|
||||||
return timedelta(seconds=duration)
|
return timedelta(seconds=duration)
|
||||||
@ -12,7 +22,7 @@ def _safe_timedelta(duration: timedelta | int | None):
|
|||||||
|
|
||||||
|
|
||||||
def format_duration(
|
def format_duration(
|
||||||
duration: timedelta | int | None, format_string: str = "%H hours"
|
duration: timedelta | int | float | None, format_string: str = "%H hours"
|
||||||
) -> str:
|
) -> str:
|
||||||
"""
|
"""
|
||||||
Format timedelta into the specified format_string.
|
Format timedelta into the specified format_string.
|
||||||
@ -70,3 +80,90 @@ def format_duration(
|
|||||||
rf"%\d*\.?\d*{pattern}", replacement, formatted_string
|
rf"%\d*\.?\d*{pattern}", replacement, formatted_string
|
||||||
)
|
)
|
||||||
return formatted_string
|
return formatted_string
|
||||||
|
|
||||||
|
|
||||||
|
def local_strftime(datetime: datetime, format: str = datetimeformat) -> str:
|
||||||
|
return timezone.localtime(datetime).strftime(format)
|
||||||
|
|
||||||
|
|
||||||
|
def daterange(start: date, end: date, end_inclusive: bool = False) -> list[date]:
|
||||||
|
time_between: timedelta = end - start
|
||||||
|
if (days_between := time_between.days) < 1:
|
||||||
|
raise ValueError("start and end have to be at least 1 day apart.")
|
||||||
|
if end_inclusive:
|
||||||
|
print(f"{end_inclusive=}")
|
||||||
|
print(f"{days_between=}")
|
||||||
|
days_between += 1
|
||||||
|
print(f"{days_between=}")
|
||||||
|
return [start + timedelta(x) for x in range(days_between)]
|
||||||
|
|
||||||
|
|
||||||
|
def streak(datelist: list[date]) -> dict[str, int | tuple[date, date]]:
|
||||||
|
if len(datelist) == 1:
|
||||||
|
return {"days": 1, "dates": (datelist[0], datelist[0])}
|
||||||
|
else:
|
||||||
|
print(f"Processing {len(datelist)} dates.")
|
||||||
|
missing = sorted(
|
||||||
|
set(
|
||||||
|
datelist[0] + timedelta(x)
|
||||||
|
for x in range((datelist[-1] - datelist[0]).days)
|
||||||
|
)
|
||||||
|
- set(datelist)
|
||||||
|
)
|
||||||
|
print(f"{len(missing)} days missing.")
|
||||||
|
datelist_with_missing = sorted(datelist + missing)
|
||||||
|
ranges = list(generate_split_ranges(datelist_with_missing, missing))
|
||||||
|
print(f"{len(ranges)} ranges calculated.")
|
||||||
|
longest_consecutive_days = timedelta(0)
|
||||||
|
longest_range: tuple[date, date] = (date(1970, 1, 1), date(1970, 1, 1))
|
||||||
|
for start, end in ranges:
|
||||||
|
if (current_streak := end - start) > longest_consecutive_days:
|
||||||
|
longest_consecutive_days = current_streak
|
||||||
|
longest_range = (start, end)
|
||||||
|
return {"days": longest_consecutive_days.days + 1, "dates": longest_range}
|
||||||
|
|
||||||
|
|
||||||
|
def streak_bruteforce(datelist: list[date]) -> dict[str, int | tuple[date, date]]:
|
||||||
|
if (datelist_length := len(datelist)) == 0:
|
||||||
|
raise ValueError("Number of dates in the list is 0.")
|
||||||
|
datelist.sort()
|
||||||
|
current_streak = 1
|
||||||
|
current_start = datelist[0]
|
||||||
|
current_end = datelist[0]
|
||||||
|
current_date = datelist[0]
|
||||||
|
highest_streak = 1
|
||||||
|
highest_streak_daterange = (current_start, current_end)
|
||||||
|
|
||||||
|
def update_highest_streak():
|
||||||
|
nonlocal highest_streak, highest_streak_daterange
|
||||||
|
if current_streak > highest_streak:
|
||||||
|
highest_streak = current_streak
|
||||||
|
highest_streak_daterange = (current_start, current_end)
|
||||||
|
|
||||||
|
def reset_streak():
|
||||||
|
nonlocal current_start, current_end, current_streak
|
||||||
|
current_start = current_end = current_date
|
||||||
|
current_streak = 1
|
||||||
|
|
||||||
|
def increment_streak():
|
||||||
|
nonlocal current_end, current_streak
|
||||||
|
current_end = current_date
|
||||||
|
current_streak += 1
|
||||||
|
|
||||||
|
for i, datelist_item in enumerate(datelist, start=1):
|
||||||
|
current_date = datelist_item
|
||||||
|
if current_date == current_start or current_date == current_end:
|
||||||
|
continue
|
||||||
|
if current_date - timedelta(1) != current_end and i != datelist_length:
|
||||||
|
update_highest_streak()
|
||||||
|
reset_streak()
|
||||||
|
elif current_date - timedelta(1) == current_end and i == datelist_length:
|
||||||
|
increment_streak()
|
||||||
|
update_highest_streak()
|
||||||
|
else:
|
||||||
|
increment_streak()
|
||||||
|
return {"days": highest_streak, "dates": highest_streak_daterange}
|
||||||
|
|
||||||
|
|
||||||
|
def available_stats_year_range():
|
||||||
|
return range(datetime.now().year, 1999, -1)
|
||||||
|
158
common/utils.py
158
common/utils.py
@ -1,3 +1,15 @@
|
|||||||
|
import operator
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from datetime import date
|
||||||
|
from functools import reduce, wraps
|
||||||
|
from typing import Any, Callable, Generator, Literal, TypeVar
|
||||||
|
from urllib.parse import urlencode
|
||||||
|
|
||||||
|
from django.db.models import Q
|
||||||
|
from django.http import HttpRequest
|
||||||
|
from django.shortcuts import redirect
|
||||||
|
|
||||||
|
|
||||||
def safe_division(numerator: int | float, denominator: int | float) -> int | float:
|
def safe_division(numerator: int | float, denominator: int | float) -> int | float:
|
||||||
"""
|
"""
|
||||||
Divides without triggering division by zero exception.
|
Divides without triggering division by zero exception.
|
||||||
@ -7,3 +19,149 @@ def safe_division(numerator: int | float, denominator: int | float) -> int | flo
|
|||||||
return numerator / denominator
|
return numerator / denominator
|
||||||
except ZeroDivisionError:
|
except ZeroDivisionError:
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
def safe_getattr(obj: object, attr_chain: str, default: Any | None = None) -> object:
|
||||||
|
"""
|
||||||
|
Safely get the nested attribute from an object.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
obj (object): The object from which to retrieve the attribute.
|
||||||
|
attr_chain (str): The chain of attributes, separated by dots.
|
||||||
|
default: The default value to return if any attribute in the chain does not exist.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The value of the nested attribute if it exists, otherwise the default value.
|
||||||
|
"""
|
||||||
|
attrs = attr_chain.split(".")
|
||||||
|
for attr in attrs:
|
||||||
|
try:
|
||||||
|
obj = getattr(obj, attr)
|
||||||
|
except AttributeError:
|
||||||
|
return default
|
||||||
|
return obj
|
||||||
|
|
||||||
|
|
||||||
|
def truncate_(input_string: str, length: int = 30, ellipsis: str = "…") -> str:
|
||||||
|
return (
|
||||||
|
(f"{input_string[: length - len(ellipsis)].rstrip()}{ellipsis}")
|
||||||
|
if len(input_string) > length
|
||||||
|
else input_string
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def truncate(
|
||||||
|
input_string: str, length: int = 30, ellipsis: str = "…", endpart: str = ""
|
||||||
|
) -> str:
|
||||||
|
max_content_length = length - len(endpart)
|
||||||
|
if max_content_length < 0:
|
||||||
|
raise ValueError("Length cannot be shorter than the length of endpart.")
|
||||||
|
|
||||||
|
if len(input_string) > max_content_length:
|
||||||
|
return f"{input_string[: max_content_length - len(ellipsis)].rstrip()}{ellipsis}{endpart}"
|
||||||
|
|
||||||
|
return (
|
||||||
|
f"{input_string}{endpart}"
|
||||||
|
if len(input_string) + len(endpart) <= length
|
||||||
|
else f"{input_string[: length - len(ellipsis) - len(endpart)].rstrip()}{ellipsis}{endpart}"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
T = TypeVar("T", str, int, date)
|
||||||
|
|
||||||
|
|
||||||
|
def generate_split_ranges(
|
||||||
|
value_list: list[T], split_points: list[T]
|
||||||
|
) -> Generator[tuple[T, T], None, None]:
|
||||||
|
for x in range(0, len(split_points) + 1):
|
||||||
|
if x == 0:
|
||||||
|
start = 0
|
||||||
|
elif x >= len(split_points):
|
||||||
|
start = value_list.index(split_points[x - 1]) + 1
|
||||||
|
else:
|
||||||
|
start = value_list.index(split_points[x - 1]) + 1
|
||||||
|
try:
|
||||||
|
end = value_list.index(split_points[x])
|
||||||
|
except IndexError:
|
||||||
|
end = len(value_list)
|
||||||
|
yield (value_list[start], value_list[end - 1])
|
||||||
|
|
||||||
|
|
||||||
|
def format_float_or_int(number: int | float):
|
||||||
|
return int(number) if float(number).is_integer() else f"{number:03.2f}"
|
||||||
|
|
||||||
|
|
||||||
|
OperatorType = Literal["|", "&"]
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class FilterEntry:
|
||||||
|
condition: Q
|
||||||
|
operator: OperatorType = "&"
|
||||||
|
|
||||||
|
|
||||||
|
def build_dynamic_filter(
|
||||||
|
filters: list[FilterEntry | Q], default_operator: OperatorType = "&"
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Constructs a Django Q filter from a list of filter conditions.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
filters (list): A list where each item is either:
|
||||||
|
- A Q object (default AND logic applied)
|
||||||
|
- A tuple of (Q object, operator) where operator is "|" (OR) or "&" (AND)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Q: A combined Q object that can be passed to Django's filter().
|
||||||
|
"""
|
||||||
|
op_map: dict[OperatorType, Callable[[Q, Q], Q]] = {
|
||||||
|
"|": operator.or_,
|
||||||
|
"&": operator.and_,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Convert all plain Q objects into (Q, "&") for default AND behavior
|
||||||
|
processed_filters = [
|
||||||
|
FilterEntry(f, default_operator) if isinstance(f, Q) else f for f in filters
|
||||||
|
]
|
||||||
|
|
||||||
|
# Reduce with dynamic operators
|
||||||
|
return reduce(
|
||||||
|
lambda combined_filters, filter: op_map[filter.operator](
|
||||||
|
combined_filters, filter.condition
|
||||||
|
),
|
||||||
|
processed_filters,
|
||||||
|
Q(),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def redirect_to(default_view: str, *default_args):
|
||||||
|
"""
|
||||||
|
A decorator that redirects the user back to the referring page or a default view if no 'next' parameter is provided.
|
||||||
|
|
||||||
|
:param default_view: The name of the default view to redirect to if 'next' is missing.
|
||||||
|
:param default_args: Any arguments required for the default view.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def decorator(view_func):
|
||||||
|
@wraps(view_func)
|
||||||
|
def wrapped_view(request: HttpRequest, *args, **kwargs):
|
||||||
|
next_url = request.GET.get("next")
|
||||||
|
if not next_url:
|
||||||
|
from django.urls import (
|
||||||
|
reverse, # Import inside function to avoid circular imports
|
||||||
|
)
|
||||||
|
|
||||||
|
next_url = reverse(default_view, args=default_args)
|
||||||
|
|
||||||
|
response = view_func(
|
||||||
|
request, *args, **kwargs
|
||||||
|
) # Execute the original view logic
|
||||||
|
return redirect(next_url)
|
||||||
|
|
||||||
|
return wrapped_view
|
||||||
|
|
||||||
|
return decorator
|
||||||
|
|
||||||
|
|
||||||
|
def add_next_param_to_url(url: str, nexturl: str) -> str:
|
||||||
|
return f"{url}?{urlencode({'next': nexturl})}"
|
||||||
|
33
contrib/scripts/get_exchange_rates_since_1990.py
Normal file
33
contrib/scripts/get_exchange_rates_since_1990.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
import requests
|
||||||
|
|
||||||
|
url = "https://data.kurzy.cz/json/meny/b[6]den[{0}].json"
|
||||||
|
date_format = "%Y%m%d"
|
||||||
|
years = range(2000, datetime.now().year + 1)
|
||||||
|
dates = [
|
||||||
|
datetime.strftime(datetime(day=1, month=1, year=year), format=date_format)
|
||||||
|
for year in years
|
||||||
|
]
|
||||||
|
for date in dates:
|
||||||
|
final_url = url.format(date)
|
||||||
|
year = date[:4]
|
||||||
|
response = requests.get(final_url)
|
||||||
|
response.raise_for_status()
|
||||||
|
data = response.json()
|
||||||
|
if kurzy := data.get("kurzy"):
|
||||||
|
with open("output.yaml", mode="a") as o:
|
||||||
|
rates = [
|
||||||
|
f"""
|
||||||
|
- model: games.exchangerate
|
||||||
|
fields:
|
||||||
|
currency_from: {currency_name}
|
||||||
|
currency_to: CZK
|
||||||
|
year: {year}
|
||||||
|
rate: {kurzy.get(currency_name, {}).get("dev_stred", 0)}
|
||||||
|
"""
|
||||||
|
for currency_name in ["EUR", "USD", "CNY"]
|
||||||
|
if kurzy.get(currency_name)
|
||||||
|
]
|
||||||
|
o.writelines(rates)
|
||||||
|
# time.sleep(0.5)
|
65
contrib/scripts/merge_exchange_records.py
Normal file
65
contrib/scripts/merge_exchange_records.py
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
import sys
|
||||||
|
|
||||||
|
import yaml
|
||||||
|
|
||||||
|
|
||||||
|
def load_yaml(filename):
|
||||||
|
with open(filename, "r", encoding="utf-8") as file:
|
||||||
|
return yaml.safe_load(file) or []
|
||||||
|
|
||||||
|
|
||||||
|
def save_yaml(filename, data):
|
||||||
|
with open(filename, "w", encoding="utf-8") as file:
|
||||||
|
yaml.safe_dump(data, file, allow_unicode=True, default_flow_style=False)
|
||||||
|
|
||||||
|
|
||||||
|
def extract_existing_combinations(data):
|
||||||
|
return {
|
||||||
|
(
|
||||||
|
entry["fields"]["currency_from"],
|
||||||
|
entry["fields"]["currency_to"],
|
||||||
|
entry["fields"]["year"],
|
||||||
|
)
|
||||||
|
for entry in data
|
||||||
|
if entry["model"] == "games.exchangerate"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def filter_new_entries(existing_combinations, additional_files):
|
||||||
|
new_entries = []
|
||||||
|
|
||||||
|
for filename in additional_files:
|
||||||
|
data = load_yaml(filename)
|
||||||
|
for entry in data:
|
||||||
|
if entry["model"] == "games.exchangerate":
|
||||||
|
key = (
|
||||||
|
entry["fields"]["currency_from"],
|
||||||
|
entry["fields"]["currency_to"],
|
||||||
|
entry["fields"]["year"],
|
||||||
|
)
|
||||||
|
if key not in existing_combinations:
|
||||||
|
new_entries.append(entry)
|
||||||
|
|
||||||
|
return new_entries
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if len(sys.argv) < 3:
|
||||||
|
print("Usage: script.py example.yaml additions1.yaml [additions2.yaml ...]")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
example_file = sys.argv[1]
|
||||||
|
additional_files = sys.argv[2:]
|
||||||
|
output_file = "filtered_output.yaml"
|
||||||
|
|
||||||
|
existing_data = load_yaml(example_file)
|
||||||
|
existing_combinations = extract_existing_combinations(existing_data)
|
||||||
|
|
||||||
|
new_entries = filter_new_entries(existing_combinations, additional_files)
|
||||||
|
|
||||||
|
save_yaml(output_file, new_entries)
|
||||||
|
print(f"Filtered data saved to {output_file}")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
24
devcontainer.Dockerfile
Normal file
24
devcontainer.Dockerfile
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
FROM python:3.13-slim
|
||||||
|
|
||||||
|
# Set up environment
|
||||||
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
WORKDIR /workspace
|
||||||
|
|
||||||
|
# Install Poetry
|
||||||
|
RUN apt-get update && apt-get install -y \
|
||||||
|
curl \
|
||||||
|
make \
|
||||||
|
npm \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
RUN curl -sSL https://install.python-poetry.org | python3 -
|
||||||
|
ENV PATH="/root/.local/bin:$PATH"
|
||||||
|
|
||||||
|
# Copy pyproject.toml and poetry.lock for dependency installation
|
||||||
|
COPY pyproject.toml poetry.lock* ./
|
||||||
|
RUN poetry install --no-root
|
||||||
|
|
||||||
|
# Copy the rest of the application code
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
# Set up Django development server
|
||||||
|
EXPOSE 8000
|
@ -10,10 +10,14 @@ poetry run python manage.py collectstatic --clear --no-input
|
|||||||
_term() {
|
_term() {
|
||||||
echo "Caught SIGTERM signal!"
|
echo "Caught SIGTERM signal!"
|
||||||
kill -SIGTERM "$gunicorn_pid"
|
kill -SIGTERM "$gunicorn_pid"
|
||||||
|
kill -SIGTERM "$django_q_pid"
|
||||||
}
|
}
|
||||||
trap _term SIGTERM
|
trap _term SIGTERM
|
||||||
|
|
||||||
|
echo "Starting Django-Q cluster"
|
||||||
|
poetry run python manage.py qcluster & django_q_pid=$!
|
||||||
|
|
||||||
echo "Starting app"
|
echo "Starting app"
|
||||||
poetry run python -m gunicorn --bind 0.0.0.0:8001 timetracker.asgi:application -k uvicorn.workers.UvicornWorker --access-logfile - --error-logfile - & gunicorn_pid=$!
|
poetry run python -m gunicorn --bind 0.0.0.0:8001 timetracker.asgi:application -k uvicorn.workers.UvicornWorker --access-logfile - --error-logfile - & gunicorn_pid=$!
|
||||||
|
|
||||||
wait "$gunicorn_pid"
|
wait "$gunicorn_pid" "$django_q_pid"
|
||||||
|
@ -1,11 +1,18 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
|
|
||||||
from games.models import Device, Edition, Game, Platform, Purchase, Session
|
from games.models import (
|
||||||
|
Device,
|
||||||
|
ExchangeRate,
|
||||||
|
Game,
|
||||||
|
Platform,
|
||||||
|
Purchase,
|
||||||
|
Session,
|
||||||
|
)
|
||||||
|
|
||||||
# Register your models here.
|
# Register your models here.
|
||||||
admin.site.register(Game)
|
admin.site.register(Game)
|
||||||
admin.site.register(Purchase)
|
admin.site.register(Purchase)
|
||||||
admin.site.register(Platform)
|
admin.site.register(Platform)
|
||||||
admin.site.register(Session)
|
admin.site.register(Session)
|
||||||
admin.site.register(Edition)
|
|
||||||
admin.site.register(Device)
|
admin.site.register(Device)
|
||||||
|
admin.site.register(ExchangeRate)
|
||||||
|
80
games/api.py
Normal file
80
games/api.py
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
from datetime import date, datetime
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django.utils.timezone import now as django_timezone_now
|
||||||
|
from ninja import Field, ModelSchema, NinjaAPI, Router, Schema
|
||||||
|
|
||||||
|
from games.models import PlayEvent
|
||||||
|
|
||||||
|
api = NinjaAPI()
|
||||||
|
playevent_router = Router()
|
||||||
|
|
||||||
|
NOW_FACTORY = django_timezone_now
|
||||||
|
|
||||||
|
|
||||||
|
class PlayEventIn(Schema):
|
||||||
|
game_id: int
|
||||||
|
started: date | None = None
|
||||||
|
ended: date | None = None
|
||||||
|
note: str = ""
|
||||||
|
days_to_finish: int | None = None
|
||||||
|
|
||||||
|
|
||||||
|
class AutoPlayEventIn(ModelSchema):
|
||||||
|
class Meta:
|
||||||
|
model = PlayEvent
|
||||||
|
fields = ["game", "started", "ended", "note"]
|
||||||
|
|
||||||
|
|
||||||
|
class UpdatePlayEventIn(Schema):
|
||||||
|
started: date | None = None
|
||||||
|
ended: date | None = None
|
||||||
|
note: str = ""
|
||||||
|
|
||||||
|
|
||||||
|
class PlayEventOut(Schema):
|
||||||
|
id: int
|
||||||
|
game: str = Field(..., alias="game.name")
|
||||||
|
started: date | None = None
|
||||||
|
ended: date | None = None
|
||||||
|
days_to_finish: int | None = None
|
||||||
|
note: str = ""
|
||||||
|
updated_at: datetime
|
||||||
|
created_at: datetime
|
||||||
|
|
||||||
|
|
||||||
|
@playevent_router.get("/", response=List[PlayEventOut])
|
||||||
|
def list_playevents(request):
|
||||||
|
return PlayEvent.objects.all()
|
||||||
|
|
||||||
|
|
||||||
|
@playevent_router.post("/", response={201: PlayEventOut})
|
||||||
|
def create_playevent(request, payload: PlayEventIn):
|
||||||
|
playevent = PlayEvent.objects.create(**payload.dict())
|
||||||
|
return playevent
|
||||||
|
|
||||||
|
|
||||||
|
@playevent_router.get("/{playevent_id}", response=PlayEventOut)
|
||||||
|
def get_playevent(request, playevent_id: int):
|
||||||
|
playevent = get_object_or_404(PlayEvent, id=playevent_id)
|
||||||
|
return playevent
|
||||||
|
|
||||||
|
|
||||||
|
@playevent_router.patch("/{playevent_id}", response=PlayEventOut)
|
||||||
|
def partial_update_playevent(request, playevent_id: int, payload: UpdatePlayEventIn):
|
||||||
|
playevent = get_object_or_404(PlayEvent, id=playevent_id)
|
||||||
|
for attr, value in payload.dict(exclude_unset=True).items():
|
||||||
|
setattr(playevent, attr, value)
|
||||||
|
playevent.save()
|
||||||
|
return playevent
|
||||||
|
|
||||||
|
|
||||||
|
@playevent_router.delete("/{playevent_id}", response={204: None})
|
||||||
|
def delete_playevent(request, playevent_id: int):
|
||||||
|
playevent = get_object_or_404(PlayEvent, id=playevent_id)
|
||||||
|
playevent.delete()
|
||||||
|
return 204, None
|
||||||
|
|
||||||
|
|
||||||
|
api.add_router("/playevent", playevent_router)
|
@ -1,6 +1,46 @@
|
|||||||
|
# from datetime import timedelta
|
||||||
|
|
||||||
from django.apps import AppConfig
|
from django.apps import AppConfig
|
||||||
|
from django.core.management import call_command
|
||||||
|
from django.db.models.signals import post_migrate
|
||||||
|
|
||||||
|
# from django.utils.timezone import now
|
||||||
|
|
||||||
|
|
||||||
class GamesConfig(AppConfig):
|
class GamesConfig(AppConfig):
|
||||||
default_auto_field = "django.db.models.BigAutoField"
|
default_auto_field = "django.db.models.BigAutoField"
|
||||||
name = "games"
|
name = "games"
|
||||||
|
|
||||||
|
def ready(self):
|
||||||
|
import games.signals # noqa: F401
|
||||||
|
|
||||||
|
post_migrate.connect(schedule_tasks, sender=self)
|
||||||
|
|
||||||
|
|
||||||
|
def schedule_tasks(sender, **kwargs):
|
||||||
|
# from django_q.models import Schedule
|
||||||
|
# from django_q.tasks import schedule
|
||||||
|
|
||||||
|
# if not Schedule.objects.filter(name="Update converted prices").exists():
|
||||||
|
# schedule(
|
||||||
|
# "games.tasks.convert_prices",
|
||||||
|
# name="Update converted prices",
|
||||||
|
# schedule_type=Schedule.MINUTES,
|
||||||
|
# next_run=now() + timedelta(seconds=30),
|
||||||
|
# catchup=False,
|
||||||
|
# )
|
||||||
|
|
||||||
|
# if not Schedule.objects.filter(name="Update price per game").exists():
|
||||||
|
# schedule(
|
||||||
|
# "games.tasks.calculate_price_per_game",
|
||||||
|
# name="Update price per game",
|
||||||
|
# schedule_type=Schedule.MINUTES,
|
||||||
|
# next_run=now() + timedelta(seconds=30),
|
||||||
|
# catchup=False,
|
||||||
|
# )
|
||||||
|
|
||||||
|
from games.models import ExchangeRate
|
||||||
|
|
||||||
|
if not ExchangeRate.objects.exists():
|
||||||
|
print("ExchangeRate table is empty. Loading fixture...")
|
||||||
|
call_command("loaddata", "exchangerates.yaml")
|
||||||
|
504
games/fixtures/exchangerates.yaml
Normal file
504
games/fixtures/exchangerates.yaml
Normal file
@ -0,0 +1,504 @@
|
|||||||
|
- model: games.exchangerate
|
||||||
|
pk: 1
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2024
|
||||||
|
rate: 23.4
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 2
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2024
|
||||||
|
rate: 3.267
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 3
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2019
|
||||||
|
rate: 22.466
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 4
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2023
|
||||||
|
rate: 22.63
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 5
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2017
|
||||||
|
rate: 25.819
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 6
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2013
|
||||||
|
rate: 19.023
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 7
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2019
|
||||||
|
rate: 3.295
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 8
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2016
|
||||||
|
rate: 3.795
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 9
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2015
|
||||||
|
rate: 3.707
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 10
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2020
|
||||||
|
rate: 3.26
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 11
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2012
|
||||||
|
rate: 25.51
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 12
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2010
|
||||||
|
rate: 26.465
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 13
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2014
|
||||||
|
rate: 27.52
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 14
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2024
|
||||||
|
rate: 25.21
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 15
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2022
|
||||||
|
rate: 24.325
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 16
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2018
|
||||||
|
rate: 3.268
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 17
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2023
|
||||||
|
rate: 3.281
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 18
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2009
|
||||||
|
rate: 26.445
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 19
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2025
|
||||||
|
rate: 3.35
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 20
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2016
|
||||||
|
rate: 27.033
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 21
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2025
|
||||||
|
rate: 25.2021966
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 22
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2017
|
||||||
|
rate: 26.33
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 23
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2000
|
||||||
|
rate: 36.13
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 24
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2000
|
||||||
|
rate: 35.979
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 25
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2001
|
||||||
|
rate: 35.09
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 26
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2001
|
||||||
|
rate: 37.813
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 27
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2002
|
||||||
|
rate: 31.98
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 28
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2002
|
||||||
|
rate: 36.259
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 29
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2003
|
||||||
|
rate: 31.6
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 30
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2003
|
||||||
|
rate: 30.141
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 31
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2004
|
||||||
|
rate: 32.405
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 32
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2004
|
||||||
|
rate: 25.654
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 33
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2005
|
||||||
|
rate: 30.465
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 34
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2005
|
||||||
|
rate: 22.365
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 35
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2006
|
||||||
|
rate: 29.005
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 36
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2006
|
||||||
|
rate: 24.588
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 37
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2006
|
||||||
|
rate: 3.047
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 38
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2007
|
||||||
|
rate: 27.495
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 39
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2007
|
||||||
|
rate: 20.876
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 40
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2007
|
||||||
|
rate: 2.674
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 41
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2008
|
||||||
|
rate: 26.62
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 42
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2008
|
||||||
|
rate: 18.078
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 43
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2008
|
||||||
|
rate: 2.475
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 44
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2009
|
||||||
|
rate: 19.346
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 45
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2009
|
||||||
|
rate: 2.836
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 46
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2010
|
||||||
|
rate: 18.368
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 47
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2010
|
||||||
|
rate: 2.691
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 48
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2011
|
||||||
|
rate: 25.06
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 49
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2011
|
||||||
|
rate: 18.751
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 50
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2011
|
||||||
|
rate: 2.845
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 51
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2012
|
||||||
|
rate: 19.94
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 52
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2012
|
||||||
|
rate: 3.168
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 53
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2013
|
||||||
|
rate: 25.14
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 54
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2013
|
||||||
|
rate: 3.059
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 55
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2014
|
||||||
|
rate: 19.894
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 56
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2014
|
||||||
|
rate: 3.286
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 57
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2015
|
||||||
|
rate: 27.725
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 58
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2015
|
||||||
|
rate: 22.834
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 59
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2016
|
||||||
|
rate: 24.824
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 60
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2017
|
||||||
|
rate: 3.693
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 61
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2018
|
||||||
|
rate: 25.54
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 62
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2018
|
||||||
|
rate: 21.291
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 63
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2019
|
||||||
|
rate: 25.725
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 64
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2020
|
||||||
|
rate: 25.41
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 65
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2020
|
||||||
|
rate: 22.621
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 66
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2021
|
||||||
|
rate: 26.245
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 67
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2021
|
||||||
|
rate: 21.387
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 68
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2021
|
||||||
|
rate: 3.273
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 69
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2022
|
||||||
|
rate: 21.951
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 70
|
||||||
|
fields:
|
||||||
|
currency_from: CNY
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2022
|
||||||
|
rate: 3.458
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 71
|
||||||
|
fields:
|
||||||
|
currency_from: EUR
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2023
|
||||||
|
rate: 24.115
|
||||||
|
- model: games.exchangerate
|
||||||
|
pk: 72
|
||||||
|
fields:
|
||||||
|
currency_from: USD
|
||||||
|
currency_to: CZK
|
||||||
|
year: 2025
|
||||||
|
rate: 24.237
|
174
games/forms.py
174
games/forms.py
@ -1,7 +1,17 @@
|
|||||||
from django import forms
|
from django import forms
|
||||||
|
from django.db import transaction
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
|
||||||
from games.models import Device, Edition, Game, Platform, Purchase, Session
|
from common.utils import safe_getattr
|
||||||
|
from games.models import (
|
||||||
|
Device,
|
||||||
|
Game,
|
||||||
|
GameStatusChange,
|
||||||
|
Platform,
|
||||||
|
PlayEvent,
|
||||||
|
Purchase,
|
||||||
|
Session,
|
||||||
|
)
|
||||||
|
|
||||||
custom_date_widget = forms.DateInput(attrs={"type": "date"})
|
custom_date_widget = forms.DateInput(attrs={"type": "date"})
|
||||||
custom_datetime_widget = forms.DateTimeInput(
|
custom_datetime_widget = forms.DateTimeInput(
|
||||||
@ -10,17 +20,37 @@ custom_datetime_widget = forms.DateTimeInput(
|
|||||||
autofocus_input_widget = forms.TextInput(attrs={"autofocus": "autofocus"})
|
autofocus_input_widget = forms.TextInput(attrs={"autofocus": "autofocus"})
|
||||||
|
|
||||||
|
|
||||||
|
class MultipleGameChoiceField(forms.ModelMultipleChoiceField):
|
||||||
|
def label_from_instance(self, obj) -> str:
|
||||||
|
return f"{obj.sort_name} ({obj.platform}, {obj.year_released})"
|
||||||
|
|
||||||
|
|
||||||
|
class SingleGameChoiceField(forms.ModelChoiceField):
|
||||||
|
def label_from_instance(self, obj) -> str:
|
||||||
|
return f"{obj.sort_name} ({obj.platform}, {obj.year_released})"
|
||||||
|
|
||||||
|
|
||||||
class SessionForm(forms.ModelForm):
|
class SessionForm(forms.ModelForm):
|
||||||
# purchase = forms.ModelChoiceField(
|
game = SingleGameChoiceField(
|
||||||
# queryset=Purchase.objects.filter(date_refunded=None).order_by("edition__name")
|
queryset=Game.objects.order_by("sort_name"),
|
||||||
# )
|
|
||||||
purchase = forms.ModelChoiceField(
|
|
||||||
queryset=Purchase.objects.order_by("edition__sort_name"),
|
|
||||||
widget=forms.Select(attrs={"autofocus": "autofocus"}),
|
widget=forms.Select(attrs={"autofocus": "autofocus"}),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
duration_manual = forms.DurationField(
|
||||||
|
required=False,
|
||||||
|
widget=forms.TextInput(
|
||||||
|
attrs={"x-mask": "99:99:99", "placeholder": "HH:MM:SS", "x-data": ""}
|
||||||
|
),
|
||||||
|
label="Manual duration",
|
||||||
|
)
|
||||||
device = forms.ModelChoiceField(queryset=Device.objects.order_by("name"))
|
device = forms.ModelChoiceField(queryset=Device.objects.order_by("name"))
|
||||||
|
|
||||||
|
mark_as_played = forms.BooleanField(
|
||||||
|
required=False,
|
||||||
|
initial={"mark_as_played": True},
|
||||||
|
label="Set game status to Played if Unplayed",
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
widgets = {
|
widgets = {
|
||||||
"timestamp_start": custom_datetime_widget,
|
"timestamp_start": custom_datetime_widget,
|
||||||
@ -28,25 +58,34 @@ class SessionForm(forms.ModelForm):
|
|||||||
}
|
}
|
||||||
model = Session
|
model = Session
|
||||||
fields = [
|
fields = [
|
||||||
"purchase",
|
"game",
|
||||||
"timestamp_start",
|
"timestamp_start",
|
||||||
"timestamp_end",
|
"timestamp_end",
|
||||||
"duration_manual",
|
"duration_manual",
|
||||||
|
"emulated",
|
||||||
"device",
|
"device",
|
||||||
"note",
|
"note",
|
||||||
|
"mark_as_played",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def save(self, commit=True):
|
||||||
class EditionChoiceField(forms.ModelChoiceField):
|
session = super().save(commit=False)
|
||||||
def label_from_instance(self, obj) -> str:
|
if self.cleaned_data.get("mark_as_played"):
|
||||||
return f"{obj.sort_name} ({obj.platform}, {obj.year_released})"
|
game_instance = session.game
|
||||||
|
if game_instance.status == "u":
|
||||||
|
game_instance.status = "p"
|
||||||
|
if commit:
|
||||||
|
game_instance.save()
|
||||||
|
if commit:
|
||||||
|
session.save()
|
||||||
|
return session
|
||||||
|
|
||||||
|
|
||||||
class IncludePlatformSelect(forms.Select):
|
class IncludePlatformSelect(forms.SelectMultiple):
|
||||||
def create_option(self, name, value, *args, **kwargs):
|
def create_option(self, name, value, *args, **kwargs):
|
||||||
option = super().create_option(name, value, *args, **kwargs)
|
option = super().create_option(name, value, *args, **kwargs)
|
||||||
if value:
|
if platform_id := safe_getattr(value, "instance.platform.id"):
|
||||||
option["attrs"]["data-platform"] = value.instance.platform.id
|
option["attrs"]["data-platform"] = platform_id
|
||||||
return option
|
return option
|
||||||
|
|
||||||
|
|
||||||
@ -55,42 +94,51 @@ class PurchaseForm(forms.ModelForm):
|
|||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
|
|
||||||
# Automatically update related_purchase <select/>
|
# Automatically update related_purchase <select/>
|
||||||
# to only include purchases of the selected edition.
|
# to only include purchases of the selected game.
|
||||||
related_purchase_by_edition_url = reverse("related_purchase_by_edition")
|
related_purchase_by_game_url = reverse("related_purchase_by_game")
|
||||||
self.fields["edition"].widget.attrs.update(
|
self.fields["games"].widget.attrs.update(
|
||||||
{
|
{
|
||||||
"hx-trigger": "load, click",
|
"hx-trigger": "load, click",
|
||||||
"hx-get": related_purchase_by_edition_url,
|
"hx-get": related_purchase_by_game_url,
|
||||||
"hx-target": "#id_related_purchase",
|
"hx-target": "#id_related_purchase",
|
||||||
"hx-swap": "outerHTML",
|
"hx-swap": "outerHTML",
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
edition = EditionChoiceField(
|
games = MultipleGameChoiceField(
|
||||||
queryset=Edition.objects.order_by("sort_name"),
|
queryset=Game.objects.order_by("sort_name"),
|
||||||
widget=IncludePlatformSelect(attrs={"autoselect": "autoselect"}),
|
widget=IncludePlatformSelect(attrs={"autoselect": "autoselect"}),
|
||||||
)
|
)
|
||||||
platform = forms.ModelChoiceField(queryset=Platform.objects.order_by("name"))
|
platform = forms.ModelChoiceField(queryset=Platform.objects.order_by("name"))
|
||||||
related_purchase = forms.ModelChoiceField(
|
related_purchase = forms.ModelChoiceField(
|
||||||
queryset=Purchase.objects.filter(type=Purchase.GAME).order_by(
|
queryset=Purchase.objects.filter(type=Purchase.GAME),
|
||||||
"edition__sort_name"
|
|
||||||
),
|
|
||||||
required=False,
|
required=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
price_currency = forms.CharField(
|
||||||
|
widget=forms.TextInput(
|
||||||
|
attrs={
|
||||||
|
"x-mask": "aaa",
|
||||||
|
"placeholder": "CZK",
|
||||||
|
"x-data": "",
|
||||||
|
"class": "uppercase",
|
||||||
|
}
|
||||||
|
),
|
||||||
|
label="Currency",
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
widgets = {
|
widgets = {
|
||||||
"date_purchased": custom_date_widget,
|
"date_purchased": custom_date_widget,
|
||||||
"date_refunded": custom_date_widget,
|
"date_refunded": custom_date_widget,
|
||||||
"date_finished": custom_date_widget,
|
|
||||||
}
|
}
|
||||||
model = Purchase
|
model = Purchase
|
||||||
fields = [
|
fields = [
|
||||||
"edition",
|
"games",
|
||||||
"platform",
|
"platform",
|
||||||
"date_purchased",
|
"date_purchased",
|
||||||
"date_refunded",
|
"date_refunded",
|
||||||
"date_finished",
|
"infinite",
|
||||||
"price",
|
"price",
|
||||||
"price_currency",
|
"price_currency",
|
||||||
"ownership_type",
|
"ownership_type",
|
||||||
@ -136,31 +184,34 @@ class GameModelChoiceField(forms.ModelChoiceField):
|
|||||||
return obj.sort_name
|
return obj.sort_name
|
||||||
|
|
||||||
|
|
||||||
class EditionForm(forms.ModelForm):
|
class GameForm(forms.ModelForm):
|
||||||
game = GameModelChoiceField(
|
|
||||||
queryset=Game.objects.order_by("sort_name"),
|
|
||||||
widget=IncludeNameSelect(attrs={"autofocus": "autofocus"}),
|
|
||||||
)
|
|
||||||
platform = forms.ModelChoiceField(
|
platform = forms.ModelChoiceField(
|
||||||
queryset=Platform.objects.order_by("name"), required=False
|
queryset=Platform.objects.order_by("name"), required=False
|
||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = Edition
|
|
||||||
fields = ["game", "name", "sort_name", "platform", "year_released", "wikidata"]
|
|
||||||
|
|
||||||
|
|
||||||
class GameForm(forms.ModelForm):
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Game
|
model = Game
|
||||||
fields = ["name", "sort_name", "year_released", "wikidata"]
|
fields = [
|
||||||
|
"name",
|
||||||
|
"sort_name",
|
||||||
|
"platform",
|
||||||
|
"original_year_released",
|
||||||
|
"year_released",
|
||||||
|
"status",
|
||||||
|
"mastered",
|
||||||
|
"wikidata",
|
||||||
|
]
|
||||||
widgets = {"name": autofocus_input_widget}
|
widgets = {"name": autofocus_input_widget}
|
||||||
|
|
||||||
|
|
||||||
class PlatformForm(forms.ModelForm):
|
class PlatformForm(forms.ModelForm):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Platform
|
model = Platform
|
||||||
fields = ["name", "group"]
|
fields = [
|
||||||
|
"name",
|
||||||
|
"icon",
|
||||||
|
"group",
|
||||||
|
]
|
||||||
widgets = {"name": autofocus_input_widget}
|
widgets = {"name": autofocus_input_widget}
|
||||||
|
|
||||||
|
|
||||||
@ -169,3 +220,48 @@ class DeviceForm(forms.ModelForm):
|
|||||||
model = Device
|
model = Device
|
||||||
fields = ["name", "type"]
|
fields = ["name", "type"]
|
||||||
widgets = {"name": autofocus_input_widget}
|
widgets = {"name": autofocus_input_widget}
|
||||||
|
|
||||||
|
|
||||||
|
class PlayEventForm(forms.ModelForm):
|
||||||
|
game = GameModelChoiceField(
|
||||||
|
queryset=Game.objects.order_by("sort_name"),
|
||||||
|
widget=forms.Select(attrs={"autofocus": "autofocus"}),
|
||||||
|
)
|
||||||
|
|
||||||
|
mark_as_finished = forms.BooleanField(
|
||||||
|
required=False,
|
||||||
|
initial={"mark_as_finished": True},
|
||||||
|
label="Set game status to Finished",
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = PlayEvent
|
||||||
|
fields = ["game", "started", "ended", "note", "mark_as_finished"]
|
||||||
|
widgets = {
|
||||||
|
"started": custom_date_widget,
|
||||||
|
"ended": custom_date_widget,
|
||||||
|
}
|
||||||
|
|
||||||
|
def save(self, commit=True):
|
||||||
|
with transaction.atomic():
|
||||||
|
session = super().save(commit=False)
|
||||||
|
if self.cleaned_data.get("mark_as_finished"):
|
||||||
|
game_instance = session.game
|
||||||
|
game_instance.status = "f"
|
||||||
|
game_instance.save()
|
||||||
|
session.save()
|
||||||
|
return session
|
||||||
|
|
||||||
|
|
||||||
|
class GameStatusChangeForm(forms.ModelForm):
|
||||||
|
class Meta:
|
||||||
|
model = GameStatusChange
|
||||||
|
fields = [
|
||||||
|
"game",
|
||||||
|
"old_status",
|
||||||
|
"new_status",
|
||||||
|
"timestamp",
|
||||||
|
]
|
||||||
|
widgets = {
|
||||||
|
"timestamp": custom_datetime_widget,
|
||||||
|
}
|
||||||
|
1
games/graphql/mutations/__init__.py
Normal file
1
games/graphql/mutations/__init__.py
Normal file
@ -0,0 +1 @@
|
|||||||
|
from .game import Mutation as GameMutation
|
29
games/graphql/mutations/game.py
Normal file
29
games/graphql/mutations/game.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Game
|
||||||
|
from games.models import Game as GameModel
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateGameMutation(graphene.Mutation):
|
||||||
|
class Arguments:
|
||||||
|
id = graphene.ID(required=True)
|
||||||
|
name = graphene.String()
|
||||||
|
year_released = graphene.Int()
|
||||||
|
wikidata = graphene.String()
|
||||||
|
|
||||||
|
game = graphene.Field(Game)
|
||||||
|
|
||||||
|
def mutate(self, info, id, name=None, year_released=None, wikidata=None):
|
||||||
|
game_instance = GameModel.objects.get(pk=id)
|
||||||
|
if name is not None:
|
||||||
|
game_instance.name = name
|
||||||
|
if year_released is not None:
|
||||||
|
game_instance.year_released = year_released
|
||||||
|
if wikidata is not None:
|
||||||
|
game_instance.wikidata = wikidata
|
||||||
|
game_instance.save()
|
||||||
|
return UpdateGameMutation(game=game_instance)
|
||||||
|
|
||||||
|
|
||||||
|
class Mutation(graphene.ObjectType):
|
||||||
|
update_game = UpdateGameMutation.Field()
|
5
games/graphql/queries/__init__.py
Normal file
5
games/graphql/queries/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
from .device import Query as DeviceQuery
|
||||||
|
from .game import Query as GameQuery
|
||||||
|
from .platform import Query as PlatformQuery
|
||||||
|
from .purchase import Query as PurchaseQuery
|
||||||
|
from .session import Query as SessionQuery
|
11
games/graphql/queries/device.py
Normal file
11
games/graphql/queries/device.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Device
|
||||||
|
from games.models import Device as DeviceModel
|
||||||
|
|
||||||
|
|
||||||
|
class Query(graphene.ObjectType):
|
||||||
|
devices = graphene.List(Device)
|
||||||
|
|
||||||
|
def resolve_devices(self, info, **kwargs):
|
||||||
|
return DeviceModel.objects.all()
|
18
games/graphql/queries/game.py
Normal file
18
games/graphql/queries/game.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Game
|
||||||
|
from games.models import Game as GameModel
|
||||||
|
|
||||||
|
|
||||||
|
class Query(graphene.ObjectType):
|
||||||
|
games = graphene.List(Game)
|
||||||
|
game_by_name = graphene.Field(Game, name=graphene.String(required=True))
|
||||||
|
|
||||||
|
def resolve_games(self, info, **kwargs):
|
||||||
|
return GameModel.objects.all()
|
||||||
|
|
||||||
|
def resolve_game_by_name(self, info, name):
|
||||||
|
try:
|
||||||
|
return GameModel.objects.get(name=name)
|
||||||
|
except GameModel.DoesNotExist:
|
||||||
|
return None
|
11
games/graphql/queries/platform.py
Normal file
11
games/graphql/queries/platform.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Platform
|
||||||
|
from games.models import Platform as PlatformModel
|
||||||
|
|
||||||
|
|
||||||
|
class Query(graphene.ObjectType):
|
||||||
|
platforms = graphene.List(Platform)
|
||||||
|
|
||||||
|
def resolve_platforms(self, info, **kwargs):
|
||||||
|
return PlatformModel.objects.all()
|
11
games/graphql/queries/purchase.py
Normal file
11
games/graphql/queries/purchase.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Purchase
|
||||||
|
from games.models import Purchase as PurchaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class Query(graphene.ObjectType):
|
||||||
|
purchases = graphene.List(Purchase)
|
||||||
|
|
||||||
|
def resolve_purchases(self, info, **kwargs):
|
||||||
|
return PurchaseModel.objects.all()
|
11
games/graphql/queries/session.py
Normal file
11
games/graphql/queries/session.py
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.types import Session
|
||||||
|
from games.models import Session as SessionModel
|
||||||
|
|
||||||
|
|
||||||
|
class Query(graphene.ObjectType):
|
||||||
|
sessions = graphene.List(Session)
|
||||||
|
|
||||||
|
def resolve_sessions(self, info, **kwargs):
|
||||||
|
return SessionModel.objects.all()
|
44
games/graphql/types.py
Normal file
44
games/graphql/types.py
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
from graphene_django import DjangoObjectType
|
||||||
|
|
||||||
|
from games.models import Device as DeviceModel
|
||||||
|
from games.models import Edition as EditionModel
|
||||||
|
from games.models import Game as GameModel
|
||||||
|
from games.models import Platform as PlatformModel
|
||||||
|
from games.models import Purchase as PurchaseModel
|
||||||
|
from games.models import Session as SessionModel
|
||||||
|
|
||||||
|
|
||||||
|
class Game(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = GameModel
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class Edition(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = EditionModel
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class Purchase(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = PurchaseModel
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class Session(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = SessionModel
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class Platform(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = PlatformModel
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class Device(DjangoObjectType):
|
||||||
|
class Meta:
|
||||||
|
model = DeviceModel
|
||||||
|
fields = "__all__"
|
24
games/management/commands/schedule_convert_prices.py
Normal file
24
games/management/commands/schedule_convert_prices.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from django.core.management.base import BaseCommand
|
||||||
|
from django.utils.timezone import now
|
||||||
|
from django_q.models import Schedule
|
||||||
|
from django_q.tasks import schedule
|
||||||
|
|
||||||
|
|
||||||
|
class Command(BaseCommand):
|
||||||
|
help = "Manually schedule the next update_converted_prices task"
|
||||||
|
|
||||||
|
def handle(self, *args, **kwargs):
|
||||||
|
if not Schedule.objects.filter(name="Update converted prices").exists():
|
||||||
|
schedule(
|
||||||
|
"games.tasks.convert_prices",
|
||||||
|
name="Update converted prices",
|
||||||
|
schedule_type=Schedule.MINUTES,
|
||||||
|
next_run=now() + timedelta(seconds=30),
|
||||||
|
)
|
||||||
|
self.stdout.write(
|
||||||
|
self.style.SUCCESS("Scheduled the update_converted_prices task.")
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
self.stdout.write(self.style.WARNING("Task is already scheduled."))
|
@ -1,5 +1,6 @@
|
|||||||
# Generated by Django 4.1.4 on 2023-01-02 18:27
|
# Generated by Django 5.1.5 on 2025-01-29 21:26
|
||||||
|
|
||||||
|
import datetime
|
||||||
import django.db.models.deletion
|
import django.db.models.deletion
|
||||||
from django.db import migrations, models
|
from django.db import migrations, models
|
||||||
|
|
||||||
@ -8,94 +9,96 @@ class Migration(migrations.Migration):
|
|||||||
|
|
||||||
initial = True
|
initial = True
|
||||||
|
|
||||||
dependencies = []
|
dependencies = [
|
||||||
|
]
|
||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Game",
|
name='Device',
|
||||||
fields=[
|
fields=[
|
||||||
(
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
"id",
|
('name', models.CharField(max_length=255)),
|
||||||
models.BigAutoField(
|
('type', models.CharField(choices=[('PC', 'PC'), ('Console', 'Console'), ('Handheld', 'Handheld'), ('Mobile', 'Mobile'), ('Single-board computer', 'Single-board computer'), ('Unknown', 'Unknown')], default='Unknown', max_length=255)),
|
||||||
auto_created=True,
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
primary_key=True,
|
|
||||||
serialize=False,
|
|
||||||
verbose_name="ID",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
("name", models.CharField(max_length=255)),
|
|
||||||
("wikidata", models.CharField(max_length=50)),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Platform",
|
name='Platform',
|
||||||
fields=[
|
fields=[
|
||||||
(
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
"id",
|
('name', models.CharField(max_length=255)),
|
||||||
models.BigAutoField(
|
('group', models.CharField(blank=True, default=None, max_length=255, null=True)),
|
||||||
auto_created=True,
|
('icon', models.SlugField(blank=True)),
|
||||||
primary_key=True,
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
serialize=False,
|
|
||||||
verbose_name="ID",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
("name", models.CharField(max_length=255)),
|
|
||||||
("group", models.CharField(max_length=255)),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Purchase",
|
name='ExchangeRate',
|
||||||
fields=[
|
fields=[
|
||||||
(
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
"id",
|
('currency_from', models.CharField(max_length=255)),
|
||||||
models.BigAutoField(
|
('currency_to', models.CharField(max_length=255)),
|
||||||
auto_created=True,
|
('year', models.PositiveIntegerField()),
|
||||||
primary_key=True,
|
('rate', models.FloatField()),
|
||||||
serialize=False,
|
],
|
||||||
verbose_name="ID",
|
options={
|
||||||
),
|
'unique_together': {('currency_from', 'currency_to', 'year')},
|
||||||
),
|
},
|
||||||
("date_purchased", models.DateField()),
|
),
|
||||||
("date_refunded", models.DateField(blank=True, null=True)),
|
migrations.CreateModel(
|
||||||
(
|
name='Game',
|
||||||
"game",
|
fields=[
|
||||||
models.ForeignKey(
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.game"
|
('name', models.CharField(max_length=255)),
|
||||||
),
|
('sort_name', models.CharField(blank=True, default=None, max_length=255, null=True)),
|
||||||
),
|
('year_released', models.IntegerField(blank=True, default=None, null=True)),
|
||||||
(
|
('wikidata', models.CharField(blank=True, default=None, max_length=50, null=True)),
|
||||||
"platform",
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
models.ForeignKey(
|
('platform', models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_DEFAULT, to='games.platform')),
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
],
|
||||||
to="games.platform",
|
options={
|
||||||
),
|
'unique_together': {('name', 'platform', 'year_released')},
|
||||||
),
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Purchase',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('date_purchased', models.DateField()),
|
||||||
|
('date_refunded', models.DateField(blank=True, null=True)),
|
||||||
|
('date_finished', models.DateField(blank=True, null=True)),
|
||||||
|
('date_dropped', models.DateField(blank=True, null=True)),
|
||||||
|
('infinite', models.BooleanField(default=False)),
|
||||||
|
('price', models.FloatField(default=0)),
|
||||||
|
('price_currency', models.CharField(default='USD', max_length=3)),
|
||||||
|
('converted_price', models.FloatField(null=True)),
|
||||||
|
('converted_currency', models.CharField(max_length=3, null=True)),
|
||||||
|
('ownership_type', models.CharField(choices=[('ph', 'Physical'), ('di', 'Digital'), ('du', 'Digital Upgrade'), ('re', 'Rented'), ('bo', 'Borrowed'), ('tr', 'Trial'), ('de', 'Demo'), ('pi', 'Pirated')], default='di', max_length=2)),
|
||||||
|
('type', models.CharField(choices=[('game', 'Game'), ('dlc', 'DLC'), ('season_pass', 'Season Pass'), ('battle_pass', 'Battle Pass')], default='game', max_length=255)),
|
||||||
|
('name', models.CharField(blank=True, default='', max_length=255, null=True)),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('games', models.ManyToManyField(blank=True, related_name='purchases', to='games.game')),
|
||||||
|
('platform', models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.CASCADE, to='games.platform')),
|
||||||
|
('related_purchase', models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='related_purchases', to='games.purchase')),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Session",
|
name='Session',
|
||||||
fields=[
|
fields=[
|
||||||
(
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
"id",
|
('timestamp_start', models.DateTimeField()),
|
||||||
models.BigAutoField(
|
('timestamp_end', models.DateTimeField(blank=True, null=True)),
|
||||||
auto_created=True,
|
('duration_manual', models.DurationField(blank=True, default=datetime.timedelta(0), null=True)),
|
||||||
primary_key=True,
|
('duration_calculated', models.DurationField(blank=True, null=True)),
|
||||||
serialize=False,
|
('note', models.TextField(blank=True, null=True)),
|
||||||
verbose_name="ID",
|
('emulated', models.BooleanField(default=False)),
|
||||||
),
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
),
|
('modified_at', models.DateTimeField(auto_now=True)),
|
||||||
("timestamp_start", models.DateTimeField()),
|
('device', models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.SET_DEFAULT, to='games.device')),
|
||||||
("timestamp_end", models.DateTimeField()),
|
('game', models.ForeignKey(blank=True, default=None, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='sessions', to='games.game')),
|
||||||
("duration_manual", models.DurationField(blank=True, null=True)),
|
|
||||||
("duration_calculated", models.DurationField(blank=True, null=True)),
|
|
||||||
("note", models.TextField(blank=True, null=True)),
|
|
||||||
(
|
|
||||||
"purchase",
|
|
||||||
models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.purchase",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
|
options={
|
||||||
|
'get_latest_by': 'timestamp_start',
|
||||||
|
},
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
# Generated by Django 4.1.4 on 2023-01-02 18:55
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0001_initial"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="duration_manual",
|
|
||||||
field=models.DurationField(
|
|
||||||
blank=True, default=datetime.timedelta(0), null=True
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
18
games/migrations/0002_purchase_price_per_game.py
Normal file
18
games/migrations/0002_purchase_price_per_game.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-01-30 11:04
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0001_initial'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='price_per_game',
|
||||||
|
field=models.FloatField(null=True),
|
||||||
|
),
|
||||||
|
]
|
@ -1,23 +0,0 @@
|
|||||||
# Generated by Django 4.1.4 on 2023-01-02 23:11
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0002_alter_session_duration_manual"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="duration_manual",
|
|
||||||
field=models.DurationField(blank=True, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="timestamp_end",
|
|
||||||
field=models.DateTimeField(blank=True, null=True),
|
|
||||||
),
|
|
||||||
]
|
|
18
games/migrations/0003_purchase_updated_at.py
Normal file
18
games/migrations/0003_purchase_updated_at.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-01-30 11:12
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0002_purchase_price_per_game'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='updated_at',
|
||||||
|
field=models.DateTimeField(auto_now=True),
|
||||||
|
),
|
||||||
|
]
|
@ -1,22 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-01-09 14:49
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0003_alter_session_duration_manual_and_more"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="duration_manual",
|
|
||||||
field=models.DurationField(
|
|
||||||
blank=True, default=datetime.timedelta(0), null=True
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
28
games/migrations/0004_purchase_num_purchases.py
Normal file
28
games/migrations/0004_purchase_num_purchases.py
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-01-30 11:57
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
from django.db.models import Count
|
||||||
|
|
||||||
|
|
||||||
|
def initialize_num_purchases(apps, schema_editor):
|
||||||
|
Purchase = apps.get_model("games", "Purchase")
|
||||||
|
purchases = Purchase.objects.annotate(num_games=Count("games"))
|
||||||
|
|
||||||
|
for purchase in purchases:
|
||||||
|
purchase.num_purchases = purchase.num_games
|
||||||
|
purchase.save(update_fields=["num_purchases"])
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("games", "0003_purchase_updated_at"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="purchase",
|
||||||
|
name="num_purchases",
|
||||||
|
field=models.IntegerField(default=0),
|
||||||
|
),
|
||||||
|
migrations.RunPython(initialize_num_purchases),
|
||||||
|
]
|
@ -1,35 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-01-09 17:43
|
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
def set_duration_calculated_none_to_zero(apps, schema_editor):
|
|
||||||
Session = apps.get_model("games", "Session")
|
|
||||||
for session in Session.objects.all():
|
|
||||||
if session.duration_calculated == None:
|
|
||||||
session.duration_calculated = timedelta(0)
|
|
||||||
session.save()
|
|
||||||
|
|
||||||
|
|
||||||
def revert_set_duration_calculated_none_to_zero(apps, schema_editor):
|
|
||||||
Session = apps.get_model("games", "Session")
|
|
||||||
for session in Session.objects.all():
|
|
||||||
if session.duration_calculated == timedelta(0):
|
|
||||||
session.duration_calculated = None
|
|
||||||
session.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0004_alter_session_duration_manual"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RunPython(
|
|
||||||
set_duration_calculated_none_to_zero,
|
|
||||||
revert_set_duration_calculated_none_to_zero,
|
|
||||||
)
|
|
||||||
]
|
|
38
games/migrations/0005_game_mastered_game_status.py
Normal file
38
games/migrations/0005_game_mastered_game_status.py
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-02-01 19:18
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
def set_finished_status(apps, schema_editor):
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
Game.objects.filter(purchases__date_finished__isnull=False).update(status="f")
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("games", "0004_purchase_num_purchases"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="game",
|
||||||
|
name="mastered",
|
||||||
|
field=models.BooleanField(default=False),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="game",
|
||||||
|
name="status",
|
||||||
|
field=models.CharField(
|
||||||
|
choices=[
|
||||||
|
("u", "Unplayed"),
|
||||||
|
("p", "Played"),
|
||||||
|
("f", "Finished"),
|
||||||
|
("r", "Retired"),
|
||||||
|
("a", "Abandoned"),
|
||||||
|
],
|
||||||
|
default="u",
|
||||||
|
max_length=1,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.RunPython(set_finished_status),
|
||||||
|
]
|
@ -0,0 +1,59 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-03-01 12:52
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0005_game_mastered_game_status'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='game',
|
||||||
|
name='sort_name',
|
||||||
|
field=models.CharField(blank=True, default='', max_length=255),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='game',
|
||||||
|
name='wikidata',
|
||||||
|
field=models.CharField(blank=True, default='', max_length=50),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='platform',
|
||||||
|
name='group',
|
||||||
|
field=models.CharField(blank=True, default='', max_length=255),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='converted_currency',
|
||||||
|
field=models.CharField(blank=True, default='', max_length=3),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='games',
|
||||||
|
field=models.ManyToManyField(related_name='purchases', to='games.game'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='name',
|
||||||
|
field=models.CharField(blank=True, default='', max_length=255),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='related_purchase',
|
||||||
|
field=models.ForeignKey(default=None, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='related_purchases', to='games.purchase'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='session',
|
||||||
|
name='game',
|
||||||
|
field=models.ForeignKey(default=None, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='sessions', to='games.game'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='session',
|
||||||
|
name='note',
|
||||||
|
field=models.TextField(blank=True, default=''),
|
||||||
|
),
|
||||||
|
]
|
@ -1,35 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-01-09 18:04
|
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
def set_duration_manual_none_to_zero(apps, schema_editor):
|
|
||||||
Session = apps.get_model("games", "Session")
|
|
||||||
for session in Session.objects.all():
|
|
||||||
if session.duration_manual == None:
|
|
||||||
session.duration_manual = timedelta(0)
|
|
||||||
session.save()
|
|
||||||
|
|
||||||
|
|
||||||
def revert_set_duration_manual_none_to_zero(apps, schema_editor):
|
|
||||||
Session = apps.get_model("games", "Session")
|
|
||||||
for session in Session.objects.all():
|
|
||||||
if session.duration_manual == timedelta(0):
|
|
||||||
session.duration_manual = None
|
|
||||||
session.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0005_auto_20230109_1843"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RunPython(
|
|
||||||
set_duration_manual_none_to_zero,
|
|
||||||
revert_set_duration_manual_none_to_zero,
|
|
||||||
)
|
|
||||||
]
|
|
@ -1,35 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-01-19 18:30
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0006_auto_20230109_1904"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="game",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.game"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="platform",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.platform"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="purchase",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.purchase"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
18
games/migrations/0007_game_updated_at.py
Normal file
18
games/migrations/0007_game_updated_at.py
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.1.5 on 2025-03-17 07:36
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0006_alter_game_sort_name_alter_game_wikidata_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='game',
|
||||||
|
name='updated_at',
|
||||||
|
field=models.DateTimeField(auto_now=True),
|
||||||
|
),
|
||||||
|
]
|
@ -1,41 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 16:29
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0007_alter_purchase_game_alter_purchase_platform_and_more"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.CreateModel(
|
|
||||||
name="Edition",
|
|
||||||
fields=[
|
|
||||||
(
|
|
||||||
"id",
|
|
||||||
models.BigAutoField(
|
|
||||||
auto_created=True,
|
|
||||||
primary_key=True,
|
|
||||||
serialize=False,
|
|
||||||
verbose_name="ID",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
("name", models.CharField(max_length=255)),
|
|
||||||
(
|
|
||||||
"game",
|
|
||||||
models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.game"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
(
|
|
||||||
"platform",
|
|
||||||
models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.platform"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
]
|
|
@ -0,0 +1,190 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-19 13:11
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
import django.db.models.expressions
|
||||||
|
from django.db import migrations, models
|
||||||
|
from django.db.models import F, Min
|
||||||
|
|
||||||
|
|
||||||
|
def copy_year_released(apps, schema_editor):
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
Game.objects.update(original_year_released=F("year_released"))
|
||||||
|
|
||||||
|
|
||||||
|
def set_abandoned_status(apps, schema_editor):
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
PlayEvent = apps.get_model("games", "PlayEvent")
|
||||||
|
|
||||||
|
Game.objects.filter(purchases__date_refunded__isnull=False).update(status="a")
|
||||||
|
Game.objects.filter(purchases__date_dropped__isnull=False).update(status="a")
|
||||||
|
|
||||||
|
finished = Game.objects.filter(purchases__date_finished__isnull=False)
|
||||||
|
|
||||||
|
for game in finished:
|
||||||
|
for purchase in game.purchases.all():
|
||||||
|
first_session = game.sessions.filter(
|
||||||
|
timestamp_start__gte=purchase.date_purchased
|
||||||
|
).aggregate(Min("timestamp_start"))["timestamp_start__min"]
|
||||||
|
first_session_date = first_session.date() if first_session else None
|
||||||
|
if purchase.date_finished:
|
||||||
|
play_event = PlayEvent(
|
||||||
|
game=game,
|
||||||
|
started=first_session_date
|
||||||
|
if first_session_date
|
||||||
|
else purchase.date_purchased,
|
||||||
|
ended=purchase.date_finished,
|
||||||
|
)
|
||||||
|
play_event.save()
|
||||||
|
|
||||||
|
|
||||||
|
def create_game_status_changes(apps, schema_editor):
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
GameStatusChange = apps.get_model("games", "GameStatusChange")
|
||||||
|
|
||||||
|
# if game has any sessions, find the earliest session and create a status change from unplayed to played with that sessions's timestamp_start
|
||||||
|
for game in Game.objects.filter(sessions__isnull=False).distinct():
|
||||||
|
if game.sessions.exists():
|
||||||
|
earliest_session = game.sessions.earliest()
|
||||||
|
GameStatusChange.objects.create(
|
||||||
|
game=game,
|
||||||
|
old_status="u",
|
||||||
|
new_status="p",
|
||||||
|
timestamp=earliest_session.timestamp_start,
|
||||||
|
)
|
||||||
|
|
||||||
|
for game in Game.objects.filter(purchases__date_dropped__isnull=False):
|
||||||
|
GameStatusChange.objects.create(
|
||||||
|
game=game,
|
||||||
|
old_status="p",
|
||||||
|
new_status="a",
|
||||||
|
timestamp=game.purchases.first().date_dropped,
|
||||||
|
)
|
||||||
|
|
||||||
|
for game in Game.objects.filter(purchases__date_refunded__isnull=False):
|
||||||
|
GameStatusChange.objects.create(
|
||||||
|
game=game,
|
||||||
|
old_status="p",
|
||||||
|
new_status="a",
|
||||||
|
timestamp=game.purchases.first().date_refunded,
|
||||||
|
)
|
||||||
|
|
||||||
|
# check if game has any playevents, if so create a status change from current status to finished based on playevent's ended date
|
||||||
|
# consider only the first playevent
|
||||||
|
for game in Game.objects.filter(playevents__isnull=False):
|
||||||
|
first_playevent = game.playevents.first()
|
||||||
|
GameStatusChange.objects.create(
|
||||||
|
game=game,
|
||||||
|
old_status="p",
|
||||||
|
new_status="f",
|
||||||
|
timestamp=first_playevent.ended,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("games", "0007_game_updated_at"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="game",
|
||||||
|
name="original_year_released",
|
||||||
|
field=models.IntegerField(blank=True, default=None, null=True),
|
||||||
|
),
|
||||||
|
migrations.RunPython(copy_year_released),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="GameStatusChange",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"id",
|
||||||
|
models.BigAutoField(
|
||||||
|
auto_created=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
verbose_name="ID",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"old_status",
|
||||||
|
models.CharField(
|
||||||
|
blank=True,
|
||||||
|
choices=[
|
||||||
|
("u", "Unplayed"),
|
||||||
|
("p", "Played"),
|
||||||
|
("f", "Finished"),
|
||||||
|
("r", "Retired"),
|
||||||
|
("a", "Abandoned"),
|
||||||
|
],
|
||||||
|
max_length=1,
|
||||||
|
null=True,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"new_status",
|
||||||
|
models.CharField(
|
||||||
|
choices=[
|
||||||
|
("u", "Unplayed"),
|
||||||
|
("p", "Played"),
|
||||||
|
("f", "Finished"),
|
||||||
|
("r", "Retired"),
|
||||||
|
("a", "Abandoned"),
|
||||||
|
],
|
||||||
|
max_length=1,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("timestamp", models.DateTimeField(null=True)),
|
||||||
|
(
|
||||||
|
"game",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="status_changes",
|
||||||
|
to="games.game",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
"ordering": ["-timestamp"],
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name="PlayEvent",
|
||||||
|
fields=[
|
||||||
|
(
|
||||||
|
"id",
|
||||||
|
models.BigAutoField(
|
||||||
|
auto_created=True,
|
||||||
|
primary_key=True,
|
||||||
|
serialize=False,
|
||||||
|
verbose_name="ID",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("started", models.DateField(blank=True, null=True)),
|
||||||
|
("ended", models.DateField(blank=True, null=True)),
|
||||||
|
(
|
||||||
|
"days_to_finish",
|
||||||
|
models.GeneratedField(
|
||||||
|
db_persist=True,
|
||||||
|
expression=django.db.models.expressions.RawSQL(
|
||||||
|
"\n COALESCE(\n CASE \n WHEN date(ended) = date(started) THEN 1\n ELSE julianday(ended) - julianday(started)\n END, 0\n )\n ",
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
output_field=models.IntegerField(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
("note", models.CharField(blank=True, default="", max_length=255)),
|
||||||
|
("created_at", models.DateTimeField(auto_now_add=True)),
|
||||||
|
("updated_at", models.DateTimeField(auto_now=True)),
|
||||||
|
(
|
||||||
|
"game",
|
||||||
|
models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
related_name="playevents",
|
||||||
|
to="games.game",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
migrations.RunPython(set_abandoned_status),
|
||||||
|
migrations.RunPython(create_game_status_changes),
|
||||||
|
]
|
@ -1,34 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 18:51
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
def create_edition_of_game(apps, schema_editor):
|
|
||||||
Game = apps.get_model("games", "Game")
|
|
||||||
Edition = apps.get_model("games", "Edition")
|
|
||||||
Platform = apps.get_model("games", "Platform")
|
|
||||||
first_platform = Platform.objects.first()
|
|
||||||
all_games = Game.objects.all()
|
|
||||||
all_editions = Edition.objects.all()
|
|
||||||
for game in all_games:
|
|
||||||
existing_edition = None
|
|
||||||
try:
|
|
||||||
existing_edition = all_editions.objects.get(game=game.id)
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
if existing_edition == None:
|
|
||||||
edition = Edition()
|
|
||||||
edition.id = game.id
|
|
||||||
edition.game = game
|
|
||||||
edition.name = game.name
|
|
||||||
edition.platform = first_platform
|
|
||||||
edition.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0008_edition"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [migrations.RunPython(create_edition_of_game)]
|
|
@ -0,0 +1,21 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-20 11:35
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0008_game_original_year_released_gamestatuschange_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='date_dropped',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='date_finished',
|
||||||
|
),
|
||||||
|
]
|
@ -1,21 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 19:06
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0009_create_editions"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="game",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
on_delete=django.db.models.deletion.CASCADE, to="games.edition"
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
17
games/migrations/0010_remove_purchase_price_per_game.py
Normal file
17
games/migrations/0010_remove_purchase_price_per_game.py
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-22 17:46
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0009_remove_purchase_date_dropped_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='price_per_game',
|
||||||
|
),
|
||||||
|
]
|
20
games/migrations/0011_purchase_price_per_game.py
Normal file
20
games/migrations/0011_purchase_price_per_game.py
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-22 17:46
|
||||||
|
|
||||||
|
import django.db.models.expressions
|
||||||
|
import django.db.models.functions.comparison
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0010_remove_purchase_price_per_game'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='purchase',
|
||||||
|
name='price_per_game',
|
||||||
|
field=models.GeneratedField(db_persist=True, expression=django.db.models.expressions.CombinedExpression(django.db.models.functions.comparison.Coalesce(models.F('converted_price'), models.F('price'), 0), '/', models.F('num_purchases')), output_field=models.FloatField()),
|
||||||
|
),
|
||||||
|
]
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 19:18
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0010_alter_purchase_game"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RenameField(
|
|
||||||
model_name="purchase",
|
|
||||||
old_name="game",
|
|
||||||
new_name="edition",
|
|
||||||
),
|
|
||||||
]
|
|
32
games/migrations/0012_alter_session_duration_calculated.py
Normal file
32
games/migrations/0012_alter_session_duration_calculated.py
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-25 20:30
|
||||||
|
|
||||||
|
import django.db.models.expressions
|
||||||
|
import django.db.models.functions.comparison
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("games", "0011_purchase_price_per_game"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="session",
|
||||||
|
name="duration_calculated",
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="session",
|
||||||
|
name="duration_calculated",
|
||||||
|
field=models.GeneratedField(
|
||||||
|
db_persist=True,
|
||||||
|
expression=django.db.models.functions.comparison.Coalesce(
|
||||||
|
django.db.models.expressions.CombinedExpression(
|
||||||
|
models.F("timestamp_end"), "-", models.F("timestamp_start")
|
||||||
|
),
|
||||||
|
0,
|
||||||
|
),
|
||||||
|
output_field=models.DurationField(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
@ -1,23 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 19:53
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0011_rename_game_purchase_edition"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="price",
|
|
||||||
field=models.IntegerField(default=0),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="price_currency",
|
|
||||||
field=models.CharField(default="USD", max_length=3),
|
|
||||||
),
|
|
||||||
]
|
|
35
games/migrations/0013_game_playtime.py
Normal file
35
games/migrations/0013_game_playtime.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-25 20:33
|
||||||
|
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
from django.db.models import F, Sum
|
||||||
|
|
||||||
|
|
||||||
|
def calculate_game_playtime(apps, schema_editor):
|
||||||
|
Game = apps.get_model("games", "Game")
|
||||||
|
games = Game.objects.all()
|
||||||
|
for game in games:
|
||||||
|
total_playtime = game.sessions.aggregate(
|
||||||
|
total_playtime=Sum(F("duration_total"))
|
||||||
|
)["total_playtime"]
|
||||||
|
if total_playtime:
|
||||||
|
game.playtime = total_playtime
|
||||||
|
game.save(update_fields=["playtime"])
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("games", "0012_alter_session_duration_calculated"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="game",
|
||||||
|
name="playtime",
|
||||||
|
field=models.DurationField(
|
||||||
|
blank=True, default=datetime.timedelta(0), editable=False
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.RunPython(calculate_game_playtime),
|
||||||
|
]
|
@ -1,31 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 19:54
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0012_purchase_price_purchase_price_currency"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="ownership_type",
|
|
||||||
field=models.CharField(
|
|
||||||
choices=[
|
|
||||||
("ph", "Physical"),
|
|
||||||
("di", "Digital"),
|
|
||||||
("du", "Digital Upgrade"),
|
|
||||||
("re", "Rented"),
|
|
||||||
("bo", "Borrowed"),
|
|
||||||
("tr", "Trial"),
|
|
||||||
("de", "Demo"),
|
|
||||||
("pi", "Pirated"),
|
|
||||||
],
|
|
||||||
default="di",
|
|
||||||
max_length=2,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,52 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-18 19:59
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0013_purchase_ownership_type"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.CreateModel(
|
|
||||||
name="Device",
|
|
||||||
fields=[
|
|
||||||
(
|
|
||||||
"id",
|
|
||||||
models.BigAutoField(
|
|
||||||
auto_created=True,
|
|
||||||
primary_key=True,
|
|
||||||
serialize=False,
|
|
||||||
verbose_name="ID",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
("name", models.CharField(max_length=255)),
|
|
||||||
(
|
|
||||||
"type",
|
|
||||||
models.CharField(
|
|
||||||
choices=[
|
|
||||||
("pc", "PC"),
|
|
||||||
("co", "Console"),
|
|
||||||
("ha", "Handheld"),
|
|
||||||
("mo", "Mobile"),
|
|
||||||
("sbc", "Single-board computer"),
|
|
||||||
],
|
|
||||||
default="pc",
|
|
||||||
max_length=3,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="session",
|
|
||||||
name="device",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.device",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
19
games/migrations/0014_session_duration_total.py
Normal file
19
games/migrations/0014_session_duration_total.py
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
# Generated by Django 5.1.7 on 2025-03-25 20:46
|
||||||
|
|
||||||
|
import django.db.models.expressions
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('games', '0013_game_playtime'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='session',
|
||||||
|
name='duration_total',
|
||||||
|
field=models.GeneratedField(db_persist=True, expression=django.db.models.expressions.CombinedExpression(models.F('duration_calculated'), '+', models.F('duration_manual')), output_field=models.DurationField()),
|
||||||
|
),
|
||||||
|
]
|
@ -1,23 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-02-20 14:55
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0014_device_session_device"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="edition",
|
|
||||||
name="wikidata",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=50, null=True),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="edition",
|
|
||||||
name="year_released",
|
|
||||||
field=models.IntegerField(default=2023),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,51 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 11:10
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0015_edition_wikidata_edition_year_released"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="edition",
|
|
||||||
name="platform",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.platform",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="edition",
|
|
||||||
name="year_released",
|
|
||||||
field=models.IntegerField(blank=True, default=None, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="game",
|
|
||||||
name="wikidata",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=50, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="platform",
|
|
||||||
name="group",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=255, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="device",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.device",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,141 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 18:14
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
def rename_duplicates(apps, schema_editor):
|
|
||||||
Edition = apps.get_model("games", "Edition")
|
|
||||||
|
|
||||||
duplicates = (
|
|
||||||
Edition.objects.values("name", "platform")
|
|
||||||
.annotate(name_count=models.Count("id"))
|
|
||||||
.filter(name_count__gt=1)
|
|
||||||
)
|
|
||||||
|
|
||||||
for duplicate in duplicates:
|
|
||||||
counter = 1
|
|
||||||
duplicate_editions = Edition.objects.filter(
|
|
||||||
name=duplicate["name"], platform_id=duplicate["platform"]
|
|
||||||
).order_by("id")
|
|
||||||
|
|
||||||
for edition in duplicate_editions[1:]: # Skip the first one
|
|
||||||
edition.name = f"{edition.name} {counter}"
|
|
||||||
edition.save()
|
|
||||||
counter += 1
|
|
||||||
|
|
||||||
|
|
||||||
def update_game_year(apps, schema_editor):
|
|
||||||
Game = apps.get_model("games", "Game")
|
|
||||||
Edition = apps.get_model("games", "Edition")
|
|
||||||
|
|
||||||
for game in Game.objects.filter(year__isnull=True):
|
|
||||||
# Try to get the first related edition with a non-null year_released
|
|
||||||
edition = Edition.objects.filter(game=game, year_released__isnull=False).first()
|
|
||||||
if edition:
|
|
||||||
# If an edition is found, update the game's year
|
|
||||||
game.year = edition.year_released
|
|
||||||
game.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
replaces = [
|
|
||||||
("games", "0016_alter_edition_platform_alter_edition_year_released_and_more"),
|
|
||||||
("games", "0017_alter_device_type_alter_purchase_platform"),
|
|
||||||
("games", "0018_auto_20231106_1825"),
|
|
||||||
("games", "0019_alter_edition_unique_together"),
|
|
||||||
("games", "0020_game_year"),
|
|
||||||
("games", "0021_auto_20231106_1909"),
|
|
||||||
("games", "0022_rename_year_game_year_released"),
|
|
||||||
]
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0015_edition_wikidata_edition_year_released"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="edition",
|
|
||||||
name="platform",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.platform",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="edition",
|
|
||||||
name="year_released",
|
|
||||||
field=models.IntegerField(blank=True, default=None, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="game",
|
|
||||||
name="wikidata",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=50, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="platform",
|
|
||||||
name="group",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=255, null=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="device",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.device",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="device",
|
|
||||||
name="type",
|
|
||||||
field=models.CharField(
|
|
||||||
choices=[
|
|
||||||
("pc", "PC"),
|
|
||||||
("co", "Console"),
|
|
||||||
("ha", "Handheld"),
|
|
||||||
("mo", "Mobile"),
|
|
||||||
("sbc", "Single-board computer"),
|
|
||||||
("un", "Unknown"),
|
|
||||||
],
|
|
||||||
default="un",
|
|
||||||
max_length=3,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="platform",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.platform",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.RunPython(
|
|
||||||
code=rename_duplicates,
|
|
||||||
),
|
|
||||||
migrations.AlterUniqueTogether(
|
|
||||||
name="edition",
|
|
||||||
unique_together={("name", "platform")},
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="game",
|
|
||||||
name="year",
|
|
||||||
field=models.IntegerField(blank=True, default=None, null=True),
|
|
||||||
),
|
|
||||||
migrations.RunPython(
|
|
||||||
code=update_game_year,
|
|
||||||
),
|
|
||||||
migrations.RenameField(
|
|
||||||
model_name="game",
|
|
||||||
old_name="year",
|
|
||||||
new_name="year_released",
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,41 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 16:53
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0016_alter_edition_platform_alter_edition_year_released_and_more"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="device",
|
|
||||||
name="type",
|
|
||||||
field=models.CharField(
|
|
||||||
choices=[
|
|
||||||
("pc", "PC"),
|
|
||||||
("co", "Console"),
|
|
||||||
("ha", "Handheld"),
|
|
||||||
("mo", "Mobile"),
|
|
||||||
("sbc", "Single-board computer"),
|
|
||||||
("un", "Unknown"),
|
|
||||||
],
|
|
||||||
default="un",
|
|
||||||
max_length=3,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="platform",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.CASCADE,
|
|
||||||
to="games.platform",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,34 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 17:25
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
def rename_duplicates(apps, schema_editor):
|
|
||||||
Edition = apps.get_model("games", "Edition")
|
|
||||||
|
|
||||||
duplicates = (
|
|
||||||
Edition.objects.values("name", "platform")
|
|
||||||
.annotate(name_count=models.Count("id"))
|
|
||||||
.filter(name_count__gt=1)
|
|
||||||
)
|
|
||||||
|
|
||||||
for duplicate in duplicates:
|
|
||||||
counter = 1
|
|
||||||
duplicate_editions = Edition.objects.filter(
|
|
||||||
name=duplicate["name"], platform_id=duplicate["platform"]
|
|
||||||
).order_by("id")
|
|
||||||
|
|
||||||
for edition in duplicate_editions[1:]: # Skip the first one
|
|
||||||
edition.name = f"{edition.name} {counter}"
|
|
||||||
edition.save()
|
|
||||||
counter += 1
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
("games", "0017_alter_device_type_alter_purchase_platform"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RunPython(rename_duplicates),
|
|
||||||
]
|
|
@ -1,17 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 17:26
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0018_auto_20231106_1825"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterUniqueTogether(
|
|
||||||
name="edition",
|
|
||||||
unique_together={("name", "platform")},
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 18:05
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0019_alter_edition_unique_together"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="game",
|
|
||||||
name="year",
|
|
||||||
field=models.IntegerField(blank=True, default=None, null=True),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,24 +0,0 @@
|
|||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
def update_game_year(apps, schema_editor):
|
|
||||||
Game = apps.get_model("games", "Game")
|
|
||||||
Edition = apps.get_model("games", "Edition")
|
|
||||||
|
|
||||||
for game in Game.objects.filter(year__isnull=True):
|
|
||||||
# Try to get the first related edition with a non-null year_released
|
|
||||||
edition = Edition.objects.filter(game=game, year_released__isnull=False).first()
|
|
||||||
if edition:
|
|
||||||
# If an edition is found, update the game's year
|
|
||||||
game.year = edition.year_released
|
|
||||||
game.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
("games", "0020_game_year"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RunPython(update_game_year),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 18:12
|
|
||||||
|
|
||||||
from django.db import migrations
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0021_auto_20231106_1909"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RenameField(
|
|
||||||
model_name="game",
|
|
||||||
old_name="year",
|
|
||||||
new_name="year_released",
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,21 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-06 18:24
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
(
|
|
||||||
"games",
|
|
||||||
"0016_alter_edition_platform_alter_edition_year_released_and_more_squashed_0022_rename_year_game_year_released",
|
|
||||||
),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="date_finished",
|
|
||||||
field=models.DateField(blank=True, null=True),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,39 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-09 09:32
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
def create_sort_name(apps, schema_editor):
|
|
||||||
Edition = apps.get_model(
|
|
||||||
"games", "Edition"
|
|
||||||
) # Replace 'your_app_name' with the actual name of your app
|
|
||||||
|
|
||||||
for edition in Edition.objects.all():
|
|
||||||
name = edition.name
|
|
||||||
# Check for articles at the beginning of the name and move them to the end
|
|
||||||
if name.lower().startswith("the "):
|
|
||||||
sort_name = f"{name[4:]}, The"
|
|
||||||
elif name.lower().startswith("a "):
|
|
||||||
sort_name = f"{name[2:]}, A"
|
|
||||||
elif name.lower().startswith("an "):
|
|
||||||
sort_name = f"{name[3:]}, An"
|
|
||||||
else:
|
|
||||||
sort_name = name
|
|
||||||
# Save the sort_name back to the database
|
|
||||||
edition.sort_name = sort_name
|
|
||||||
edition.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
("games", "0023_purchase_date_finished"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="edition",
|
|
||||||
name="sort_name",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=255, null=True),
|
|
||||||
),
|
|
||||||
migrations.RunPython(create_sort_name),
|
|
||||||
]
|
|
@ -1,39 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-09 09:32
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
def create_sort_name(apps, schema_editor):
|
|
||||||
Game = apps.get_model(
|
|
||||||
"games", "Game"
|
|
||||||
) # Replace 'your_app_name' with the actual name of your app
|
|
||||||
|
|
||||||
for game in Game.objects.all():
|
|
||||||
name = game.name
|
|
||||||
# Check for articles at the beginning of the name and move them to the end
|
|
||||||
if name.lower().startswith("the "):
|
|
||||||
sort_name = f"{name[4:]}, The"
|
|
||||||
elif name.lower().startswith("a "):
|
|
||||||
sort_name = f"{name[2:]}, A"
|
|
||||||
elif name.lower().startswith("an "):
|
|
||||||
sort_name = f"{name[3:]}, An"
|
|
||||||
else:
|
|
||||||
sort_name = name
|
|
||||||
# Save the sort_name back to the database
|
|
||||||
game.sort_name = sort_name
|
|
||||||
game.save()
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
("games", "0024_edition_sort_name"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="game",
|
|
||||||
name="sort_name",
|
|
||||||
field=models.CharField(blank=True, default=None, max_length=255, null=True),
|
|
||||||
),
|
|
||||||
migrations.RunPython(create_sort_name),
|
|
||||||
]
|
|
@ -1,27 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-14 08:35
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0025_game_sort_name"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="type",
|
|
||||||
field=models.CharField(
|
|
||||||
choices=[
|
|
||||||
("game", "Game"),
|
|
||||||
("dlc", "DLC"),
|
|
||||||
("season_pass", "Season Pass"),
|
|
||||||
("battle_pass", "Battle Pass"),
|
|
||||||
],
|
|
||||||
default="game",
|
|
||||||
max_length=255,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,25 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-14 08:41
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0026_purchase_type"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="related_purchase",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.SET_NULL,
|
|
||||||
to="games.purchase",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,26 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-14 11:05
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
from games.models import Purchase
|
|
||||||
|
|
||||||
|
|
||||||
def null_game_name(apps, schema_editor):
|
|
||||||
Purchase.objects.filter(type=Purchase.GAME).update(name=None)
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
("games", "0027_purchase_related_purchase"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="name",
|
|
||||||
field=models.CharField(
|
|
||||||
blank=True, default="Unknown Name", max_length=255, null=True
|
|
||||||
),
|
|
||||||
),
|
|
||||||
migrations.RunPython(null_game_name),
|
|
||||||
]
|
|
@ -1,26 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-14 21:19
|
|
||||||
|
|
||||||
import django.db.models.deletion
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0028_purchase_name"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="related_purchase",
|
|
||||||
field=models.ForeignKey(
|
|
||||||
blank=True,
|
|
||||||
default=None,
|
|
||||||
null=True,
|
|
||||||
on_delete=django.db.models.deletion.SET_NULL,
|
|
||||||
related_name="related_purchases",
|
|
||||||
to="games.purchase",
|
|
||||||
),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-15 12:04
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0029_alter_purchase_related_purchase"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="name",
|
|
||||||
field=models.CharField(blank=True, default="", max_length=255, null=True),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,44 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-15 13:51
|
|
||||||
|
|
||||||
import django.utils.timezone
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0030_alter_purchase_name"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="device",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="edition",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="game",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="platform",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="session",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(default=django.utils.timezone.now),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,52 +0,0 @@
|
|||||||
# Generated by Django 4.1.5 on 2023-11-15 18:02
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
("games", "0031_device_created_at_edition_created_at_game_created_at_and_more"),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterModelOptions(
|
|
||||||
name="session",
|
|
||||||
options={"get_latest_by": "timestamp_start"},
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name="session",
|
|
||||||
name="modified_at",
|
|
||||||
field=models.DateTimeField(auto_now=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="device",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="edition",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="game",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="platform",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="purchase",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name="session",
|
|
||||||
name="created_at",
|
|
||||||
field=models.DateTimeField(auto_now_add=True),
|
|
||||||
),
|
|
||||||
]
|
|
439
games/models.py
439
games/models.py
@ -1,65 +1,112 @@
|
|||||||
|
import logging
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
|
|
||||||
|
import requests
|
||||||
from django.core.exceptions import ValidationError
|
from django.core.exceptions import ValidationError
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.models import F, Manager, Sum
|
from django.db.models import F, Sum
|
||||||
|
from django.db.models.expressions import RawSQL
|
||||||
|
from django.db.models.fields.generated import GeneratedField
|
||||||
|
from django.db.models.functions import Coalesce
|
||||||
|
from django.template.defaultfilters import floatformat, pluralize, slugify
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from common.time import format_duration
|
from common.time import format_duration
|
||||||
|
|
||||||
|
logger = logging.getLogger("games")
|
||||||
|
|
||||||
|
|
||||||
class Game(models.Model):
|
class Game(models.Model):
|
||||||
|
class Meta:
|
||||||
|
unique_together = [["name", "platform", "year_released"]]
|
||||||
|
|
||||||
name = models.CharField(max_length=255)
|
name = models.CharField(max_length=255)
|
||||||
sort_name = models.CharField(max_length=255, null=True, blank=True, default=None)
|
sort_name = models.CharField(max_length=255, blank=True, default="")
|
||||||
year_released = models.IntegerField(null=True, blank=True, default=None)
|
year_released = models.IntegerField(null=True, blank=True, default=None)
|
||||||
wikidata = models.CharField(max_length=50, null=True, blank=True, default=None)
|
original_year_released = models.IntegerField(null=True, blank=True, default=None)
|
||||||
|
wikidata = models.CharField(max_length=50, blank=True, default="")
|
||||||
|
platform = models.ForeignKey(
|
||||||
|
"Platform", on_delete=models.SET_DEFAULT, null=True, blank=True, default=None
|
||||||
|
)
|
||||||
|
|
||||||
|
playtime = models.DurationField(blank=True, editable=False, default=timedelta(0))
|
||||||
|
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
class Status(models.TextChoices):
|
||||||
|
UNPLAYED = (
|
||||||
|
"u",
|
||||||
|
"Unplayed",
|
||||||
|
)
|
||||||
|
PLAYED = (
|
||||||
|
"p",
|
||||||
|
"Played",
|
||||||
|
)
|
||||||
|
FINISHED = (
|
||||||
|
"f",
|
||||||
|
"Finished",
|
||||||
|
)
|
||||||
|
RETIRED = (
|
||||||
|
"r",
|
||||||
|
"Retired",
|
||||||
|
)
|
||||||
|
ABANDONED = (
|
||||||
|
"a",
|
||||||
|
"Abandoned",
|
||||||
|
)
|
||||||
|
|
||||||
|
status = models.CharField(max_length=1, choices=Status, default=Status.UNPLAYED)
|
||||||
|
mastered = models.BooleanField(default=False)
|
||||||
|
|
||||||
|
session_average: float | int | timedelta | None
|
||||||
|
session_count: int | None
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
def finished(self):
|
||||||
|
return self.status == self.Status.FINISHED
|
||||||
|
|
||||||
|
def abandoned(self):
|
||||||
|
return self.status == self.Status.ABANDONED
|
||||||
|
|
||||||
|
def retired(self):
|
||||||
|
return self.status == self.Status.RETIRED
|
||||||
|
|
||||||
|
def played(self):
|
||||||
|
return self.status == self.Status.PLAYED
|
||||||
|
|
||||||
|
def unplayed(self):
|
||||||
|
return self.status == self.Status.UNPLAYED
|
||||||
|
|
||||||
|
def playtime_formatted(self):
|
||||||
|
return format_duration(self.playtime, "%2.1H")
|
||||||
|
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
if self.platform is None:
|
||||||
|
self.platform = get_sentinel_platform()
|
||||||
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
def get_sentinel_platform():
|
||||||
|
return Platform.objects.get_or_create(
|
||||||
|
name="Unspecified", icon="unspecified", group="Unspecified"
|
||||||
|
)[0]
|
||||||
|
|
||||||
|
|
||||||
|
class Platform(models.Model):
|
||||||
|
name = models.CharField(max_length=255)
|
||||||
|
group = models.CharField(max_length=255, blank=True, default="")
|
||||||
|
icon = models.SlugField(blank=True)
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
def get_sort_name(name):
|
if not self.icon:
|
||||||
articles = ["a", "an", "the"]
|
self.icon = slugify(self.name)
|
||||||
name_parts = name.split()
|
|
||||||
first_word = name_parts[0].lower()
|
|
||||||
if first_word in articles:
|
|
||||||
return f"{' '.join(name_parts[1:])}, {name_parts[0]}"
|
|
||||||
else:
|
|
||||||
return name
|
|
||||||
|
|
||||||
self.sort_name = get_sort_name(self.name)
|
|
||||||
super().save(*args, **kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
class Edition(models.Model):
|
|
||||||
class Meta:
|
|
||||||
unique_together = [["name", "platform"]]
|
|
||||||
|
|
||||||
game = models.ForeignKey("Game", on_delete=models.CASCADE)
|
|
||||||
name = models.CharField(max_length=255)
|
|
||||||
sort_name = models.CharField(max_length=255, null=True, blank=True, default=None)
|
|
||||||
platform = models.ForeignKey(
|
|
||||||
"Platform", on_delete=models.CASCADE, null=True, blank=True, default=None
|
|
||||||
)
|
|
||||||
year_released = models.IntegerField(null=True, blank=True, default=None)
|
|
||||||
wikidata = models.CharField(max_length=50, null=True, blank=True, default=None)
|
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.sort_name
|
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
|
||||||
def get_sort_name(name):
|
|
||||||
articles = ["a", "an", "the"]
|
|
||||||
name_parts = name.split()
|
|
||||||
first_word = name_parts[0].lower()
|
|
||||||
if first_word in articles:
|
|
||||||
return f"{' '.join(name_parts[1:])}, {name_parts[0]}"
|
|
||||||
else:
|
|
||||||
return name
|
|
||||||
|
|
||||||
self.sort_name = get_sort_name(self.name)
|
|
||||||
super().save(*args, **kwargs)
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
@ -70,9 +117,6 @@ class PurchaseQueryset(models.QuerySet):
|
|||||||
def not_refunded(self):
|
def not_refunded(self):
|
||||||
return self.filter(date_refunded__isnull=True)
|
return self.filter(date_refunded__isnull=True)
|
||||||
|
|
||||||
def finished(self):
|
|
||||||
return self.filter(date_finished__isnull=False)
|
|
||||||
|
|
||||||
def games_only(self):
|
def games_only(self):
|
||||||
return self.filter(type=Purchase.GAME)
|
return self.filter(type=Purchase.GAME)
|
||||||
|
|
||||||
@ -109,63 +153,106 @@ class Purchase(models.Model):
|
|||||||
|
|
||||||
objects = PurchaseQueryset().as_manager()
|
objects = PurchaseQueryset().as_manager()
|
||||||
|
|
||||||
edition = models.ForeignKey("Edition", on_delete=models.CASCADE)
|
games = models.ManyToManyField(Game, related_name="purchases")
|
||||||
|
|
||||||
platform = models.ForeignKey(
|
platform = models.ForeignKey(
|
||||||
"Platform", on_delete=models.CASCADE, default=None, null=True, blank=True
|
Platform, on_delete=models.CASCADE, default=None, null=True, blank=True
|
||||||
)
|
)
|
||||||
date_purchased = models.DateField()
|
date_purchased = models.DateField(verbose_name="Purchased")
|
||||||
date_refunded = models.DateField(blank=True, null=True)
|
date_refunded = models.DateField(blank=True, null=True, verbose_name="Refunded")
|
||||||
date_finished = models.DateField(blank=True, null=True)
|
infinite = models.BooleanField(default=False)
|
||||||
price = models.IntegerField(default=0)
|
price = models.FloatField(default=0)
|
||||||
price_currency = models.CharField(max_length=3, default="USD")
|
price_currency = models.CharField(max_length=3, default="USD")
|
||||||
|
converted_price = models.FloatField(null=True)
|
||||||
|
converted_currency = models.CharField(max_length=3, blank=True, default="")
|
||||||
|
price_per_game = GeneratedField(
|
||||||
|
expression=Coalesce(F("converted_price"), F("price"), 0) / F("num_purchases"),
|
||||||
|
output_field=models.FloatField(),
|
||||||
|
db_persist=True,
|
||||||
|
editable=False,
|
||||||
|
)
|
||||||
|
num_purchases = models.IntegerField(default=0)
|
||||||
ownership_type = models.CharField(
|
ownership_type = models.CharField(
|
||||||
max_length=2, choices=OWNERSHIP_TYPES, default=DIGITAL
|
max_length=2, choices=OWNERSHIP_TYPES, default=DIGITAL
|
||||||
)
|
)
|
||||||
type = models.CharField(max_length=255, choices=TYPES, default=GAME)
|
type = models.CharField(max_length=255, choices=TYPES, default=GAME)
|
||||||
name = models.CharField(max_length=255, default="", null=True, blank=True)
|
name = models.CharField(max_length=255, blank=True, default="")
|
||||||
related_purchase = models.ForeignKey(
|
related_purchase = models.ForeignKey(
|
||||||
"Purchase",
|
"self",
|
||||||
on_delete=models.SET_NULL,
|
on_delete=models.SET_NULL,
|
||||||
default=None,
|
default=None,
|
||||||
null=True,
|
null=True,
|
||||||
blank=True,
|
|
||||||
related_name="related_purchases",
|
related_name="related_purchases",
|
||||||
)
|
)
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def standardized_price(self):
|
||||||
|
return (
|
||||||
|
f"{floatformat(self.converted_price, 0)} {self.converted_currency}"
|
||||||
|
if self.converted_price
|
||||||
|
else None
|
||||||
|
)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def has_one_item(self):
|
||||||
|
return self.games.count() == 1
|
||||||
|
|
||||||
|
@property
|
||||||
|
def standardized_name(self):
|
||||||
|
return self.name or self.first_game.name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def first_game(self):
|
||||||
|
return self.games.first()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
|
return self.standardized_name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def full_name(self):
|
||||||
additional_info = [
|
additional_info = [
|
||||||
self.get_type_display() if self.type != Purchase.GAME else "",
|
str(item)
|
||||||
f"{self.edition.platform} version on {self.platform}"
|
for item in [
|
||||||
if self.platform != self.edition.platform
|
f"{self.num_purchases} game{pluralize(self.num_purchases)}",
|
||||||
else self.platform,
|
self.date_purchased,
|
||||||
self.edition.year_released,
|
self.standardized_price,
|
||||||
self.get_ownership_type_display(),
|
]
|
||||||
|
if item
|
||||||
]
|
]
|
||||||
return f"{self.edition} ({', '.join(filter(None, map(str, additional_info)))})"
|
return f"{self.standardized_name} ({', '.join(additional_info)})"
|
||||||
|
|
||||||
def is_game(self):
|
def is_game(self):
|
||||||
return self.type == self.GAME
|
return self.type == self.GAME
|
||||||
|
|
||||||
|
def price_or_currency_differ_from(self, purchase_to_compare):
|
||||||
|
return (
|
||||||
|
self.price != purchase_to_compare.price
|
||||||
|
or self.price_currency != purchase_to_compare.price_currency
|
||||||
|
)
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
if self.type == Purchase.GAME:
|
if self.type != Purchase.GAME and not self.related_purchase:
|
||||||
self.name = ""
|
|
||||||
elif self.type != Purchase.GAME and not self.related_purchase:
|
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
f"{self.get_type_display()} must have a related purchase."
|
f"{self.get_type_display()} must have a related purchase."
|
||||||
)
|
)
|
||||||
|
if self.pk is not None:
|
||||||
|
# Retrieve the existing instance from the database
|
||||||
|
existing_purchase = Purchase.objects.get(pk=self.pk)
|
||||||
|
# If price has changed, reset converted fields
|
||||||
|
if existing_purchase.price_or_currency_differ_from(self):
|
||||||
|
from games.tasks import currency_to
|
||||||
|
|
||||||
|
exchange_rate = get_or_create_rate(
|
||||||
|
self.price_currency, currency_to, self.date_purchased.year
|
||||||
|
)
|
||||||
|
if exchange_rate:
|
||||||
|
self.converted_price = floatformat(self.price * exchange_rate, 0)
|
||||||
|
self.converted_currency = currency_to
|
||||||
super().save(*args, **kwargs)
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class Platform(models.Model):
|
|
||||||
name = models.CharField(max_length=255)
|
|
||||||
group = models.CharField(max_length=255, null=True, blank=True, default=None)
|
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
|
|
||||||
class SessionQuerySet(models.QuerySet):
|
class SessionQuerySet(models.QuerySet):
|
||||||
def total_duration_formatted(self):
|
def total_duration_formatted(self):
|
||||||
return format_duration(self.total_duration_unformatted())
|
return format_duration(self.total_duration_unformatted())
|
||||||
@ -176,32 +263,66 @@ class SessionQuerySet(models.QuerySet):
|
|||||||
)
|
)
|
||||||
return result["duration"]
|
return result["duration"]
|
||||||
|
|
||||||
|
def calculated_duration_formatted(self):
|
||||||
|
return format_duration(self.calculated_duration_unformatted())
|
||||||
|
|
||||||
|
def calculated_duration_unformatted(self):
|
||||||
|
result = self.aggregate(duration=Sum(F("duration_calculated")))
|
||||||
|
return result["duration"]
|
||||||
|
|
||||||
|
def without_manual(self):
|
||||||
|
return self.exclude(duration_calculated__iexact=0)
|
||||||
|
|
||||||
|
def only_manual(self):
|
||||||
|
return self.filter(duration_calculated__iexact=0)
|
||||||
|
|
||||||
|
|
||||||
class Session(models.Model):
|
class Session(models.Model):
|
||||||
class Meta:
|
class Meta:
|
||||||
get_latest_by = "timestamp_start"
|
get_latest_by = "timestamp_start"
|
||||||
|
|
||||||
purchase = models.ForeignKey("Purchase", on_delete=models.CASCADE)
|
game = models.ForeignKey(
|
||||||
timestamp_start = models.DateTimeField()
|
Game,
|
||||||
timestamp_end = models.DateTimeField(blank=True, null=True)
|
on_delete=models.CASCADE,
|
||||||
duration_manual = models.DurationField(blank=True, null=True, default=timedelta(0))
|
null=True,
|
||||||
duration_calculated = models.DurationField(blank=True, null=True)
|
default=None,
|
||||||
|
related_name="sessions",
|
||||||
|
)
|
||||||
|
timestamp_start = models.DateTimeField(verbose_name="Start")
|
||||||
|
timestamp_end = models.DateTimeField(blank=True, null=True, verbose_name="End")
|
||||||
|
duration_manual = models.DurationField(
|
||||||
|
blank=True, null=True, default=timedelta(0), verbose_name="Manual duration"
|
||||||
|
)
|
||||||
|
duration_calculated = GeneratedField(
|
||||||
|
expression=Coalesce(F("timestamp_end") - F("timestamp_start"), 0),
|
||||||
|
output_field=models.DurationField(),
|
||||||
|
db_persist=True,
|
||||||
|
editable=False,
|
||||||
|
)
|
||||||
|
duration_total = GeneratedField(
|
||||||
|
expression=F("duration_calculated") + F("duration_manual"),
|
||||||
|
output_field=models.DurationField(),
|
||||||
|
db_persist=True,
|
||||||
|
editable=False,
|
||||||
|
)
|
||||||
device = models.ForeignKey(
|
device = models.ForeignKey(
|
||||||
"Device",
|
"Device",
|
||||||
on_delete=models.CASCADE,
|
on_delete=models.SET_DEFAULT,
|
||||||
null=True,
|
null=True,
|
||||||
blank=True,
|
blank=True,
|
||||||
default=None,
|
default=None,
|
||||||
)
|
)
|
||||||
note = models.TextField(blank=True, null=True)
|
note = models.TextField(blank=True, default="")
|
||||||
|
emulated = models.BooleanField(default=False)
|
||||||
|
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
modified_at = models.DateTimeField(auto_now=True)
|
modified_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
objects = SessionQuerySet.as_manager()
|
objects = SessionQuerySet.as_manager()
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
mark = ", manual" if self.is_manual() else ""
|
mark = "*" if self.is_manual() else ""
|
||||||
return f"{str(self.purchase)} {str(self.timestamp_start.date())} ({self.duration_formatted()}{mark})"
|
return f"{str(self.game)} {str(self.timestamp_start.date())} ({self.duration_formatted()}{mark})"
|
||||||
|
|
||||||
def finish_now(self):
|
def finish_now(self):
|
||||||
self.timestamp_end = timezone.now()
|
self.timestamp_end = timezone.now()
|
||||||
@ -209,31 +330,20 @@ class Session(models.Model):
|
|||||||
def start_now():
|
def start_now():
|
||||||
self.timestamp_start = timezone.now()
|
self.timestamp_start = timezone.now()
|
||||||
|
|
||||||
def duration_seconds(self) -> timedelta:
|
|
||||||
manual = timedelta(0)
|
|
||||||
calculated = timedelta(0)
|
|
||||||
if self.is_manual():
|
|
||||||
manual = self.duration_manual
|
|
||||||
if self.timestamp_end != None and self.timestamp_start != None:
|
|
||||||
calculated = self.timestamp_end - self.timestamp_start
|
|
||||||
return timedelta(seconds=(manual + calculated).total_seconds())
|
|
||||||
|
|
||||||
def duration_formatted(self) -> str:
|
def duration_formatted(self) -> str:
|
||||||
result = format_duration(self.duration_seconds(), "%02.0H:%02.0m")
|
result = format_duration(self.duration_total, "%02.1H")
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
def duration_formatted_with_mark(self) -> str:
|
||||||
|
mark = "*" if self.is_manual() else ""
|
||||||
|
return f"{self.duration_formatted()}{mark}"
|
||||||
|
|
||||||
def is_manual(self) -> bool:
|
def is_manual(self) -> bool:
|
||||||
return not self.duration_manual == timedelta(0)
|
return not self.duration_manual == timedelta(0)
|
||||||
|
|
||||||
@property
|
def save(self, *args, **kwargs) -> None:
|
||||||
def duration_sum(self) -> str:
|
if not isinstance(self.duration_manual, timedelta):
|
||||||
return Session.objects.all().total_duration_formatted()
|
self.duration_manual = timedelta(0)
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
|
||||||
if self.timestamp_start != None and self.timestamp_end != None:
|
|
||||||
self.duration_calculated = self.timestamp_end - self.timestamp_start
|
|
||||||
else:
|
|
||||||
self.duration_calculated = timedelta(0)
|
|
||||||
|
|
||||||
if not self.device:
|
if not self.device:
|
||||||
default_device, _ = Device.objects.get_or_create(
|
default_device, _ = Device.objects.get_or_create(
|
||||||
@ -244,12 +354,12 @@ class Session(models.Model):
|
|||||||
|
|
||||||
|
|
||||||
class Device(models.Model):
|
class Device(models.Model):
|
||||||
PC = "pc"
|
PC = "PC"
|
||||||
CONSOLE = "co"
|
CONSOLE = "Console"
|
||||||
HANDHELD = "ha"
|
HANDHELD = "Handheld"
|
||||||
MOBILE = "mo"
|
MOBILE = "Mobile"
|
||||||
SBC = "sbc"
|
SBC = "Single-board computer"
|
||||||
UNKNOWN = "un"
|
UNKNOWN = "Unknown"
|
||||||
DEVICE_TYPES = [
|
DEVICE_TYPES = [
|
||||||
(PC, "PC"),
|
(PC, "PC"),
|
||||||
(CONSOLE, "Console"),
|
(CONSOLE, "Console"),
|
||||||
@ -259,8 +369,115 @@ class Device(models.Model):
|
|||||||
(UNKNOWN, "Unknown"),
|
(UNKNOWN, "Unknown"),
|
||||||
]
|
]
|
||||||
name = models.CharField(max_length=255)
|
name = models.CharField(max_length=255)
|
||||||
type = models.CharField(max_length=3, choices=DEVICE_TYPES, default=UNKNOWN)
|
type = models.CharField(max_length=255, choices=DEVICE_TYPES, default=UNKNOWN)
|
||||||
created_at = models.DateTimeField(auto_now_add=True)
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"{self.name} ({self.get_type_display()})"
|
return f"{self.name} ({self.type})"
|
||||||
|
|
||||||
|
|
||||||
|
class ExchangeRate(models.Model):
|
||||||
|
currency_from = models.CharField(max_length=255)
|
||||||
|
currency_to = models.CharField(max_length=255)
|
||||||
|
year = models.PositiveIntegerField()
|
||||||
|
rate = models.FloatField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = ("currency_from", "currency_to", "year")
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.currency_from}/{self.currency_to} - {self.rate} ({self.year})"
|
||||||
|
|
||||||
|
|
||||||
|
def get_or_create_rate(currency_from: str, currency_to: str, year: int) -> float | None:
|
||||||
|
exchange_rate = None
|
||||||
|
result = ExchangeRate.objects.filter(
|
||||||
|
currency_from=currency_from, currency_to=currency_to, year=year
|
||||||
|
)
|
||||||
|
if result:
|
||||||
|
exchange_rate = result[0].rate
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
# this API endpoint only accepts lowercase currency string
|
||||||
|
response = requests.get(
|
||||||
|
f"https://cdn.jsdelivr.net/npm/@fawazahmed0/currency-api@{year}-01-01/v1/currencies/{currency_from.lower()}.json"
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
data = response.json()
|
||||||
|
currency_from_data = data.get(currency_from.lower())
|
||||||
|
rate = currency_from_data.get(currency_to.lower())
|
||||||
|
|
||||||
|
if rate:
|
||||||
|
logger.info(f"[convert_prices]: Got {rate}, saving...")
|
||||||
|
exchange_rate = ExchangeRate.objects.create(
|
||||||
|
currency_from=currency_from,
|
||||||
|
currency_to=currency_to,
|
||||||
|
year=year,
|
||||||
|
rate=floatformat(rate, 2),
|
||||||
|
)
|
||||||
|
exchange_rate = exchange_rate.rate
|
||||||
|
else:
|
||||||
|
logger.info("[convert_prices]: Could not get an exchange rate.")
|
||||||
|
except requests.RequestException as e:
|
||||||
|
logger.info(
|
||||||
|
f"[convert_prices]: Failed to fetch exchange rate for {currency_from}->{currency_to} in {year}: {e}"
|
||||||
|
)
|
||||||
|
return exchange_rate
|
||||||
|
|
||||||
|
|
||||||
|
class PlayEvent(models.Model):
|
||||||
|
game = models.ForeignKey(Game, related_name="playevents", on_delete=models.CASCADE)
|
||||||
|
started = models.DateField(null=True, blank=True)
|
||||||
|
ended = models.DateField(null=True, blank=True)
|
||||||
|
days_to_finish = GeneratedField(
|
||||||
|
# special cases:
|
||||||
|
# missing ended, started, or both = 0
|
||||||
|
# same day = 1 day to finish
|
||||||
|
expression=RawSQL(
|
||||||
|
"""
|
||||||
|
COALESCE(
|
||||||
|
CASE
|
||||||
|
WHEN date(ended) = date(started) THEN 1
|
||||||
|
ELSE julianday(ended) - julianday(started)
|
||||||
|
END, 0
|
||||||
|
)
|
||||||
|
""",
|
||||||
|
[],
|
||||||
|
),
|
||||||
|
output_field=models.IntegerField(),
|
||||||
|
db_persist=True,
|
||||||
|
editable=False,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
note = models.CharField(max_length=255, blank=True, default="")
|
||||||
|
|
||||||
|
created_at = models.DateTimeField(auto_now_add=True)
|
||||||
|
updated_at = models.DateTimeField(auto_now=True)
|
||||||
|
|
||||||
|
|
||||||
|
# class PlayMarker(models.Model):
|
||||||
|
# game = models.ForeignKey(Game, related_name="markers", on_delete=models.CASCADE)
|
||||||
|
# played_since = models.DurationField()
|
||||||
|
# played_total = models.DurationField()
|
||||||
|
# note = models.CharField(max_length=255)
|
||||||
|
|
||||||
|
|
||||||
|
class GameStatusChange(models.Model):
|
||||||
|
"""
|
||||||
|
Tracks changes to the status of a Game.
|
||||||
|
"""
|
||||||
|
|
||||||
|
game = models.ForeignKey(
|
||||||
|
Game, on_delete=models.CASCADE, related_name="status_changes"
|
||||||
|
)
|
||||||
|
old_status = models.CharField(
|
||||||
|
max_length=1, choices=Game.Status.choices, blank=True, null=True
|
||||||
|
)
|
||||||
|
new_status = models.CharField(max_length=1, choices=Game.Status.choices)
|
||||||
|
timestamp = models.DateTimeField(null=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.game.name}: {self.old_status or 'None'} -> {self.new_status} at {self.timestamp}"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
ordering = ["-timestamp"]
|
||||||
|
30
games/schema.py
Normal file
30
games/schema.py
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import graphene
|
||||||
|
|
||||||
|
from games.graphql.mutations import GameMutation
|
||||||
|
from games.graphql.queries import (
|
||||||
|
DeviceQuery,
|
||||||
|
EditionQuery,
|
||||||
|
GameQuery,
|
||||||
|
PlatformQuery,
|
||||||
|
PurchaseQuery,
|
||||||
|
SessionQuery,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Query(
|
||||||
|
GameQuery,
|
||||||
|
EditionQuery,
|
||||||
|
DeviceQuery,
|
||||||
|
PlatformQuery,
|
||||||
|
PurchaseQuery,
|
||||||
|
SessionQuery,
|
||||||
|
graphene.ObjectType,
|
||||||
|
):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Mutation(GameMutation, graphene.ObjectType):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
schema = graphene.Schema(query=Query, mutation=Mutation)
|
58
games/signals.py
Normal file
58
games/signals.py
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import logging
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from django.db.models import F, Sum
|
||||||
|
from django.db.models.signals import m2m_changed, post_delete, post_save, pre_save
|
||||||
|
from django.dispatch import receiver
|
||||||
|
from django.utils.timezone import now
|
||||||
|
|
||||||
|
from games.models import Game, GameStatusChange, Purchase, Session
|
||||||
|
|
||||||
|
logger = logging.getLogger("games")
|
||||||
|
|
||||||
|
|
||||||
|
@receiver(m2m_changed, sender=Purchase.games.through)
|
||||||
|
def update_num_purchases(sender, instance, **kwargs):
|
||||||
|
instance.num_purchases = instance.games.count()
|
||||||
|
instance.updated_at = now()
|
||||||
|
instance.save(update_fields=["num_purchases"])
|
||||||
|
|
||||||
|
|
||||||
|
@receiver([post_save, post_delete], sender=Session)
|
||||||
|
def update_game_playtime(sender, instance, **kwargs):
|
||||||
|
game = instance.game
|
||||||
|
total_playtime = game.sessions.aggregate(
|
||||||
|
total_playtime=Sum(F("duration_calculated") + F("duration_manual"))
|
||||||
|
)["total_playtime"]
|
||||||
|
game.playtime = total_playtime if total_playtime else timedelta(0)
|
||||||
|
game.save(update_fields=["playtime"])
|
||||||
|
|
||||||
|
|
||||||
|
@receiver(pre_save, sender=Game)
|
||||||
|
def game_status_changed(sender, instance, **kwargs):
|
||||||
|
"""
|
||||||
|
Signal handler to create a GameStatusChange record whenever a Game's status is updated.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
old_instance = sender.objects.get(pk=instance.pk)
|
||||||
|
old_status = old_instance.status
|
||||||
|
logger.info("[game_status_changed]: Previous status exists.")
|
||||||
|
except sender.DoesNotExist:
|
||||||
|
# Handle the case where the instance was deleted before the signal was sent
|
||||||
|
logger.info("[game_status_changed]: Previous status does not exist.")
|
||||||
|
return
|
||||||
|
|
||||||
|
if old_status != instance.status:
|
||||||
|
logger.info(
|
||||||
|
"[game_status_changed]: Status changed from {} to {}".format(
|
||||||
|
old_status, instance.status
|
||||||
|
)
|
||||||
|
)
|
||||||
|
GameStatusChange.objects.create(
|
||||||
|
game=instance,
|
||||||
|
old_status=old_status,
|
||||||
|
new_status=instance.status,
|
||||||
|
timestamp=now(),
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
logger.info("[game_status_changed]: Status has not changed")
|
File diff suppressed because it is too large
Load Diff
BIN
games/static/fonts/IBMPlexSansCondensed-Regular.woff2
Normal file
BIN
games/static/fonts/IBMPlexSansCondensed-Regular.woff2
Normal file
Binary file not shown.
BIN
games/static/fonts/IBMPlexSerif-Bold.woff2
Normal file
BIN
games/static/fonts/IBMPlexSerif-Bold.woff2
Normal file
Binary file not shown.
@ -7,7 +7,7 @@ import {
|
|||||||
|
|
||||||
let syncData = [
|
let syncData = [
|
||||||
{
|
{
|
||||||
source: "#id_edition",
|
source: "#id_games",
|
||||||
source_value: "dataset.platform",
|
source_value: "dataset.platform",
|
||||||
target: "#id_platform",
|
target: "#id_platform",
|
||||||
target_value: "value",
|
target_value: "value",
|
||||||
@ -21,11 +21,6 @@ function setupElementHandlers() {
|
|||||||
"#id_name",
|
"#id_name",
|
||||||
"#id_related_purchase",
|
"#id_related_purchase",
|
||||||
]);
|
]);
|
||||||
disableElementsWhenValueNotEqual(
|
|
||||||
"#id_type",
|
|
||||||
["game", "dlc"],
|
|
||||||
["#id_date_finished"]
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener("DOMContentLoaded", setupElementHandlers);
|
document.addEventListener("DOMContentLoaded", setupElementHandlers);
|
||||||
@ -36,8 +31,8 @@ getEl("#id_type").onchange = () => {
|
|||||||
|
|
||||||
document.body.addEventListener("htmx:beforeRequest", function (event) {
|
document.body.addEventListener("htmx:beforeRequest", function (event) {
|
||||||
// Assuming 'Purchase1' is the element that triggers the HTMX request
|
// Assuming 'Purchase1' is the element that triggers the HTMX request
|
||||||
if (event.target.id === "id_edition") {
|
if (event.target.id === "id_games") {
|
||||||
var idEditionValue = document.getElementById("id_edition").value;
|
var idEditionValue = document.getElementById("id_games").value;
|
||||||
|
|
||||||
// Condition to check - replace this with your actual logic
|
// Condition to check - replace this with your actual logic
|
||||||
if (idEditionValue != "") {
|
if (idEditionValue != "") {
|
||||||
|
@ -36,7 +36,7 @@ function addToggleButton(targetNode) {
|
|||||||
targetNode.parentElement.appendChild(manualToggleButton);
|
targetNode.parentElement.appendChild(manualToggleButton);
|
||||||
}
|
}
|
||||||
|
|
||||||
const toggleableFields = ["#id_game", "#id_edition", "#id_platform"];
|
const toggleableFields = ["#id_games", "#id_platform"];
|
||||||
|
|
||||||
toggleableFields.map((selector) => {
|
toggleableFields.map((selector) => {
|
||||||
addToggleButton(document.querySelector(selector));
|
addToggleButton(document.querySelector(selector));
|
||||||
|
94
games/tasks.py
Normal file
94
games/tasks.py
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
import requests
|
||||||
|
from django.db.models import ExpressionWrapper, F, FloatField, Q
|
||||||
|
from django.template.defaultfilters import floatformat
|
||||||
|
from django.utils.timezone import now
|
||||||
|
from django_q.models import Task
|
||||||
|
import logging
|
||||||
|
|
||||||
|
logger = logging.getLogger("games")
|
||||||
|
|
||||||
|
from games.models import ExchangeRate, Purchase
|
||||||
|
|
||||||
|
# fixme: save preferred currency in user model
|
||||||
|
currency_to = "CZK"
|
||||||
|
currency_to = currency_to.upper()
|
||||||
|
|
||||||
|
|
||||||
|
def save_converted_info(purchase, converted_price, converted_currency):
|
||||||
|
logger.info(
|
||||||
|
f"Setting converted price of {purchase} to {converted_price} {converted_currency} (originally {purchase.price} {purchase.price_currency})"
|
||||||
|
)
|
||||||
|
purchase.converted_price = converted_price
|
||||||
|
purchase.converted_currency = converted_currency
|
||||||
|
purchase.save()
|
||||||
|
|
||||||
|
|
||||||
|
def convert_prices():
|
||||||
|
purchases = Purchase.objects.filter(
|
||||||
|
converted_price__isnull=True, converted_currency=""
|
||||||
|
)
|
||||||
|
if purchases.count() == 0:
|
||||||
|
logger.info("[convert_prices]: No prices to convert.")
|
||||||
|
|
||||||
|
for purchase in purchases:
|
||||||
|
if purchase.price_currency.upper() == currency_to or purchase.price == 0:
|
||||||
|
save_converted_info(purchase, purchase.price, currency_to)
|
||||||
|
continue
|
||||||
|
year = purchase.date_purchased.year
|
||||||
|
currency_from = purchase.price_currency.upper()
|
||||||
|
|
||||||
|
exchange_rate = ExchangeRate.objects.filter(
|
||||||
|
currency_from=currency_from, currency_to=currency_to, year=year
|
||||||
|
).first()
|
||||||
|
logger.info(f"[convert_prices]: Looking for exchange rate in database: {currency_from}->{currency_to}")
|
||||||
|
if not exchange_rate:
|
||||||
|
logger.info(
|
||||||
|
f"[convert_prices]: Getting exchange rate from {currency_from} to {currency_to} for {year}..."
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
# this API endpoint only accepts lowercase currency string
|
||||||
|
response = requests.get(
|
||||||
|
f"https://cdn.jsdelivr.net/npm/@fawazahmed0/currency-api@{year}-01-01/v1/currencies/{currency_from.lower()}.json"
|
||||||
|
)
|
||||||
|
response.raise_for_status()
|
||||||
|
data = response.json()
|
||||||
|
currency_from_data = data.get(currency_from.lower())
|
||||||
|
rate = currency_from_data.get(currency_to.lower())
|
||||||
|
|
||||||
|
if rate:
|
||||||
|
logger.info(f"[convert_prices]: Got {rate}, saving...")
|
||||||
|
exchange_rate = ExchangeRate.objects.create(
|
||||||
|
currency_from=currency_from,
|
||||||
|
currency_to=currency_to,
|
||||||
|
year=year,
|
||||||
|
rate=floatformat(rate, 2),
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
logger.info("[convert_prices]: Could not get an exchange rate.")
|
||||||
|
except requests.RequestException as e:
|
||||||
|
logger.info(
|
||||||
|
f"[convert_prices]: Failed to fetch exchange rate for {currency_from}->{currency_to} in {year}: {e}"
|
||||||
|
)
|
||||||
|
if exchange_rate:
|
||||||
|
save_converted_info(
|
||||||
|
purchase,
|
||||||
|
floatformat(purchase.price * exchange_rate.rate, 0),
|
||||||
|
currency_to,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def calculate_price_per_game():
|
||||||
|
try:
|
||||||
|
last_task = Task.objects.filter(group="Update price per game").first()
|
||||||
|
last_run = last_task.started
|
||||||
|
except Task.DoesNotExist or AttributeError:
|
||||||
|
last_run = now()
|
||||||
|
purchases = Purchase.objects.filter(converted_price__isnull=False).filter(
|
||||||
|
Q(updated_at__gte=last_run) | Q(price_per_game__isnull=True)
|
||||||
|
)
|
||||||
|
logger.info(f"[calculate_price_per_game]: Updating {purchases.count()} purchases.")
|
||||||
|
purchases.update(
|
||||||
|
price_per_game=ExpressionWrapper(
|
||||||
|
F("converted_price") / F("num_purchases"), output_field=FloatField()
|
||||||
|
)
|
||||||
|
)
|
@ -1,24 +1,2 @@
|
|||||||
{% extends "base.html" %}
|
<c-layouts.add>
|
||||||
{% load static %}
|
</c-layouts.add>
|
||||||
{% block title %}
|
|
||||||
{{ title }}
|
|
||||||
{% endblock title %}
|
|
||||||
{% block content %}
|
|
||||||
<form method="post" enctype="multipart/form-data">
|
|
||||||
<table class="mx-auto">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form.as_table }}
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit" value="Submit" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
</form>
|
|
||||||
{% endblock content %}
|
|
||||||
{% block scripts %}
|
|
||||||
{% if script_name %}
|
|
||||||
<script type="module" src="{% static 'js/'|add:script_name %}"></script>
|
|
||||||
{% endif %}
|
|
||||||
{% endblock scripts %}
|
|
||||||
|
@ -1,32 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% load static %}
|
|
||||||
{% block title %}
|
|
||||||
{{ title }}
|
|
||||||
{% endblock title %}
|
|
||||||
{% block content %}
|
|
||||||
<form method="post" enctype="multipart/form-data">
|
|
||||||
<table class="mx-auto">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form.as_table }}
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit" name="submit" value="Submit" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit"
|
|
||||||
name="submit_and_redirect"
|
|
||||||
value="Submit & Create Purchase" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
</form>
|
|
||||||
{% endblock content %}
|
|
||||||
{% block scripts %}
|
|
||||||
{% if script_name %}
|
|
||||||
<script type="module" src="{% static 'js/'|add:script_name %}"></script>
|
|
||||||
{% endif %}
|
|
||||||
{% endblock scripts %}
|
|
@ -1,32 +1,7 @@
|
|||||||
{% extends "base.html" %}
|
<c-layouts.add>
|
||||||
{% load static %}
|
<c-slot name="additional_row">
|
||||||
{% block title %}
|
<input type="submit"
|
||||||
{{ title }}
|
name="submit_and_redirect"
|
||||||
{% endblock title %}
|
value="Submit & Create Purchase" />
|
||||||
{% block content %}
|
</c-slot>
|
||||||
<form method="post" enctype="multipart/form-data">
|
</c-layouts.add>
|
||||||
<table class="mx-auto">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form.as_table }}
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit" name="submit" value="Submit" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit"
|
|
||||||
name="submit_and_redirect"
|
|
||||||
value="Submit & Create Edition" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
</form>
|
|
||||||
{% endblock content %}
|
|
||||||
{% block scripts %}
|
|
||||||
{% if script_name %}
|
|
||||||
<script type="module" src="{% static 'js/'|add:script_name %}"></script>
|
|
||||||
{% endif %}
|
|
||||||
{% endblock scripts %}
|
|
||||||
|
@ -1,32 +1,12 @@
|
|||||||
{% extends "base.html" %}
|
<c-layouts.add>
|
||||||
{% load static %}
|
<c-slot name="additional_row">
|
||||||
{% block title %}
|
<tr>
|
||||||
{{ title }}
|
<td></td>
|
||||||
{% endblock title %}
|
<td>
|
||||||
{% block content %}
|
<input type="submit"
|
||||||
<form method="post" enctype="multipart/form-data">
|
name="submit_and_redirect"
|
||||||
<table class="mx-auto">
|
value="Submit & Create Session" />
|
||||||
{% csrf_token %}
|
</td>
|
||||||
{{ form.as_table }}
|
</tr>
|
||||||
<tr>
|
</c-slot>
|
||||||
<td></td>
|
</c-layouts.add>
|
||||||
<td>
|
|
||||||
<input type="submit" name="submit" value="Submit" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td></td>
|
|
||||||
<td>
|
|
||||||
<input type="submit"
|
|
||||||
name="submit_and_redirect"
|
|
||||||
value="Submit & Create Session" />
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
</form>
|
|
||||||
{% endblock content %}
|
|
||||||
{% block scripts %}
|
|
||||||
{% if script_name %}
|
|
||||||
<script type="module" src="{% static 'js/'|add:script_name %}"></script>
|
|
||||||
{% endif %}
|
|
||||||
{% endblock scripts %}
|
|
||||||
|
@ -1,40 +1,36 @@
|
|||||||
{% extends "base.html" %}
|
<c-layouts.add>
|
||||||
{% block title %}
|
<c-slot name="form_content">
|
||||||
{{ title }}
|
<form method="post" enctype="multipart/form-data">
|
||||||
{% endblock title %}
|
<table class="mx-auto">
|
||||||
{% block content %}
|
{% csrf_token %}
|
||||||
<form method="post" enctype="multipart/form-data">
|
{% for field in form %}
|
||||||
<table class="mx-auto">
|
|
||||||
{% csrf_token %}
|
|
||||||
{% for field in form %}
|
|
||||||
<tr>
|
|
||||||
<th>{{ field.label_tag }}</th>
|
|
||||||
{% if field.name == "note" %}
|
|
||||||
<td>{{ field }}</td>
|
|
||||||
{% else %}
|
|
||||||
<td>{{ field }}</td>
|
|
||||||
{% endif %}
|
|
||||||
{% if field.name == "timestamp_start" or field.name == "timestamp_end" %}
|
|
||||||
<td>
|
|
||||||
<div class="basic-button-container">
|
|
||||||
<button class="basic-button" data-target="{{ field.name }}" data-type="now">Set to now</button>
|
|
||||||
<button class="basic-button"
|
|
||||||
data-target="{{ field.name }}"
|
|
||||||
data-type="toggle">Toggle text</button>
|
|
||||||
<button class="basic-button" data-target="{{ field.name }}" data-type="copy">Copy</button>
|
|
||||||
</div>
|
|
||||||
</td>
|
|
||||||
{% endif %}
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
<tr>
|
<tr>
|
||||||
<td></td>
|
<th>{{ field.label_tag }}</th>
|
||||||
<td>
|
{% if field.name == "note" %}
|
||||||
<input type="submit" value="Submit" />
|
<td>{{ field }}</td>
|
||||||
</td>
|
{% else %}
|
||||||
|
<td>{{ field }}</td>
|
||||||
|
{% endif %}
|
||||||
|
{% if field.name == "timestamp_start" or field.name == "timestamp_end" %}
|
||||||
|
<td>
|
||||||
|
<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 }}"
|
||||||
|
data-type="toggle">Toggle text</button>
|
||||||
|
<button class="basic-button" data-target="{{ field.name }}" data-type="copy">Copy</button>
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
{% endfor %}
|
||||||
</form>
|
<tr>
|
||||||
{% load static %}
|
<td></td>
|
||||||
<script type="module" src="{% static 'js/add_session.js' %}"></script>
|
<td>
|
||||||
{% endblock content %}
|
<input type="submit" value="Submit" />
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</form>
|
||||||
|
</c-slot>
|
||||||
|
</c-layouts.add>
|
||||||
|
@ -1,101 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
{% load static %}
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="description" content="Self-hosted time-tracker." />
|
|
||||||
<meta name="keywords" content="time, tracking, video games, self-hosted" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<title>Timetracker -
|
|
||||||
{% block title %}
|
|
||||||
Untitled
|
|
||||||
{% endblock title %}
|
|
||||||
</title>
|
|
||||||
<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">
|
|
||||||
<img id="indicator"
|
|
||||||
src="{% static 'icons/loading.png' %}"
|
|
||||||
class="absolute right-3 top-3 animate-spin htmx-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" />
|
|
||||||
</span>
|
|
||||||
<span class="self-center text-xl font-semibold whitespace-nowrap text-white">Timetracker</span>
|
|
||||||
</a>
|
|
||||||
<div class="w-full md:block md:w-auto">
|
|
||||||
<ul class="flex flex-col md:flex-row p-4 mt-4 dark:text-white">
|
|
||||||
<li class="relative group">
|
|
||||||
<a class="block py-2 pl-3 pr-4 hover:underline"
|
|
||||||
href="{% url 'add_game' %}">New</a>
|
|
||||||
<ul class="absolute hidden text-gray-700 pt-1 group-hover:block w-auto whitespace-nowrap">
|
|
||||||
{% if purchase_available %}
|
|
||||||
<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>
|
|
||||||
{% endif %}
|
|
||||||
<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>
|
|
||||||
{% if game_available and platform_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>
|
|
||||||
{% endif %}
|
|
||||||
<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>
|
|
||||||
{% 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 %}
|
|
||||||
<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>
|
|
||||||
{% endif %}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
{% block content %}
|
|
||||||
No content here.
|
|
||||||
{% endblock content %}
|
|
||||||
</div>
|
|
||||||
{% load version %}
|
|
||||||
<span class="fixed left-2 bottom-2 text-xs text-slate-300 dark:text-slate-600">{% version %} ({% version_date %})</span>
|
|
||||||
{% block scripts %}
|
|
||||||
{% endblock scripts %}
|
|
||||||
</body>
|
|
||||||
</html>
|
|
@ -1,13 +0,0 @@
|
|||||||
<a href="{{ edit_url }}">
|
|
||||||
<button type="button"
|
|
||||||
title="Edit"
|
|
||||||
class="ml-1 py-1 px-2 flex justify-center items-center bg-violet-600 hover:bg-violet-700 focus:ring-violet-500 focus:ring-offset-violet-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 w-7 h-4 rounded-lg">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg"
|
|
||||||
viewBox="0 0 20 20"
|
|
||||||
fill="currentColor"
|
|
||||||
class="w-5 h-5">
|
|
||||||
<path d="M5.433 13.917l1.262-3.155A4 4 0 017.58 9.42l6.92-6.918a2.121 2.121 0 013 3l-6.92 6.918c-.383.383-.84.685-1.343.886l-3.154 1.262a.5.5 0 01-.65-.65z" />
|
|
||||||
<path d="M3.5 5.75c0-.69.56-1.25 1.25-1.25H10A.75.75 0 0010 3H4.75A2.75 2.75 0 002 5.75v9.5A2.75 2.75 0 004.75 18h9.5A2.75 2.75 0 0017 15.25V10a.75.75 0 00-1.5 0v5.25c0 .69-.56 1.25-1.25 1.25h-9.5c-.69 0-1.25-.56-1.25-1.25v-9.5z" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
</a>
|
|
6
games/templates/cotton/button.html
Normal file
6
games/templates/cotton/button.html
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<c-vars color="blue" size="base" type="button" />
|
||||||
|
<button type="{{ type }}"
|
||||||
|
title="{{ title }}"
|
||||||
|
class="{{ class }} {% if color == "blue" %} bg-blue-700 dark:bg-blue-600 dark:focus:ring-blue-800 dark:hover:bg-blue-700 focus:ring-blue-300 hover:bg-blue-800 text-white {% elif color == "red" %} bg-red-700 dark:bg-red-600 dark:focus:ring-red-900 dark:hover:bg-red-700 focus:ring-red-300 hover:bg-red-800 text-white {% elif color == "gray" %} bg-white border-gray-200 dark:bg-gray-800 dark:border-gray-600 dark:focus:ring-gray-700 dark:hover:bg-gray-700 dark:hover:text-white dark:text-gray-400 focus:ring-gray-100 hover:bg-gray-100 hover:text-blue-700 text-gray-900 border {% elif color == "green" %} bg-green-700 dark:bg-green-600 dark:focus:ring-green-800 dark:hover:bg-green-700 focus:ring-green-300 hover:bg-green-800 text-white {% endif %} focus:outline-none focus:ring-4 font-medium mb-2 me-2 rounded-lg {% if size == "xs" %} px-3 py-2 text-xs {% elif size == "sm" %} px-3 py-2 text-sm {% elif size == "base" %} px-5 py-2.5 text-sm {% elif size == "lg" %} px-5 py-3 text-base {% elif size == "xl" %} px-6 py-3.5 text-base {% endif %} {% if icon %} inline-flex text-center items-center gap-2 {% else %} {% endif %} ">
|
||||||
|
{{ slot }}
|
||||||
|
</button>
|
8
games/templates/cotton/button_group.html
Normal file
8
games/templates/cotton/button_group.html
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
<div class="inline-flex rounded-md shadow-sm" role="group">
|
||||||
|
{% if slot %}{{ slot }}{% endif %}
|
||||||
|
{% for button in buttons %}
|
||||||
|
{% if button.slot %}
|
||||||
|
<c-button-group-button-sm :href=button.href :slot=button.slot :color=button.color :hover=button.hover :title=button.title />
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user