From 7b7902f706e1b920526b493db23df692bb26064b Mon Sep 17 00:00:00 2001 From: Flatlogic Bot Date: Fri, 19 Dec 2025 12:36:27 +0000 Subject: [PATCH] Simple CRM --- core/__pycache__/admin.cpython-311.pyc | Bin 212 -> 361 bytes core/__pycache__/forms.cpython-311.pyc | Bin 0 -> 855 bytes core/__pycache__/models.cpython-311.pyc | Bin 209 -> 1423 bytes core/__pycache__/urls.cpython-311.pyc | Bin 347 -> 898 bytes core/__pycache__/views.cpython-311.pyc | Bin 1364 -> 1803 bytes core/admin.py | 3 +- core/forms.py | 7 + core/migrations/0001_initial.py | 28 +++ .../__pycache__/0001_initial.cpython-311.pyc | Bin 0 -> 1657 bytes core/models.py | 20 +- core/templates/base.html | 46 +++-- .../core/contact_confirm_delete.html | 13 ++ core/templates/core/contact_form.html | 13 ++ core/templates/core/index.html | 187 +++++------------- core/urls.py | 15 +- core/views.py | 44 +++-- static/css/custom.css | 41 +++- staticfiles/css/custom.css | 58 ++++-- 18 files changed, 266 insertions(+), 209 deletions(-) create mode 100644 core/__pycache__/forms.cpython-311.pyc create mode 100644 core/forms.py create mode 100644 core/migrations/0001_initial.py create mode 100644 core/migrations/__pycache__/0001_initial.cpython-311.pyc create mode 100644 core/templates/core/contact_confirm_delete.html create mode 100644 core/templates/core/contact_form.html diff --git a/core/__pycache__/admin.cpython-311.pyc b/core/__pycache__/admin.cpython-311.pyc index cd6f855b12f4883b1ba9de01c54245c53aacd714..7a88848ec4ed5c94e3274f21c76a8196f31e45c0 100644 GIT binary patch delta 267 zcmcb@_>!r9IWI340}w>Jxn@QI>Bk@r3@}3(pM!vm=?p0hDU3M`xr|Yaj0`DEDa<)c zxy(__j0~xaSu8Lu3s_++WIC0386yM3Y9NLHpnjHM22IwN3Lw5F<1N<2l-$g`mmmc| z9=mgXUP)qdiJvChM1N7)TWq=cDXBTdMIf_pu@q;Pq~78vN=?r!E=et#nBvIA3ls-w rEq0x_Uds3egY*Sd^nii2h3h(l@FfP}3GNpeWUesCT!5h>ZlFd0=$1DS delta 117 zcmaFKbcNA>IWI340}y=Q^((_3NIwQ~V1NnA_$&luOlL@8NMX!j$YqRTWMoKT3TDt` wehE^b$#{!3F(o%M&rg$S;(XDG7ah4cfWnMGTr4|ThEa<90)qk&6|n(D09Qm7z5oCK diff --git a/core/__pycache__/forms.cpython-311.pyc b/core/__pycache__/forms.cpython-311.pyc new file mode 100644 index 0000000000000000000000000000000000000000..ba490e00e9c0b904df1ad1a7f515c8188f34bb76 GIT binary patch literal 855 zcmZ`%F>ll`6t?3e*E^_mMS@m<*pRq^<_}aXz%jK;nIa*}#J)Bqi4QvwbYo!X$bi(H zp$PG3ZbB!oOx>=!PB-yh(g2a#XZy>0_Iuy+d(QXu^#K8SGk%%=g70rMS_!U{3=3NQ1r-egU=WtKVdZz2Qu6zqux21BkF>QHn}~g zjn#Xui&7+I(l|HRGA*BgeM%IGsU)#iXQPnKJM$m|tVUuc8D#L?tt+xIWatpKuFLx! zLd9bL080Bt;gTLuu+{KCitSxhic#nWlhoLf7a~`#pNUotR4&rY1=B-aDA!ALJ{85W z3vDUN%I-nn8n1&mcLUCIEvrm{ew*|6mB^5ebAz3B0UWB#g_Bfe(*7VzENt~OI@Gy} zs<%~9R+0LslB(1u`k>7_I~9qF5^Ypew{v@X?0meg=|d!P+D8DN{*L)@^CXyW?9b|X z65=rTz{P9rtqSk8xAIOdY{v}v=UhtYO$n|(p?_b_<+N3~s z>{=*~L~)=E{y{FRJ7b$R^>?vN_`1ziy4PE+?KC}pW*$N){$Mr*STIWIoNUdyb50)2 Zx^v+(iX~i+UmkyrF6y((8$9YV{s7Fy#~J_t literal 0 HcmV?d00001 diff --git a/core/__pycache__/models.cpython-311.pyc b/core/__pycache__/models.cpython-311.pyc index 9aa598b9d5171b2bb0d207f045e07f9fbbb95a64..e90c69b505033d99ef2460df1113df1bae6ba177 100644 GIT binary patch literal 1423 zcmah}y>Ht_6hD6Giza2!Xq~pO6Q@Z~sDot#Dr^`4>uGL=Z~G(Xo6b&S%c;lWQ~ zuAbr*gffR%BnvN*Q6eg|R!ed<*YUCCm*ujU$X`=TSQgxaQkbkz!g8r4Ukg0n?NY{g zH^z9^^H~9ubHXh`qIB^y)27aW-#&j2OwF>$7GO>?`!;r-q`sxpl(X7y$D&@6BeaDB z+lT!s4t&>i+@~fc1oZq=w`NBM-uMmS?1rAfiY6A=Q*9<^H@-?nzSUTH&>m0&+`{GdXgQ45k{m8ULy;`OjaE0)V> zCB2O6z2`~JG@%Wq`C@U>vh|69cfk7yScry8%Rg13#VWkN-2JVdN>RRg!PBmn^(1eO zvog&@F-`Ct*c4>dG`|n9&GQAV*l2#)eAF;&Uww1Gw%cI52XIbp%=S9eCXB~p?{lNF z+r=GwY)vwh%Soo!b{(4JEVtXk&hsSe`PdJljKs`xc7&!ZtdELl>El<+v3|4Pi1f{Yz8UMA;e%mm z?Nv4|t@UeBX=6~@h)Wyc{h_*gv=^(Z1NBC%-UxSx>I&Ca`r_H`NZlH!Td}$Y=Hm6E zR$RQ^zjfA(iaUeiPF&mx?+vxpqsOtf+Fv^R?$5`O)*NWfSZjuRL;Vg<;Lce!(zgft zcC2qh0uy83kM!z5uf}>cd@z!Q%x5EXm8I45!s}%@DUd_#9JuRbKRt-dL;ib8`|=5I z&a79L`Oni6{epc83;dRO4}py&K@f&$C3^cE$x8ynzsinUKidCr`@ip49EbZ4DbZNx literal 209 zcmZ3^%ge<81S>LrWmp2~#~=<2FhLogg@BCd3@HpLj5!Rsj8Tk?3@J>(44TX@K?*b( zZ?Wa(r=;c-`)M-W;!Md(%uCPLOGzqX21>4E_zY6>OHV%|KQ~psG^sSNq*On(A~m_R zB)>?%JijQrxF9h(RX;huC{-U~j9x+IFAf_ZyEG@&u80Guoe_wOWr4&8W=2NF8w@fR Ku%RM0pb7xTsWln^ diff --git a/core/__pycache__/urls.cpython-311.pyc b/core/__pycache__/urls.cpython-311.pyc index 1f807fae1897c89361c99c5e868a26d00799b73f..ff8e2b9480c4f1a0f529a793a0aa64d1ab34d759 100644 GIT binary patch literal 898 zcmbV~Pixdb6u@V)ll{|eS`{fSh#=_}R-}Ql>p|8~XjK-(i>odiq!0FQoziOiwoDhRzT^wxv;vb#C7b}vqn$NTqtFY{(TY1#sU_3p`I=O;$! zS88OdI8)C4fbtPhL~$Rv*uof5h35K-o3nB*v52c$s++g+7%7NhNodm|XLiUgxLywEW!EWtk~4?yjDuebkFv7 z{Pl?U1D>8tqP~ay1Y8B3=TniMCr__LQm}(=8coWO4&w+J_ku6z3}Wc~zOn1O%!uAb zUKkndfc2u#=f=?I`@z8OF{9^m2t%Ls!H4r=)Aec8XC_|(IGt+n8^kz=F+M@J#l?55 zs8|`J{}b1cS~}c5x*@dHqps9eV{J`pYoA{W(uhezlEw)sLi~=XY>3J|S=opyby=x@ zer!vVx+!(@#~VS~F=av7r-85vTTQkZj? za+#x;85x)uQW;ZNQkhd&*RU;PW?)zi#1N3q7{!vp9?YQ0@e(AU$#{#UAh9IlB_ouR zA)23?dW)e5WD!U*FEKaOPm^^rJCodGTPEKaKA<>AZLt@S_`uA_$asT6<^n2uz#x7B p6@6f1VC8IZ=@9A&nZb5}MgAg-{1q1Y56n#bOby&1Si}P~4gj~)HeCP! diff --git a/core/__pycache__/views.cpython-311.pyc b/core/__pycache__/views.cpython-311.pyc index 6867ddf286a519cf1754ac97038f9f1e4605fbf2..ba79e6d672d50585a0f607a54cb363d470e0c3c3 100644 GIT binary patch literal 1803 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 literal 0 HcmV?d00001 diff --git a/core/models.py b/core/models.py index 71a8362..9c7d86b 100644 --- a/core/models.py +++ b/core/models.py @@ -1,3 +1,21 @@ from django.db import models -# Create your models here. +class Contact(models.Model): + STATUS_CHOICES = [ + ('Lead', 'Lead'), + ('Contacted', 'Contacted'), + ('Customer', 'Customer'), + ('Lost', 'Lost'), + ] + + first_name = models.CharField(max_length=100) + last_name = models.CharField(max_length=100) + email = models.EmailField() + phone = models.CharField(max_length=20, blank=True) + company = models.CharField(max_length=100, blank=True) + status = models.CharField(max_length=20, choices=STATUS_CHOICES, default='Lead') + created_at = models.DateTimeField(auto_now_add=True) + updated_at = models.DateTimeField(auto_now=True) + + def __str__(self): + return f"{self.first_name} {self.last_name}" \ No newline at end of file diff --git a/core/templates/base.html b/core/templates/base.html index 1e7e5fb..ef53b17 100644 --- a/core/templates/base.html +++ b/core/templates/base.html @@ -1,25 +1,33 @@ - - - {% block title %}Knowledge Base{% endblock %} - {% if project_description %} - - - - {% endif %} - {% if project_image_url %} - - - {% endif %} - {% load static %} - - {% block head %}{% endblock %} + + + Contacts & Deals Manager + + + {% load static %} + - - {% block content %}{% endblock %} - + - +
+ {% block content %} + {% endblock %} +
+ + + + \ No newline at end of file diff --git a/core/templates/core/contact_confirm_delete.html b/core/templates/core/contact_confirm_delete.html new file mode 100644 index 0000000..0f1f547 --- /dev/null +++ b/core/templates/core/contact_confirm_delete.html @@ -0,0 +1,13 @@ +{% extends 'base.html' %} + +{% block content %} +
+

Delete Contact

+

Are you sure you want to delete {{ contact.first_name }} {{ contact.last_name }}?

+
+ {% csrf_token %} + + Cancel +
+
+{% endblock %} diff --git a/core/templates/core/contact_form.html b/core/templates/core/contact_form.html new file mode 100644 index 0000000..727b1d5 --- /dev/null +++ b/core/templates/core/contact_form.html @@ -0,0 +1,13 @@ +{% extends 'base.html' %} + +{% block content %} +
+

{% if form.instance.pk %}Edit Contact{% else %}Add Contact{% endif %}

+
+ {% csrf_token %} + {{ form.as_p }} + + Cancel +
+
+{% endblock %} diff --git a/core/templates/core/index.html b/core/templates/core/index.html index faec813..cb5d17a 100644 --- a/core/templates/core/index.html +++ b/core/templates/core/index.html @@ -1,145 +1,48 @@ -{% extends "base.html" %} - -{% block title %}{{ project_name }}{% endblock %} - -{% block head %} - - - - -{% endblock %} +{% extends 'base.html' %} {% block content %} -
-
-

Analyzing your requirements and generating your app…

-
- Loading… +
+

Contacts & Deals Manager

+
+

A lightweight back office to track contacts and deals. Add and edit records, filter and sort lists, and see simple status counters.

+
-

AppWizzy AI is collecting your requirements and applying the first changes.

-

This page will refresh automatically as the plan is implemented.

-

- Runtime: Django {{ django_version }} · Python {{ python_version }} - — UTC {{ current_time|date:"Y-m-d H:i:s" }} -

-
-
-
- Page updated: {{ current_time|date:"Y-m-d H:i:s" }} (UTC) -
-{% endblock %} \ No newline at end of file + + +
+

Contacts

+ + + + + + + + + + + + + {% for contact in contacts %} + + + + + + + + + {% empty %} + + + + {% endfor %} + +
NameEmailPhoneCompanyStatusActions
{{ contact.first_name }} {{ contact.last_name }}{{ contact.email }}{{ contact.phone }}{{ contact.company }}{{ contact.status }} + Edit + Delete +
No contacts found.
+
+{% endblock %} diff --git a/core/urls.py b/core/urls.py index 6299e3d..a9f1e92 100644 --- a/core/urls.py +++ b/core/urls.py @@ -1,7 +1,14 @@ from django.urls import path - -from .views import home +from .views import ( + ContactListView, + ContactCreateView, + ContactUpdateView, + ContactDeleteView, +) urlpatterns = [ - path("", home, name="home"), -] + path('', ContactListView.as_view(), name='index'), + path('contact/new/', ContactCreateView.as_view(), name='contact_new'), + path('contact//edit/', ContactUpdateView.as_view(), name='contact_edit'), + path('contact//delete/', ContactDeleteView.as_view(), name='contact_delete'), +] \ No newline at end of file diff --git a/core/views.py b/core/views.py index c9aed12..675273e 100644 --- a/core/views.py +++ b/core/views.py @@ -1,25 +1,27 @@ -import os -import platform - -from django import get_version as django_version from django.shortcuts import render -from django.utils import timezone +from django.urls import reverse_lazy +from django.views.generic import ListView, CreateView, UpdateView, DeleteView +from .models import Contact +from .forms import ContactForm +class ContactListView(ListView): + model = Contact + template_name = 'core/index.html' + context_object_name = 'contacts' -def home(request): - """Render the landing screen with loader and environment details.""" - host_name = request.get_host().lower() - agent_brand = "AppWizzy" if host_name == "appwizzy.com" else "Flatlogic" - now = timezone.now() +class ContactCreateView(CreateView): + model = Contact + form_class = ContactForm + template_name = 'core/contact_form.html' + success_url = reverse_lazy('index') - context = { - "project_name": "New Style", - "agent_brand": agent_brand, - "django_version": django_version(), - "python_version": platform.python_version(), - "current_time": now, - "host_name": host_name, - "project_description": os.getenv("PROJECT_DESCRIPTION", ""), - "project_image_url": os.getenv("PROJECT_IMAGE_URL", ""), - } - return render(request, "core/index.html", context) +class ContactUpdateView(UpdateView): + model = Contact + form_class = ContactForm + template_name = 'core/contact_form.html' + success_url = reverse_lazy('index') + +class ContactDeleteView(DeleteView): + model = Contact + template_name = 'core/contact_confirm_delete.html' + success_url = reverse_lazy('index') \ No newline at end of file diff --git a/static/css/custom.css b/static/css/custom.css index 925f6ed..d3facf2 100644 --- a/static/css/custom.css +++ b/static/css/custom.css @@ -1,4 +1,41 @@ -/* Custom styles for the application */ body { - font-family: system-ui, -apple-system, sans-serif; + font-family: 'Lato', sans-serif; + background-color: #F8F9FA; } + +h1, h2, h3, h4, h5, h6, .navbar-brand, .display-5 { + font-family: 'Poppins', sans-serif; +} + +.btn-primary { + background-color: #4A90E2; + border-color: #4A90E2; +} + +.btn-primary:hover { + background-color: #357ABD; + border-color: #357ABD; +} + +.btn-outline-primary { + color: #4A90E2; + border-color: #4A90E2; +} + +.btn-outline-primary:hover { + background-color: #4A90E2; + color: #fff; +} + +.badge.bg-secondary { + background-color: #F5A623 !important; +} + +.table-hover tbody tr:hover { + background-color: #e9ecef; +} + +.navbar-light .navbar-brand { + color: #333; + font-weight: 600; +} \ No newline at end of file diff --git a/staticfiles/css/custom.css b/staticfiles/css/custom.css index 108056f..d3facf2 100644 --- a/staticfiles/css/custom.css +++ b/staticfiles/css/custom.css @@ -1,21 +1,41 @@ - -:root { - --bg-color-start: #6a11cb; - --bg-color-end: #2575fc; - --text-color: #ffffff; - --card-bg-color: rgba(255, 255, 255, 0.01); - --card-border-color: rgba(255, 255, 255, 0.1); -} body { - margin: 0; - font-family: 'Inter', sans-serif; - background: linear-gradient(45deg, var(--bg-color-start), var(--bg-color-end)); - color: var(--text-color); - display: flex; - justify-content: center; - align-items: center; - min-height: 100vh; - text-align: center; - overflow: hidden; - position: relative; + font-family: 'Lato', sans-serif; + background-color: #F8F9FA; } + +h1, h2, h3, h4, h5, h6, .navbar-brand, .display-5 { + font-family: 'Poppins', sans-serif; +} + +.btn-primary { + background-color: #4A90E2; + border-color: #4A90E2; +} + +.btn-primary:hover { + background-color: #357ABD; + border-color: #357ABD; +} + +.btn-outline-primary { + color: #4A90E2; + border-color: #4A90E2; +} + +.btn-outline-primary:hover { + background-color: #4A90E2; + color: #fff; +} + +.badge.bg-secondary { + background-color: #F5A623 !important; +} + +.table-hover tbody tr:hover { + background-color: #e9ecef; +} + +.navbar-light .navbar-brand { + color: #333; + font-weight: 600; +} \ No newline at end of file