diff options
author | luxagraf <sng@luxagraf.net> | 2022-11-11 16:57:21 -0600 |
---|---|---|
committer | luxagraf <sng@luxagraf.net> | 2022-11-11 16:57:21 -0600 |
commit | 2db52b9aecda64220777983a5a4ce26b9eb237f6 (patch) | |
tree | 25c091d2dbb3f7312c05c53f9a4193a8e43e849b | |
parent | be0cc14d112afc136e54806d7105ded0a7c3f15a (diff) |
bdgt: added simple spending tracker
-rw-r--r-- | app/budget/admin.py | 17 | ||||
-rw-r--r-- | app/budget/migrations/0001_initial.py | 36 | ||||
-rw-r--r-- | app/budget/migrations/__init__.py | 0 | ||||
-rw-r--r-- | app/budget/models.py | 31 | ||||
-rw-r--r-- | app/budget/templates/budget/base.html | 34 | ||||
-rw-r--r-- | app/budget/templates/budget/create_form.html | 26 | ||||
-rw-r--r-- | app/budget/urls.py | 24 | ||||
-rw-r--r-- | app/budget/views.py | 13 | ||||
-rw-r--r-- | config/base_urls.py | 1 |
9 files changed, 182 insertions, 0 deletions
diff --git a/app/budget/admin.py b/app/budget/admin.py new file mode 100644 index 0000000..fc84fa0 --- /dev/null +++ b/app/budget/admin.py @@ -0,0 +1,17 @@ +from django.contrib import admin + +from .models import LuxSource, LuxPurchase + + +@admin.register(LuxSource) +class SourceAdmin(admin.ModelAdmin): + list_display = ('name',) + +@admin.register(LuxPurchase) +class PurchaseAdmin(admin.ModelAdmin): + list_display = ('source', 'amount', 'category' ) + search_fields = ['source', 'amount'] + list_filter = ('category',) + + class Media: + js = ('next-prev-links.js',) diff --git a/app/budget/migrations/0001_initial.py b/app/budget/migrations/0001_initial.py new file mode 100644 index 0000000..30bd5a1 --- /dev/null +++ b/app/budget/migrations/0001_initial.py @@ -0,0 +1,36 @@ +# Generated by Django 4.0.6 on 2022-11-11 17:50 + +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + initial = True + + dependencies = [ + ] + + operations = [ + migrations.CreateModel( + name='LuxSource', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('name', models.CharField(max_length=200)), + ('date_recorded', models.DateTimeField(auto_now_add=True)), + ], + ), + migrations.CreateModel( + name='LuxPurchase', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('amount', models.IntegerField()), + ('category', models.IntegerField(choices=[(0, 'Grocery and Home'), (1, 'Gas'), (2, 'Bus'), (3, 'Lodging'), (4, 'Books'), (5, 'Clothes'), (6, 'Eating Out'), (7, 'Misc')], default=0)), + ('date_recorded', models.DateTimeField(auto_now_add=True)), + ('source', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='budget.luxsource')), + ], + options={ + 'ordering': ('-date_recorded',), + }, + ), + ] diff --git a/app/budget/migrations/__init__.py b/app/budget/migrations/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/app/budget/migrations/__init__.py diff --git a/app/budget/models.py b/app/budget/models.py new file mode 100644 index 0000000..acf2b69 --- /dev/null +++ b/app/budget/models.py @@ -0,0 +1,31 @@ +from django.db import models + +class LuxSource(models.Model): + name = models.CharField(max_length=200) + date_recorded = models.DateTimeField(auto_now_add=True) + + def __str__(self): + return self.name + +class LuxPurchase(models.Model): + amount = models.IntegerField() + source = models.ForeignKey(LuxSource, on_delete=models.CASCADE) + CATEGORY = ( + (0, 'Grocery and Home'), + (1, 'Gas'), + (2, 'Bus'), + (3, 'Lodging'), + (4, 'Books'), + (5, 'Clothes'), + (6, 'Eating Out'), + (7, 'Misc'), + ) + category = models.IntegerField(choices=CATEGORY, default=0) + date_recorded = models.DateTimeField(auto_now_add=True) + + class Meta: + ordering = ('-date_recorded',) + + def __str__(self): + return self.name + diff --git a/app/budget/templates/budget/base.html b/app/budget/templates/budget/base.html new file mode 100644 index 0000000..e17813a --- /dev/null +++ b/app/budget/templates/budget/base.html @@ -0,0 +1,34 @@ +<html> +<head> + <title>{% block pagetitle %}Luxagraf - Trading{% endblock %}</title> + <meta charset="utf-8"> + <meta http-equiv="x-ua-compatible" content="ie=edge"> + <meta name="viewport" content="width=device-width, initial-scale=1"> + {%block stylesheet%}<link rel="stylesheet" + href="/media/trading.css{%comment%}?{% now "u" %}{%endcomment%}" + media="screen">{%endblock%} + <link rel="shortcut icon" href="/favicon.ico" type="image/x-icon"> +<style> + select { + border: 1px solid #dedddd; + border-radius: 4px; + padding: 2.2rem 0 0.75rem 0.75rem; + width: 96%; + font-size: 24px; + font-size: 1.5rem; + } +</style> + {%block extrahead%}{%endblock%} +</head> + </head> + <body> + <nav> + <span class="nav-item"><a href="{% url 'luxtrade:list' %}">Home</a></span> + </nav> + {% block content %} + {% endblock %} + </body> + {% block js %} + {% endblock %} +</html> + diff --git a/app/budget/templates/budget/create_form.html b/app/budget/templates/budget/create_form.html new file mode 100644 index 0000000..2d38acf --- /dev/null +++ b/app/budget/templates/budget/create_form.html @@ -0,0 +1,26 @@ +{% extends 'budget/base.html' %} +{% load typogrify_tags %} + {% block pagetitle %}Luxagraf - Record Purchase{% endblock %} + {% block content %} + <form id="id_form" action="{% url 'luxbudget:createview' %}" method="post" class="big">{% csrf_token %} + {% for field in form %} + <fieldset> + {{ field.errors }} + {% if field.name == 'status'%} + <label class="hide" for="id_status">Status:</label>{{ field }} + {% else %} + {{ field.label_tag }} {{ field }} + {% endif %} + {% if field.help_text %} + <p class="help">{{ field.help_text|safe }}</p> + {% endif %} + </fieldset> +{% endfor %} + <div class="flex"> + <input type="submit" name="post" class="btn" value="record purchase"/> + </div> + </form> + {% endblock %} + + {% block js %} + {% endblock %} diff --git a/app/budget/urls.py b/app/budget/urls.py new file mode 100644 index 0000000..86c56a2 --- /dev/null +++ b/app/budget/urls.py @@ -0,0 +1,24 @@ +from django.urls import path, re_path + +from . import views + +app_name = "luxbudget" + +urlpatterns = [ + path( + 'record', + views.PurchaseModelFormView.as_view(), + name='createview' + ), + path( + 'purchase/<pk>', + views.PurchaseModelFormView.as_view(), + name='detail' + ), + path( + '', + views.LuxPurchaseListView.as_view(), + {'page':1}, + name='list' + ), +] diff --git a/app/budget/views.py b/app/budget/views.py new file mode 100644 index 0000000..3743eaf --- /dev/null +++ b/app/budget/views.py @@ -0,0 +1,13 @@ +from django.shortcuts import render +from django.views.generic.edit import CreateView, UpdateView +from utils.views import PaginatedListView +from .models import LuxPurchase + +class PurchaseModelFormView(CreateView): + model = LuxPurchase + fields = ['amount', 'source', 'category'] + success_url = '/spending/' + template_name = 'budget/create_form.html' + +class LuxPurchaseListView(PaginatedListView): + pass diff --git a/config/base_urls.py b/config/base_urls.py index a149038..7314d1c 100644 --- a/config/base_urls.py +++ b/config/base_urls.py @@ -34,6 +34,7 @@ urlpatterns = [ #path(r'admin/income/invoice/monthlyview/<str:slug>/', MonthlyInvoiceView.as_view(), name="monthly-invoice"), path(r'admin/', admin.site.urls), path(r'trading/', include('trading.urls')), + path(r'spending/', include('budget.urls')), path(r'planner/', include('planner.urls')), path(r'luximages/insert/', utils.views.insert_image), path(r'luxproduct/insert/', products.views.insert_products), |