Netdeploy2 Babyyyy

This commit is contained in:
Ben Mosley
2026-02-03 17:41:29 -06:00
commit c77674b86d
23 changed files with 659 additions and 0 deletions

22
routes/clients.py Normal file
View File

@@ -0,0 +1,22 @@
from flask import Blueprint, render_template, request, redirect, url_for
from extensions import db
from models.client import Client
bp = Blueprint("clients", __name__, url_prefix="/clients")
@bp.route("/")
def list_clients():
return render_template("clients/list.html", clients=Client.query.all())
@bp.route("/new", methods=["GET","POST"])
def new_client():
if request.method == "POST":
c = Client(
name=request.form["name"],
email=request.form["email"]
)
db.session.add(c)
db.session.commit()
return redirect(url_for("clients.list_clients"))
return render_template("clients/new.html")

27
routes/invoices.py Normal file
View File

@@ -0,0 +1,27 @@
from flask import Blueprint, render_template, redirect, url_for
from extensions import db
from models.quote import Quote
from models.invoice import Invoice
from models.payment import Payment
from services.invoicing import invoice_from_quote
bp = Blueprint("invoices", __name__, url_prefix="/invoices")
@bp.route("/")
def list_invoices():
return render_template("invoices/list.html", invoices=Invoice.query.all())
@bp.route("/from-quote/<int:quote_id>", methods=["POST"])
def create_from_quote(quote_id):
q = Quote.query.get_or_404(quote_id)
inv = invoice_from_quote(q)
db.session.add(inv)
db.session.commit()
return redirect(url_for("invoices.view_invoice", invoice_id=inv.id))
@bp.route("/<int:invoice_id>")
def view_invoice(invoice_id):
inv = Invoice.query.get_or_404(invoice_id)
return render_template("invoices/view.html", inv=inv)

67
routes/quotes.py Normal file
View File

@@ -0,0 +1,67 @@
from flask import Blueprint, render_template, request, redirect, url_for
from extensions import db
from models.quote import Quote, QuoteItem
from services.totals import compute_quote_totals
bp = Blueprint("quotes", __name__, url_prefix="/quotes")
@bp.route("/")
def dashboard():
quotes = Quote.query.all()
return render_template("quotes/dashboard.html", quotes=quotes)
@bp.route("/new", methods=["GET","POST"])
@bp.route("/new", methods=["GET","POST"])
def new_quote():
from models.client import Client
if request.method == "POST":
q = Quote(
title=request.form["title"],
category=request.form["category"],
client_id=request.form["client_id"]
)
db.session.add(q)
db.session.commit()
return redirect(url_for("quotes.edit_quote", quote_id=q.id))
clients = Client.query.all()
return render_template("quotes/new.html", clients=clients)
@bp.route("/<int:quote_id>", methods=["GET", "POST"])
def edit_quote(quote_id):
q = Quote.query.get_or_404(quote_id)
if request.method == "POST":
q.title = request.form.get("title", q.title)
item_ids = request.form.getlist("item_id")
descriptions = request.form.getlist("description")
qtys = request.form.getlist("qty")
prices = request.form.getlist("unit_price")
from decimal import Decimal
for idx, item_id in enumerate(item_ids):
item = QuoteItem.query.get(int(item_id))
if not item or item.quote_id != q.id:
continue
item.description = descriptions[idx]
item.qty = Decimal(qtys[idx] or "0")
item.unit_price = Decimal(prices[idx] or "0")
compute_quote_totals(q)
db.session.commit()
return redirect(url_for("quotes.edit_quote", quote_id=q.id))
return render_template("quotes/edit.html", q=q)
@bp.route("/<int:quote_id>/add-item", methods=["POST"])
def add_item(quote_id):
item = QuoteItem(quote_id=quote_id, description="New Item", qty=1, unit_price=0)
db.session.add(item)
db.session.commit()
return redirect(url_for("quotes.edit_quote", quote_id=quote_id))