Compare commits

..

1 Commits

Author SHA1 Message Date
Herculino Trotta 2f99021d0b feat: initial commit 2025-08-07 22:55:25 -03:00
300 changed files with 28864 additions and 38496 deletions
+45 -29
View File
@@ -12,7 +12,7 @@ on:
required: true required: true
type: string type: string
ref: ref:
description: 'Git ref to checkout' description: 'Git ref to checkout (branch, tag, or SHA)'
required: true required: true
default: 'main' default: 'main'
type: string type: string
@@ -29,57 +29,73 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
permissions: permissions:
contents: read contents: read
packages: write # Needed if you switch to GHCR, good practice
steps: steps:
- name: Checkout code - name: Checkout code
uses: actions/checkout@v4 uses: actions/checkout@v4
with: with:
ref: ${{ inputs.ref || github.ref }} ref: ${{ github.event.inputs.ref }}
if: github.event_name == 'workflow_dispatch'
- name: Checkout code (non-manual)
uses: actions/checkout@v4
if: github.event_name != 'workflow_dispatch'
- name: Log in to Docker Hub - name: Log in to Docker Hub
uses: docker/login-action@v3 uses: docker/login-action@65b78e6e13532edd9afa3aa52ac7964289d1a9c1
with: with:
username: ${{ secrets.DOCKERHUB_USERNAME }} username: ${{ secrets.DOCKERHUB_USERNAME }}
password: ${{ secrets.DOCKERHUB_TOKEN }} password: ${{ secrets.DOCKERHUB_TOKEN }}
# This action handles all the logic for tags (nightly vs release vs custom)
- name: Docker Metadata
id: meta
uses: docker/metadata-action@v5
with:
images: ${{ secrets.DOCKERHUB_USERNAME }}/${{ env.IMAGE_NAME }}
tags: |
# Logic for Push to Main -> nightly
type=raw,value=nightly,enable=${{ github.event_name == 'push' }}
# Logic for Release -> semver and latest
type=semver,pattern={{version}},enable=${{ github.event_name == 'release' }}
type=raw,value=latest,enable=${{ github.event_name == 'release' }}
# Logic for Manual Dispatch -> custom input
type=raw,value=${{ inputs.tag }},enable=${{ github.event_name == 'workflow_dispatch' }}
- name: Set up QEMU - name: Set up QEMU
uses: docker/setup-qemu-action@v3 uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx - name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3 uses: docker/setup-buildx-action@v3
- name: Build and push - name: Build and push nightly image
if: github.event_name == 'push'
uses: docker/build-push-action@v6 uses: docker/build-push-action@v6
with: with:
context: . context: .
file: ./docker/prod/django/Dockerfile file: ./docker/prod/django/Dockerfile
push: true push: true
provenance: false provenance: false
# Pass the calculated tags from the meta step
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
build-args: | build-args: |
VERSION=${{ steps.meta.outputs.version }} VERSION=nightly
tags: ${{ secrets.DOCKERHUB_USERNAME }}/${{ env.IMAGE_NAME }}:nightly
platforms: linux/amd64,linux/arm64 platforms: linux/amd64,linux/arm64
cache-from: type=gha
cache-to: type=gha,mode=max
# --- CACHE CONFIGURATION --- - name: Build and push release image
# We set a specific 'scope' key. if: github.event_name == 'release'
# This allows the Release tag to see the cache created by the Main branch. uses: docker/build-push-action@v6
cache-from: type=gha,scope=build-cache with:
cache-to: type=gha,mode=max,scope=build-cache context: .
file: ./docker/prod/django/Dockerfile
push: true
provenance: false
build-args: |
VERSION=${{ github.event.release.tag_name }}
tags: |
${{ secrets.DOCKERHUB_USERNAME }}/${{ env.IMAGE_NAME }}:latest
${{ secrets.DOCKERHUB_USERNAME }}/${{ env.IMAGE_NAME }}:${{ github.event.release.tag_name }}
platforms: linux/amd64,linux/arm64
cache-from: type=gha
cache-to: type=gha,mode=max
- name: Build and push custom image
if: github.event_name == 'workflow_dispatch'
uses: docker/build-push-action@v6
with:
context: .
file: ./docker/prod/django/Dockerfile
push: true
provenance: false
build-args: |
VERSION=${{ github.event.inputs.tag }}
tags: ${{ secrets.DOCKERHUB_USERNAME }}/${{ env.IMAGE_NAME }}:${{ github.event.inputs.tag }}
platforms: linux/amd64,linux/arm64
cache-from: type=gha
cache-to: type=gha,mode=max
-5
View File
@@ -123,7 +123,6 @@ celerybeat.pid
# Environments # Environments
.env .env
.prod.env
.venv .venv
env/ env/
venv/ venv/
@@ -161,7 +160,3 @@ cython_debug/
# and can be added to the global gitignore or merged into this file. For a more nuclear # and can be added to the global gitignore or merged into this file. For a more nuclear
# option (not recommended) you can uncomment the following to ignore the entire idea folder. # option (not recommended) you can uncomment the following to ignore the entire idea folder.
.idea/ .idea/
node_modules/
postgres_data/
.prod.env
-8
View File
@@ -1,8 +0,0 @@
{
"djlint.showInstallError": false,
"files.associations": {
"*.css": "tailwindcss"
},
"tailwindCSS.experimental.configFile": "frontend/src/styles/tailwind.css",
"djlint.profile": "django",
}
-6
View File
@@ -13,7 +13,6 @@
<a href="#key-features">Features</a> • <a href="#key-features">Features</a> •
<a href="#how-to-use">Usage</a> • <a href="#how-to-use">Usage</a> •
<a href="#how-it-works">How</a> • <a href="#how-it-works">How</a> •
<a href="#mcp-server">MCP Server</a> •
<a href="#help-us-translate-wygiwyh">Translate</a> • <a href="#help-us-translate-wygiwyh">Translate</a> •
<a href="#caveats-and-warnings">Caveats and Warnings</a> • <a href="#caveats-and-warnings">Caveats and Warnings</a> •
<a href="#built-with">Built with</a> <a href="#built-with">Built with</a>
@@ -127,7 +126,6 @@ To create the first user, open the container's console using Unraid's UI, by cli
| variable | type | default | explanation | | variable | type | default | explanation |
|-------------------------------|-------------|-----------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| |-------------------------------|-------------|-----------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| INTERNAL_PORT | int | 8000 | The port on which the app listens on. Defaults to 8000 if not set. |
| DJANGO_ALLOWED_HOSTS | string | localhost 127.0.0.1 | A list of space separated domains and IPs representing the host/domain names that WYGIWYH site can serve. [Click here](https://docs.djangoproject.com/en/5.1/ref/settings/#allowed-hosts) for more details | | DJANGO_ALLOWED_HOSTS | string | localhost 127.0.0.1 | A list of space separated domains and IPs representing the host/domain names that WYGIWYH site can serve. [Click here](https://docs.djangoproject.com/en/5.1/ref/settings/#allowed-hosts) for more details |
| HTTPS_ENABLED | true\|false | false | Whether to use secure cookies. If this is set to true, the cookie will be marked as “secure”, which means browsers may ensure that the cookie is only sent under an HTTPS connection | | HTTPS_ENABLED | true\|false | false | Whether to use secure cookies. If this is set to true, the cookie will be marked as “secure”, which means browsers may ensure that the cookie is only sent under an HTTPS connection |
| URL | string | http://localhost http://127.0.0.1 | A list of space separated domains and IPs (with the protocol) representing the trusted origins for unsafe requests (e.g. POST). [Click here](https://docs.djangoproject.com/en/5.1/ref/settings/#csrf-trusted-origins ) for more details | | URL | string | http://localhost http://127.0.0.1 | A list of space separated domains and IPs (with the protocol) representing the trusted origins for unsafe requests (e.g. POST). [Click here](https://docs.djangoproject.com/en/5.1/ref/settings/#csrf-trusted-origins ) for more details |
@@ -184,10 +182,6 @@ Check out our [Wiki](https://github.com/eitchtee/WYGIWYH/wiki) for more informat
> [!NOTE] > [!NOTE]
> Login with your github account > Login with your github account
# MCP Server
[IZIme07](https://github.com/IZIme07) has kindly created an MCP Server for WYGIWYH that you can self-host. [Check it out at MCP-WYGIWYH](https://github.com/ReNewator/MCP-WYGIWYH)!
# Caveats and Warnings # Caveats and Warnings
- I'm not an accountant, some terms and even calculations might be wrong. Make sure to open an issue if you see anything that could be improved. - I'm not an accountant, some terms and even calculations might be wrong. Make sure to open an issue if you see anything that could be improved.
+8 -22
View File
@@ -11,7 +11,6 @@ https://docs.djangoproject.com/en/5.1/ref/settings/
""" """
import os import os
import re
import sys import sys
from pathlib import Path from pathlib import Path
@@ -47,7 +46,7 @@ INSTALLED_APPS = [
"django.contrib.sites", "django.contrib.sites",
"whitenoise.runserver_nostatic", "whitenoise.runserver_nostatic",
"django.contrib.staticfiles", "django.contrib.staticfiles",
"django_vite", "webpack_boilerplate",
"django.contrib.humanize", "django.contrib.humanize",
"django.contrib.postgres", "django.contrib.postgres",
"django_browser_reload", "django_browser_reload",
@@ -129,14 +128,6 @@ STORAGES = {
WHITENOISE_MANIFEST_STRICT = False WHITENOISE_MANIFEST_STRICT = False
def immutable_file_test(path, url):
# Match vite (rollup)-generated hashes, à la, `some_file-CSliV9zW.js`
return re.match(r"^.+[.-][0-9a-zA-Z_-]{8,12}\..+$", url)
WHITENOISE_IMMUTABLE_FILE_TEST = immutable_file_test
WSGI_APPLICATION = "WYGIWYH.wsgi.application" WSGI_APPLICATION = "WYGIWYH.wsgi.application"
@@ -298,7 +289,7 @@ STATIC_URL = "static/"
STATIC_ROOT = BASE_DIR / "static_files" STATIC_ROOT = BASE_DIR / "static_files"
STATICFILES_DIRS = [ STATICFILES_DIRS = [
ROOT_DIR / "frontend" / "build", ROOT_DIR / "frontend/build",
BASE_DIR / "static", BASE_DIR / "static",
] ]
@@ -314,11 +305,9 @@ CACHES = {
} }
} }
DJANGO_VITE_ASSETS_PATH = STATIC_ROOT WEBPACK_LOADER = {
DJANGO_VITE_MANIFEST_PATH = DJANGO_VITE_ASSETS_PATH / "manifest.json" "MANIFEST_FILE": ROOT_DIR / "frontend/build/manifest.json",
DJANGO_VITE_DEV_MODE = DEBUG }
DJANGO_VITE_DEV_SERVER_PORT = 5173
DJANGO_VITE_DEV_SERVER_HOST = "localhost"
# Default primary key field type # Default primary key field type
# https://docs.djangoproject.com/en/5.1/ref/settings/#default-auto-field # https://docs.djangoproject.com/en/5.1/ref/settings/#default-auto-field
@@ -365,11 +354,8 @@ ACCOUNT_ADAPTER = "allauth.account.adapter.DefaultAccountAdapter"
SOCIALACCOUNT_ADAPTER = "allauth.socialaccount.adapter.DefaultSocialAccountAdapter" SOCIALACCOUNT_ADAPTER = "allauth.socialaccount.adapter.DefaultSocialAccountAdapter"
# CRISPY FORMS # CRISPY FORMS
CRISPY_ALLOWED_TEMPLATE_PACKS = [ CRISPY_ALLOWED_TEMPLATE_PACKS = ["bootstrap5", "crispy_forms/pure_text"]
"crispy_forms/pure_text", CRISPY_TEMPLATE_PACK = "bootstrap5"
"crispy-daisyui",
]
CRISPY_TEMPLATE_PACK = "crispy-daisyui"
SESSION_EXPIRE_AT_BROWSER_CLOSE = False SESSION_EXPIRE_AT_BROWSER_CLOSE = False
SESSION_COOKIE_AGE = int(os.getenv("SESSION_EXPIRY_TIME", 2678400)) # 31 days SESSION_COOKIE_AGE = int(os.getenv("SESSION_EXPIRY_TIME", 2678400)) # 31 days
@@ -393,7 +379,7 @@ DEBUG_TOOLBAR_PANELS = [
"debug_toolbar.panels.signals.SignalsPanel", "debug_toolbar.panels.signals.SignalsPanel",
"debug_toolbar.panels.redirects.RedirectsPanel", "debug_toolbar.panels.redirects.RedirectsPanel",
"debug_toolbar.panels.profiling.ProfilingPanel", "debug_toolbar.panels.profiling.ProfilingPanel",
# "cachalot.panels.CachalotPanel", "cachalot.panels.CachalotPanel",
] ]
INTERNAL_IPS = [ INTERNAL_IPS = [
"127.0.0.1", "127.0.0.1",
+28 -31
View File
@@ -1,21 +1,21 @@
from apps.accounts.models import Account, AccountGroup from crispy_bootstrap5.bootstrap5 import Switch
from apps.common.fields.forms.dynamic_select import (
DynamicModelChoiceField,
DynamicModelMultipleChoiceField,
)
from apps.common.widgets.crispy.daisyui import Switch
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
from apps.common.widgets.tom_select import TomSelect
from apps.currencies.models import Currency
from apps.transactions.models import TransactionCategory, TransactionTag
from crispy_forms.bootstrap import FormActions from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import Column, Field, Layout, Row from crispy_forms.layout import Layout, Field, Column, Row
from django import forms from django import forms
from django.db.models import Q
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.accounts.models import Account
from apps.accounts.models import AccountGroup
from apps.common.fields.forms.dynamic_select import (
DynamicModelMultipleChoiceField,
DynamicModelChoiceField,
)
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.tom_select import TomSelect
from apps.transactions.models import TransactionCategory, TransactionTag
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
class AccountGroupForm(forms.ModelForm): class AccountGroupForm(forms.ModelForm):
class Meta: class Meta:
@@ -36,13 +36,17 @@ class AccountGroupForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -75,18 +79,6 @@ class AccountForm(forms.ModelForm):
self.fields["group"].queryset = AccountGroup.objects.all() self.fields["group"].queryset = AccountGroup.objects.all()
if self.instance.id:
qs = Currency.objects.filter(
Q(is_archived=False) | Q(accounts=self.instance.id)
).distinct()
self.fields["currency"].queryset = qs
self.fields["exchange_currency"].queryset = qs
else:
qs = Currency.objects.filter(Q(is_archived=False))
self.fields["currency"].queryset = qs
self.fields["exchange_currency"].queryset = qs
self.helper = FormHelper() self.helper = FormHelper()
self.helper.form_tag = False self.helper.form_tag = False
self.helper.form_method = "post" self.helper.form_method = "post"
@@ -102,13 +94,17 @@ class AccountForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -146,8 +142,9 @@ class AccountBalanceForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
"new_balance", "new_balance",
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Field("account_id"), Field("account_id"),
) )
@@ -1,20 +0,0 @@
# Generated by Django 5.2.4 on 2025-08-09 05:52
from django.conf import settings
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('accounts', '0015_alter_account_owner_alter_account_shared_with_and_more'),
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
]
operations = [
migrations.AddField(
model_name='account',
name='untracked_by',
field=models.ManyToManyField(blank=True, related_name='untracked_accounts', to=settings.AUTH_USER_MODEL),
),
]
+2 -11
View File
@@ -1,11 +1,11 @@
from django.conf import settings from django.conf import settings
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 Q
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.middleware.thread_local import get_current_user
from apps.common.models import SharedObject, SharedObjectManager
from apps.transactions.models import Transaction from apps.transactions.models import Transaction
from apps.common.models import SharedObject, SharedObjectManager
class AccountGroup(SharedObject): class AccountGroup(SharedObject):
@@ -62,11 +62,6 @@ class Account(SharedObject):
verbose_name=_("Archived"), verbose_name=_("Archived"),
help_text=_("Archived accounts don't show up nor count towards your net worth"), help_text=_("Archived accounts don't show up nor count towards your net worth"),
) )
untracked_by = models.ManyToManyField(
settings.AUTH_USER_MODEL,
blank=True,
related_name="untracked_accounts",
)
objects = SharedObjectManager() objects = SharedObjectManager()
all_objects = models.Manager() # Unfiltered manager all_objects = models.Manager() # Unfiltered manager
@@ -80,10 +75,6 @@ class Account(SharedObject):
def __str__(self): def __str__(self):
return self.name return self.name
def is_untracked_by(self):
user = get_current_user()
return self.untracked_by.filter(pk=user.pk).exists()
def clean(self): def clean(self):
super().clean() super().clean()
if self.exchange_currency == self.currency: if self.exchange_currency == self.currency:
-5
View File
@@ -31,11 +31,6 @@ urlpatterns = [
views.account_take_ownership, views.account_take_ownership,
name="account_take_ownership", name="account_take_ownership",
), ),
path(
"account/<int:pk>/toggle-untracked/",
views.account_toggle_untracked,
name="account_toggle_untracked",
),
path("account-groups/", views.account_groups_index, name="account_groups_index"), path("account-groups/", views.account_groups_index, name="account_groups_index"),
path("account-groups/list/", views.account_groups_list, name="account_groups_list"), path("account-groups/list/", views.account_groups_list, name="account_groups_list"),
path("account-groups/add/", views.account_group_add, name="account_group_add"), path("account-groups/add/", views.account_group_add, name="account_group_add"),
-20
View File
@@ -155,26 +155,6 @@ def account_delete(request, pk):
) )
@only_htmx
@login_required
@require_http_methods(["GET"])
def account_toggle_untracked(request, pk):
account = get_object_or_404(Account, id=pk)
if account.is_untracked_by():
account.untracked_by.remove(request.user)
messages.success(request, _("Account is now tracked"))
else:
account.untracked_by.add(request.user)
messages.success(request, _("Account is now untracked"))
return HttpResponse(
status=204,
headers={
"HX-Trigger": "updated",
},
)
@only_htmx @only_htmx
@login_required @login_required
@require_http_methods(["GET"]) @require_http_methods(["GET"])
-1
View File
@@ -138,7 +138,6 @@ class RecurringTransactionSerializer(serializers.ModelSerializer):
def update(self, instance, validated_data): def update(self, instance, validated_data):
instance = super().update(instance, validated_data) instance = super().update(instance, validated_data)
instance.update_unpaid_transactions() instance.update_unpaid_transactions()
instance.generate_upcoming_transactions()
return instance return instance
+1 -4
View File
@@ -1,5 +1,3 @@
from copy import deepcopy
from rest_framework import viewsets from rest_framework import viewsets
from apps.api.custom.pagination import CustomPageNumberPagination from apps.api.custom.pagination import CustomPageNumberPagination
@@ -32,9 +30,8 @@ class TransactionViewSet(viewsets.ModelViewSet):
transaction_created.send(sender=instance) transaction_created.send(sender=instance)
def perform_update(self, serializer): def perform_update(self, serializer):
old_data = deepcopy(Transaction.objects.get(pk=serializer.data["pk"]))
instance = serializer.save() instance = serializer.save()
transaction_updated.send(sender=instance, old_data=old_data) transaction_updated.send(sender=instance)
def partial_update(self, request, *args, **kwargs): def partial_update(self, request, *args, **kwargs):
kwargs["partial"] = True kwargs["partial"] = True
@@ -139,6 +139,7 @@ class DynamicModelMultipleChoiceField(forms.ModelMultipleChoiceField):
instance.save() instance.save()
return instance return instance
except Exception as e: except Exception as e:
print(e)
raise ValidationError(_("Error creating new instance")) raise ValidationError(_("Error creating new instance"))
def clean(self, value): def clean(self, value):
+14 -9
View File
@@ -1,13 +1,14 @@
from apps.common.models import SharedObject
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.tom_select import TomSelect, TomSelectMultiple
from crispy_forms.bootstrap import FormActions from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Div, Field, Layout, Submit
from django import forms from django import forms
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
from django.core.exceptions import ValidationError
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from django.core.exceptions import ValidationError
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Field, Submit, Div, HTML
from apps.common.widgets.tom_select import TomSelect, TomSelectMultiple
from apps.common.models import SharedObject
from apps.common.widgets.crispy.submit import NoClassSubmit
User = get_user_model() User = get_user_model()
@@ -38,7 +39,6 @@ class SharedObjectForm(forms.Form):
choices=SharedObject.Visibility.choices, choices=SharedObject.Visibility.choices,
required=True, required=True,
label=_("Visibility"), label=_("Visibility"),
widget=TomSelect(clear_button=False),
help_text=_( help_text=_(
"Private: Only shown for the owner and shared users. Only editable by the owner." "Private: Only shown for the owner and shared users. Only editable by the owner."
"<br/>" "<br/>"
@@ -48,6 +48,9 @@ class SharedObjectForm(forms.Form):
class Meta: class Meta:
fields = ["visibility", "shared_with_users"] fields = ["visibility", "shared_with_users"]
widgets = {
"visibility": TomSelect(clear_button=False),
}
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
# Get the current user to filter available sharing options # Get the current user to filter available sharing options
@@ -70,10 +73,12 @@ class SharedObjectForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
Field("owner"), Field("owner"),
Field("visibility"), Field("visibility"),
HTML('<hr class="hr my-3">'), HTML("<hr>"),
Field("shared_with_users"), Field("shared_with_users"),
FormActions( FormActions(
NoClassSubmit("submit", _("Save"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Save"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
-3
View File
@@ -9,8 +9,5 @@ def truncate_decimal(value, decimal_places):
:param decimal_places: The number of decimal places to keep :param decimal_places: The number of decimal places to keep
:return: Truncated Decimal value :return: Truncated Decimal value
""" """
if isinstance(value, (int, float)):
value = Decimal(str(value))
multiplier = Decimal(10**decimal_places) multiplier = Decimal(10**decimal_places)
return (value * multiplier).to_integral_value(rounding=ROUND_DOWN) / multiplier return (value * multiplier).to_integral_value(rounding=ROUND_DOWN) / multiplier
+3 -10
View File
@@ -5,12 +5,7 @@ from django.utils.formats import get_format as original_get_format
def get_format(format_type=None, lang=None, use_l10n=None): def get_format(format_type=None, lang=None, use_l10n=None):
user = get_current_user() user = get_current_user()
if ( if user and user.is_authenticated and hasattr(user, "settings") and use_l10n:
user
and user.is_authenticated
and hasattr(user, "settings")
and use_l10n is not False
):
user_settings = user.settings user_settings = user.settings
if format_type == "THOUSAND_SEPARATOR": if format_type == "THOUSAND_SEPARATOR":
number_format = getattr(user_settings, "number_format", None) number_format = getattr(user_settings, "number_format", None)
@@ -18,13 +13,11 @@ def get_format(format_type=None, lang=None, use_l10n=None):
return "." return "."
elif number_format == "CD": elif number_format == "CD":
return "," return ","
elif number_format == "SD" or number_format == "SC":
return " "
elif format_type == "DECIMAL_SEPARATOR": elif format_type == "DECIMAL_SEPARATOR":
number_format = getattr(user_settings, "number_format", None) number_format = getattr(user_settings, "number_format", None)
if number_format == "DC" or number_format == "SC": if number_format == "DC":
return "," return ","
elif number_format == "CD" or number_format == "SD": elif number_format == "CD":
return "." return "."
elif format_type == "SHORT_DATE_FORMAT": elif format_type == "SHORT_DATE_FORMAT":
date_format = getattr(user_settings, "date_format", None) date_format = getattr(user_settings, "date_format", None)
+1 -1
View File
@@ -23,7 +23,7 @@ async def remove_old_jobs(context, timestamp):
return await builtin_tasks.remove_old_jobs( return await builtin_tasks.remove_old_jobs(
context, context,
max_hours=744, max_hours=744,
remove_failed=True, remove_error=True,
remove_cancelled=True, remove_cancelled=True,
remove_aborted=True, remove_aborted=True,
) )
@@ -1,13 +0,0 @@
from django import forms, template
register = template.Library()
@register.filter
def is_input(field):
return isinstance(field.field.widget, forms.TextInput)
@register.filter
def is_textarea(field):
return isinstance(field.field.widget, forms.Textarea)
+1 -1
View File
@@ -11,7 +11,7 @@ def toast_bg(tags):
elif "warning" in tags: elif "warning" in tags:
return "warning" return "warning"
elif "error" in tags: elif "error" in tags:
return "error" return "danger"
elif "info" in tags: elif "info" in tags:
return "info" return "info"
-7
View File
@@ -91,12 +91,6 @@ def month_year_picker(request):
for date in all_months for date in all_months
] ]
today_url = (
reverse(url, kwargs={"month": current_date.month, "year": current_date.year})
if url
else ""
)
return render( return render(
request, request,
"common/fragments/month_year_picker.html", "common/fragments/month_year_picker.html",
@@ -104,7 +98,6 @@ def month_year_picker(request):
"month_year_data": result, "month_year_data": result,
"current_month": current_month, "current_month": current_month,
"current_year": current_year, "current_year": current_year,
"today_url": today_url,
}, },
) )
@@ -1,5 +0,0 @@
from crispy_forms.layout import Field
class Switch(Field):
template = "crispy-daisyui/layout/switch.html"
+6 -7
View File
@@ -1,14 +1,15 @@
import datetime import datetime
from apps.common.functions.format import get_format from django.forms import widgets
from django.utils import formats, translation, dates
from django.utils.translation import gettext_lazy as _
from apps.common.utils.django import ( from apps.common.utils.django import (
django_to_python_datetime,
django_to_airdatepicker_datetime, django_to_airdatepicker_datetime,
django_to_airdatepicker_datetime_separated, django_to_airdatepicker_datetime_separated,
django_to_python_datetime,
) )
from django.forms import widgets from apps.common.functions.format import get_format
from django.utils import dates, formats, translation
from django.utils.translation import gettext_lazy as _
class AirDatePickerInput(widgets.DateInput): class AirDatePickerInput(widgets.DateInput):
@@ -51,8 +52,6 @@ class AirDatePickerInput(widgets.DateInput):
def build_attrs(self, base_attrs, extra_attrs=None): def build_attrs(self, base_attrs, extra_attrs=None):
attrs = super().build_attrs(base_attrs, extra_attrs) attrs = super().build_attrs(base_attrs, extra_attrs)
attrs["class"] = attrs.get("class", "") + " input"
attrs["data-now-button-txt"] = _("Today") attrs["data-now-button-txt"] = _("Today")
attrs["data-auto-close"] = str(self.auto_close).lower() attrs["data-auto-close"] = str(self.auto_close).lower()
attrs["data-clear-button"] = str(self.clear_button).lower() attrs["data-clear-button"] = str(self.clear_button).lower()
+3 -2
View File
@@ -35,8 +35,9 @@ class ArbitraryDecimalDisplayNumberInput(forms.TextInput):
self.attrs.update( self.attrs.update(
{ {
"x-data": "", "x-data": "",
"x-mask:dynamic": f"$money($input, '{get_format('DECIMAL_SEPARATOR')}', '{get_format('THOUSAND_SEPARATOR')}', '30')", "x-mask:dynamic": f"$money($input, '{get_format('DECIMAL_SEPARATOR')}', "
"x-on:keyup": "if (!['Control', 'Shift', 'Alt', 'Meta'].includes($event.key) && !(($event.ctrlKey || $event.metaKey) && $event.key.toLowerCase() === 'a')) $el.dispatchEvent(new Event('input'))", f"'{get_format('THOUSAND_SEPARATOR')}', '30')",
"x-on:keyup": "$el.dispatchEvent(new Event('input'))",
} }
) )
+2 -2
View File
@@ -1,4 +1,4 @@
from django.forms import SelectMultiple, widgets from django.forms import widgets, SelectMultiple
from django.urls import reverse from django.urls import reverse
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
@@ -17,7 +17,7 @@ class TomSelect(widgets.Select):
checkboxes=False, checkboxes=False,
group_by=None, group_by=None,
*args, *args,
**kwargs, **kwargs
): ):
super().__init__(attrs, *args, **kwargs) super().__init__(attrs, *args, **kwargs)
self.remove_button = remove_button self.remove_button = remove_button
+24 -61
View File
@@ -4,7 +4,13 @@ from datetime import timedelta
from django.db.models import QuerySet from django.db.models import QuerySet
from django.utils import timezone from django.utils import timezone
import apps.currencies.exchange_rates.providers as providers from apps.currencies.exchange_rates.providers import (
SynthFinanceProvider,
SynthFinanceStockProvider,
CoinGeckoFreeProvider,
CoinGeckoProProvider,
TransitiveRateProvider,
)
from apps.currencies.models import ExchangeRateService, ExchangeRate, Currency from apps.currencies.models import ExchangeRateService, ExchangeRate, Currency
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
@@ -12,12 +18,11 @@ logger = logging.getLogger(__name__)
# Map service types to provider classes # Map service types to provider classes
PROVIDER_MAPPING = { PROVIDER_MAPPING = {
"coingecko_free": providers.CoinGeckoFreeProvider, "synth_finance": SynthFinanceProvider,
"coingecko_pro": providers.CoinGeckoProProvider, "synth_finance_stock": SynthFinanceStockProvider,
"transitive": providers.TransitiveRateProvider, "coingecko_free": CoinGeckoFreeProvider,
"frankfurter": providers.FrankfurterProvider, "coingecko_pro": CoinGeckoProProvider,
"twelvedata": providers.TwelveDataProvider, "transitive": TransitiveRateProvider,
"twelvedatamarkets": providers.TwelveDataMarketsProvider,
} }
@@ -198,63 +203,21 @@ class ExchangeRateFetcher:
if provider.rates_inverted: if provider.rates_inverted:
# If rates are inverted, we need to swap currencies # If rates are inverted, we need to swap currencies
if service.singleton: ExchangeRate.objects.create(
# Try to get the last automatically created exchange rate from_currency=to_currency,
exchange_rate = ( to_currency=from_currency,
ExchangeRate.objects.filter( rate=rate,
automatic=True, date=timezone.now(),
from_currency=to_currency, )
to_currency=from_currency,
)
.order_by("-date")
.first()
)
else:
exchange_rate = None
if not exchange_rate:
ExchangeRate.objects.create(
automatic=True,
from_currency=to_currency,
to_currency=from_currency,
rate=rate,
date=timezone.now(),
)
else:
exchange_rate.rate = rate
exchange_rate.date = timezone.now()
exchange_rate.save()
processed_pairs.add((to_currency.id, from_currency.id)) processed_pairs.add((to_currency.id, from_currency.id))
else: else:
# If rates are not inverted, we can use them as is # If rates are not inverted, we can use them as is
if service.singleton: ExchangeRate.objects.create(
# Try to get the last automatically created exchange rate from_currency=from_currency,
exchange_rate = ( to_currency=to_currency,
ExchangeRate.objects.filter( rate=rate,
automatic=True, date=timezone.now(),
from_currency=from_currency, )
to_currency=to_currency,
)
.order_by("-date")
.first()
)
else:
exchange_rate = None
if not exchange_rate:
ExchangeRate.objects.create(
automatic=True,
from_currency=from_currency,
to_currency=to_currency,
rate=rate,
date=timezone.now(),
)
else:
exchange_rate.rate = rate
exchange_rate.date = timezone.now()
exchange_rate.save()
processed_pairs.add((from_currency.id, to_currency.id)) processed_pairs.add((from_currency.id, to_currency.id))
service.last_fetch = timezone.now() service.last_fetch = timezone.now()
+129 -326
View File
@@ -13,6 +13,70 @@ from apps.currencies.exchange_rates.base import ExchangeRateProvider
logger = logging.getLogger(__name__) logger = logging.getLogger(__name__)
class SynthFinanceProvider(ExchangeRateProvider):
"""Implementation for Synth Finance API (synthfinance.com)"""
BASE_URL = "https://api.synthfinance.com/rates/live"
rates_inverted = False # SynthFinance returns non-inverted rates
def __init__(self, api_key: str = None):
super().__init__(api_key)
self.session = requests.Session()
self.session.headers.update({"Authorization": f"Bearer {self.api_key}"})
def get_rates(
self, target_currencies: QuerySet, exchange_currencies: set
) -> List[Tuple[Currency, Currency, Decimal]]:
results = []
currency_groups = {}
for currency in target_currencies:
if currency.exchange_currency in exchange_currencies:
group = currency_groups.setdefault(currency.exchange_currency.code, [])
group.append(currency)
for base_currency, currencies in currency_groups.items():
try:
to_currencies = ",".join(
currency.code
for currency in currencies
if currency.code != base_currency
)
response = self.session.get(
f"{self.BASE_URL}",
params={"from": base_currency, "to": to_currencies},
)
response.raise_for_status()
data = response.json()
rates = data["data"]["rates"]
for currency in currencies:
if currency.code == base_currency:
rate = Decimal("1")
else:
rate = Decimal(str(rates[currency.code]))
# Return the rate as is, without inversion
results.append((currency.exchange_currency, currency, rate))
credits_used = data["meta"]["credits_used"]
credits_remaining = data["meta"]["credits_remaining"]
logger.info(
f"Synth Finance API call: {credits_used} credits used, {credits_remaining} remaining"
)
except requests.RequestException as e:
logger.error(
f"Error fetching rates from Synth Finance API for base {base_currency}: {e}"
)
except KeyError as e:
logger.error(
f"Unexpected response structure from Synth Finance API for base {base_currency}: {e}"
)
except Exception as e:
logger.error(
f"Unexpected error processing Synth Finance data for base {base_currency}: {e}"
)
return results
class CoinGeckoFreeProvider(ExchangeRateProvider): class CoinGeckoFreeProvider(ExchangeRateProvider):
"""Implementation for CoinGecko Free API""" """Implementation for CoinGecko Free API"""
@@ -88,6 +152,71 @@ class CoinGeckoProProvider(CoinGeckoFreeProvider):
self.session.headers.update({"x-cg-pro-api-key": api_key}) self.session.headers.update({"x-cg-pro-api-key": api_key})
class SynthFinanceStockProvider(ExchangeRateProvider):
"""Implementation for Synth Finance API Real-Time Prices endpoint (synthfinance.com)"""
BASE_URL = "https://api.synthfinance.com/tickers"
rates_inverted = True
def __init__(self, api_key: str = None):
super().__init__(api_key)
self.session = requests.Session()
self.session.headers.update(
{"Authorization": f"Bearer {self.api_key}", "accept": "application/json"}
)
def get_rates(
self, target_currencies: QuerySet, exchange_currencies: set
) -> List[Tuple[Currency, Currency, Decimal]]:
results = []
for currency in target_currencies:
if currency.exchange_currency not in exchange_currencies:
continue
try:
# Same currency has rate of 1
if currency.code == currency.exchange_currency.code:
rate = Decimal("1")
results.append((currency.exchange_currency, currency, rate))
continue
# Fetch real-time price for this ticker
response = self.session.get(
f"{self.BASE_URL}/{currency.code}/real-time"
)
response.raise_for_status()
data = response.json()
# Use fair market value as the rate
rate = Decimal(data["data"]["fair_market_value"])
results.append((currency.exchange_currency, currency, rate))
# Log API usage
credits_used = data["meta"]["credits_used"]
credits_remaining = data["meta"]["credits_remaining"]
logger.info(
f"Synth Finance API call for {currency.code}: {credits_used} credits used, {credits_remaining} remaining"
)
except requests.RequestException as e:
logger.error(
f"Error fetching rate from Synth Finance API for ticker {currency.code}: {e}",
exc_info=True,
)
except KeyError as e:
logger.error(
f"Unexpected response structure from Synth Finance API for ticker {currency.code}: {e}",
exc_info=True,
)
except Exception as e:
logger.error(
f"Unexpected error processing Synth Finance data for ticker {currency.code}: {e}",
exc_info=True,
)
return results
class TransitiveRateProvider(ExchangeRateProvider): class TransitiveRateProvider(ExchangeRateProvider):
"""Calculates exchange rates through paths of existing rates""" """Calculates exchange rates through paths of existing rates"""
@@ -177,329 +306,3 @@ class TransitiveRateProvider(ExchangeRateProvider):
queue.append((neighbor, path + [neighbor], current_rate * rate)) queue.append((neighbor, path + [neighbor], current_rate * rate))
return None, None return None, None
class FrankfurterProvider(ExchangeRateProvider):
"""Implementation for the Frankfurter API (frankfurter.dev)"""
BASE_URL = "https://api.frankfurter.dev/v1/latest"
rates_inverted = (
False # Frankfurter returns non-inverted rates (e.g., 1 EUR = 1.1 USD)
)
def __init__(self, api_key: str = None):
"""
Initializes the provider. The Frankfurter API does not require an API key,
so the api_key parameter is ignored.
"""
super().__init__(api_key)
self.session = requests.Session()
@classmethod
def requires_api_key(cls) -> bool:
return False
def get_rates(
self, target_currencies: QuerySet, exchange_currencies: set
) -> List[Tuple[Currency, Currency, Decimal]]:
results = []
currency_groups = {}
# Group target currencies by their exchange (base) currency to minimize API calls
for currency in target_currencies:
if currency.exchange_currency in exchange_currencies:
group = currency_groups.setdefault(currency.exchange_currency.code, [])
group.append(currency)
# Make one API call for each base currency
for base_currency, currencies in currency_groups.items():
try:
# Create a comma-separated list of target currency codes
to_currencies = ",".join(
currency.code
for currency in currencies
if currency.code != base_currency
)
# If there are no target currencies other than the base, skip the API call
if not to_currencies:
# Handle the case where the only request is for the base rate (e.g., USD to USD)
for currency in currencies:
if currency.code == base_currency:
results.append(
(currency.exchange_currency, currency, Decimal("1"))
)
continue
response = self.session.get(
self.BASE_URL,
params={"base": base_currency, "symbols": to_currencies},
)
response.raise_for_status()
data = response.json()
rates = data["rates"]
# Process the returned rates
for currency in currencies:
if currency.code == base_currency:
# The rate for the base currency to itself is always 1
rate = Decimal("1")
else:
rate = Decimal(str(rates[currency.code]))
results.append((currency.exchange_currency, currency, rate))
except requests.RequestException as e:
logger.error(
f"Error fetching rates from Frankfurter API for base {base_currency}: {e}"
)
except KeyError as e:
logger.error(
f"Unexpected response structure from Frankfurter API for base {base_currency}: {e}"
)
except Exception as e:
logger.error(
f"Unexpected error processing Frankfurter data for base {base_currency}: {e}"
)
return results
class TwelveDataProvider(ExchangeRateProvider):
"""Implementation for the Twelve Data API (twelvedata.com)"""
BASE_URL = "https://api.twelvedata.com/exchange_rate"
rates_inverted = (
False # The API returns direct rates, e.g., for EUR/USD it's 1 EUR = X USD
)
def __init__(self, api_key: str):
"""
Initializes the provider with an API key and a requests session.
"""
super().__init__(api_key)
self.session = requests.Session()
@classmethod
def requires_api_key(cls) -> bool:
"""This provider requires an API key."""
return True
def get_rates(
self, target_currencies: QuerySet, exchange_currencies: set
) -> List[Tuple[Currency, Currency, Decimal]]:
"""
Fetches exchange rates from the Twelve Data API for the given currency pairs.
This provider makes one API call for each requested currency pair.
"""
results = []
for target_currency in target_currencies:
# Ensure the target currency's exchange currency is one we're interested in
if target_currency.exchange_currency not in exchange_currencies:
continue
base_currency = target_currency.exchange_currency
# The exchange rate for the same currency is always 1
if base_currency.code == target_currency.code:
rate = Decimal("1")
results.append((base_currency, target_currency, rate))
continue
# Construct the symbol in the format "BASE/TARGET", e.g., "EUR/USD"
symbol = f"{base_currency.code}/{target_currency.code}"
try:
params = {
"symbol": symbol,
"apikey": self.api_key,
}
response = self.session.get(self.BASE_URL, params=params)
response.raise_for_status() # Raise an HTTPError for bad responses (4xx or 5xx)
data = response.json()
# The API may return an error message in a JSON object
if "rate" not in data:
error_message = data.get("message", "Rate not found in response.")
logger.error(
f"Could not fetch rate for {symbol} from Twelve Data: {error_message}"
)
continue
# Convert the rate to a Decimal for precision
rate = Decimal(str(data["rate"]))
results.append((base_currency, target_currency, rate))
logger.info(f"Successfully fetched rate for {symbol} from Twelve Data.")
time.sleep(
60
) # We sleep every pair as to not step over TwelveData's minute limit
except requests.RequestException as e:
logger.error(
f"Error fetching rate from Twelve Data API for symbol {symbol}: {e}"
)
except KeyError as e:
logger.error(
f"Unexpected response structure from Twelve Data API for symbol {symbol}: Missing key {e}"
)
except Exception as e:
logger.error(
f"An unexpected error occurred while processing Twelve Data for {symbol}: {e}"
)
return results
class TwelveDataMarketsProvider(ExchangeRateProvider):
"""
Provides prices for market instruments (stocks, ETFs, etc.) using the Twelve Data API.
This provider performs a multi-step process:
1. Parses instrument codes which can be symbols, FIGI, CUSIP, or ISIN.
2. For CUSIPs, it defaults the currency to USD. For all others, it searches
for the instrument to determine its native trading currency.
3. Fetches the latest price for the instrument in its native currency.
4. Converts the price to the requested target exchange currency.
"""
SYMBOL_SEARCH_URL = "https://api.twelvedata.com/symbol_search"
PRICE_URL = "https://api.twelvedata.com/price"
EXCHANGE_RATE_URL = "https://api.twelvedata.com/exchange_rate"
rates_inverted = True
def __init__(self, api_key: str):
super().__init__(api_key)
self.session = requests.Session()
@classmethod
def requires_api_key(cls) -> bool:
return True
def _parse_code(self, raw_code: str) -> Tuple[str, str]:
"""Parses the raw code to determine its type and value."""
if raw_code.startswith("figi:"):
return "figi", raw_code.removeprefix("figi:")
if raw_code.startswith("cusip:"):
return "cusip", raw_code.removeprefix("cusip:")
if raw_code.startswith("isin:"):
return "isin", raw_code.removeprefix("isin:")
return "symbol", raw_code
def get_rates(
self, target_currencies: QuerySet, exchange_currencies: set
) -> List[Tuple[Currency, Currency, Decimal]]:
results = []
for asset in target_currencies:
if asset.exchange_currency not in exchange_currencies:
continue
code_type, code_value = self._parse_code(asset.code)
original_currency_code = None
try:
# Determine the instrument's native currency
if code_type == "cusip":
# CUSIP codes always default to USD
original_currency_code = "USD"
logger.info(f"Defaulting CUSIP {code_value} to USD currency.")
else:
# For all other types, find currency via symbol search
search_params = {"symbol": code_value, "apikey": "demo"}
search_res = self.session.get(
self.SYMBOL_SEARCH_URL, params=search_params
)
search_res.raise_for_status()
search_data = search_res.json()
if not search_data.get("data"):
logger.warning(
f"TwelveDataMarkets: Symbol search for '{code_value}' returned no results."
)
continue
instrument_data = search_data["data"][0]
original_currency_code = instrument_data.get("currency")
if not original_currency_code:
logger.error(
f"TwelveDataMarkets: Could not determine original currency for '{code_value}'."
)
continue
# Get the instrument's price in its native currency
price_params = {code_type: code_value, "apikey": self.api_key}
price_res = self.session.get(self.PRICE_URL, params=price_params)
price_res.raise_for_status()
price_data = price_res.json()
if "price" not in price_data:
error_message = price_data.get(
"message", "Price key not found in response"
)
logger.error(
f"TwelveDataMarkets: Could not get price for {code_type} '{code_value}': {error_message}"
)
continue
price_in_original_currency = Decimal(price_data["price"])
# Convert price to the target exchange currency
target_exchange_currency = asset.exchange_currency
if (
original_currency_code.upper()
== target_exchange_currency.code.upper()
):
final_price = price_in_original_currency
else:
rate_symbol = (
f"{original_currency_code}/{target_exchange_currency.code}"
)
rate_params = {"symbol": rate_symbol, "apikey": self.api_key}
rate_res = self.session.get(
self.EXCHANGE_RATE_URL, params=rate_params
)
rate_res.raise_for_status()
rate_data = rate_res.json()
if "rate" not in rate_data:
error_message = rate_data.get(
"message", "Rate key not found in response"
)
logger.error(
f"TwelveDataMarkets: Could not get conversion rate for '{rate_symbol}': {error_message}"
)
continue
conversion_rate = Decimal(str(rate_data["rate"]))
final_price = price_in_original_currency * conversion_rate
results.append((target_exchange_currency, asset, final_price))
logger.info(
f"Successfully processed price for {asset.code} as {final_price} {target_exchange_currency.code}"
)
time.sleep(
60
) # We sleep every pair as to not step over TwelveData's minute limit
except requests.RequestException as e:
logger.error(
f"TwelveDataMarkets: API request failed for {code_value}: {e}"
)
except (KeyError, IndexError) as e:
logger.error(
f"TwelveDataMarkets: Error processing API response for {code_value}: {e}"
)
except Exception as e:
logger.error(
f"TwelveDataMarkets: An unexpected error occurred for {code_value}: {e}"
)
return results
+28 -19
View File
@@ -1,15 +1,16 @@
from apps.common.widgets.crispy.daisyui import Switch from crispy_bootstrap5.bootstrap5 import Switch
from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Row, Column
from django import forms
from django.forms import CharField
from django.utils.translation import gettext_lazy as _
from apps.common.widgets.crispy.submit import NoClassSubmit from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.datepicker import AirDateTimePickerInput from apps.common.widgets.datepicker import AirDateTimePickerInput
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
from apps.common.widgets.tom_select import TomSelect from apps.common.widgets.tom_select import TomSelect
from apps.currencies.models import Currency, ExchangeRate, ExchangeRateService from apps.currencies.models import Currency, ExchangeRate, ExchangeRateService
from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Column, Layout, Row
from django import forms
from django.forms import CharField
from django.utils.translation import gettext_lazy as _
class CurrencyForm(forms.ModelForm): class CurrencyForm(forms.ModelForm):
@@ -25,7 +26,6 @@ class CurrencyForm(forms.ModelForm):
"suffix", "suffix",
"code", "code",
"exchange_currency", "exchange_currency",
"is_archived",
] ]
widgets = { widgets = {
"exchange_currency": TomSelect(), "exchange_currency": TomSelect(),
@@ -40,7 +40,6 @@ class CurrencyForm(forms.ModelForm):
self.helper.layout = Layout( self.helper.layout = Layout(
"code", "code",
"name", "name",
Switch("is_archived"),
"decimal_places", "decimal_places",
"prefix", "prefix",
"suffix", "suffix",
@@ -50,13 +49,17 @@ class CurrencyForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -84,13 +87,17 @@ class ExchangeRateForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -107,7 +114,6 @@ class ExchangeRateServiceForm(forms.ModelForm):
"fetch_interval", "fetch_interval",
"target_currencies", "target_currencies",
"target_accounts", "target_accounts",
"singleton",
] ]
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
@@ -120,11 +126,10 @@ class ExchangeRateServiceForm(forms.ModelForm):
"name", "name",
"service_type", "service_type",
Switch("is_active"), Switch("is_active"),
Switch("singleton"),
"api_key", "api_key",
Row( Row(
Column("interval_type"), Column("interval_type", css_class="form-group col-md-6"),
Column("fetch_interval"), Column("fetch_interval", css_class="form-group col-md-6"),
), ),
"target_currencies", "target_currencies",
"target_accounts", "target_accounts",
@@ -133,12 +138,16 @@ class ExchangeRateServiceForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -1,23 +0,0 @@
# Generated by Django 5.2.4 on 2025-08-08 02:18
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0014_alter_currency_options'),
]
operations = [
migrations.AddField(
model_name='exchangerate',
name='automatic',
field=models.BooleanField(default=False, verbose_name='Automatic'),
),
migrations.AddField(
model_name='exchangerateservice',
name='singleton',
field=models.BooleanField(default=False, help_text='Create one exchange rate and keep updating it. Avoids database clutter.', verbose_name='Single exchange rate'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.4 on 2025-08-08 02:38
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0015_exchangerate_automatic_exchangerateservice_singleton'),
]
operations = [
migrations.AlterField(
model_name='exchangerate',
name='automatic',
field=models.BooleanField(default=False, verbose_name='Auto'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-16 22:18
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0016_alter_exchangerate_automatic'),
]
operations = [
migrations.AlterField(
model_name='exchangerateservice',
name='service_type',
field=models.CharField(choices=[('synth_finance', 'Synth Finance'), ('synth_finance_stock', 'Synth Finance Stock'), ('coingecko_free', 'CoinGecko (Demo/Free)'), ('coingecko_pro', 'CoinGecko (Pro)'), ('transitive', 'Transitive (Calculated from Existing Rates)'), ('frankfurter', 'Frankfurter')], max_length=255, verbose_name='Service Type'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-17 03:54
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0017_alter_exchangerateservice_service_type'),
]
operations = [
migrations.AlterField(
model_name='exchangerateservice',
name='service_type',
field=models.CharField(choices=[('synth_finance', 'Synth Finance'), ('synth_finance_stock', 'Synth Finance Stock'), ('coingecko_free', 'CoinGecko (Demo/Free)'), ('coingecko_pro', 'CoinGecko (Pro)'), ('transitive', 'Transitive (Calculated from Existing Rates)'), ('frankfurter', 'Frankfurter'), ('twelvedata', 'TwelveData')], max_length=255, verbose_name='Service Type'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-17 06:01
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0018_alter_exchangerateservice_service_type'),
]
operations = [
migrations.AlterField(
model_name='exchangerateservice',
name='service_type',
field=models.CharField(choices=[('synth_finance', 'Synth Finance'), ('synth_finance_stock', 'Synth Finance Stock'), ('coingecko_free', 'CoinGecko (Demo/Free)'), ('coingecko_pro', 'CoinGecko (Pro)'), ('transitive', 'Transitive (Calculated from Existing Rates)'), ('frankfurter', 'Frankfurter'), ('twelvedata', 'TwelveData'), ('twelvedatamarkets', 'TwelveData Markets')], max_length=255, verbose_name='Service Type'),
),
]
@@ -1,51 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-17 06:25
from django.db import migrations
# The new value we are migrating to
NEW_SERVICE_TYPE = "frankfurter"
# The old values we are deprecating
OLD_SERVICE_TYPE_TO_UPDATE = "synth_finance"
OLD_SERVICE_TYPE_TO_DELETE = "synth_finance_stock"
def forwards_func(apps, schema_editor):
"""
Forward migration:
- Deletes all ExchangeRateService instances with service_type 'synth_finance_stock'.
- Updates all ExchangeRateService instances with service_type 'synth_finance' to 'frankfurter'.
"""
ExchangeRateService = apps.get_model("currencies", "ExchangeRateService")
db_alias = schema_editor.connection.alias
# 1. Delete the SYNTH_FINANCE_STOCK entries
ExchangeRateService.objects.using(db_alias).filter(
service_type=OLD_SERVICE_TYPE_TO_DELETE
).delete()
# 2. Update the SYNTH_FINANCE entries to FRANKFURTER
ExchangeRateService.objects.using(db_alias).filter(
service_type=OLD_SERVICE_TYPE_TO_UPDATE
).update(service_type=NEW_SERVICE_TYPE, api_key=None)
def backwards_func(apps, schema_editor):
"""
Backward migration: This operation is not safely reversible.
- We cannot know which 'frankfurter' services were originally 'synth_finance'.
- The deleted 'synth_finance_stock' services cannot be recovered.
We will leave this function empty to allow migrating backwards without doing anything.
"""
pass
class Migration(migrations.Migration):
dependencies = [
# Add the previous migration file here
("currencies", "0019_alter_exchangerateservice_service_type"),
]
operations = [
migrations.RunPython(forwards_func, reverse_code=backwards_func),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-17 06:29
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0020_migrate_synth_finance_services'),
]
operations = [
migrations.AlterField(
model_name='exchangerateservice',
name='service_type',
field=models.CharField(choices=[('coingecko_free', 'CoinGecko (Demo/Free)'), ('coingecko_pro', 'CoinGecko (Pro)'), ('transitive', 'Transitive (Calculated from Existing Rates)'), ('frankfurter', 'Frankfurter'), ('twelvedata', 'TwelveData'), ('twelvedatamarkets', 'TwelveData Markets')], max_length=255, verbose_name='Service Type'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-30 00:47
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('currencies', '0021_alter_exchangerateservice_service_type'),
]
operations = [
migrations.AddField(
model_name='currency',
name='is_archived',
field=models.BooleanField(default=False, verbose_name='Archived'),
),
]
+2 -18
View File
@@ -32,11 +32,6 @@ class Currency(models.Model):
help_text=_("Default currency for exchange calculations"), help_text=_("Default currency for exchange calculations"),
) )
is_archived = models.BooleanField(
default=False,
verbose_name=_("Archived"),
)
def __str__(self): def __str__(self):
return self.name return self.name
@@ -75,8 +70,6 @@ class ExchangeRate(models.Model):
) )
date = models.DateTimeField(verbose_name=_("Date and Time")) date = models.DateTimeField(verbose_name=_("Date and Time"))
automatic = models.BooleanField(verbose_name=_("Auto"), default=False)
class Meta: class Meta:
verbose_name = _("Exchange Rate") verbose_name = _("Exchange Rate")
verbose_name_plural = _("Exchange Rates") verbose_name_plural = _("Exchange Rates")
@@ -99,12 +92,11 @@ class ExchangeRateService(models.Model):
"""Configuration for exchange rate services""" """Configuration for exchange rate services"""
class ServiceType(models.TextChoices): class ServiceType(models.TextChoices):
SYNTH_FINANCE = "synth_finance", "Synth Finance"
SYNTH_FINANCE_STOCK = "synth_finance_stock", "Synth Finance Stock"
COINGECKO_FREE = "coingecko_free", "CoinGecko (Demo/Free)" COINGECKO_FREE = "coingecko_free", "CoinGecko (Demo/Free)"
COINGECKO_PRO = "coingecko_pro", "CoinGecko (Pro)" COINGECKO_PRO = "coingecko_pro", "CoinGecko (Pro)"
TRANSITIVE = "transitive", "Transitive (Calculated from Existing Rates)" TRANSITIVE = "transitive", "Transitive (Calculated from Existing Rates)"
FRANKFURTER = "frankfurter", "Frankfurter"
TWELVEDATA = "twelvedata", "TwelveData"
TWELVEDATA_MARKETS = "twelvedatamarkets", "TwelveData Markets"
class IntervalType(models.TextChoices): class IntervalType(models.TextChoices):
ON = "on", _("On") ON = "on", _("On")
@@ -156,14 +148,6 @@ class ExchangeRateService(models.Model):
blank=True, blank=True,
) )
singleton = models.BooleanField(
verbose_name=_("Single exchange rate"),
default=False,
help_text=_(
"Create one exchange rate and keep updating it. Avoids database clutter."
),
)
class Meta: class Meta:
verbose_name = _("Exchange Rate Service") verbose_name = _("Exchange Rate Service")
verbose_name_plural = _("Exchange Rate Services") verbose_name_plural = _("Exchange Rate Services")
+6
View File
@@ -40,6 +40,12 @@ class CurrencyTests(TestCase):
with self.assertRaises(ValidationError): with self.assertRaises(ValidationError):
currency.full_clean() currency.full_clean()
def test_currency_unique_code(self):
"""Test that currency codes must be unique"""
Currency.objects.create(code="USD", name="US Dollar", decimal_places=2)
with self.assertRaises(IntegrityError):
Currency.objects.create(code="USD", name="Another Dollar", decimal_places=2)
def test_currency_unique_name(self): def test_currency_unique_name(self):
"""Test that currency names must be unique""" """Test that currency names must be unique"""
Currency.objects.create(code="USD", name="US Dollar", decimal_places=2) Currency.objects.create(code="USD", name="US Dollar", decimal_places=2)
+47 -25
View File
@@ -1,20 +1,22 @@
from crispy_bootstrap5.bootstrap5 import Switch, BS5Accordion
from crispy_forms.bootstrap import FormActions, AccordionGroup
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Row, Column, HTML
from django import forms
from django.utils.translation import gettext_lazy as _
from apps.accounts.models import Account from apps.accounts.models import Account
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.datepicker import AirDatePickerInput
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
from apps.common.widgets.tom_select import TomSelect
from apps.dca.models import DCAStrategy, DCAEntry
from apps.common.widgets.tom_select import TransactionSelect
from apps.transactions.models import Transaction, TransactionTag, TransactionCategory
from apps.common.fields.forms.dynamic_select import ( from apps.common.fields.forms.dynamic_select import (
DynamicModelChoiceField, DynamicModelChoiceField,
DynamicModelMultipleChoiceField, DynamicModelMultipleChoiceField,
) )
from apps.common.widgets.crispy.daisyui import Switch
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.datepicker import AirDatePickerInput
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
from apps.common.widgets.tom_select import TomSelect, TransactionSelect
from apps.dca.models import DCAEntry, DCAStrategy
from apps.transactions.models import Transaction, TransactionCategory, TransactionTag
from crispy_forms.bootstrap import AccordionGroup, FormActions, Accordion
from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Column, Layout, Row
from django import forms
from django.utils.translation import gettext_lazy as _
class DCAStrategyForm(forms.ModelForm): class DCAStrategyForm(forms.ModelForm):
@@ -34,8 +36,8 @@ class DCAStrategyForm(forms.ModelForm):
self.helper.layout = Layout( self.helper.layout = Layout(
"name", "name",
Row( Row(
Column("payment_currency"), Column("payment_currency", css_class="form-group col-md-6"),
Column("target_currency"), Column("target_currency", css_class="form-group col-md-6"),
), ),
"notes", "notes",
) )
@@ -43,13 +45,17 @@ class DCAStrategyForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -149,11 +155,11 @@ class DCAEntryForm(forms.ModelForm):
self.helper.layout = Layout( self.helper.layout = Layout(
"date", "date",
Row( Row(
Column("amount_paid"), Column("amount_paid", css_class="form-group col-md-6"),
Column("amount_received"), Column("amount_received", css_class="form-group col-md-6"),
), ),
"notes", "notes",
Accordion( BS5Accordion(
AccordionGroup( AccordionGroup(
_("Create transaction"), _("Create transaction"),
Switch("create_transaction"), Switch("create_transaction"),
@@ -162,11 +168,19 @@ class DCAEntryForm(forms.ModelForm):
Row( Row(
Column( Column(
"from_account", "from_account",
css_class="form-group",
), ),
css_class="form-row",
), ),
Row( Row(
Column("from_category"), Column(
Column("from_tags"), "from_category",
css_class="form-group col-md-6 mb-0",
),
Column(
"from_tags", css_class="form-group col-md-6 mb-0"
),
css_class="form-row",
), ),
), ),
css_class="p-1 mx-1 my-3 border rounded-3", css_class="p-1 mx-1 my-3 border rounded-3",
@@ -178,10 +192,14 @@ class DCAEntryForm(forms.ModelForm):
"to_account", "to_account",
css_class="form-group", css_class="form-group",
), ),
css_class="form-row",
), ),
Row( Row(
Column("to_category"), Column(
Column("to_tags"), "to_category", css_class="form-group col-md-6 mb-0"
),
Column("to_tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
), ),
css_class="p-1 mx-1 my-3 border rounded-3", css_class="p-1 mx-1 my-3 border rounded-3",
@@ -202,13 +220,17 @@ class DCAEntryForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
+10 -5
View File
@@ -1,10 +1,11 @@
from apps.common.widgets.crispy.submit import NoClassSubmit
from crispy_forms.bootstrap import FormActions from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Layout from crispy_forms.layout import Layout, HTML
from django import forms from django import forms
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.widgets.crispy.submit import NoClassSubmit
class ExportForm(forms.Form): class ExportForm(forms.Form):
users = forms.BooleanField( users = forms.BooleanField(
@@ -114,7 +115,9 @@ class ExportForm(forms.Form):
"dca", "dca",
"import_profiles", "import_profiles",
FormActions( FormActions(
NoClassSubmit("submit", _("Export"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Export"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -159,7 +162,7 @@ class RestoreForm(forms.Form):
self.helper.form_method = "post" self.helper.form_method = "post"
self.helper.layout = Layout( self.helper.layout = Layout(
"zip_file", "zip_file",
HTML('<hr class="hr my-3"/>'), HTML("<hr />"),
"users", "users",
"accounts", "accounts",
"currencies", "currencies",
@@ -178,7 +181,9 @@ class RestoreForm(forms.Form):
"dca_entries", "dca_entries",
"import_profiles", "import_profiles",
FormActions( FormActions(
NoClassSubmit("submit", _("Restore"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Restore"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
+12 -5
View File
@@ -1,5 +1,3 @@
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.import_app.models import ImportProfile
from crispy_forms.bootstrap import FormActions from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import ( from crispy_forms.layout import (
@@ -8,6 +6,9 @@ from crispy_forms.layout import (
from django import forms from django import forms
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.import_app.models import ImportProfile
from apps.common.widgets.crispy.submit import NoClassSubmit
class ImportProfileForm(forms.ModelForm): class ImportProfileForm(forms.ModelForm):
class Meta: class Meta:
@@ -29,13 +30,17 @@ class ImportProfileForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -52,6 +57,8 @@ class ImportRunFileUploadForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
"file", "file",
FormActions( FormActions(
NoClassSubmit("submit", _("Import"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Import"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
+15 -14
View File
@@ -1,15 +1,16 @@
from apps.common.widgets.datepicker import (
AirDatePickerInput,
AirMonthYearPickerInput,
AirYearPickerInput,
)
from apps.common.widgets.tom_select import TomSelect
from apps.transactions.models import TransactionCategory
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import Column, Field, Layout, Row from crispy_forms.layout import Layout, Field, Row, Column
from django import forms from django import forms
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.widgets.datepicker import (
AirMonthYearPickerInput,
AirYearPickerInput,
AirDatePickerInput,
)
from apps.transactions.models import TransactionCategory
from apps.common.widgets.tom_select import TomSelect
class SingleMonthForm(forms.Form): class SingleMonthForm(forms.Form):
month = forms.DateField( month = forms.DateField(
@@ -58,8 +59,8 @@ class MonthRangeForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
Row( Row(
Column("month_from"), Column("month_from", css_class="form-group col-md-6"),
Column("month_to"), Column("month_to", css_class="form-group col-md-6"),
), ),
) )
@@ -81,8 +82,8 @@ class YearRangeForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
Row( Row(
Column("year_from"), Column("year_from", css_class="form-group col-md-6"),
Column("year_to"), Column("year_to", css_class="form-group col-md-6"),
), ),
) )
@@ -104,8 +105,8 @@ class DateRangeForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
Row( Row(
Column("date_from"), Column("date_from", css_class="form-group col-md-6"),
Column("date_to"), Column("date_to", css_class="form-group col-md-6"),
css_class="mb-0", css_class="mb-0",
), ),
) )
+8 -190
View File
@@ -9,13 +9,8 @@ from apps.currencies.models import Currency
from apps.currencies.utils.convert import convert from apps.currencies.utils.convert import convert
def get_categories_totals( def get_categories_totals(transactions_queryset, ignore_empty=False):
transactions_queryset, ignore_empty=False, show_entities=False # First get the category totals as before
):
# Step 1: Aggregate transaction data by category and currency.
# This query calculates the total current and projected income/expense for each
# category by grouping transactions and summing up their amounts based on their
# type (income/expense) and payment status (paid/unpaid).
category_currency_metrics = ( category_currency_metrics = (
transactions_queryset.values( transactions_queryset.values(
"category", "category",
@@ -79,10 +74,7 @@ def get_categories_totals(
.order_by("category__name") .order_by("category__name")
) )
# Step 2: Aggregate transaction data by tag, category, and currency. # Get tag totals within each category with currency details
# This is similar to the category metrics but adds tags to the grouping,
# allowing for a breakdown of totals by tag within each category. It also
# handles untagged transactions, where the 'tags' field is None.
tag_metrics = transactions_queryset.values( tag_metrics = transactions_queryset.values(
"category", "category",
"tags", "tags",
@@ -137,12 +129,10 @@ def get_categories_totals(
), ),
) )
# Step 3: Initialize the main dictionary to structure the final results. # Process the results to structure by category
# The data will be organized hierarchically: category -> currency -> tags -> entities.
result = {} result = {}
# Step 4: Process the aggregated category metrics to build the initial result structure. # Process category totals first
# This loop iterates through each category's metrics and populates the `result` dict.
for metric in category_currency_metrics: for metric in category_currency_metrics:
# Skip empty categories if ignore_empty is True # Skip empty categories if ignore_empty is True
if ignore_empty and all( if ignore_empty and all(
@@ -193,7 +183,7 @@ def get_categories_totals(
"total_final": total_final, "total_final": total_final,
} }
# Step 4a: Handle currency conversion for category totals if an exchange currency is defined. # Add exchanged values if exchange_currency exists
if metric["account__currency__exchange_currency"]: if metric["account__currency__exchange_currency"]:
from_currency = Currency.objects.get(id=currency_id) from_currency = Currency.objects.get(id=currency_id)
exchange_currency = Currency.objects.get( exchange_currency = Currency.objects.get(
@@ -232,7 +222,7 @@ def get_categories_totals(
result[category_id]["currencies"][currency_id] = currency_data result[category_id]["currencies"][currency_id] = currency_data
# Step 5: Process the aggregated tag metrics and integrate them into the result structure. # Process tag totals and add them to the result, including untagged
for tag_metric in tag_metrics: for tag_metric in tag_metrics:
category_id = tag_metric["category"] category_id = tag_metric["category"]
tag_id = tag_metric["tags"] # Will be None for untagged transactions tag_id = tag_metric["tags"] # Will be None for untagged transactions
@@ -250,7 +240,6 @@ def get_categories_totals(
result[category_id]["tags"][tag_key] = { result[category_id]["tags"][tag_key] = {
"name": tag_name, "name": tag_name,
"currencies": {}, "currencies": {},
"entities": {},
} }
currency_id = tag_metric["account__currency"] currency_id = tag_metric["account__currency"]
@@ -289,7 +278,7 @@ def get_categories_totals(
"total_final": tag_total_final, "total_final": tag_total_final,
} }
# Step 5a: Handle currency conversion for tag totals. # Add exchange currency support for tags
if tag_metric["account__currency__exchange_currency"]: if tag_metric["account__currency__exchange_currency"]:
from_currency = Currency.objects.get(id=currency_id) from_currency = Currency.objects.get(id=currency_id)
exchange_currency = Currency.objects.get( exchange_currency = Currency.objects.get(
@@ -330,175 +319,4 @@ def get_categories_totals(
currency_id currency_id
] = tag_currency_data ] = tag_currency_data
# Step 6: If requested, aggregate and process entity-level data.
if show_entities:
entity_metrics = transactions_queryset.values(
"category",
"tags",
"entities",
"entities__name",
"account__currency",
"account__currency__code",
"account__currency__name",
"account__currency__decimal_places",
"account__currency__prefix",
"account__currency__suffix",
"account__currency__exchange_currency",
).annotate(
expense_current=Coalesce(
Sum(
Case(
When(
type=Transaction.Type.EXPENSE, is_paid=True, then="amount"
),
default=Value(0),
output_field=models.DecimalField(),
)
),
Decimal("0"),
),
expense_projected=Coalesce(
Sum(
Case(
When(
type=Transaction.Type.EXPENSE, is_paid=False, then="amount"
),
default=Value(0),
output_field=models.DecimalField(),
)
),
Decimal("0"),
),
income_current=Coalesce(
Sum(
Case(
When(type=Transaction.Type.INCOME, is_paid=True, then="amount"),
default=Value(0),
output_field=models.DecimalField(),
)
),
Decimal("0"),
),
income_projected=Coalesce(
Sum(
Case(
When(
type=Transaction.Type.INCOME, is_paid=False, then="amount"
),
default=Value(0),
output_field=models.DecimalField(),
)
),
Decimal("0"),
),
)
for entity_metric in entity_metrics:
category_id = entity_metric["category"]
tag_id = entity_metric["tags"]
entity_id = entity_metric["entities"]
if category_id in result:
tag_key = tag_id if tag_id is not None else "untagged"
if tag_key in result[category_id]["tags"]:
entity_key = entity_id if entity_id is not None else "no_entity"
entity_name = (
entity_metric["entities__name"]
if entity_id is not None
else None
)
if "entities" not in result[category_id]["tags"][tag_key]:
result[category_id]["tags"][tag_key]["entities"] = {}
if (
entity_key
not in result[category_id]["tags"][tag_key]["entities"]
):
result[category_id]["tags"][tag_key]["entities"][entity_key] = {
"name": entity_name,
"currencies": {},
}
currency_id = entity_metric["account__currency"]
entity_total_current = (
entity_metric["income_current"]
- entity_metric["expense_current"]
)
entity_total_projected = (
entity_metric["income_projected"]
- entity_metric["expense_projected"]
)
entity_total_income = (
entity_metric["income_current"]
+ entity_metric["income_projected"]
)
entity_total_expense = (
entity_metric["expense_current"]
+ entity_metric["expense_projected"]
)
entity_total_final = entity_total_current + entity_total_projected
entity_currency_data = {
"currency": {
"code": entity_metric["account__currency__code"],
"name": entity_metric["account__currency__name"],
"decimal_places": entity_metric[
"account__currency__decimal_places"
],
"prefix": entity_metric["account__currency__prefix"],
"suffix": entity_metric["account__currency__suffix"],
},
"expense_current": entity_metric["expense_current"],
"expense_projected": entity_metric["expense_projected"],
"total_expense": entity_total_expense,
"income_current": entity_metric["income_current"],
"income_projected": entity_metric["income_projected"],
"total_income": entity_total_income,
"total_current": entity_total_current,
"total_projected": entity_total_projected,
"total_final": entity_total_final,
}
if entity_metric["account__currency__exchange_currency"]:
from_currency = Currency.objects.get(id=currency_id)
exchange_currency = Currency.objects.get(
id=entity_metric["account__currency__exchange_currency"]
)
exchanged = {}
for field in [
"expense_current",
"expense_projected",
"income_current",
"income_projected",
"total_income",
"total_expense",
"total_current",
"total_projected",
"total_final",
]:
amount, prefix, suffix, decimal_places = convert(
amount=entity_currency_data[field],
from_currency=from_currency,
to_currency=exchange_currency,
)
if amount is not None:
exchanged[field] = amount
if "currency" not in exchanged:
exchanged["currency"] = {
"prefix": prefix,
"suffix": suffix,
"decimal_places": decimal_places,
"code": exchange_currency.code,
"name": exchange_currency.name,
}
if exchanged:
entity_currency_data["exchanged"] = exchanged
result[category_id]["tags"][tag_key]["entities"][entity_key][
"currencies"
][currency_id] = entity_currency_data
return result return result
+1 -10
View File
@@ -13,9 +13,7 @@ from apps.insights.forms import (
) )
def get_transactions( def get_transactions(request, include_unpaid=True, include_silent=False):
request, include_unpaid=True, include_silent=False, include_untracked_accounts=False
):
transactions = Transaction.objects.all() transactions = Transaction.objects.all()
filter_type = request.GET.get("type", None) filter_type = request.GET.get("type", None)
@@ -97,11 +95,4 @@ def get_transactions(
Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True) Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True)
) )
if not include_untracked_accounts:
transactions = transactions.exclude(
account__in=request.user.untracked_accounts.all()
)
transactions = transactions.exclude(account__currency__is_archived=True)
return transactions return transactions
+6 -22
View File
@@ -74,7 +74,7 @@ def index(request):
def sankey_by_account(request): def sankey_by_account(request):
# Get filtered transactions # Get filtered transactions
transactions = get_transactions(request, include_untracked_accounts=True) transactions = get_transactions(request)
# Generate Sankey data # Generate Sankey data
sankey_data = generate_sankey_data_by_account(transactions) sankey_data = generate_sankey_data_by_account(transactions)
@@ -180,14 +180,6 @@ def category_overview(request):
else: else:
show_tags = request.session.get("insights_category_explorer_show_tags", True) show_tags = request.session.get("insights_category_explorer_show_tags", True)
if "show_entities" in request.GET:
show_entities = request.GET["show_entities"] == "on"
request.session["insights_category_explorer_show_entities"] = show_entities
else:
show_entities = request.session.get(
"insights_category_explorer_show_entities", False
)
if "showing" in request.GET: if "showing" in request.GET:
showing = request.GET["showing"] showing = request.GET["showing"]
request.session["insights_category_explorer_showing"] = showing request.session["insights_category_explorer_showing"] = showing
@@ -198,9 +190,7 @@ def category_overview(request):
transactions = get_transactions(request, include_silent=True) transactions = get_transactions(request, include_silent=True)
total_table = get_categories_totals( total_table = get_categories_totals(
transactions_queryset=transactions, transactions_queryset=transactions, ignore_empty=False
ignore_empty=False,
show_entities=show_entities,
) )
return render( return render(
@@ -210,7 +200,6 @@ def category_overview(request):
"total_table": total_table, "total_table": total_table,
"view_type": view_type, "view_type": view_type,
"show_tags": show_tags, "show_tags": show_tags,
"show_entities": show_entities,
"showing": showing, "showing": showing,
}, },
) )
@@ -250,14 +239,10 @@ def late_transactions(request):
@login_required @login_required
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def emergency_fund(request): def emergency_fund(request):
transactions_currency_queryset = ( transactions_currency_queryset = Transaction.objects.filter(
Transaction.objects.filter( is_paid=True, account__is_archived=False, account__is_asset=False
is_paid=True, account__is_archived=False, account__is_asset=False ).order_by(
) "account__currency__name",
.exclude(account__in=request.user.untracked_accounts.all())
.order_by(
"account__currency__name",
)
) )
currency_net_worth = calculate_currency_totals( currency_net_worth = calculate_currency_totals(
transactions_queryset=transactions_currency_queryset, ignore_empty=False transactions_queryset=transactions_currency_queryset, ignore_empty=False
@@ -277,7 +262,6 @@ def emergency_fund(request):
category__mute=False, category__mute=False,
mute=False, mute=False,
) )
.exclude(account__in=request.user.untracked_accounts.all())
.values("reference_date", "account__currency") .values("reference_date", "account__currency")
.annotate(monthly_total=Sum("amount")) .annotate(monthly_total=Sum("amount"))
) )
+7 -17
View File
@@ -107,15 +107,9 @@ def transactions_list(request, month: int, year: int):
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def monthly_summary(request, month: int, year: int): def monthly_summary(request, month: int, year: int):
# Base queryset with all required filters # Base queryset with all required filters
base_queryset = ( base_queryset = Transaction.objects.filter(
Transaction.objects.filter( reference_date__year=year, reference_date__month=month, account__is_asset=False
reference_date__year=year, ).exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True))
reference_date__month=month,
account__is_asset=False,
)
.exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True))
.exclude(account__in=request.user.untracked_accounts.all())
)
data = calculate_currency_totals(base_queryset, ignore_empty=True) data = calculate_currency_totals(base_queryset, ignore_empty=True)
percentages = calculate_percentage_distribution(data) percentages = calculate_percentage_distribution(data)
@@ -171,14 +165,10 @@ def monthly_account_summary(request, month: int, year: int):
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def monthly_currency_summary(request, month: int, year: int): def monthly_currency_summary(request, month: int, year: int):
# Base queryset with all required filters # Base queryset with all required filters
base_queryset = ( base_queryset = Transaction.objects.filter(
Transaction.objects.filter( reference_date__year=year,
reference_date__year=year, reference_date__month=month,
reference_date__month=month, ).exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True))
)
.exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True))
.exclude(account__in=request.user.untracked_accounts.all())
)
currency_data = calculate_currency_totals(base_queryset.all(), ignore_empty=True) currency_data = calculate_currency_totals(base_queryset.all(), ignore_empty=True)
currency_percentages = calculate_percentage_distribution(currency_data) currency_percentages = calculate_percentage_distribution(currency_data)
@@ -30,7 +30,6 @@ def calculate_historical_currency_net_worth(queryset):
| Q(accounts__visibility="private", accounts__owner=None), | Q(accounts__visibility="private", accounts__owner=None),
accounts__is_archived=False, accounts__is_archived=False,
accounts__isnull=False, accounts__isnull=False,
is_archived=False,
) )
.values_list("name", flat=True) .values_list("name", flat=True)
.distinct() .distinct()
@@ -182,29 +181,3 @@ def calculate_historical_account_balance(queryset):
historical_account_balance[date_filter(end_date, "b Y")] = month_data historical_account_balance[date_filter(end_date, "b Y")] = month_data
return historical_account_balance return historical_account_balance
def calculate_monthly_net_worth_difference(historical_net_worth):
diff_dict = OrderedDict()
if not historical_net_worth:
return diff_dict
# Get all currencies
currencies = set()
for data in historical_net_worth.values():
currencies.update(data.keys())
# Initialize prev_values for all currencies
prev_values = {currency: Decimal("0.00") for currency in currencies}
for month, values in historical_net_worth.items():
diff_values = {}
for currency in sorted(list(currencies)):
current_val = values.get(currency, Decimal("0.00"))
prev_val = prev_values.get(currency, Decimal("0.00"))
diff_values[currency] = current_val - prev_val
diff_dict[month] = diff_values
prev_values = values.copy()
return diff_dict
+9 -46
View File
@@ -8,7 +8,6 @@ from django.views.decorators.http import require_http_methods
from apps.net_worth.utils.calculate_net_worth import ( from apps.net_worth.utils.calculate_net_worth import (
calculate_historical_currency_net_worth, calculate_historical_currency_net_worth,
calculate_historical_account_balance, calculate_historical_account_balance,
calculate_monthly_net_worth_difference,
) )
from apps.transactions.models import Transaction from apps.transactions.models import Transaction
from apps.transactions.utils.calculations import ( from apps.transactions.utils.calculations import (
@@ -21,18 +20,17 @@ from apps.transactions.utils.calculations import (
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def net_worth(request): def net_worth(request):
if "view_type" in request.GET: if "view_type" in request.GET:
print(request.GET["view_type"])
view_type = request.GET["view_type"] view_type = request.GET["view_type"]
request.session["networth_view_type"] = view_type request.session["networth_view_type"] = view_type
else: else:
view_type = request.session.get("networth_view_type", "current") view_type = request.session.get("networth_view_type", "current")
if view_type == "current": if view_type == "current":
transactions_currency_queryset = ( transactions_currency_queryset = Transaction.objects.filter(
Transaction.objects.filter(is_paid=True, account__is_archived=False) is_paid=True, account__is_archived=False
.order_by( ).order_by(
"account__currency__name", "account__currency__name",
)
.exclude(account__in=request.user.untracked_accounts.all())
) )
transactions_account_queryset = Transaction.objects.filter( transactions_account_queryset = Transaction.objects.filter(
is_paid=True, account__is_archived=False is_paid=True, account__is_archived=False
@@ -41,12 +39,10 @@ def net_worth(request):
"account__name", "account__name",
) )
else: else:
transactions_currency_queryset = ( transactions_currency_queryset = Transaction.objects.filter(
Transaction.objects.filter(account__is_archived=False) account__is_archived=False
.order_by( ).order_by(
"account__currency__name", "account__currency__name",
)
.exclude(account__in=request.user.untracked_accounts.all())
) )
transactions_account_queryset = Transaction.objects.filter( transactions_account_queryset = Transaction.objects.filter(
account__is_archived=False account__is_archived=False
@@ -97,38 +93,6 @@ def net_worth(request):
chart_data_currency_json = json.dumps(chart_data_currency, cls=DjangoJSONEncoder) chart_data_currency_json = json.dumps(chart_data_currency, cls=DjangoJSONEncoder)
monthly_difference_data = calculate_monthly_net_worth_difference(
historical_net_worth=historical_currency_net_worth
)
diff_labels = (
list(monthly_difference_data.keys()) if monthly_difference_data else []
)
diff_currencies = (
list(monthly_difference_data[diff_labels[0]].keys())
if monthly_difference_data and diff_labels
else []
)
diff_datasets = []
for i, currency in enumerate(diff_currencies):
data = [
float(month_data.get(currency, 0))
for month_data in monthly_difference_data.values()
]
diff_datasets.append(
{
"label": currency,
"data": data,
"borderWidth": 3,
}
)
chart_data_monthly_difference = {"labels": diff_labels, "datasets": diff_datasets}
chart_data_monthly_difference_json = json.dumps(
chart_data_monthly_difference, cls=DjangoJSONEncoder
)
historical_account_balance = calculate_historical_account_balance( historical_account_balance = calculate_historical_account_balance(
queryset=transactions_account_queryset queryset=transactions_account_queryset
) )
@@ -173,7 +137,6 @@ def net_worth(request):
"chart_data_accounts_json": chart_data_accounts_json, "chart_data_accounts_json": chart_data_accounts_json,
"accounts": accounts, "accounts": accounts,
"type": view_type, "type": view_type,
"chart_data_monthly_difference_json": chart_data_monthly_difference_json,
}, },
) )
+54 -171
View File
@@ -1,21 +1,16 @@
from apps.common.fields.forms.dynamic_select import DynamicModelChoiceField from crispy_bootstrap5.bootstrap5 import Switch, BS5Accordion
from apps.common.widgets.crispy.daisyui import Switch from crispy_forms.bootstrap import FormActions, AccordionGroup
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.tom_select import TomSelect, TransactionSelect
from apps.rules.models import (
TransactionRule,
TransactionRuleAction,
UpdateOrCreateTransactionRuleAction,
)
from apps.transactions.forms import BulkEditTransactionForm
from apps.transactions.models import Transaction
from crispy_forms.bootstrap import AccordionGroup, FormActions, Accordion
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Column, Field, Layout, Row from crispy_forms.layout import Layout, Field, Row, Column
from django import forms from django import forms
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.tom_select import TomSelect
from apps.rules.models import TransactionRule, UpdateOrCreateTransactionRuleAction
from apps.rules.models import TransactionRuleAction
class TransactionRuleForm(forms.ModelForm): class TransactionRuleForm(forms.ModelForm):
class Meta: class Meta:
@@ -45,8 +40,6 @@ class TransactionRuleForm(forms.ModelForm):
Column(Switch("on_create")), Column(Switch("on_create")),
Column(Switch("on_delete")), Column(Switch("on_delete")),
), ),
"order",
Switch("sequenced"),
"description", "description",
"trigger", "trigger",
) )
@@ -54,13 +47,17 @@ class TransactionRuleForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -68,11 +65,10 @@ class TransactionRuleForm(forms.ModelForm):
class TransactionRuleActionForm(forms.ModelForm): class TransactionRuleActionForm(forms.ModelForm):
class Meta: class Meta:
model = TransactionRuleAction model = TransactionRuleAction
fields = ("value", "field", "order") fields = ("value", "field")
labels = { labels = {
"field": _("Set field"), "field": _("Set field"),
"value": _("To"), "value": _("To"),
"order": _("Order"),
} }
widgets = {"field": TomSelect(clear_button=False)} widgets = {"field": TomSelect(clear_button=False)}
@@ -86,7 +82,6 @@ class TransactionRuleActionForm(forms.ModelForm):
self.helper.form_method = "post" self.helper.form_method = "post"
# TO-DO: Add helper with available commands # TO-DO: Add helper with available commands
self.helper.layout = Layout( self.helper.layout = Layout(
"order",
"field", "field",
"value", "value",
) )
@@ -94,13 +89,17 @@ class TransactionRuleActionForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -148,11 +147,9 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
"search_category_operator": TomSelect(clear_button=False), "search_category_operator": TomSelect(clear_button=False),
"search_internal_note_operator": TomSelect(clear_button=False), "search_internal_note_operator": TomSelect(clear_button=False),
"search_internal_id_operator": TomSelect(clear_button=False), "search_internal_id_operator": TomSelect(clear_button=False),
"search_mute_operator": TomSelect(clear_button=False),
} }
labels = { labels = {
"order": _("Order"),
"search_account_operator": _("Operator"), "search_account_operator": _("Operator"),
"search_type_operator": _("Operator"), "search_type_operator": _("Operator"),
"search_is_paid_operator": _("Operator"), "search_is_paid_operator": _("Operator"),
@@ -166,7 +163,6 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
"search_internal_id_operator": _("Operator"), "search_internal_id_operator": _("Operator"),
"search_tags_operator": _("Operator"), "search_tags_operator": _("Operator"),
"search_entities_operator": _("Operator"), "search_entities_operator": _("Operator"),
"search_mute_operator": _("Operator"),
"search_account": _("Account"), "search_account": _("Account"),
"search_type": _("Type"), "search_type": _("Type"),
"search_is_paid": _("Paid"), "search_is_paid": _("Paid"),
@@ -180,7 +176,6 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
"search_internal_id": _("Internal ID"), "search_internal_id": _("Internal ID"),
"search_tags": _("Tags"), "search_tags": _("Tags"),
"search_entities": _("Entities"), "search_entities": _("Entities"),
"search_mute": _("Mute"),
"set_account": _("Account"), "set_account": _("Account"),
"set_type": _("Type"), "set_type": _("Type"),
"set_is_paid": _("Paid"), "set_is_paid": _("Paid"),
@@ -194,7 +189,6 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
"set_category": _("Category"), "set_category": _("Category"),
"set_internal_note": _("Internal Note"), "set_internal_note": _("Internal Note"),
"set_internal_id": _("Internal ID"), "set_internal_id": _("Internal ID"),
"set_mute": _("Mute"),
} }
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
@@ -206,149 +200,138 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
self.helper.form_method = "post" self.helper.form_method = "post"
self.helper.layout = Layout( self.helper.layout = Layout(
"order", BS5Accordion(
Accordion(
AccordionGroup( AccordionGroup(
_("Search Criteria"), _("Search Criteria"),
Field("filter", rows=1), Field("filter", rows=1),
Row( Row(
Column( Column(
Field("search_type_operator"), Field("search_type_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_type", rows=1), Field("search_type", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_is_paid_operator"), Field("search_is_paid_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_is_paid", rows=1), Field("search_is_paid", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
),
),
Row(
Column(
Field("search_mute_operator"),
css_class="col-span-12 md:col-span-4",
),
Column(
Field("search_mute", rows=1),
css_class="col-span-12 md:col-span-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_account_operator"), Field("search_account_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_account", rows=1), Field("search_account", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_entities_operator"), Field("search_entities_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_entities", rows=1), Field("search_entities", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_date_operator"), Field("search_date_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_date", rows=1), Field("search_date", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_reference_date_operator"), Field("search_reference_date_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_reference_date", rows=1), Field("search_reference_date", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_description_operator"), Field("search_description_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_description", rows=1), Field("search_description", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_amount_operator"), Field("search_amount_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_amount", rows=1), Field("search_amount", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_category_operator"), Field("search_category_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_category", rows=1), Field("search_category", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_tags_operator"), Field("search_tags_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_tags", rows=1), Field("search_tags", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_notes_operator"), Field("search_notes_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_notes", rows=1), Field("search_notes", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_internal_note_operator"), Field("search_internal_note_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_internal_note", rows=1), Field("search_internal_note", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
Row( Row(
Column( Column(
Field("search_internal_id_operator"), Field("search_internal_id_operator"),
css_class="col-span-12 md:col-span-4", css_class="form-group col-md-4",
), ),
Column( Column(
Field("search_internal_id", rows=1), Field("search_internal_id", rows=1),
css_class="col-span-12 md:col-span-8", css_class="form-group col-md-8",
), ),
), ),
active=True, active=True,
@@ -357,7 +340,6 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
_("Set Values"), _("Set Values"),
Field("set_type", rows=1), Field("set_type", rows=1),
Field("set_is_paid", rows=1), Field("set_is_paid", rows=1),
Field("set_mute", rows=1),
Field("set_account", rows=1), Field("set_account", rows=1),
Field("set_entities", rows=1), Field("set_entities", rows=1),
Field("set_date", rows=1), Field("set_date", rows=1),
@@ -379,13 +361,17 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -395,106 +381,3 @@ class UpdateOrCreateTransactionRuleActionForm(forms.ModelForm):
if commit: if commit:
instance.save() instance.save()
return instance return instance
class DryRunCreatedTransacion(forms.Form):
transaction = DynamicModelChoiceField(
model=Transaction,
to_field_name="id",
label=_("Transaction"),
required=True,
queryset=Transaction.objects.none(),
widget=TransactionSelect(clear_button=False, income=True, expense=True),
help_text=_("Type to search for a transaction"),
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.helper = FormHelper()
self.helper.form_tag = False
self.helper.layout = Layout(
"transaction",
FormActions(
NoClassSubmit("submit", _("Test"), css_class="btn btn-primary"),
),
)
if self.data.get("transaction"):
try:
transaction = Transaction.objects.get(id=self.data.get("transaction"))
except Transaction.DoesNotExist:
transaction = None
if transaction:
self.fields["transaction"].queryset = Transaction.objects.filter(
id=transaction.id
)
class DryRunDeletedTransacion(forms.Form):
transaction = DynamicModelChoiceField(
model=Transaction,
to_field_name="id",
label=_("Transaction"),
required=True,
queryset=Transaction.objects.none(),
widget=TransactionSelect(clear_button=False, income=True, expense=True),
help_text=_("Type to search for a transaction"),
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.helper = FormHelper()
self.helper.form_tag = False
self.helper.layout = Layout(
"transaction",
FormActions(
NoClassSubmit("submit", _("Test"), css_class="btn btn-primary"),
),
)
if self.data.get("transaction"):
try:
transaction = Transaction.objects.get(id=self.data.get("transaction"))
except Transaction.DoesNotExist:
transaction = None
if transaction:
self.fields["transaction"].queryset = Transaction.objects.filter(
id=transaction.id
)
class DryRunUpdatedTransactionForm(BulkEditTransactionForm):
transaction = DynamicModelChoiceField(
model=Transaction,
to_field_name="id",
label=_("Transaction"),
required=True,
queryset=Transaction.objects.none(),
widget=TransactionSelect(clear_button=False, income=True, expense=True),
help_text=_("Type to search for a transaction"),
)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.helper.layout.insert(0, "transaction")
self.helper.layout.insert(1, HTML('<hr class="hr my-3" />'))
# Change submit button
self.helper.layout[-1] = FormActions(
NoClassSubmit("submit", _("Test"), css_class="btn btn-primary")
)
if self.data.get("transaction"):
try:
transaction = Transaction.objects.get(id=self.data.get("transaction"))
except Transaction.DoesNotExist:
transaction = None
if transaction:
self.fields["transaction"].queryset = Transaction.objects.filter(
id=transaction.id
)
@@ -1,39 +0,0 @@
# Generated by Django 5.2 on 2025-08-30 18:48
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
("rules", "0014_alter_transactionrule_owner_and_more"),
]
operations = [
migrations.AlterModelOptions(
name="transactionruleaction",
options={
"ordering": ["order"],
"verbose_name": "Edit transaction action",
"verbose_name_plural": "Edit transaction actions",
},
),
migrations.AlterModelOptions(
name="updateorcreatetransactionruleaction",
options={
"ordering": ["order"],
"verbose_name": "Update or create transaction action",
"verbose_name_plural": "Update or create transaction actions",
},
),
migrations.AddField(
model_name="transactionruleaction",
name="order",
field=models.PositiveIntegerField(default=0, verbose_name="Order"),
),
migrations.AddField(
model_name="updateorcreatetransactionruleaction",
name="order",
field=models.PositiveIntegerField(default=0, verbose_name="Order"),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-31 18:42
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('rules', '0015_alter_transactionruleaction_options_and_more'),
]
operations = [
migrations.AddField(
model_name='transactionrule',
name='sequenced',
field=models.BooleanField(default=False, verbose_name='Sequenced'),
),
]
@@ -1,33 +0,0 @@
# Generated by Django 5.2.5 on 2025-08-31 19:09
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('rules', '0016_transactionrule_sequenced'),
]
operations = [
migrations.AddField(
model_name='updateorcreatetransactionruleaction',
name='search_mute',
field=models.TextField(blank=True, verbose_name='Search Mute'),
),
migrations.AddField(
model_name='updateorcreatetransactionruleaction',
name='search_mute_operator',
field=models.CharField(choices=[('exact', 'is exactly'), ('contains', 'contains'), ('startswith', 'starts with'), ('endswith', 'ends with'), ('eq', 'equals'), ('gt', 'greater than'), ('lt', 'less than'), ('gte', 'greater than or equal'), ('lte', 'less than or equal')], default='exact', max_length=10, verbose_name='Mute Operator'),
),
migrations.AddField(
model_name='updateorcreatetransactionruleaction',
name='set_mute',
field=models.TextField(blank=True, verbose_name='Mute'),
),
migrations.AlterField(
model_name='transactionruleaction',
name='field',
field=models.CharField(choices=[('account', 'Account'), ('type', 'Type'), ('is_paid', 'Paid'), ('date', 'Date'), ('reference_date', 'Reference Date'), ('mute', 'Mute'), ('amount', 'Amount'), ('description', 'Description'), ('notes', 'Notes'), ('category', 'Category'), ('tags', 'Tags'), ('entities', 'Entities'), ('internal_nome', 'Internal Note'), ('internal_id', 'Internal ID')], max_length=50, verbose_name='Field'),
),
]
@@ -1,18 +0,0 @@
# Generated by Django 5.2.5 on 2025-09-02 14:52
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('rules', '0017_updateorcreatetransactionruleaction_search_mute_and_more'),
]
operations = [
migrations.AddField(
model_name='transactionrule',
name='order',
field=models.PositiveIntegerField(default=0, verbose_name='Order'),
),
]
-40
View File
@@ -13,11 +13,6 @@ class TransactionRule(SharedObject):
name = models.CharField(max_length=100, verbose_name=_("Name")) name = models.CharField(max_length=100, verbose_name=_("Name"))
description = models.TextField(blank=True, null=True, verbose_name=_("Description")) description = models.TextField(blank=True, null=True, verbose_name=_("Description"))
trigger = models.TextField(verbose_name=_("Trigger")) trigger = models.TextField(verbose_name=_("Trigger"))
sequenced = models.BooleanField(
verbose_name=_("Sequenced"),
default=False,
)
order = models.PositiveIntegerField(default=0, verbose_name=_("Order"))
objects = SharedObjectManager() objects = SharedObjectManager()
all_objects = models.Manager() # Unfiltered manager all_objects = models.Manager() # Unfiltered manager
@@ -37,15 +32,12 @@ class TransactionRuleAction(models.Model):
is_paid = "is_paid", _("Paid") is_paid = "is_paid", _("Paid")
date = "date", _("Date") date = "date", _("Date")
reference_date = "reference_date", _("Reference Date") reference_date = "reference_date", _("Reference Date")
mute = "mute", _("Mute")
amount = "amount", _("Amount") amount = "amount", _("Amount")
description = "description", _("Description") description = "description", _("Description")
notes = "notes", _("Notes") notes = "notes", _("Notes")
category = "category", _("Category") category = "category", _("Category")
tags = "tags", _("Tags") tags = "tags", _("Tags")
entities = "entities", _("Entities") entities = "entities", _("Entities")
internal_note = "internal_nome", _("Internal Note")
internal_id = "internal_id", _("Internal ID")
rule = models.ForeignKey( rule = models.ForeignKey(
TransactionRule, TransactionRule,
@@ -59,7 +51,6 @@ class TransactionRuleAction(models.Model):
verbose_name=_("Field"), verbose_name=_("Field"),
) )
value = models.TextField(verbose_name=_("Value")) value = models.TextField(verbose_name=_("Value"))
order = models.PositiveIntegerField(default=0, verbose_name=_("Order"))
def __str__(self): def __str__(self):
return f"{self.rule} - {self.field} - {self.value}" return f"{self.rule} - {self.field} - {self.value}"
@@ -68,11 +59,6 @@ class TransactionRuleAction(models.Model):
verbose_name = _("Edit transaction action") verbose_name = _("Edit transaction action")
verbose_name_plural = _("Edit transaction actions") verbose_name_plural = _("Edit transaction actions")
unique_together = (("rule", "field"),) unique_together = (("rule", "field"),)
ordering = ["order"]
@property
def action_type(self):
return "edit_transaction"
class UpdateOrCreateTransactionRuleAction(models.Model): class UpdateOrCreateTransactionRuleAction(models.Model):
@@ -251,17 +237,6 @@ class UpdateOrCreateTransactionRuleAction(models.Model):
verbose_name="Internal ID Operator", verbose_name="Internal ID Operator",
) )
search_mute = models.TextField(
verbose_name="Search Mute",
blank=True,
)
search_mute_operator = models.CharField(
max_length=10,
choices=SearchOperator.choices,
default=SearchOperator.EXACT,
verbose_name="Mute Operator",
)
# Set fields # Set fields
set_account = models.TextField( set_account = models.TextField(
verbose_name=_("Account"), verbose_name=_("Account"),
@@ -315,21 +290,10 @@ class UpdateOrCreateTransactionRuleAction(models.Model):
verbose_name=_("Tags"), verbose_name=_("Tags"),
blank=True, blank=True,
) )
set_mute = models.TextField(
verbose_name=_("Mute"),
blank=True,
)
order = models.PositiveIntegerField(default=0, verbose_name=_("Order"))
class Meta: class Meta:
verbose_name = _("Update or create transaction action") verbose_name = _("Update or create transaction action")
verbose_name_plural = _("Update or create transaction actions") verbose_name_plural = _("Update or create transaction actions")
ordering = ["order"]
@property
def action_type(self):
return "update_or_create_transaction"
def __str__(self): def __str__(self):
return f"Update or create transaction action for {self.rule}" return f"Update or create transaction action for {self.rule}"
@@ -361,10 +325,6 @@ class UpdateOrCreateTransactionRuleAction(models.Model):
value = simple.eval(self.search_is_paid) value = simple.eval(self.search_is_paid)
search_query &= add_to_query("is_paid", value, self.search_is_paid_operator) search_query &= add_to_query("is_paid", value, self.search_is_paid_operator)
if self.search_mute:
value = simple.eval(self.search_mute)
search_query &= add_to_query("mute", value, self.search_mute_operator)
if self.search_date: if self.search_date:
value = simple.eval(self.search_date) value = simple.eval(self.search_date)
search_query &= add_to_query("date", value, self.search_date_operator) search_query &= add_to_query("date", value, self.search_date_operator)
+26 -7
View File
@@ -9,17 +9,40 @@ from apps.transactions.models import (
) )
from apps.rules.tasks import check_for_transaction_rules from apps.rules.tasks import check_for_transaction_rules
from apps.common.middleware.thread_local import get_current_user from apps.common.middleware.thread_local import get_current_user
from apps.rules.utils.transactions import serialize_transaction
@receiver(transaction_created) @receiver(transaction_created)
@receiver(transaction_updated) @receiver(transaction_updated)
@receiver(transaction_deleted) @receiver(transaction_deleted)
def transaction_changed_receiver(sender: Transaction, signal, **kwargs): def transaction_changed_receiver(sender: Transaction, signal, **kwargs):
old_data = kwargs.get("old_data")
if signal is transaction_deleted: if signal is transaction_deleted:
# Serialize transaction data for processing # Serialize transaction data for processing
transaction_data = serialize_transaction(sender, deleted=True) transaction_data = {
"id": sender.id,
"account": (sender.account.id, sender.account.name),
"account_group": (
sender.account.group.id if sender.account.group else None,
sender.account.group.name if sender.account.group else None,
),
"type": str(sender.type),
"is_paid": sender.is_paid,
"is_asset": sender.account.is_asset,
"is_archived": sender.account.is_archived,
"category": (
sender.category.id if sender.category else None,
sender.category.name if sender.category else None,
),
"date": sender.date.isoformat(),
"reference_date": sender.reference_date.isoformat(),
"amount": str(sender.amount),
"description": sender.description,
"notes": sender.notes,
"tags": list(sender.tags.values_list("id", "name")),
"entities": list(sender.entities.values_list("id", "name")),
"deleted": True,
"internal_note": sender.internal_note,
"internal_id": sender.internal_id,
}
check_for_transaction_rules.defer( check_for_transaction_rules.defer(
transaction_data=transaction_data, transaction_data=transaction_data,
@@ -36,9 +59,6 @@ def transaction_changed_receiver(sender: Transaction, signal, **kwargs):
dca_entry.amount_received = sender.amount dca_entry.amount_received = sender.amount
dca_entry.save() dca_entry.save()
if signal is transaction_updated and old_data:
old_data = serialize_transaction(old_data, deleted=False)
check_for_transaction_rules.defer( check_for_transaction_rules.defer(
instance_id=sender.id, instance_id=sender.id,
user_id=get_current_user().id, user_id=get_current_user().id,
@@ -47,5 +67,4 @@ def transaction_changed_receiver(sender: Transaction, signal, **kwargs):
if signal is transaction_created if signal is transaction_created
else "transaction_updated" else "transaction_updated"
), ),
old_data=old_data,
) )
+335 -679
View File
File diff suppressed because it is too large Load Diff
-15
View File
@@ -42,21 +42,6 @@ urlpatterns = [
views.transaction_rule_take_ownership, views.transaction_rule_take_ownership,
name="transaction_rule_take_ownership", name="transaction_rule_take_ownership",
), ),
path(
"rules/transaction/<int:pk>/dry-run/created/",
views.dry_run_rule_created,
name="transaction_rule_dry_run_created",
),
path(
"rules/transaction/<int:pk>/dry-run/deleted/",
views.dry_run_rule_deleted,
name="transaction_rule_dry_run_deleted",
),
path(
"rules/transaction/<int:pk>/dry-run/updated/",
views.dry_run_rule_updated,
name="transaction_rule_dry_run_updated",
),
path( path(
"rules/transaction/<int:pk>/share/", "rules/transaction/<int:pk>/share/",
views.transaction_rule_share, views.transaction_rule_share,
-101
View File
@@ -1,101 +0,0 @@
import logging
from decimal import Decimal
from django.db.models import Sum, Value, DecimalField, Case, When, F
from django.db.models.functions import Coalesce
from apps.transactions.models import (
Transaction,
)
logger = logging.getLogger(__name__)
class TransactionsGetter:
def __init__(self, **filters):
self.__queryset = Transaction.objects.filter(**filters)
def exclude(self, **exclude_filters):
self.__queryset = self.__queryset.exclude(**exclude_filters)
return self
@property
def sum(self):
return self.__queryset.aggregate(
total=Coalesce(
Sum("amount"), Value(Decimal("0")), output_field=DecimalField()
)
)["total"]
@property
def balance(self):
return abs(
self.__queryset.aggregate(
balance=Coalesce(
Sum(
Case(
When(type=Transaction.Type.EXPENSE, then=-F("amount")),
default=F("amount"),
output_field=DecimalField(),
)
),
Value(Decimal("0")),
output_field=DecimalField(),
)
)["balance"]
)
@property
def raw_balance(self):
return self.__queryset.aggregate(
balance=Coalesce(
Sum(
Case(
When(type=Transaction.Type.EXPENSE, then=-F("amount")),
default=F("amount"),
output_field=DecimalField(),
)
),
Value(Decimal("0")),
output_field=DecimalField(),
)
)["balance"]
def serialize_transaction(sender: Transaction, deleted: bool):
return {
"id": sender.id,
"account": (sender.account.id, sender.account.name),
"account_group": (
sender.account.group.id if sender.account.group else None,
sender.account.group.name if sender.account.group else None,
),
"type": str(sender.type),
"is_paid": sender.is_paid,
"is_asset": sender.account.is_asset,
"is_archived": sender.account.is_archived,
"category": (
sender.category.id if sender.category else None,
sender.category.name if sender.category else None,
),
"date": sender.date.isoformat(),
"reference_date": sender.reference_date.isoformat(),
"amount": str(sender.amount),
"description": sender.description,
"notes": sender.notes,
"tags": list(sender.tags.values_list("id", "name")),
"entities": list(sender.entities.values_list("id", "name")),
"deleted": deleted,
"internal_note": sender.internal_note,
"internal_id": sender.internal_id,
"mute": sender.mute,
"installment_id": sender.installment_id if sender.installment_plan else None,
"installment_total": (
sender.installment_plan.number_of_installments
if sender.installment_plan is not None
else None
),
"installment": sender.installment_plan is not None,
"recurring_transaction": sender.recurring_transaction is not None,
}
+2 -178
View File
@@ -1,10 +1,5 @@
from itertools import chain
from copy import deepcopy
from django.contrib import messages from django.contrib import messages
from django.contrib.auth.decorators import login_required from django.contrib.auth.decorators import login_required
from django.db import transaction
from django.http import HttpResponse from django.http import HttpResponse
from django.shortcuts import render, get_object_or_404, redirect from django.shortcuts import render, get_object_or_404, redirect
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
@@ -15,9 +10,6 @@ from apps.rules.forms import (
TransactionRuleForm, TransactionRuleForm,
TransactionRuleActionForm, TransactionRuleActionForm,
UpdateOrCreateTransactionRuleActionForm, UpdateOrCreateTransactionRuleActionForm,
DryRunCreatedTransacion,
DryRunDeletedTransacion,
DryRunUpdatedTransactionForm,
) )
from apps.rules.models import ( from apps.rules.models import (
TransactionRule, TransactionRule,
@@ -27,11 +19,6 @@ from apps.rules.models import (
from apps.common.models import SharedObject from apps.common.models import SharedObject
from apps.common.forms import SharedObjectForm from apps.common.forms import SharedObjectForm
from apps.common.decorators.demo import disabled_on_demo from apps.common.decorators.demo import disabled_on_demo
from apps.rules.tasks import check_for_transaction_rules
from apps.common.middleware.thread_local import get_current_user
from apps.rules.signals import transaction_created, transaction_updated
from apps.rules.utils.transactions import serialize_transaction
from apps.transactions.models import Transaction
@login_required @login_required
@@ -49,7 +36,7 @@ def rules_index(request):
@disabled_on_demo @disabled_on_demo
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def rules_list(request): def rules_list(request):
transaction_rules = TransactionRule.objects.all().order_by("order", "id") transaction_rules = TransactionRule.objects.all().order_by("id")
return render( return render(
request, request,
"rules/fragments/list.html", "rules/fragments/list.html",
@@ -153,20 +140,10 @@ def transaction_rule_edit(request, transaction_rule_id):
def transaction_rule_view(request, transaction_rule_id): def transaction_rule_view(request, transaction_rule_id):
transaction_rule = get_object_or_404(TransactionRule, id=transaction_rule_id) transaction_rule = get_object_or_404(TransactionRule, id=transaction_rule_id)
edit_actions = transaction_rule.transaction_actions.all()
update_or_create_actions = (
transaction_rule.update_or_create_transaction_actions.all()
)
all_actions = sorted(
chain(edit_actions, update_or_create_actions),
key=lambda a: a.order,
)
return render( return render(
request, request,
"rules/fragments/transaction_rule/view.html", "rules/fragments/transaction_rule/view.html",
{"transaction_rule": transaction_rule, "all_actions": all_actions}, {"transaction_rule": transaction_rule},
) )
@@ -429,156 +406,3 @@ def update_or_create_transaction_rule_action_delete(request, pk):
"HX-Trigger": "updated, hide_offcanvas", "HX-Trigger": "updated, hide_offcanvas",
}, },
) )
@only_htmx
@login_required
@disabled_on_demo
@require_http_methods(["GET", "POST"])
def dry_run_rule_created(request, pk):
rule = get_object_or_404(TransactionRule, id=pk)
logs = None
results = None
if request.method == "POST":
form = DryRunCreatedTransacion(request.POST)
if form.is_valid():
try:
with transaction.atomic():
logs, results = check_for_transaction_rules(
instance_id=form.cleaned_data["transaction"].id,
signal="transaction_created",
dry_run=True,
rule_id=rule.id,
user_id=get_current_user().id,
)
logs = "\n".join(logs)
response = render(
request,
"rules/fragments/transaction_rule/dry_run/created.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
raise Exception("ROLLBACK")
except Exception:
pass
return response
else:
form = DryRunCreatedTransacion()
return render(
request,
"rules/fragments/transaction_rule/dry_run/created.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
@only_htmx
@login_required
@disabled_on_demo
@require_http_methods(["GET", "POST"])
def dry_run_rule_deleted(request, pk):
rule = get_object_or_404(TransactionRule, id=pk)
logs = None
results = None
if request.method == "POST":
form = DryRunDeletedTransacion(request.POST)
if form.is_valid():
try:
with transaction.atomic():
logs, results = check_for_transaction_rules(
instance_id=form.cleaned_data["transaction"].id,
signal="transaction_deleted",
dry_run=True,
rule_id=rule.id,
user_id=get_current_user().id,
)
logs = "\n".join(logs)
response = render(
request,
"rules/fragments/transaction_rule/dry_run/created.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
raise Exception("ROLLBACK")
except Exception:
pass
return response
else:
form = DryRunDeletedTransacion()
return render(
request,
"rules/fragments/transaction_rule/dry_run/deleted.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
@only_htmx
@login_required
@disabled_on_demo
@require_http_methods(["GET", "POST"])
def dry_run_rule_updated(request, pk):
rule = get_object_or_404(TransactionRule, id=pk)
logs = None
results = None
if request.method == "POST":
form = DryRunUpdatedTransactionForm(request.POST)
if form.is_valid():
base_transaction = Transaction.objects.get(
id=request.POST.get("transaction")
)
old_data = deepcopy(base_transaction)
try:
with transaction.atomic():
for field_name, value in form.cleaned_data.items():
if value or isinstance(
value, bool
): # Only update fields that have been filled in the form
if field_name == "tags":
base_transaction.tags.set(value)
elif field_name == "entities":
base_transaction.entities.set(value)
else:
setattr(base_transaction, field_name, value)
base_transaction.save()
logs, results = check_for_transaction_rules(
instance_id=base_transaction.id,
signal="transaction_updated",
dry_run=True,
rule_id=rule.id,
user_id=get_current_user().id,
old_data=old_data,
)
logs = "\n".join(logs) if logs else ""
response = render(
request,
"rules/fragments/transaction_rule/dry_run/updated.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
# This will rollback the transaction
raise Exception("ROLLBACK")
except Exception:
pass
return response
else:
form = DryRunUpdatedTransactionForm(initial={"is_paid": None, "type": None})
return render(
request,
"rules/fragments/transaction_rule/dry_run/updated.html",
{"form": form, "rule": rule, "logs": logs, "results": results},
)
+29 -108
View File
@@ -1,4 +1,11 @@
import django_filters import django_filters
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Field, Row, Column
from django import forms
from django.db.models import Q
from django.utils.translation import gettext_lazy as _
from django_filters import Filter
from apps.accounts.models import Account from apps.accounts.models import Account
from apps.common.fields.month_year import MonthYearFormField from apps.common.fields.month_year import MonthYearFormField
from apps.common.widgets.datepicker import AirDatePickerInput from apps.common.widgets.datepicker import AirDatePickerInput
@@ -8,15 +15,9 @@ from apps.currencies.models import Currency
from apps.transactions.models import ( from apps.transactions.models import (
Transaction, Transaction,
TransactionCategory, TransactionCategory,
TransactionEntity,
TransactionTag, TransactionTag,
TransactionEntity,
) )
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Column, Field, Layout, Row
from django import forms
from django.db.models import Q
from django.utils.translation import gettext_lazy as _
from django_filters import Filter
SITUACAO_CHOICES = ( SITUACAO_CHOICES = (
("1", _("Paid")), ("1", _("Paid")),
@@ -59,20 +60,26 @@ class TransactionsFilter(django_filters.FilterSet):
label=_("Currencies"), label=_("Currencies"),
widget=TomSelectMultiple(checkboxes=True, remove_button=True), widget=TomSelectMultiple(checkboxes=True, remove_button=True),
) )
category = django_filters.MultipleChoiceFilter( category = django_filters.ModelMultipleChoiceFilter(
field_name="category__name",
queryset=TransactionCategory.objects.all(),
to_field_name="name",
label=_("Categories"), label=_("Categories"),
widget=TomSelectMultiple(checkboxes=True, remove_button=True), widget=TomSelectMultiple(checkboxes=True, remove_button=True),
method="filter_category",
) )
tags = django_filters.MultipleChoiceFilter( tags = django_filters.ModelMultipleChoiceFilter(
field_name="tags__name",
queryset=TransactionTag.objects.all(),
to_field_name="name",
label=_("Tags"), label=_("Tags"),
widget=TomSelectMultiple(checkboxes=True, remove_button=True), widget=TomSelectMultiple(checkboxes=True, remove_button=True),
method="filter_tags",
) )
entities = django_filters.MultipleChoiceFilter( entities = django_filters.ModelMultipleChoiceFilter(
field_name="entities__name",
queryset=TransactionEntity.objects.all(),
to_field_name="name",
label=_("Entities"), label=_("Entities"),
widget=TomSelectMultiple(checkboxes=True, remove_button=True), widget=TomSelectMultiple(checkboxes=True, remove_button=True),
method="filter_entities",
) )
is_paid = django_filters.MultipleChoiceFilter( is_paid = django_filters.MultipleChoiceFilter(
choices=SITUACAO_CHOICES, choices=SITUACAO_CHOICES,
@@ -118,7 +125,6 @@ class TransactionsFilter(django_filters.FilterSet):
"is_paid", "is_paid",
"category", "category",
"tags", "tags",
"entities",
"date_start", "date_start",
"date_end", "date_end",
"reference_date_start", "reference_date_start",
@@ -158,12 +164,14 @@ class TransactionsFilter(django_filters.FilterSet):
Field("description"), Field("description"),
Row(Column("date_start"), Column("date_end")), Row(Column("date_start"), Column("date_end")),
Row( Row(
Column("reference_date_start"), Column("reference_date_start", css_class="form-group col-md-6 mb-0"),
Column("reference_date_end"), Column("reference_date_end", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Row( Row(
Column("from_amount"), Column("from_amount", css_class="form-group col-md-6 mb-0"),
Column("to_amount"), Column("to_amount", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Field("account", size=1), Field("account", size=1),
Field("currency", size=1), Field("currency", size=1),
@@ -178,93 +186,6 @@ class TransactionsFilter(django_filters.FilterSet):
self.form.fields["date_end"].widget = AirDatePickerInput() self.form.fields["date_end"].widget = AirDatePickerInput()
self.form.fields["account"].queryset = Account.objects.all() self.form.fields["account"].queryset = Account.objects.all()
category_choices = list( self.form.fields["category"].queryset = TransactionCategory.objects.all()
TransactionCategory.objects.values_list("name", "name").order_by("name") self.form.fields["tags"].queryset = TransactionTag.objects.all()
) self.form.fields["entities"].queryset = TransactionEntity.objects.all()
custom_choices = [
("any", _("Categorized")),
("uncategorized", _("Uncategorized")),
]
self.form.fields["category"].choices = custom_choices + category_choices
tag_choices = list(
TransactionTag.objects.values_list("name", "name").order_by("name")
)
custom_tag_choices = [("any", _("Tagged")), ("untagged", _("Untagged"))]
self.form.fields["tags"].choices = custom_tag_choices + tag_choices
entity_choices = list(
TransactionEntity.objects.values_list("name", "name").order_by("name")
)
custom_entity_choices = [
("any", _("Any entity")),
("no_entity", _("No entity")),
]
self.form.fields["entities"].choices = custom_entity_choices + entity_choices
@staticmethod
def filter_category(queryset, name, value):
if not value:
return queryset
value = list(value)
if "any" in value:
return queryset.filter(category__isnull=False)
q = Q()
if "uncategorized" in value:
q |= Q(category__isnull=True)
value.remove("uncategorized")
if value:
q |= Q(category__name__in=value)
if q.children:
return queryset.filter(q)
return queryset
@staticmethod
def filter_tags(queryset, name, value):
if not value:
return queryset
value = list(value)
if "any" in value:
return queryset.filter(tags__isnull=False).distinct()
q = Q()
if "untagged" in value:
q |= Q(tags__isnull=True)
value.remove("untagged")
if value:
q |= Q(tags__name__in=value)
if q.children:
return queryset.filter(q).distinct()
return queryset
@staticmethod
def filter_entities(queryset, name, value):
if not value:
return queryset
value = list(value)
if "any" in value:
return queryset.filter(entities__isnull=False).distinct()
q = Q()
if "no_entity" in value:
q |= Q(entities__isnull=True)
value.remove("no_entity")
if value:
q |= Q(entities__name__in=value)
if q.children:
return queryset.filter(q).distinct()
return queryset
+185 -191
View File
@@ -1,38 +1,37 @@
from copy import deepcopy from crispy_bootstrap5.bootstrap5 import Switch, BS5Accordion
from crispy_forms.bootstrap import FormActions, AccordionGroup, AppendedText
from crispy_forms.helper import FormHelper
from crispy_forms.layout import (
Layout,
Row,
Column,
Field,
Div,
HTML,
)
from django import forms
from django.db.models import Q
from django.utils.translation import gettext_lazy as _
from apps.accounts.models import Account from apps.accounts.models import Account
from apps.common.fields.forms.dynamic_select import ( from apps.common.fields.forms.dynamic_select import (
DynamicModelChoiceField, DynamicModelChoiceField,
DynamicModelMultipleChoiceField, DynamicModelMultipleChoiceField,
) )
from apps.common.widgets.crispy.daisyui import Switch
from apps.common.widgets.crispy.submit import NoClassSubmit from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.common.widgets.datepicker import AirDatePickerInput, AirMonthYearPickerInput from apps.common.widgets.datepicker import AirDatePickerInput, AirMonthYearPickerInput
from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput from apps.common.widgets.decimal import ArbitraryDecimalDisplayNumberInput
from apps.common.widgets.tom_select import TomSelect from apps.common.widgets.tom_select import TomSelect
from apps.rules.signals import transaction_created, transaction_updated from apps.rules.signals import transaction_created, transaction_updated
from apps.transactions.models import ( from apps.transactions.models import (
InstallmentPlan,
QuickTransaction,
RecurringTransaction,
Transaction, Transaction,
TransactionCategory, TransactionCategory,
TransactionEntity,
TransactionTag, TransactionTag,
InstallmentPlan,
RecurringTransaction,
TransactionEntity,
QuickTransaction,
) )
from crispy_forms.bootstrap import AccordionGroup, AppendedText, FormActions, Accordion
from crispy_forms.helper import FormHelper
from crispy_forms.layout import (
HTML,
Column,
Div,
Field,
Layout,
Row,
)
from django import forms
from django.db.models import Q
from django.utils.translation import gettext_lazy as _
class TransactionForm(forms.ModelForm): class TransactionForm(forms.ModelForm):
@@ -133,18 +132,21 @@ class TransactionForm(forms.ModelForm):
), ),
Field("is_paid", template="transactions/widgets/paid_toggle_button.html"), Field("is_paid", template="transactions/widgets/paid_toggle_button.html"),
Row( Row(
Column("account"), Column("account", css_class="form-group col-md-6 mb-0"),
Column("entities"), Column("entities", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Row( Row(
Column(Field("date")), Column(Field("date"), css_class="form-group col-md-6 mb-0"),
Column(Field("reference_date")), Column(Field("reference_date"), css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"description", "description",
Field("amount", inputmode="decimal"), Field("amount", inputmode="decimal"),
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"notes", "notes",
) )
@@ -160,18 +162,20 @@ class TransactionForm(forms.ModelForm):
Field("is_paid", template="transactions/widgets/paid_toggle_button.html"), Field("is_paid", template="transactions/widgets/paid_toggle_button.html"),
"account", "account",
Row( Row(
Column(Field("date")), Column(Field("date"), css_class="form-group col-md-6 mb-0"),
Column(Field("reference_date")), Column(Field("reference_date"), css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"description", "description",
Field("amount", inputmode="decimal"), Field("amount", inputmode="decimal"),
Accordion( BS5Accordion(
AccordionGroup( AccordionGroup(
_("More"), _("More"),
"entities", "entities",
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"notes", "notes",
active=False, active=False,
@@ -181,7 +185,9 @@ class TransactionForm(forms.ModelForm):
css_class="mb-3", css_class="mb-3",
), ),
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -194,25 +200,29 @@ class TransactionForm(forms.ModelForm):
) )
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.fields["amount"].widget = ArbitraryDecimalDisplayNumberInput() self.fields["amount"].widget = ArbitraryDecimalDisplayNumberInput()
self.helper.layout.append( self.helper.layout.append(
Div( Div(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary"
),
NoClassSubmit( NoClassSubmit(
"submit_and_similar", "submit_and_similar",
_("Save and add similar"), _("Save and add similar"),
css_class="btn btn-primary btn-soft", css_class="btn btn-outline-primary",
), ),
NoClassSubmit( NoClassSubmit(
"submit_and_another", "submit_and_another",
_("Save and add another"), _("Save and add another"),
css_class="btn btn-primary btn-soft", css_class="btn btn-outline-primary",
), ),
css_class="flex flex-col gap-2 mt-3", css_class="d-grid gap-2",
), ),
) )
@@ -229,16 +239,11 @@ class TransactionForm(forms.ModelForm):
def save(self, **kwargs): def save(self, **kwargs):
is_new = not self.instance.id is_new = not self.instance.id
if not is_new:
old_data = deepcopy(Transaction.objects.get(pk=self.instance.id))
else:
old_data = None
instance = super().save(**kwargs) instance = super().save(**kwargs)
if is_new: if is_new:
transaction_created.send(sender=instance) transaction_created.send(sender=instance)
else: else:
transaction_updated.send(sender=instance, old_data=old_data) transaction_updated.send(sender=instance)
return instance return instance
@@ -336,16 +341,23 @@ class QuickTransactionForm(forms.ModelForm):
), ),
Field("is_paid", template="transactions/widgets/paid_toggle_button.html"), Field("is_paid", template="transactions/widgets/paid_toggle_button.html"),
"name", "name",
HTML('<hr class="hr my-3" />'), HTML("<hr />"),
Row( Row(
Column("account"), Column("account", css_class="form-group col-md-6 mb-0"),
Column("entities"), Column("entities", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
),
Row(
Column(Field("date"), css_class="form-group col-md-6 mb-0"),
Column(Field("reference_date"), css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"description", "description",
Field("amount", inputmode="decimal"), Field("amount", inputmode="decimal"),
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"notes", "notes",
Switch("mute"), Switch("mute"),
@@ -358,131 +370,58 @@ class QuickTransactionForm(forms.ModelForm):
) )
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.fields["amount"].widget = ArbitraryDecimalDisplayNumberInput() self.fields["amount"].widget = ArbitraryDecimalDisplayNumberInput()
self.helper.layout.append( self.helper.layout.append(
FormActions( Div(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary"
),
css_class="d-grid gap-2",
), ),
) )
class BulkEditTransactionForm(forms.Form): class BulkEditTransactionForm(TransactionForm):
type = forms.ChoiceField( is_paid = forms.NullBooleanField(required=False)
choices=(Transaction.Type.choices),
required=False,
label=_("Type"),
)
is_paid = forms.NullBooleanField(
required=False,
label=_("Paid"),
)
account = DynamicModelChoiceField(
model=Account,
required=False,
label=_("Account"),
queryset=Account.objects.filter(is_archived=False),
widget=TomSelect(clear_button=False, group_by="group"),
)
date = forms.DateField(
label=_("Date"),
required=False,
widget=AirDatePickerInput(clear_button=False),
)
reference_date = forms.DateField(
widget=AirMonthYearPickerInput(),
label=_("Reference Date"),
required=False,
)
amount = forms.DecimalField(
max_digits=42,
decimal_places=30,
required=False,
label=_("Amount"),
widget=ArbitraryDecimalDisplayNumberInput(),
)
description = forms.CharField(
max_length=500, required=False, label=_("Description")
)
notes = forms.CharField(
required=False,
widget=forms.Textarea(attrs={"rows": 3}),
label=_("Notes"),
)
category = DynamicModelChoiceField(
create_field="name",
model=TransactionCategory,
required=False,
label=_("Category"),
queryset=TransactionCategory.objects.filter(active=True),
)
tags = DynamicModelMultipleChoiceField(
model=TransactionTag,
to_field_name="name",
create_field="name",
required=False,
label=_("Tags"),
queryset=TransactionTag.objects.filter(active=True),
)
entities = DynamicModelMultipleChoiceField(
model=TransactionEntity,
to_field_name="name",
create_field="name",
required=False,
label=_("Entities"),
queryset=TransactionEntity.objects.all(),
)
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
# Make all fields optional
for field_name, field in self.fields.items():
field.required = False
self.fields["account"].queryset = Account.objects.filter( del self.helper.layout[-1] # Remove button
is_archived=False, del self.helper.layout[0:2] # Remove type, is_paid field
)
self.fields["category"].queryset = TransactionCategory.objects.filter( self.helper.layout.insert(
active=True 0,
)
self.fields["tags"].queryset = TransactionTag.objects.filter(active=True)
self.fields["entities"].queryset = TransactionEntity.objects.all()
self.helper = FormHelper()
self.helper.form_tag = False
self.helper.form_method = "post"
self.helper.layout = Layout(
Field( Field(
"type", "type",
template="transactions/widgets/unselectable_income_expense_toggle_buttons.html", template="transactions/widgets/unselectable_income_expense_toggle_buttons.html",
), ),
)
self.helper.layout.insert(
1,
Field( Field(
"is_paid", "is_paid",
template="transactions/widgets/unselectable_paid_toggle_button.html", template="transactions/widgets/unselectable_paid_toggle_button.html",
), ),
Row(
Column("account"),
Column("entities"),
),
Row(
Column(Field("date")),
Column(Field("reference_date")),
),
"description",
Field("amount", inputmode="decimal"),
Row(
Column("category"),
Column("tags"),
),
"notes",
FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"),
),
) )
self.fields["amount"].widget = ArbitraryDecimalDisplayNumberInput() self.helper.layout.append(
self.fields["date"].widget = AirDatePickerInput(clear_button=False) FormActions(
NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
),
)
class TransferForm(forms.Form): class TransferForm(forms.Form):
@@ -576,34 +515,62 @@ class TransferForm(forms.Form):
self.helper.layout = Layout( self.helper.layout = Layout(
Row( Row(
Column(Field("date")), Column(Field("date"), css_class="form-group col-md-6 mb-0"),
Column( Column(
Field("reference_date"), Field("reference_date"),
css_class="form-group col-md-6 mb-0",
), ),
css_class="form-row",
), ),
Field("description"), Field("description"),
Field("notes"), Field("notes"),
Switch("mute"), Switch("mute"),
Row( Row(
Column("from_account"), Column(
Column(Field("from_amount")), Row(
Column("from_category"), Column(
Column("from_tags"), "from_account",
css_class="bg-base-100 rounded-box p-4 border-base-content/60 border my-3", css_class="form-group col-md-6 mb-0",
),
Column(
Field("from_amount"),
css_class="form-group col-md-6 mb-0",
),
css_class="form-row",
),
Row(
Column("from_category", css_class="form-group col-md-6 mb-0"),
Column("from_tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
),
),
css_class="p-1 mx-1 my-3 border rounded-3",
), ),
Row( Row(
Column( Column(
"to_account", Row(
Column(
"to_account",
css_class="form-group col-md-6 mb-0",
),
Column(
Field("to_amount"),
css_class="form-group col-md-6 mb-0",
),
css_class="form-row",
),
Row(
Column("to_category", css_class="form-group col-md-6 mb-0"),
Column("to_tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
),
), ),
Column( css_class="p-1 mx-1 my-3 border rounded-3",
Field("to_amount"),
),
Column("to_category"),
Column("to_tags"),
css_class="bg-base-100 rounded-box p-4 border-base-content/60 border",
), ),
FormActions( FormActions(
NoClassSubmit("submit", _("Transfer"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Transfer"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -789,26 +756,30 @@ class InstallmentPlanForm(forms.ModelForm):
template="transactions/widgets/income_expense_toggle_buttons.html", template="transactions/widgets/income_expense_toggle_buttons.html",
), ),
Row( Row(
Column("account"), Column("account", css_class="form-group col-md-6 mb-0"),
Column("entities"), Column("entities", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"description", "description",
Switch("add_description_to_transaction"), Switch("add_description_to_transaction"),
"notes", "notes",
Switch("add_notes_to_transaction"), Switch("add_notes_to_transaction"),
Row( Row(
Column("number_of_installments"), Column("number_of_installments", css_class="form-group col-md-6 mb-0"),
Column("installment_start"), Column("installment_start", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Row( Row(
Column("start_date", css_class="col-span-12 md:col-span-4"), Column("start_date", css_class="form-group col-md-4 mb-0"),
Column("reference_date", css_class="col-span-12 md:col-span-4"), Column("reference_date", css_class="form-group col-md-4 mb-0"),
Column("recurrence", css_class="col-span-12 md:col-span-4"), Column("recurrence", css_class="form-group col-md-4 mb-0"),
css_class="form-row",
), ),
"installment_amount", "installment_amount",
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
) )
@@ -818,13 +789,17 @@ class InstallmentPlanForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -857,13 +832,17 @@ class TransactionTagForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -885,13 +864,17 @@ class TransactionEntityForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -916,13 +899,17 @@ class TransactionCategoryForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -1031,26 +1018,30 @@ class RecurringTransactionForm(forms.ModelForm):
template="transactions/widgets/income_expense_toggle_buttons.html", template="transactions/widgets/income_expense_toggle_buttons.html",
), ),
Row( Row(
Column("account"), Column("account", css_class="form-group col-md-6 mb-0"),
Column("entities"), Column("entities", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"description", "description",
Switch("add_description_to_transaction"), Switch("add_description_to_transaction"),
"amount", "amount",
Row( Row(
Column("category"), Column("category", css_class="form-group col-md-6 mb-0"),
Column("tags"), Column("tags", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
"notes", "notes",
Switch("add_notes_to_transaction"), Switch("add_notes_to_transaction"),
Row( Row(
Column("start_date"), Column("start_date", css_class="form-group col-md-6 mb-0"),
Column("reference_date"), Column("reference_date", css_class="form-group col-md-6 mb-0"),
css_class="form-row",
), ),
Row( Row(
Column("recurrence_interval", css_class="col-span-12 md:col-span-4"), Column("recurrence_interval", css_class="form-group col-md-4 mb-0"),
Column("recurrence_type", css_class="col-span-12 md:col-span-4"), Column("recurrence_type", css_class="form-group col-md-4 mb-0"),
Column("end_date", css_class="col-span-12 md:col-span-4"), Column("end_date", css_class="form-group col-md-4 mb-0"),
css_class="form-row",
), ),
AppendedText("keep_at_most", _("future transactions")), AppendedText("keep_at_most", _("future transactions")),
) )
@@ -1062,13 +1053,17 @@ class RecurringTransactionForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -1090,6 +1085,5 @@ class RecurringTransactionForm(forms.ModelForm):
instance.create_upcoming_transactions() instance.create_upcoming_transactions()
else: else:
instance.update_unpaid_transactions() instance.update_unpaid_transactions()
instance.generate_upcoming_transactions()
return instance return instance
+21 -91
View File
@@ -1,30 +1,28 @@
import decimal
import logging import logging
from copy import deepcopy
from apps.common.fields.month_year import MonthYearModelField
from apps.common.functions.decimals import truncate_decimal
from apps.common.middleware.thread_local import get_current_user
from apps.common.models import (
OwnedObject,
OwnedObjectManager,
SharedObject,
SharedObjectManager,
)
from apps.common.templatetags.decimal import drop_trailing_zeros, localize_number
from apps.currencies.utils.convert import convert
from apps.transactions.validators import validate_decimal_places, validate_non_negative
from dateutil.relativedelta import relativedelta from dateutil.relativedelta import relativedelta
from django.conf import settings from django.conf import settings
from django.core.validators import MinValueValidator from django.core.validators import MinValueValidator
from django.db import models, transaction from django.db import models, transaction
from django.db.models import Q from django.db.models import Q
from django.dispatch import Signal from django.dispatch import Signal
from django.forms import ValidationError
from django.template.defaultfilters import date from django.template.defaultfilters import date
from django.utils import timezone from django.utils import timezone
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.fields.month_year import MonthYearModelField
from apps.common.functions.decimals import truncate_decimal
from apps.common.templatetags.decimal import localize_number, drop_trailing_zeros
from apps.currencies.utils.convert import convert
from apps.transactions.validators import validate_decimal_places, validate_non_negative
from apps.common.middleware.thread_local import get_current_user
from apps.common.models import (
SharedObject,
SharedObjectManager,
OwnedObject,
OwnedObjectManager,
)
logger = logging.getLogger() logger = logging.getLogger()
@@ -35,13 +33,13 @@ transaction_deleted = Signal()
class SoftDeleteQuerySet(models.QuerySet): class SoftDeleteQuerySet(models.QuerySet):
@staticmethod @staticmethod
def _emit_signals(instances, created=False, old_data=None): def _emit_signals(instances, created=False):
"""Helper to emit signals for multiple instances""" """Helper to emit signals for multiple instances"""
for i, instance in enumerate(instances): for instance in instances:
if created: if created:
transaction_created.send(sender=instance) transaction_created.send(sender=instance)
else: else:
transaction_updated.send(sender=instance, old_data=old_data[i]) transaction_updated.send(sender=instance)
def bulk_create(self, objs, emit_signal=True, **kwargs): def bulk_create(self, objs, emit_signal=True, **kwargs):
instances = super().bulk_create(objs, **kwargs) instances = super().bulk_create(objs, **kwargs)
@@ -52,25 +50,22 @@ class SoftDeleteQuerySet(models.QuerySet):
return instances return instances
def bulk_update(self, objs, fields, emit_signal=True, **kwargs): def bulk_update(self, objs, fields, emit_signal=True, **kwargs):
old_data = deepcopy(objs)
result = super().bulk_update(objs, fields, **kwargs) result = super().bulk_update(objs, fields, **kwargs)
if emit_signal: if emit_signal:
self._emit_signals(objs, created=False, old_data=old_data) self._emit_signals(objs, created=False)
return result return result
def update(self, emit_signal=True, **kwargs): def update(self, emit_signal=True, **kwargs):
# Get instances before update # Get instances before update
instances = list(self) instances = list(self)
old_data = deepcopy(instances)
result = super().update(**kwargs) result = super().update(**kwargs)
if emit_signal: if emit_signal:
# Refresh instances to get new values # Refresh instances to get new values
refreshed = self.model.objects.filter(pk__in=[obj.pk for obj in instances]) refreshed = self.model.objects.filter(pk__in=[obj.pk for obj in instances])
self._emit_signals(refreshed, created=False, old_data=old_data) self._emit_signals(refreshed, created=False)
return result return result
@@ -381,35 +376,16 @@ class Transaction(OwnedObject):
db_table = "transactions" db_table = "transactions"
default_manager_name = "objects" default_manager_name = "objects"
def clean(self): def save(self, *args, **kwargs):
super().clean()
# Only process amount and reference_date if account exists
# If account is missing, Django's required field validation will handle it
try:
account = self.account
except Transaction.account.RelatedObjectDoesNotExist:
# Account doesn't exist, skip processing that depends on it
# Django will add the required field error
return
# Validate and normalize amount
if isinstance(self.amount, (str, int, float)):
self.amount = decimal.Decimal(str(self.amount))
self.amount = truncate_decimal( self.amount = truncate_decimal(
value=self.amount, decimal_places=account.currency.decimal_places value=self.amount, decimal_places=self.account.currency.decimal_places
) )
# Normalize reference_date
if self.reference_date: if self.reference_date:
self.reference_date = self.reference_date.replace(day=1) self.reference_date = self.reference_date.replace(day=1)
elif not self.reference_date and self.date: elif not self.reference_date and self.date:
self.reference_date = self.date.replace(day=1) self.reference_date = self.date.replace(day=1)
def save(self, *args, **kwargs):
# This is not recommended as it will run twice on some cases like form and API saves.
# We only do this here because we forgot to independently call it on multiple places.
self.full_clean() self.full_clean()
super().save(*args, **kwargs) super().save(*args, **kwargs)
@@ -467,58 +443,12 @@ class Transaction(OwnedObject):
type_display = self.get_type_display() type_display = self.get_type_display()
frmt_date = date(self.date, "SHORT_DATE_FORMAT") frmt_date = date(self.date, "SHORT_DATE_FORMAT")
account = self.account account = self.account
tags = ( tags = ", ".join([x.name for x in self.tags.all()]) or _("No tags")
", ".join([x.name for x in self.tags.all()])
if self.id
else None or _("No tags")
)
category = self.category or _("No category") category = self.category or _("No category")
amount = localize_number(drop_trailing_zeros(self.amount)) amount = localize_number(drop_trailing_zeros(self.amount))
description = self.description or _("No description") description = self.description or _("No description")
return f"[{frmt_date}][{type_display}][{account}] {description}{category}{tags}{amount}" return f"[{frmt_date}][{type_display}][{account}] {description}{category}{tags}{amount}"
def deepcopy(self, memo=None):
"""
Creates a deep copy of the transaction instance.
This method returns a new, unsaved Transaction instance with the same
values as the original, including its many-to-many relationships.
The primary key and any other unique fields are reset to avoid
database integrity errors upon saving.
"""
if memo is None:
memo = {}
# Create a new instance of the class
new_obj = self.__class__()
memo[id(self)] = new_obj
# Copy all concrete fields from the original to the new object
for field in self._meta.concrete_fields:
# Skip the primary key to allow the database to generate a new one
if field.primary_key:
continue
# Reset any unique fields to None to avoid constraint violations
if field.unique and field.name == "internal_id":
setattr(new_obj, field.name, None)
continue
# Copy the value of the field
setattr(new_obj, field.name, getattr(self, field.name))
# Save the new object to the database to get a primary key
new_obj.save()
# Copy the many-to-many relationships
for field in self._meta.many_to_many:
source_manager = getattr(self, field.name)
destination_manager = getattr(new_obj, field.name)
# Set the M2M relationships for the new object
destination_manager.set(source_manager.all())
return new_obj
class InstallmentPlan(models.Model): class InstallmentPlan(models.Model):
class Recurrence(models.TextChoices): class Recurrence(models.TextChoices):
@@ -3,6 +3,7 @@ from decimal import Decimal
from django import template from django import template
from django.utils.formats import number_format from django.utils.formats import number_format
register = template.Library() register = template.Library()
@@ -12,27 +13,13 @@ def _format_string(prefix, amount, decimal_places, suffix):
value=abs(amount), decimal_pos=decimal_places, force_grouping=True value=abs(amount), decimal_pos=decimal_places, force_grouping=True
) )
if amount < 0: if amount < 0:
return "-", prefix, formatted_amount, suffix
return f"-{prefix}{formatted_amount}{suffix}" return f"-{prefix}{formatted_amount}{suffix}"
else: else:
return "", prefix, formatted_amount, suffix
return f"{prefix}{formatted_amount}{suffix}" return f"{prefix}{formatted_amount}{suffix}"
else: else:
return "", "", "ERR", "" return "ERR"
@register.simple_tag(name="currency_display") @register.simple_tag(name="currency_display")
def currency_display(amount, prefix, suffix, decimal_places, string=False): def currency_display(amount, prefix, suffix, decimal_places):
sign, prefix, amount, suffix = _format_string( return _format_string(prefix, amount, decimal_places, suffix)
prefix, amount, decimal_places, suffix
)
if string:
return f"{sign}{prefix}{amount}{suffix}"
return {
"sign": sign,
"prefix": prefix,
"amount": amount,
"suffix": suffix,
}
+1 -1
View File
@@ -175,6 +175,6 @@ class RecurringTransactionTests(TestCase):
recurrence_type=RecurringTransaction.RecurrenceType.MONTH, recurrence_type=RecurringTransaction.RecurrenceType.MONTH,
recurrence_interval=1, recurrence_interval=1,
) )
self.assertFalse(recurring.is_paused) self.assertFalse(recurring.paused)
self.assertEqual(recurring.recurrence_interval, 1) self.assertEqual(recurring.recurrence_interval, 1)
self.assertEqual(recurring.account.currency.code, "USD") self.assertEqual(recurring.account.currency.code, "USD")
@@ -137,7 +137,6 @@ def quick_transaction_add_as_transaction(request, quick_transaction_id):
"category", "category",
"tags", "tags",
"entities", "entities",
"internal_id",
], ],
) )
@@ -207,7 +206,6 @@ def quick_transaction_add_as_quick_transaction(request, transaction_id):
"recurring_transaction", "recurring_transaction",
"deleted", "deleted",
"deleted_at", "deleted_at",
"internal_id",
], ],
) )
+7 -18
View File
@@ -213,7 +213,6 @@ def transactions_bulk_edit(request):
if form.is_valid(): if form.is_valid():
# Apply changes from the form to all selected transactions # Apply changes from the form to all selected transactions
for transaction in transactions: for transaction in transactions:
old_data = deepcopy(transaction)
for field_name, value in form.cleaned_data.items(): for field_name, value in form.cleaned_data.items():
if value or isinstance( if value or isinstance(
value, bool value, bool
@@ -226,7 +225,7 @@ def transactions_bulk_edit(request):
setattr(transaction, field_name, value) setattr(transaction, field_name, value)
transaction.save() transaction.save()
transaction_updated.send(sender=transaction, old_data=old_data) transaction_updated.send(sender=transaction)
messages.success( messages.success(
request, request,
@@ -374,13 +373,10 @@ def transactions_transfer(request):
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def transaction_pay(request, transaction_id): def transaction_pay(request, transaction_id):
transaction = get_object_or_404(Transaction, pk=transaction_id) transaction = get_object_or_404(Transaction, pk=transaction_id)
old_data = deepcopy(transaction)
new_is_paid = False if transaction.is_paid else True new_is_paid = False if transaction.is_paid else True
transaction.is_paid = new_is_paid transaction.is_paid = new_is_paid
transaction.save() transaction.save()
transaction_updated.send(sender=transaction)
transaction_updated.send(sender=transaction, old_data=old_data)
response = render( response = render(
request, request,
@@ -398,12 +394,11 @@ def transaction_pay(request, transaction_id):
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def transaction_mute(request, transaction_id): def transaction_mute(request, transaction_id):
transaction = get_object_or_404(Transaction, pk=transaction_id) transaction = get_object_or_404(Transaction, pk=transaction_id)
old_data = deepcopy(transaction)
new_mute = False if transaction.mute else True new_mute = False if transaction.mute else True
transaction.mute = new_mute transaction.mute = new_mute
transaction.save() transaction.save()
transaction_updated.send(sender=transaction, old_data=old_data) transaction_updated.send(sender=transaction)
response = render( response = render(
request, request,
@@ -419,20 +414,19 @@ def transaction_mute(request, transaction_id):
@require_http_methods(["GET"]) @require_http_methods(["GET"])
def transaction_change_month(request, transaction_id, change_type): def transaction_change_month(request, transaction_id, change_type):
transaction: Transaction = get_object_or_404(Transaction, pk=transaction_id) transaction: Transaction = get_object_or_404(Transaction, pk=transaction_id)
old_data = deepcopy(transaction)
if change_type == "next": if change_type == "next":
transaction.reference_date = transaction.reference_date + relativedelta( transaction.reference_date = transaction.reference_date + relativedelta(
months=1 months=1
) )
transaction.save() transaction.save()
transaction_updated.send(sender=transaction, old_data=old_data) transaction_updated.send(sender=transaction)
elif change_type == "previous": elif change_type == "previous":
transaction.reference_date = transaction.reference_date - relativedelta( transaction.reference_date = transaction.reference_date - relativedelta(
months=1 months=1
) )
transaction.save() transaction.save()
transaction_updated.send(sender=transaction, old_data=old_data) transaction_updated.send(sender=transaction)
return HttpResponse( return HttpResponse(
status=204, status=204,
@@ -446,11 +440,9 @@ def transaction_change_month(request, transaction_id, change_type):
def transaction_move_to_today(request, transaction_id): def transaction_move_to_today(request, transaction_id):
transaction: Transaction = get_object_or_404(Transaction, pk=transaction_id) transaction: Transaction = get_object_or_404(Transaction, pk=transaction_id)
old_data = deepcopy(transaction)
transaction.date = timezone.localdate(timezone.now()) transaction.date = timezone.localdate(timezone.now())
transaction.save() transaction.save()
transaction_updated.send(sender=transaction, old_data=old_data) transaction_updated.send(sender=transaction)
return HttpResponse( return HttpResponse(
status=204, status=204,
@@ -597,10 +589,7 @@ def transaction_all_currency_summary(request):
f = TransactionsFilter(request.GET, queryset=transactions) f = TransactionsFilter(request.GET, queryset=transactions)
currency_data = calculate_currency_totals( currency_data = calculate_currency_totals(f.qs.all(), ignore_empty=True)
f.qs.exclude(account__in=request.user.untracked_accounts.all()),
ignore_empty=True,
)
currency_percentages = calculate_percentage_distribution(currency_data) currency_percentages = calculate_percentage_distribution(currency_data)
context = { context = {
+28 -28
View File
@@ -1,43 +1,35 @@
from apps.common.middleware.thread_local import get_current_user
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.users.models import UserSettings
from crispy_forms.bootstrap import ( from crispy_forms.bootstrap import (
FormActions, FormActions,
) )
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Column, Div, Field, Layout, Row, Submit from crispy_forms.layout import Layout, Submit, Row, Column, Field, Div, HTML
from django import forms from django import forms
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
from django.contrib.auth.forms import ( from django.contrib.auth.forms import (
UsernameField,
AuthenticationForm, AuthenticationForm,
UserCreationForm, UserCreationForm,
UsernameField,
) )
from django.db import transaction from django.db import transaction
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from apps.common.widgets.crispy.submit import NoClassSubmit
from apps.users.models import UserSettings
from apps.common.middleware.thread_local import get_current_user
class LoginForm(AuthenticationForm): class LoginForm(AuthenticationForm):
username = UsernameField( username = UsernameField(
label=_("E-mail"), label=_("E-mail"),
widget=forms.EmailInput( widget=forms.EmailInput(
attrs={ attrs={"class": "form-control", "placeholder": "E-mail", "name": "email"}
"class": "input",
"placeholder": _("E-mail"),
"name": "email",
"autocomplete": "email",
}
), ),
) )
password = forms.CharField( password = forms.CharField(
label=_("Password"), label=_("Password"),
strip=False, strip=False,
widget=forms.PasswordInput( widget=forms.PasswordInput(
attrs={ attrs={"class": "form-control", "placeholder": "Senha"}
"class": "input",
"placeholder": _("Password"),
"autocomplete": "current-password",
}
), ),
) )
@@ -53,7 +45,7 @@ class LoginForm(AuthenticationForm):
self.helper.layout = Layout( self.helper.layout = Layout(
"username", "username",
"password", "password",
Submit("Submit", "Login", css_class="w-full mt-3"), Submit("Submit", "Login", css_class="btn btn-primary w-100"),
) )
@@ -97,8 +89,6 @@ class UserSettingsForm(forms.ModelForm):
("AA", _("Default")), ("AA", _("Default")),
("DC", "1.234,50"), ("DC", "1.234,50"),
("CD", "1,234.50"), ("CD", "1,234.50"),
("SD", "1 234.50"),
("SC", "1 234,50"),
] ]
date_format = forms.ChoiceField( date_format = forms.ChoiceField(
@@ -146,7 +136,9 @@ class UserSettingsForm(forms.ModelForm):
HTML("<hr />"), HTML("<hr />"),
"volume", "volume",
FormActions( FormActions(
NoClassSubmit("submit", _("Save"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Save"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -197,8 +189,8 @@ class UserUpdateForm(forms.ModelForm):
# Define the layout using Crispy Forms, including the new fields # Define the layout using Crispy Forms, including the new fields
self.helper.layout = Layout( self.helper.layout = Layout(
Row( Row(
Column("first_name"), Column("first_name", css_class="form-group col-md-6"),
Column("last_name"), Column("last_name", css_class="form-group col-md-6"),
css_class="row", css_class="row",
), ),
Field("email"), Field("email"),
@@ -219,13 +211,17 @@ class UserUpdateForm(forms.ModelForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
@@ -356,8 +352,8 @@ class UserAddForm(UserCreationForm):
self.helper.layout = Layout( self.helper.layout = Layout(
Field("email"), Field("email"),
Row( Row(
Column("first_name"), Column("first_name", css_class="form-group col-md-6"),
Column("last_name"), Column("last_name", css_class="form-group col-md-6"),
css_class="row", css_class="row",
), ),
# UserCreationForm provides 'password1' and 'password2' fields # UserCreationForm provides 'password1' and 'password2' fields
@@ -377,13 +373,17 @@ class UserAddForm(UserCreationForm):
if self.instance and self.instance.pk: if self.instance and self.instance.pk:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Update"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Update"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
else: else:
self.helper.layout.append( self.helper.layout.append(
FormActions( FormActions(
NoClassSubmit("submit", _("Add"), css_class="btn btn-primary"), NoClassSubmit(
"submit", _("Add"), css_class="btn btn-outline-primary w-100"
),
), ),
) )
+1 -10
View File
@@ -4,6 +4,7 @@ from . import views
urlpatterns = [ urlpatterns = [
path("", views.index, name="index"), path("", views.index, name="index"),
path("setup/", views.setup, name="setup"),
path("login/", views.UserLoginView.as_view(), name="login"), path("login/", views.UserLoginView.as_view(), name="login"),
# path("login/fallback/", views.UserLoginView.as_view(), name="fallback_login"), # path("login/fallback/", views.UserLoginView.as_view(), name="fallback_login"),
path("logout/", views.logout_view, name="logout"), path("logout/", views.logout_view, name="logout"),
@@ -17,16 +18,6 @@ urlpatterns = [
views.toggle_sound_playing, views.toggle_sound_playing,
name="toggle_sound_playing", name="toggle_sound_playing",
), ),
path(
"user/session/toggle-sidebar/",
views.toggle_sidebar_status,
name="toggle_sidebar_status",
),
path(
"user/session/toggle-theme/",
views.toggle_theme,
name="toggle_theme",
),
path( path(
"user/settings/", "user/settings/",
views.update_settings, views.update_settings,
+37 -49
View File
@@ -1,26 +1,29 @@
from apps.common.decorators.demo import disabled_on_demo
from apps.common.decorators.htmx import only_htmx
from apps.common.decorators.user import htmx_login_required, is_superuser
from apps.users.forms import (
LoginForm,
UserAddForm,
UserSettingsForm,
UserUpdateForm,
)
from apps.users.models import UserSettings
from django.contrib import messages from django.contrib import messages
from django.contrib.auth import get_user_model, logout from django.contrib.auth import logout, get_user_model
from django.contrib.auth.decorators import login_required from django.contrib.auth.decorators import login_required
from django.contrib.auth.views import ( from django.contrib.auth.views import (
LoginView, LoginView,
) )
from django.core.exceptions import PermissionDenied from django.core.exceptions import PermissionDenied
from django.http import HttpResponse from django.http import HttpResponse
from django.shortcuts import get_object_or_404, redirect, render from django.shortcuts import redirect, render, get_object_or_404
from django.urls import reverse from django.urls import reverse
from django.utils.translation import gettext_lazy as _ from django.utils.translation import gettext_lazy as _
from django.views.decorators.http import require_http_methods from django.views.decorators.http import require_http_methods
from apps.common.decorators.htmx import only_htmx
from apps.common.decorators.user import is_superuser, htmx_login_required
from apps.users.forms import (
LoginForm,
UserSettingsForm,
UserUpdateForm,
UserAddForm,
)
from apps.users.models import UserSettings
from apps.common.decorators.demo import disabled_on_demo
from apps.currencies.models import Currency
from apps.accounts.models import Account
def logout_view(request): def logout_view(request):
logout(request) logout(request)
@@ -47,6 +50,28 @@ def index(request):
return redirect(reverse("monthly_index")) return redirect(reverse("monthly_index"))
@login_required
def setup(request):
has_currency = Currency.objects.exists()
has_account = Account.objects.exists()
# return render(
# request,
# "users/setup/setup.html",
# {"has_currency": has_currency, "has_account": has_account},
# )
if not has_currency or not has_account:
return render(
request,
"users/setup/setup.html",
{"has_currency": has_currency, "has_account": has_account},
)
else:
return HttpResponse(
status=200,
headers={"HX-Reswap": "delete"},
)
class UserLoginView(LoginView): class UserLoginView(LoginView):
form_class = LoginForm form_class = LoginForm
template_name = "users/login.html" template_name = "users/login.html"
@@ -115,43 +140,6 @@ def update_settings(request):
return render(request, "users/fragments/user_settings.html", {"form": form}) return render(request, "users/fragments/user_settings.html", {"form": form})
@only_htmx
@htmx_login_required
@require_http_methods(["GET"])
def toggle_sidebar_status(request):
if not request.session.get("sidebar_status"):
request.session["sidebar_status"] = "floating"
if request.session["sidebar_status"] == "floating":
request.session["sidebar_status"] = "fixed"
elif request.session["sidebar_status"] == "fixed":
request.session["sidebar_status"] = "floating"
else:
request.session["sidebar_status"] = "fixed"
return HttpResponse(
status=204,
)
@htmx_login_required
@require_http_methods(["GET"])
def toggle_theme(request):
if not request.session.get("theme"):
request.session["theme"] = "wygiwyh_dark"
if request.session["theme"] == "wygiwyh_dark":
request.session["theme"] = "wygiwyh_light"
elif request.session["theme"] == "wygiwyh_light":
request.session["theme"] = "wygiwyh_dark"
else:
request.session["theme"] = "wygiwyh_light"
return HttpResponse(
status=204,
)
@htmx_login_required @htmx_login_required
@is_superuser @is_superuser
@require_http_methods(["GET"]) @require_http_methods(["GET"])
+1 -2
View File
@@ -79,7 +79,7 @@ def yearly_overview_by_currency(request, year: int):
currency = request.GET.get("currency") currency = request.GET.get("currency")
# Base query filter # Base query filter
filter_params = {"reference_date__year": year} filter_params = {"reference_date__year": year, "account__is_archived": False}
# Add month filter if provided # Add month filter if provided
if month: if month:
@@ -95,7 +95,6 @@ def yearly_overview_by_currency(request, year: int):
transactions = ( transactions = (
Transaction.objects.filter(**filter_params) Transaction.objects.filter(**filter_params)
.exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True)) .exclude(Q(Q(category__mute=True) & ~Q(category=None)) | Q(mute=True))
.exclude(account__in=request.user.untracked_accounts.all())
.order_by("account__currency__name") .order_by("account__currency__name")
) )
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
File diff suppressed because it is too large Load Diff
+1 -3
View File
@@ -1,3 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 375 374.999991" version="1.0"> <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="500" zoomAndPan="magnify" viewBox="0 0 375 374.999991" height="500" preserveAspectRatio="xMidYMid meet" version="1.0"><defs><clipPath id="a2373d79ec"><path d="M 1.980469 1.980469 L 373 1.980469 L 373 373 L 1.980469 373 Z M 1.980469 1.980469 " clip-rule="nonzero"/></clipPath></defs><g clip-path="url(#a2373d79ec)"><path fill="#fbb700" d="M 239.671875 301.757812 L 79.152344 141.238281 L 118.234375 102.152344 L 239.671875 223.589844 L 355.179688 108.078125 C 325.429688 45.34375 261.519531 1.957031 187.472656 1.957031 C 113.375 1.957031 49.433594 45.410156 19.707031 108.210938 L 174.503906 263.003906 L 135.757812 301.757812 L 2.882812 168.878906 C 2.273438 174.996094 1.957031 181.199219 1.957031 187.472656 C 1.957031 289.929688 85.015625 372.988281 187.472656 372.988281 C 289.929688 372.988281 372.988281 289.929688 372.988281 187.472656 C 372.988281 181.347656 372.679688 175.296875 372.101562 169.320312 L 239.671875 301.757812 " fill-opacity="1" fill-rule="nonzero"/></g></svg>
<path fill="#000000" d="M 239.671875 301.757812 L 79.152344 141.238281 L 118.234375 102.152344 L 239.671875 223.589844 L 355.179688 108.078125 C 325.429688 45.34375 261.519531 1.957031 187.472656 1.957031 C 113.375 1.957031 49.433594 45.410156 19.707031 108.210938 L 174.503906 263.003906 L 135.757812 301.757812 L 2.882812 168.878906 C 2.273438 174.996094 1.957031 181.199219 1.957031 187.472656 C 1.957031 289.929688 85.015625 372.988281 187.472656 372.988281 C 289.929688 372.988281 372.988281 289.929688 372.988281 187.472656 C 372.988281 181.347656 372.679688 175.296875 372.101562 169.320312 L 239.671875 301.757812 "/>
</svg>

Before

Width:  |  Height:  |  Size: 718 B

After

Width:  |  Height:  |  Size: 1.1 KiB

@@ -1,72 +1,78 @@
{% load i18n %} {% load i18n %}
<c-ui.fab-single-action <div class="container px-md-3 py-3 column-gap-5">
url="{% url 'account_group_add' %}" <div class="tw:text-3xl fw-bold font-monospace tw:w-full mb-3">
hx_target="#generic-offcanvas">
</c-ui.fab-single-action>
<div class="container">
<div class="text-3xl font-bold font-mono w-full mb-3">
{% spaceless %} {% spaceless %}
<div>{% translate 'Account Groups' %}</div> <div>{% translate 'Account Groups' %}<span>
<a class="text-decoration-none tw:text-2xl p-1 category-action"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Add" %}"
hx-get="{% url 'account_group_add' %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-circle-plus fa-fw"></i></a>
</span></div>
{% endspaceless %} {% endspaceless %}
</div> </div>
<div class="card bg-base-100 shadow-xl"> <div class="card">
<div class="card-body"> <div class="card-body table-responsive">
{% if account_groups %} {% if account_groups %}
<c-config.search></c-config.search> <c-config.search></c-config.search>
<div class="overflow-x-auto"> <table class="table table-hover">
<table class="table table-zebra"> <thead>
<thead> <tr>
<tr> <th scope="col" class="col-auto"></th>
<th scope="col" class="table-col-auto"></th> <th scope="col" class="col">{% translate 'Name' %}</th>
<th scope="col">{% translate 'Name' %}</th> </tr>
</thead>
<tbody>
{% for account_group in account_groups %}
<tr class="account_group">
<td class="col-auto">
<div class="btn-group" role="group" aria-label="{% translate 'Actions' %}">
<a class="btn btn-secondary btn-sm"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Edit" %}"
hx-get="{% url 'account_group_edit' pk=account_group.id %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-pencil fa-fw"></i></a>
<a class="btn btn-secondary btn-sm text-danger"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Delete" %}"
hx-delete="{% url 'account_group_delete' pk=account_group.id %}"
hx-trigger='confirmed'
data-bypass-on-ctrl="true"
data-title="{% translate "Are you sure?" %}"
data-text="{% translate "You won't be able to revert this!" %}"
data-confirm-text="{% translate "Yes, delete it!" %}"
_="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a>
{% if not account_group.owner %}
<a class="btn btn-secondary btn-sm text-warning"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Take ownership" %}"
hx-get="{% url 'account_group_take_ownership' pk=account_group.id %}">
<i class="fa-solid fa-crown fa-fw"></i></a>
{% endif %}
{% if user == account_group.owner %}
<a class="btn btn-secondary btn-sm text-primary"
role="button"
hx-target="#generic-offcanvas"
hx-swap="innerHTML"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Share" %}"
hx-get="{% url 'account_group_share_settings' pk=account_group.id %}">
<i class="fa-solid fa-share fa-fw"></i></a>
{% endif %}
</div>
</td>
<td class="col">{{ account_group.name }}</td>
</tr> </tr>
</thead> {% endfor %}
<tbody> </tbody>
{% for account_group in account_groups %} </table>
<tr class="account_group">
<td class="table-col-auto">
<div class="join" role="group" aria-label="{% translate 'Actions' %}">
<a class="btn btn-secondary btn-sm join-item"
role="button"
data-tippy-content="{% translate "Edit" %}"
hx-get="{% url 'account_group_edit' pk=account_group.id %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-pencil fa-fw"></i></a>
{% if not account_group.owner %}
<a class="btn btn-secondary btn-sm join-item"
role="button"
data-tippy-content="{% translate "Take ownership" %}"
hx-get="{% url 'account_group_take_ownership' pk=account_group.id %}">
<i class="fa-solid fa-crown fa-fw"></i></a>
{% endif %}
{% if user == account_group.owner %}
<a class="btn btn-secondary btn-sm join-item"
role="button"
hx-target="#generic-offcanvas"
hx-swap="innerHTML"
data-tippy-content="{% translate "Share" %}"
hx-get="{% url 'account_group_share_settings' pk=account_group.id %}">
<i class="fa-solid fa-share fa-fw"></i></a>
{% endif %}
<a class="btn btn-error btn-sm join-item"
role="button"
data-tippy-content="{% translate "Delete" %}"
hx-delete="{% url 'account_group_delete' pk=account_group.id %}"
hx-trigger='confirmed'
data-bypass-on-ctrl="true"
data-title="{% translate "Are you sure?" %}"
data-text="{% translate "You won't be able to revert this!" %}"
data-confirm-text="{% translate "Yes, delete it!" %}"
_="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a>
</div>
</td>
<td>{{ account_group.name }}</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>
{% else %} {% else %}
<c-msg.empty title="{% translate "No account groups" %}" remove-padding></c-msg.empty> <c-msg.empty title="{% translate "No account groups" %}" remove-padding></c-msg.empty>
{% endif %} {% endif %}
@@ -9,59 +9,65 @@
<form hx-post="{% url 'account_reconciliation' %}"> <form hx-post="{% url 'account_reconciliation' %}">
{% csrf_token %} {% csrf_token %}
{{ form.management_form }} {{ form.management_form }}
<div class="join join-vertical w-full" id="balanceAccordionFlush"> <div class="accordion accordion-flush" id="balanceAccordionFlush">
{% for form in form.forms %} {% for form in form.forms %}
<c-ui.components.collapse> <div class="accordion-item">
<c-slot name="title"> <h2 class="accordion-header">
{% if form.account_group %}<span class="badge badge-primary badge-outline me-2">{{ form.account_group.name }}</span>{% endif %}{{ form.account_name }} <button class="accordion-button collapsed" type="button" data-bs-toggle="collapse"
</c-slot> data-bs-target="#flush-collapse-{{ forloop.counter0 }}" aria-expanded="false"
<c-slot name="content"> aria-controls="flush-collapseOne">
<div class="fieldset"> {% if form.account_group %}<span class="badge text-bg-primary me-2">{{ form.account_group.name }}</span>{% endif %}{{ form.account_name }}
<span class="fieldset-legend">{% translate 'Current balance' %}</span> </button>
<div data-amount="{{ form.current_balance|floatformat:"-40u" }}" </h2>
data-decimal-places="{{ form.currency_decimal_places }}" <div id="flush-collapse-{{ forloop.counter0 }}" class="accordion-collapse collapse">
id="amount-{{ forloop.counter0 }}" class="text-base"> <div class="accordion-body">
<c-amount.display <div class="mb-3">
:amount="form.current_balance" <div class="form-label">
:prefix="form.currency_prefix" {% translate 'Current balance' %}
:suffix="form.currency_suffix" </div>
:decimal_places="form.currency_decimal_places" <div data-amount="{{ form.current_balance|floatformat:"-40u" }}"
color="auto"></c-amount.display> data-decimal-places="{{ form.currency_decimal_places }}"
id="amount-{{ forloop.counter0 }}">
{% currency_display amount=form.current_balance prefix=form.currency_prefix suffix=form.currency_suffix decimal_places=form.currency_decimal_places %}
</div>
</div>
<div>
{% crispy form %}
</div>
<div class="mb-3">
<div class="form-label">
{% translate 'Difference' %}
</div>
<div _="on input from #id_form-{{ forloop.counter0 }}-new_balance
set original_amount to parseFloat('{{ form.current_balance|floatformat:"-40u" }}')
then set prefix to '{{ form.currency_prefix }}'
then set suffix to '{{ form.currency_suffix }}'
then set decimal_places to {{ form.currency_decimal_places }}
then call parseLocaleNumber(#id_form-{{ forloop.counter0 }}-new_balance.value)
then set new_amount to result
then set diff to (Math.round((new_amount - original_amount) * Math.pow(10, decimal_places))) / Math.pow(10, decimal_places)
then log diff
then set format_new_amount to
Intl.NumberFormat(
undefined,
{
minimumFractionDigits: decimal_places,
maximumFractionDigits: decimal_places,
roundingMode: 'trunc'
}
).format(diff)
then set formatted_string to `${prefix}${format_new_amount}${suffix}`
then put formatted_string into me if diff else
put '-' into me">-</div>
</div> </div>
</div> </div>
<div> </div>
{% crispy form %} </div>
</div>
<div class="fieldset">
<span class="fieldset-legend">{% translate 'Difference' %}</span>
<div class="text-base"
_="on input from #id_form-{{ forloop.counter0 }}-new_balance
set original_amount to parseFloat('{{ form.current_balance|floatformat:"-40u" }}')
then set prefix to '{{ form.currency_prefix }}'
then set suffix to '{{ form.currency_suffix }}'
then set decimal_places to {{ form.currency_decimal_places }}
then call parseLocaleNumber(#id_form-{{ forloop.counter0 }}-new_balance.value)
then set new_amount to result
then set diff to (Math.round((new_amount - original_amount) * Math.pow(10, decimal_places))) / Math.pow(10, decimal_places)
then set format_new_amount to
Intl.NumberFormat(
undefined,
{
minimumFractionDigits: decimal_places,
maximumFractionDigits: decimal_places,
roundingMode: 'trunc'
}
).format(diff)
then set formatted_string to `${prefix}${format_new_amount}${suffix}`
then put formatted_string into me if diff else put '-' into me">-</div>
</div>
</c-slot>
</c-ui.components.collapse>
{% endfor %} {% endfor %}
</div> </div>
<div class="mt-3"> <div class="mt-3">
<div> <div>
<input type="submit" name="submit" value="{% translate 'Reconcile balances' %}" class="btn btn-primary w-full" id="submit-id-submit"> <input type="submit" name="submit" value="{% translate 'Reconcile balances' %}" class="btn btn-outline-primary w-100" id="submit-id-submit">
</div> </div>
</div> </div>
</form> </form>
+80 -86
View File
@@ -1,96 +1,90 @@
{% load i18n %} {% load i18n %}
<c-ui.fab-single-action <div class="container px-md-3 py-3 column-gap-5">
url="{% url 'account_add' %}" <div class="tw:text-3xl fw-bold font-monospace tw:w-full mb-3">
hx_target="#generic-offcanvas">
</c-ui.fab-single-action>
<div class="container">
<div class="text-3xl font-bold font-mono w-full mb-3">
{% spaceless %} {% spaceless %}
<div>{% translate 'Accounts' %}</div> <div>{% translate 'Accounts' %}<span>
<a class="text-decoration-none tw:text-2xl p-1 category-action"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Add" %}"
hx-get="{% url 'account_add' %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-circle-plus fa-fw"></i></a>
</span></div>
{% endspaceless %} {% endspaceless %}
</div> </div>
<div class="card bg-base-100 shadow-xl"> <div class="card">
<div class="card-body"> <div class="card-body table-responsive">
{% if accounts %} {% if accounts %}
<c-config.search></c-config.search> <c-config.search></c-config.search>
<div class="overflow-x-auto"> <table class="table table-hover text-nowrap">
<table class="table table-zebra"> <thead>
<thead> <tr>
<tr> <th scope="col" class="col-auto"></th>
<th scope="col" class="table-col-auto"></th> <th scope="col" class="col">{% translate 'Name' %}</th>
<th scope="col">{% translate 'Name' %}</th> <th scope="col" class="col">{% translate 'Group' %}</th>
<th scope="col">{% translate 'Group' %}</th> <th scope="col" class="col">{% translate 'Currency' %}</th>
<th scope="col">{% translate 'Currency' %}</th> <th scope="col" class="col">{% translate 'Exchange Currency' %}</th>
<th scope="col">{% translate 'Exchange Currency' %}</th> <th scope="col" class="col">{% translate 'Is Asset' %}</th>
<th scope="col">{% translate 'Is Asset' %}</th> <th scope="col" class="col">{% translate 'Archived' %}</th>
<th scope="col">{% translate 'Archived' %}</th> </tr>
</tr> </thead>
</thead> <tbody>
<tbody> {% for account in accounts %}
{% for account in accounts %} <tr class="account">
<tr class="account"> <td class="col-auto">
<td class="table-col-auto"> <div class="btn-group" role="group" aria-label="{% translate 'Actions' %}">
<div class="join" role="group" aria-label="{% translate 'Actions' %}"> <a class="btn btn-secondary btn-sm"
<a class="btn btn-secondary btn-sm join-item" role="button"
role="button" data-bs-toggle="tooltip"
data-tippy-content="{% translate "Edit" %}" data-bs-title="{% translate "Edit" %}"
hx-get="{% url 'account_edit' pk=account.id %}" hx-get="{% url 'account_edit' pk=account.id %}"
hx-target="#generic-offcanvas"> hx-target="#generic-offcanvas">
<i class="fa-solid fa-pencil fa-fw"></i></a> <i class="fa-solid fa-pencil fa-fw"></i></a>
{% if not account.owner %} <a class="btn btn-secondary btn-sm text-danger"
<a class="btn btn-secondary btn-sm join-item" role="button"
role="button" data-bs-toggle="tooltip"
data-tippy-content="{% translate "Take ownership" %}" data-bs-title="{% translate "Delete" %}"
hx-get="{% url 'account_take_ownership' pk=account.id %}"> hx-delete="{% url 'account_delete' pk=account.id %}"
<i class="fa-solid fa-crown fa-fw"></i></a> hx-trigger='confirmed'
{% endif %} data-bypass-on-ctrl="true"
{% if user == account.owner %} data-title="{% translate "Are you sure?" %}"
<a class="btn btn-secondary btn-sm join-item" data-text="{% translate "You won't be able to revert this!" %}"
role="button" data-confirm-text="{% translate "Yes, delete it!" %}"
hx-target="#generic-offcanvas" _="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a>
hx-swap="innerHTML" {% if not account.owner %}
data-tippy-content="{% translate "Share" %}" <a class="btn btn-secondary btn-sm text-primary"
hx-get="{% url 'account_share_settings' pk=account.id %}"> role="button"
<i class="fa-solid fa-share fa-fw"></i></a> data-bs-toggle="tooltip"
{% endif %} data-bs-title="{% translate "Take ownership" %}"
<a class="btn btn-secondary btn-sm join-item" hx-get="{% url 'account_take_ownership' pk=account.id %}">
role="button" <i class="fa-solid fa-crown fa-fw"></i></a>
hx-get="{% url 'account_toggle_untracked' pk=account.id %}" {% endif %}
data-tippy-content=" {% if user == account.owner %}
<a class="btn btn-secondary btn-sm text-primary"
{% if account.is_untracked_by %}{% translate "Track" %}{% else %}{% translate "Untrack" %}{% endif %}"> role="button"
{% if account.is_untracked_by %} hx-target="#generic-offcanvas"
<i class="fa-solid fa-eye fa-fw"></i> hx-swap="innerHTML"
{% else %} data-bs-toggle="tooltip"
<i class="fa-solid fa-eye-slash fa-fw"></i> data-bs-title="{% translate "Share" %}"
{% endif %} hx-get="{% url 'account_share_settings' pk=account.id %}">
</a> <i class="fa-solid fa-share fa-fw"></i></a>
<a class="btn btn-error btn-sm join-item" {% endif %}
role="button" </div>
data-tippy-content="{% translate "Delete" %}" </td>
hx-delete="{% url 'account_delete' pk=account.id %}" <td class="col">{{ account.name }}</td>
hx-trigger='confirmed' <td class="col">{{ account.group.name }}</td>
data-bypass-on-ctrl="true" <td class="col">{{ account.currency }}</td>
data-title="{% translate "Are you sure?" %}" <td class="col">{% if account.exchange_currency %}{{ account.exchange_currency }}{% else %}-{% endif %}</td>
data-text="{% translate "You won't be able to revert this!" %}" <td class="col">{% if account.is_asset %}<i class="fa-solid fa-solid fa-check text-success"></i>{% endif %}</td>
data-confirm-text="{% translate "Yes, delete it!" %}" <td class="col">{% if account.is_archived %}<i class="fa-solid fa-solid fa-check text-success"></i>{% endif %}</td>
_="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a> </tr>
</div> {% endfor %}
</td> </tbody>
<td>{{ account.name }}</td> </table>
<td>{{ account.group.name }}</td>
<td>{{ account.currency }}</td>
<td>{% if account.exchange_currency %}{{ account.exchange_currency }}{% else %}-{% endif %}</td>
<td>{% if account.is_asset %}<i class="fa-solid fa-solid fa-check text-success"></i>{% endif %}</td>
<td>{% if account.is_archived %}<i class="fa-solid fa-solid fa-check text-success"></i>{% endif %}</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>
{% else %} {% else %}
<c-msg.empty title="{% translate "No accounts" %}" remove-padding></c-msg.empty> <c-msg.empty title="{% translate "No accounts" %}" remove-padding></c-msg.empty>
{% endif %} {% endif %}
</div> </div>
</div> </div>
+23 -23
View File
@@ -2,67 +2,67 @@
{% load i18n %} {% load i18n %}
<div> <div>
<div class="hidden lg:grid lg:grid-cols-7 gap-4 lg:gap-0 bg-base-200"> <div class="tw:hidden tw:lg:grid tw:lg:grid-cols-7 tw:gap-4 tw:lg:gap-0">
<div class="border-l border-t border-b border-base-300 p-2 text-center"> <div class="border-start border-top border-bottom p-2 text-center">
{% translate 'MON' %} {% translate 'MON' %}
</div> </div>
<div class="border-t border-b border-base-300 p-2 text-center"> <div class="border-top border-bottom p-2 text-center">
{% translate 'TUE' %} {% translate 'TUE' %}
</div> </div>
<div class="border-t border-b border-base-300 p-2 text-center"> <div class="border-top border-bottom p-2 text-center">
{% translate 'WED' %} {% translate 'WED' %}
</div> </div>
<div class="border-t border-b border-base-300 p-2 text-center"> <div class="border-top border-bottom p-2 text-center">
{% translate 'THU' %} {% translate 'THU' %}
</div> </div>
<div class="border-t border-b border-base-300 p-2 text-center"> <div class="border-top border-bottom p-2 text-center">
{% translate 'FRI' %} {% translate 'FRI' %}
</div> </div>
<div class="border-t border-b border-base-300 p-2 text-center"> <div class="border-top border-bottom p-2 text-center">
{% translate 'SAT' %} {% translate 'SAT' %}
</div> </div>
<div class="border-r border-t border-b border-base-300 p-2 text-center"> <div class="border-end border-top border-bottom p-2 text-center">
{% translate 'SUN' %} {% translate 'SUN' %}
</div> </div>
</div> </div>
<div class="grid grid-cols-1 grid-rows-1 lg:grid-cols-7 lg:grid-rows-6 gap-4 lg:gap-0"> <div class="tw:grid tw:grid-cols-1 tw:grid-rows-1 tw:lg:grid-cols-7 tw:lg:grid-rows-6 tw:gap-4 tw:lg:gap-0">
{% for date in dates %} {% for date in dates %}
{% if date %} {% if date %}
<div class="card bg-base-100 h-full hover:bg-base-200! border border-base-content/30 rounded-none {% if not date.transactions %}hidden! lg:flex!{% endif %}{% if today == date.date %} border-2 border-primary{% endif %} cursor-pointer" role="button" <div class="card h-100 tw:hover:bg-zinc-900! rounded-0{% if not date.transactions %} tw:hidden! tw:lg:flex!{% endif %}{% if today == date.date %} tw:border-yellow-300 border-primary{% endif %} " role="button"
hx-get="{% url 'calendar_transactions_list' day=date.date.day month=date.date.month year=date.date.year %}" hx-get="{% url 'calendar_transactions_list' day=date.date.day month=date.date.month year=date.date.year %}"
hx-target="#persistent-generic-offcanvas-left"> hx-target="#persistent-generic-offcanvas-left">
<div class="card-header border-0 bg-transparent text-end flex justify-between p-2 w-full"> <div class="card-header border-0 bg-transparent text-end tw:flex justify-content-between p-2 w-100">
<div class="lg:hidden text-start w-full">{{ date.date|date:"l"|lower }}</div> <div class="tw:lg:hidden text-start w-100">{{ date.date|date:"l"|lower }}</div>
<div class="text-end w-full">{{ date.day }}</div> <div class="text-end w-100">{{ date.day }}</div>
</div> </div>
<div class="card-body p-2 flex flex-row flex-wrap gap-1"> <div class="card-body p-2">
{% for transaction in date.transactions %} {% for transaction in date.transactions %}
{% if transaction.is_paid %} {% if transaction.is_paid %}
{% if transaction.type == "IN" and not transaction.account.is_asset %} {% if transaction.type == "IN" and not transaction.account.is_asset %}
<i class="fa-solid fa-circle-check text-success" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i> <i class="fa-solid fa-circle-check tw:text-green-400" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i>
{% elif transaction.type == "IN" and transaction.account.is_asset %} {% elif transaction.type == "IN" and transaction.account.is_asset %}
<i class="fa-solid fa-circle-check text-success/80" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i> <i class="fa-solid fa-circle-check tw:text-green-300" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i>
{% elif transaction.type == "EX" and not transaction.account.is_asset %} {% elif transaction.type == "EX" and not transaction.account.is_asset %}
<i class="fa-solid fa-circle-check text-error" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i> <i class="fa-solid fa-circle-check tw:text-red-400" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i>
{% elif transaction.type == "EX" and transaction.account.is_asset %} {% elif transaction.type == "EX" and transaction.account.is_asset %}
<i class="fa-solid fa-circle-check text-error/80" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i> <i class="fa-solid fa-circle-check tw:text-red-300" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i>
{% endif %} {% endif %}
{% else %} {% else %}
{% if transaction.type == "IN" and not transaction.account.is_asset %} {% if transaction.type == "IN" and not transaction.account.is_asset %}
<i class="fa-regular fa-circle text-success" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i> <i class="fa-regular fa-circle tw:text-green-400" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i>
{% elif transaction.type == "IN" and transaction.account.is_asset %} {% elif transaction.type == "IN" and transaction.account.is_asset %}
<i class="fa-regular fa-circle text-success/80" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i> <i class="fa-regular fa-circle tw:text-green-300" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Income' %}{% endif %}"></i>
{% elif transaction.type == "EX" and not transaction.account.is_asset %} {% elif transaction.type == "EX" and not transaction.account.is_asset %}
<i class="fa-regular fa-circle text-error" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i> <i class="fa-regular fa-circle tw:text-red-400" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i>
{% elif transaction.type == "EX" and transaction.account.is_asset %} {% elif transaction.type == "EX" and transaction.account.is_asset %}
<i class="fa-regular fa-circle text-error/80" data-tippy-content="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i> <i class="fa-regular fa-circle tw:text-red-300" data-bs-toggle="tooltip" data-bs-title="{% if transaction.description %}{{ transaction.description }}{% else %}{% trans 'Expense' %}{% endif %}"></i>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</div> </div>
</div> </div>
{% else %} {% else %}
<div class="hidden! lg:block! card bg-base-300 h-full rounded-none"></div> <div class="tw:hidden! tw:lg:block! card h-100 rounded-0"></div>
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</div> </div>
+31 -20
View File
@@ -3,6 +3,7 @@
{% load i18n %} {% load i18n %}
{% load month_name %} {% load month_name %}
{% load static %} {% load static %}
{% load webpack_loader %}
{% block title %}{% translate 'Monthly Overview' %} :: {{ month|month_name }}/{{ year }}{% endblock %} {% block title %}{% translate 'Monthly Overview' %} :: {{ month|month_name }}/{{ year }}{% endblock %}
@@ -12,35 +13,45 @@
{% endblock %} {% endblock %}
{% block content %} {% block content %}
<div class="container"> <div class="container px-md-3 py-3 column-gap-5">
<div class="flex flex-wrap mb-4 gap-x-xl-4 gap-y-3"> <div class="row mb-3 gx-xl-4 gy-3 mb-4">
{# Date picker#} {# Date picker#}
<div class="w-full xl:w-4/12 flex-row items-center flex"> <div class="col-12 col-xl-4 flex-row align-items-center d-flex">
<a role="button" <div class="tw:text-base h-100 align-items-center d-flex">
hx-boost="true" <a role="button"
class="btn btn-ghost" class="pe-4 py-2"
hx-trigger="click, previous_month from:window" hx-boost="true"
href="{% url 'calendar' month=previous_month year=previous_year %}"> hx-trigger="click, previous_month from:window"
<i class="fa-solid fa-chevron-left"></i> href="{% url 'calendar' month=previous_month year=previous_year %}"><i
</a> class="fa-solid fa-chevron-left"></i></a>
<div class="text-2xl font-bold btn btn-ghost flex-1 text-center whitespace-normal flex-wrap h-auto min-w-0 1flex flex-" </div>
<div class="tw:text-3xl fw-bold font-monospace tw:w-full text-center"
hx-get="{% url 'month_year_picker' %}" hx-get="{% url 'month_year_picker' %}"
hx-target="#generic-offcanvas-left" hx-target="#generic-offcanvas-left"
hx-trigger="click, date_picker from:window" hx-trigger="click, date_picker from:window"
hx-vals='{"month": {{ month }}, "year": {{ year }}, "for": "calendar", "field": "date"}' role="button"> hx-vals='{"month": {{ month }}, "year": {{ year }}, "for": "calendar", "field": "date"}' role="button">
{{ month|month_name }} {{ year }} {{ month|month_name }} {{ year }}
</div> </div>
<a role="button" <div class="tw:text-base mx-2 h-100 align-items-center d-flex">
hx-boost="true" <a role="button"
class="btn btn-ghost" class="ps-3 py-2"
hx-trigger="click, next_month from:window" hx-boost="true"
href="{% url 'calendar' month=next_month year=next_year %}"> hx-trigger="click, next_month from:window"
<i class="fa-solid fa-chevron-right"></i> href="{% url 'calendar' month=next_month year=next_year %}">
</a> <i class="fa-solid fa-chevron-right"></i>
</a>
</div>
</div>
{# Action buttons#}
<div class="col-12 col-xl-8">
{# <c-ui.quick-transactions-buttons#}
{# :year="year"#}
{# :month="month"#}
{# ></c-ui.quick-transactions-buttons>#}
</div> </div>
</div> </div>
<div class="flex flex-wrap"> <div class="row">
<div class="show-loading w-full" hx-get="{% url 'calendar_list' month=month year=year %}" <div class="show-loading" hx-get="{% url 'calendar_list' month=month year=year %}"
hx-trigger="load, updated from:window, selective_update from:window, every 10m"></div> hx-trigger="load, updated from:window, selective_update from:window, every 10m"></div>
</div> </div>
</div> </div>
+22 -19
View File
@@ -1,29 +1,32 @@
{% load i18n %} {% load i18n %}
<c-ui.fab-single-action <div class="container px-md-3 py-3 column-gap-5">
url="{% url 'category_add' %}" <div class="tw:text-3xl fw-bold font-monospace tw:w-full mb-3">
hx_target="#generic-offcanvas">
</c-ui.fab-single-action>
<div class="container">
<div class="text-3xl font-bold font-mono w-full mb-3">
{% spaceless %} {% spaceless %}
<div>{% translate 'Categories' %}</div> <div>{% translate 'Categories' %}<span>
<a class="text-decoration-none tw:text-2xl p-1 category-action"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Add" %}"
hx-get="{% url 'category_add' %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-circle-plus fa-fw"></i></a>
</span></div>
{% endspaceless %} {% endspaceless %}
</div> </div>
<div class="card bg-base-100 shadow-xl"> <div class="card">
<div class="card-header bg-base-200 p-4 rounded-box"> <div class="card-header">
<div role="tablist" class="tabs tabs-border"> <ul class="nav nav-pills card-header-pills" id="myTab" role="tablist">
<input type="radio" name="installment_plan_tabs" class="tab" aria-label="{% translate 'Active' %}" <li class="nav-item" role="presentation">
checked="checked" <button class="nav-link active" data-bs-toggle="tab" type="button" role="tab" aria-selected="true" hx-get="{% url 'categories_table_active' %}" hx-trigger="load, click" hx-target="#categories-table">{% translate 'Active' %}</button>
hx-get="{% url 'categories_table_active' %}" hx-trigger="load, click" hx-target="#categories-table" </li>
hx-indicator="#categories-table"/> <li class="nav-item" role="presentation">
<input type="radio" name="installment_plan_tabs" class="tab" aria-label="{% translate 'Archived' %}" <button class="nav-link" hx-get="{% url 'categories_table_archived' %}" hx-target="#categories-table" data-bs-toggle="tab" type="button" role="tab" aria-selected="false">{% translate 'Archived' %}</button>
hx-get="{% url 'categories_table_archived' %}" hx-trigger="click" hx-target="#categories-table" </li>
hx-indicator="#categories-table"/> </ul>
</div>
</div> </div>
<div class="card-body"> <div class="card-body">
<div id="categories-table" class="show-loading"></div> <div id="categories-table"></div>
</div> </div>
</div> </div>
</div> </div>
+65 -63
View File
@@ -6,70 +6,72 @@
<div class="show-loading" hx-get="{% url 'categories_table_archived' %}" hx-trigger="updated from:window" <div class="show-loading" hx-get="{% url 'categories_table_archived' %}" hx-trigger="updated from:window"
hx-swap="outerHTML"> hx-swap="outerHTML">
{% endif %} {% endif %}
{% if categories %} {% if categories %}
<div> <div class="table-responsive">
<c-config.search></c-config.search> <c-config.search></c-config.search>
<div class="overflow-x-auto"> <table class="table table-hover">
<table class="table table-zebra"> <thead>
<thead> <tr>
<tr> <th scope="col" class="col-auto"></th>
<th scope="col" class="table-col-auto"></th> <th scope="col" class="col">{% translate 'Name' %}</th>
<th scope="col">{% translate 'Name' %}</th> <th scope="col" class="col">{% translate 'Muted' %}</th>
<th scope="col">{% translate 'Muted' %}</th> </tr>
</thead>
<tbody>
{% for category in categories %}
<tr class="category">
<td class="col-auto text-center">
<div class="btn-group" role="group" aria-label="{% translate 'Actions' %}">
<a class="btn btn-secondary btn-sm"
role="button"
data-bs-toggle="tooltip"
hx-swap="innerHTML"
data-bs-title="{% translate "Edit" %}"
hx-get="{% url 'category_edit' category_id=category.id %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-pencil fa-fw"></i></a>
<a class="btn btn-secondary btn-sm text-danger"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Delete" %}"
hx-delete="{% url 'category_delete' category_id=category.id %}"
hx-trigger='confirmed'
hx-swap="innerHTML"
data-bypass-on-ctrl="true"
data-title="{% translate "Are you sure?" %}"
data-text="{% translate "You won't be able to revert this!" %}"
data-confirm-text="{% translate "Yes, delete it!" %}"
_="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a>
{% if not category.owner %}
<a class="btn btn-secondary btn-sm text-primary"
role="button"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Take ownership" %}"
hx-get="{% url 'category_take_ownership' category_id=category.id %}">
<i class="fa-solid fa-crown fa-fw"></i></a>
{% endif %}
{% if user == category.owner %}
<a class="btn btn-secondary btn-sm text-primary"
role="button"
hx-target="#generic-offcanvas"
hx-swap="innerHTML"
data-bs-toggle="tooltip"
data-bs-title="{% translate "Share" %}"
hx-get="{% url 'category_share_settings' pk=category.id %}">
<i class="fa-solid fa-share fa-fw"></i></a>
{% endif %}
</div>
</td>
<td class="col">{{ category.name }}</td>
<td class="col">
{% if category.mute %}<i class="fa-solid fa-check text-success"></i>{% endif %}
</td>
</tr> </tr>
</thead> {% endfor %}
<tbody> </tbody>
{% for category in categories %} </table>
<tr class="category">
<td class="table-col-auto text-center">
<div class="join" role="group" aria-label="{% translate 'Actions' %}">
<a class="btn btn-secondary btn-sm join-item"
role="button"
hx-swap="innerHTML"
data-tippy-content="{% translate "Edit" %}"
hx-get="{% url 'category_edit' category_id=category.id %}"
hx-target="#generic-offcanvas">
<i class="fa-solid fa-pencil fa-fw"></i></a>
{% if not category.owner %}
<a class="btn btn-secondary btn-sm join-item"
role="button"
data-tippy-content="{% translate "Take ownership" %}"
hx-get="{% url 'category_take_ownership' category_id=category.id %}">
<i class="fa-solid fa-crown fa-fw"></i></a>
{% endif %}
{% if user == category.owner %}
<a class="btn btn-secondary btn-sm join-item"
role="button"
hx-target="#generic-offcanvas"
hx-swap="innerHTML"
data-tippy-content="{% translate "Share" %}"
hx-get="{% url 'category_share_settings' pk=category.id %}">
<i class="fa-solid fa-share fa-fw"></i></a>
{% endif %}
<a class="btn btn-error btn-sm join-item"
role="button"
data-tippy-content="{% translate "Delete" %}"
hx-delete="{% url 'category_delete' category_id=category.id %}"
hx-trigger='confirmed'
hx-swap="innerHTML"
data-bypass-on-ctrl="true"
data-title="{% translate "Are you sure?" %}"
data-text="{% translate "You won't be able to revert this!" %}"
data-confirm-text="{% translate "Yes, delete it!" %}"
_="install prompt_swal"><i class="fa-solid fa-trash fa-fw"></i></a>
</div>
</td>
<td>{{ category.name }}</td>
<td>
{% if category.mute %}<i class="fa-solid fa-check text-success"></i>{% endif %}
</td>
</tr>
{% endfor %}
</tbody>
</table>
</div>
</div> </div>
{% else %} {% else %}
<c-msg.empty title="{% translate "No categories" %}" remove-padding></c-msg.empty> <c-msg.empty title="{% translate "No categories" %}" remove-padding></c-msg.empty>
{% endif %} {% endif %}
</div> </div>
@@ -5,35 +5,47 @@
{% block title %}{% translate 'Pick a month' %}{% endblock %} {% block title %}{% translate 'Pick a month' %}{% endblock %}
{% block body %} {% block body %}
{% regroup month_year_data by year as years_list %} {% regroup month_year_data by year as years_list %}
<div role="tablist" class="tabs tabs-border w-full" id="yearTabs"> <ul class="nav nav-pills nav-fill" id="yearTabs" role="tablist">
{% for x in years_list %} {% for x in years_list %}
<input type="radio" <li class="nav-item" role="presentation">
name="year_tabs" <button class="nav-link{% if x.grouper == current_year %} active{% endif %}"
role="tab" id="{{ x.grouper }}"
class="tab" data-bs-toggle="tab"
aria-label="{{ x.grouper }}" data-bs-target="#{{ x.grouper }}-pane"
id="tab-{{ x.grouper }}" type="button"
{% if x.grouper == current_year %}checked="checked"{% endif %} /> role="tab"
<div role="tabpanel" class="tab-content" id="{{ x.grouper }}-pane"> aria-controls="{{ x.grouper }}-pane"
<ul class="menu bg-base-100 w-full" id="month-year-list" hx-boost="true"> aria-selected="{% if x.grouper == current_year %}true{% else %}false{% endif %}">
{% for month_data in x.list %} {{ x.grouper }}
<li {% if month_data.month == current_month and month_data.year == current_year %}class="disabled"{% endif %}> </button>
<a class="{% if month_data.month == current_month and month_data.year == current_year %}menu-active{% endif %}" </li>
href={{ month_data.url }} {% endfor %}
{% if month_data.month == current_month and month_data.year == current_year %}aria-disabled="true"{% endif %}> </ul>
<span class="flex-1">{{ month_data.month|month_name }}</span> <div class="tab-content" id="yearTabsContent" hx-boost="true">
<span class="badge badge-primary">{{ month_data.transaction_count }}</span> {% for x in years_list %}
</a> <div class="tab-pane fade{% if x.grouper == current_year %} show active{% endif %} mt-2"
</li> id="{{ x.grouper }}-pane"
{% endfor %} role="tabpanel"
</ul> aria-labelledby="{{ x.grouper }}"
</div> tabindex="0">
{% endfor %} <ul class="list-group list-group-flush" id="month-year-list">
</div> {% for month_data in x.list %}
<hr class="hr my-4"> <li class="list-group-item tw:hover:bg-zinc-900
<div class="w-full text-end"> {% if month_data.month == current_month and month_data.year == current_year %} disabled bg-primary{% endif %}"
<a class="btn btn-outline btn-primary btn-sm" href="{{ today_url }}" role="button" hx-boost="true">{% trans 'Today' %}</a> {% if month_data.month == current_month and month_data.year == current_year %}aria-disabled="true"{% endif %}>
<div class="d-flex justify-content-between">
<a class="text-decoration-none stretched-link {% if month_data.month == current_month and month_data.year == current_year %} text-black{% endif %}"
href={{ month_data.url }}>
{{ month_data.month|month_name }}</a>
<span class="badge text-bg-secondary">{{ month_data.transaction_count }}</span>
</div>
</li>
{% endfor %}
</ul>
</div> </div>
{% endfor %}
</div>
{% endblock %} {% endblock %}
+8 -14
View File
@@ -2,25 +2,19 @@
{% load toast_bg %} {% load toast_bg %}
{% if messages %} {% if messages %}
{% for message in messages %} {% for message in messages %}
<div class="toasty alert alert-{{ message.tags | toast_bg }}" <div class="toast align-items-center text-bg-{{ message.tags | toast_bg }} border-0"
role="alert" role="alert"
aria-live="assertive" aria-live="assertive"
aria-atomic="true"> aria-atomic="true">
<div class="flex items-center justify-between w-full"> <div class="toast-header">
<div class="flex items-center gap-2"> <i class="{{ message.tags | toast_icon }} fa-fw me-1"></i>
<i class="{{ message.tags | toast_icon }} fa-fw"></i> <strong class="me-auto">{{ message.tags | toast_title }}</strong>
<div>
<strong>{{ message.tags | toast_title }}</strong>
<div>{{ message }}</div>
</div>
</div>
<button type="button" <button type="button"
class="btn btn-ghost btn-sm btn-circle" class="btn-close"
_="on click remove closest .toasty" data-bs-dismiss="toast"
aria-label={% translate 'Close' %}> aria-label={% translate 'Close' %}></button>
<i class="fa-solid fa-xmark"></i>
</button>
</div> </div>
<div class="toast-body">{{ message }}</div>
</div> </div>
{% endfor %} {% endfor %}
{% endif %} {% endif %}
-26
View File
@@ -1,26 +0,0 @@
{#This is here so we can add dynamic Tailwind classes that will be required via JS/hyperscript but Tailwind has no knowledge of#}
<div class="lg:w-[15vw]"></div>
<div class="lg:ml-[16vw]"></div>
<div class="grid-cols-12"></div>
<div class="md:col-span-1"></div>
<div class="md:col-span-2"></div>
<div class="md:col-span-3"></div>
<div class="md:col-span-4"></div>
<div class="md:col-span-5"></div>
<div class="md:col-span-6"></div>
<div class="md:col-span-7"></div>
<div class="md:col-span-8"></div>
<div class="md:col-span-9"></div>
<div class="md:col-span-10"></div>
<div class="md:col-span-11"></div>
<div class="md:col-span-12"></div>
<div class="col-span-12"></div>
<div class="alert-error"></div>
<div class="alert-info"></div>
<div class="alert-success"></div>
<div class="alert-warning"></div>
<div class="textarea"></div>
<div class="border-base-content/60"></div>
<div class="bg-error/20"></div>
<div class="bg-success/20"></div>
<div class="checkbox checkbox-xs"></div>
+2 -15
View File
@@ -1,23 +1,10 @@
{% load currency_display %} {% load currency_display %}
{% currency_display amount=amount prefix=prefix suffix=suffix decimal_places=decimal_places as formatted_amount %}
{% if not divless %} {% if not divless %}
<div class="{% if text_end %}text-end{% elif text_start %}text-start{% endif %}"> <div class="{% if text_end %}text-end{% elif text_start %}text-start{% endif %}">
{% endif %} {% endif %}
<span class="amount <span class="amount{% if color == 'grey' or color == "gray" %} tw:text-gray-500{% elif color == 'green' %} tw:text-green-400{% elif color == 'red' %} tw:text-red-400{% endif %} {{ custom_class }}"
{% if color == 'grey' or color == "gray" %} text-exchange-rate data-original-value="{% currency_display amount=amount prefix=prefix suffix=suffix decimal_places=decimal_places %}"
{% elif color == 'green' %} text-income {% elif color == 'red' %} text-expense
{% elif color == 'auto' %}
{% if amount > 0 %} text-income
{% elif amount < 0 %} text-expense
{% else %} text-base-content {% endif %}
{% endif %}
font-medium {{ custom_class }}"
data-original-sign="{{ formatted_amount.sign }}"
data-original-prefix="{{ formatted_amount.prefix }}"
data-original-amount="{{ formatted_amount.amount }}"
data-original-suffix="{{ formatted_amount.suffix }}"
data-amount="{{ amount|floatformat:"-40u" }}"> data-amount="{{ amount|floatformat:"-40u" }}">
</span><span>{{ slot }}</span> </span><span>{{ slot }}</span>
{% if not divless %} {% if not divless %}
+29 -10
View File
@@ -1,14 +1,33 @@
{% load i18n%} <div class="tw:min-h-16">
<div
id="fab-wrapper"
class="tw:fixed tw:bottom-5 tw:right-5 tw:ml-auto tw:w-max tw:flex tw:flex-col tw:items-end mt-5 tw:z-20">
<div
id="menu"
class="tw:flex tw:flex-col tw:items-end tw:space-y-6 tw:transition-all tw:duration-300 tw:ease-in-out tw:opacity-0 tw:invisible tw:hidden tw:mb-2">
<div class="fab"> {{ slot }}
<div tabindex="0" role="button" class="btn btn-lg btn-circle btn-primary">
<i class="fa-solid fa-plus text-2xl fa-fw"></i>
</div> </div>
<div class="fab-close"> <button
{% trans 'Close' %} <span class="btn btn-circle btn-lg btn-error"><i class="fa-solid fa-xmark text-2xl fa-fw"></i></span> class="btn btn-primary rounded-circle p-0 tw:w-12 tw:h-12 tw:flex tw:items-center tw:justify-center tw:shadow-lg tw:hover:shadow-xl tw:focus:shadow-xl tw:transition-all tw:duration-300 tw:ease-in-out"
</div> _="
on click or focusout
{{ slot }} if #menu.classList.contains('tw:invisible') and event.type === 'click'
add .{'tw:rotate-45'} to #fab-icon
remove .{'tw:invisible'} from #menu
remove .{'tw:hidden'} from #menu
remove .{'tw:opacity-0'} from #menu
else
wait 0.2s
remove .{'tw:rotate-45'} from #fab-icon
add .{'tw:invisible'} to #menu
add .{'tw:hidden'} to #menu
add .{'tw:opacity-0'} to #menu
end
"
>
<i id="fab-icon" class="fa-solid fa-plus tw:text-3xl tw:transition-transform tw:duration-300 tw:ease-in-out"></i>
</button>
</div>
</div> </div>
@@ -1,6 +1,11 @@
<div hx-get="{{ url }}" {% load i18n %}
hx-trigger="{{ hx_trigger }}" <div class="tw:relative fab-item">
hx-target="{{ hx_target }}" <button class="btn btn-sm btn-{{ color }}"
hx-vals='{{ hx_vals }}'> hx-get="{{ url }}"
<span class="bg-neutral/60 text-neutral-content rounded-box p-2">{{ title }}</span> hx-trigger="{{ hx_trigger }}"
<button class="btn btn-lg btn-circle btn-{{color}}"><i class="{{ icon }} fa-fw"></i></button></div> hx-target="{{ hx_target }}"
hx-vals='{{ hx_vals }}'>
<i class="{{ icon }} me-2"></i>
{{ title }}
</button>
</div>
@@ -1,12 +1,12 @@
<li class="lg:hidden lg:group-hover:block"> <li class="tw:lg:hidden tw:lg:group-hover:block">
<div class="flex items-center" data-bs-toggle="collapse" href="#{{ title|slugify }}" role="button" <div class="d-flex align-items-center" data-bs-toggle="collapse" href="#{{ title|slugify }}" role="button"
aria-expanded="false" aria-controls="{{ title|slugify }}"> aria-expanded="false" aria-controls="{{ title|slugify }}">
<span <span
class="text-base-content/60 text-sm font-bold uppercase lg:hidden lg:group-hover:inline me-2">{{ title }}</span> class="text-muted small fw-bold text-uppercase tw:lg:hidden tw:lg:group-hover:inline me-2">{{ title }}</span>
<hr class="flex-grow"/> <hr class="flex-grow-1"/>
<i class="fas fa-chevron-down text-base-content/60 lg:before:hidden lg:group-hover:before:inline ml-2 lg:ml-0 lg:group-hover:ml-2"></i> <i class="fas fa-chevron-down text-muted tw:lg:before:hidden tw:lg:group-hover:before:inline tw:ml-2 tw:lg:ml-0 tw:lg:group-hover:ml-2"></i>
</div> </div>
</li> </li>
<div class="collapse lg:hidden lg:group-hover:block" id="{{ title|slugify }}"> <div class="collapse tw:lg:hidden tw:lg:group-hover:block" id="{{ title|slugify }}">
{{ slot }} {{ slot }}
</div> </div>
@@ -1,8 +1,6 @@
<li> <li>
<div class="flex items-center min-h-6"> <div class="d-flex align-items-center">
{% if title %} <span class="text-muted small fw-bold text-uppercase tw:lg:hidden tw:lg:group-hover:inline me-2">{{ title }}</span>
<span class="sidebar-menu-header text-base-content/60 text-xs font-bold uppercase mr-3">{{ title }}</span> <hr class="flex-grow-1"/>
{% endif %}
<hr class="hr grow"/>
</div> </div>
</li> </li>
@@ -1,13 +1,14 @@
{% load active_link %} {% load active_link %}
<li> <li>
<a href="{% url url %}" <a href="{% url url %}"
class="text-xs flex items-center no-underline ps-3 p-2 rounded-box sidebar-item {% active_link views=active css_class="sidebar-active" %}" class="tw:text-wrap tw:lg:text-nowrap tw:lg:text-sm d-flex align-items-center text-decoration-none p-2 rounded-3 sidebar-item {% active_link views=active css_class="sidebar-active" %}"
{% if tooltip %} {% if tooltip %}
data-tippy-placement="right" data-bs-placement="right"
data-tippy-content="{{ tooltip }}" data-bs-toggle="tooltip"
data-bs-title="{{ tooltip }}"
{% endif %}> {% endif %}>
<i class="{{ icon }} fa-fw"></i> <i class="{{ icon }} fa-fw"></i>
<span <span
class="ms-3 font-medium">{{ title }}</span> class="ms-3 fw-medium tw:lg:invisible tw:lg:group-hover:visible tw:lg:group-hover:truncate tw:lg:group-focus:truncate tw:lg:group-hover:text-ellipsis tw:lg:group-focus:text-ellipsis">{{ title }}</span>
</a> </a>
</li> </li>

Some files were not shown because too many files have changed in this diff Show More