Compare commits
57 commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
6b9e3a6e96 | ||
![]() |
19c4a0b6ed | ||
![]() |
9589bcf48d | ||
![]() |
d354708ffe | ||
![]() |
2ec833fe42 | ||
![]() |
039ee1ecb8 | ||
![]() |
460fe14306 | ||
![]() |
f67b5f1de0 | ||
![]() |
8d2a3cf89f | ||
![]() |
e6be787798 | ||
![]() |
34fc3d6426 | ||
![]() |
b3cc0085ff | ||
![]() |
b4311f1401 | ||
![]() |
272df57664 | ||
![]() |
18249653f9 | ||
![]() |
10aed1560b | ||
![]() |
686a31b96a | ||
3d38bb56c0 | |||
![]() |
99db86153d | ||
![]() |
8741d6d9b6 | ||
![]() |
c0c2d40286 | ||
28290f20de | |||
![]() |
44a2c20f95 | ||
![]() |
84dadd9147 | ||
![]() |
34d2db8623 | ||
![]() |
dacafb8c70 | ||
![]() |
e77c4d46d8 | ||
![]() |
f378b1516a | ||
![]() |
66aeea6ad0 | ||
![]() |
5b95cfe843 | ||
![]() |
2a3b7d467c | ||
![]() |
6b7bd335d0 | ||
![]() |
9b6cc7604c | ||
![]() |
193114e1e9 | ||
![]() |
20823580a9 | ||
![]() |
b57ebafca8 | ||
![]() |
f242870d50 | ||
![]() |
8b79492508 | ||
![]() |
39dcaf637d | ||
![]() |
d20cafe5db | ||
![]() |
4181588a3a | ||
![]() |
bcfb62e740 | ||
4fe09ea223 | |||
![]() |
12f91b55bd | ||
![]() |
d2053c7da9 | ||
a1c179f612 | |||
d67bbc7d8c | |||
91d2918586 | |||
51a6966390 | |||
![]() |
b9e66a5f2b | ||
![]() |
6b3b361e07 | ||
![]() |
b03d37ee7c | ||
![]() |
5b473f0cb3 | ||
5e5a462e45 | |||
447fd88026 | |||
![]() |
59e07f370b | ||
e5de39e568 |
4
.gitignore
vendored
|
@ -3,10 +3,8 @@ coverage.xml
|
||||||
.coverage
|
.coverage
|
||||||
la_chariotte.egg-info/
|
la_chariotte.egg-info/
|
||||||
node_modules
|
node_modules
|
||||||
/static/*
|
|
||||||
/media/*
|
|
||||||
local_settings.py
|
local_settings.py
|
||||||
.venv
|
.venv
|
||||||
|
*venv*
|
||||||
mails.sqlite
|
mails.sqlite
|
||||||
package-lock.json
|
package-lock.json
|
||||||
static
|
|
||||||
|
|
|
@ -26,8 +26,7 @@ tests:
|
||||||
- pip cache purge
|
- pip cache purge
|
||||||
- pip install -e ".[dev]"
|
- pip install -e ".[dev]"
|
||||||
- pytest --create-db --cov --cov-report=xml
|
- pytest --create-db --cov --cov-report=xml
|
||||||
- if [ "$CI_COMMIT_REF_NAME" = 'main' ] ; then exit 0 ; fi
|
- if [ "$CI_COMMIT_REF_NAME" = 'develop' ] ; then git fetch origin develop ; diff-cover coverage.xml --fail-under=90 --compare-branch origin/develop --diff-range-notation '..' && exit 0 ; fi
|
||||||
- if [ "$CI_COMMIT_REF_NAME" = 'develop' ] ; then git fetch origin main ; diff-cover coverage.xml --fail-under=90 --compare-branch origin/main --diff-range-notation '..' && exit 0 ; fi
|
|
||||||
- git fetch origin develop ; diff-cover coverage.xml --fail-under=90 --compare-branch origin/develop --diff-range-notation '..'
|
- git fetch origin develop ; diff-cover coverage.xml --fail-under=90 --compare-branch origin/develop --diff-range-notation '..'
|
||||||
- echo "Tests done."
|
- echo "Tests done."
|
||||||
|
|
||||||
|
|
97
.gitlab/issue_templates/default.md
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
# title
|
||||||
|
|
||||||
|
Short and specific: **a clear summary of the bug**.
|
||||||
|
|
||||||
|
It should also be **searchable**, so developers can find it in a pinch.
|
||||||
|
|
||||||
|
> Tips on writing great titles:
|
||||||
|
>
|
||||||
|
> 1. Keep it simple, but descriptive.
|
||||||
|
> 2. Don’t abbreviate.
|
||||||
|
> 3. Make it searchable.
|
||||||
|
> 4. Focus on the technical problem.
|
||||||
|
> 5. Highlight the specific feature/component you have issues with.
|
||||||
|
|
||||||
|
## description
|
||||||
|
|
||||||
|
Small text to describe the bug with human words.
|
||||||
|
|
||||||
|
## steps to reproduce
|
||||||
|
|
||||||
|
**Describe how you found the bug**, so the developer can try to reproduce it.
|
||||||
|
|
||||||
|
> Tips on writing reproducible steps:
|
||||||
|
>
|
||||||
|
> 1. Use a numbered list so it’s easy to follow.
|
||||||
|
> 2. This is your chance to be comprehensive and (reasonably) verbose. Don’t leave out any details!
|
||||||
|
|
||||||
|
## expected vs. actual results
|
||||||
|
|
||||||
|
Take some time to explain what **should** happen vs. what **actually** happened.
|
||||||
|
|
||||||
|
If you just describe the bug, some people might think you’re describing the expected behavior.
|
||||||
|
|
||||||
|
### expected results
|
||||||
|
|
||||||
|
### actual results
|
||||||
|
|
||||||
|
> Tips for writing expected vs. actual results:
|
||||||
|
>
|
||||||
|
> 1. Use a direct comparison format. For example, "*The button should turn green*" vs. "*The button is turning blue*".
|
||||||
|
> 2. Be precise. Simply stating "*it went wrong*" instead of *"the page loads indefinitely*" means you are leaving out very valuable information!
|
||||||
|
|
||||||
|
## visual proof/screenshot
|
||||||
|
|
||||||
|
Screenshots and annotations help developers **visualize the bug**, and pinpoint its location on the page.
|
||||||
|
|
||||||
|
> Tips on taking great screenshots:
|
||||||
|
>
|
||||||
|
> 1. Annotations go a long way towards driving your point across.
|
||||||
|
> 2. Highlight the problematic element. Don’t be ambiguous.
|
||||||
|
> 3. Use big fonts, different colors, etc. - the bug needs to be even more obvious here than in your summary.
|
||||||
|
> 4. For complicated issues, record a short video! This adds a ton of helpful context when trying to reproduce bugs.
|
||||||
|
|
||||||
|
## priority
|
||||||
|
|
||||||
|
The urgency and **potential impact of the bug**. Determines how quickly it needs to be fixed.
|
||||||
|
|
||||||
|
> How to determine priority/severity:
|
||||||
|
>
|
||||||
|
> 1. Critical: blocking bugs that **directly prevent business**. Example: a checkout page not loading.
|
||||||
|
> 2. High: affects **major features**, but non-breaking. E.g.: the search bar on an e-commerce website.
|
||||||
|
> 3. Medium: noticeable bugs that **disrupt normal use**. E.g.: broken link, long loading times.
|
||||||
|
> 4. Low: **small issues** and enhancements: typos, missing images...
|
||||||
|
|
||||||
|
## environment
|
||||||
|
|
||||||
|
For developers to reproduce and fix your bug, they’ll need to know your **browser version**, **screen size**, **operating system**...
|
||||||
|
|
||||||
|
Some bugs only occur within specific environments.
|
||||||
|
|
||||||
|
- source url:
|
||||||
|
- operating system:
|
||||||
|
- browser:
|
||||||
|
- viewport:
|
||||||
|
|
||||||
|
> How to find your environment info:
|
||||||
|
>
|
||||||
|
> 1. **Browser and version**: look for a “Help” or “About” option in your browser’s menu.
|
||||||
|
> 2. **Operating system**: on PC, press the Windows key + Pause/break. On a Mac, click the Apple logo and choose “About this Mac”.
|
||||||
|
> 3. **Screen size**: look in your computer’s display settings, or search online for “screen resolution” along with your device model.
|
||||||
|
|
||||||
|
## console logs
|
||||||
|
|
||||||
|
This is where your web browser **shows errors or warnings**.
|
||||||
|
|
||||||
|
Console logs can help developers figure out what went wrong.
|
||||||
|
|
||||||
|
```log
|
||||||
|
console log
|
||||||
|
```
|
||||||
|
|
||||||
|
> How to access your console logs:
|
||||||
|
>
|
||||||
|
> 1. Right-click the page, select “Inspect” or “Inspect Element”, then click on the “Console” tab.
|
||||||
|
> 2. Try to make the bug happen again and see if any messages pop up there.
|
||||||
|
|
||||||
|
## credits https://www.perfectbugreport.io/
|
43
CHANGELOG.md
|
@ -1,4 +1,43 @@
|
||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
- v1 :
|
All notable changes to this project will be documented in this file.
|
||||||
- Basic functionalities for grouped orders
|
|
||||||
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/)
|
||||||
|
since 0.4.1, and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## 1.3.0 (2024-11-02)
|
||||||
|
|
||||||
|
This is a small release, with a few features and bugfixes. The code is now
|
||||||
|
hosted on https://framagit.org rather than https://gitlab.com.
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Changed hour and date format in the generated CSV files
|
||||||
|
- Added the date in the grouped order dashboard
|
||||||
|
- Changed the display of the last possible date and hour to place an order
|
||||||
|
- Added the total number of ordered items in the grouped orders overview
|
||||||
|
- Removed header line and column for grouped order name in the CSV export for emails
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- The "reset password" form is now fixed, it was broken since the last release.
|
||||||
|
|
||||||
|
## 1.2.0 (2024-07-12)
|
||||||
|
|
||||||
|
A small release, with a few features, a new footer and a `/stats` endpoint.
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Staff users are now able to see the overview page for a grouped order.
|
||||||
|
- Personal information are now autocompleted automatically
|
||||||
|
- A `/stats` endpoint has been added, making it possible to see some information
|
||||||
|
about the current instance.
|
||||||
|
- The footer has been changed to include more elements, links to the project and
|
||||||
|
documentation.
|
||||||
|
|
||||||
|
### Development
|
||||||
|
|
||||||
|
- The `main` branch is not used anymore, all the development happens on the
|
||||||
|
`develop` branch instead.
|
||||||
|
- All the tests have been grouped in the `tests` folder
|
||||||
|
- The `requirements.txt` file have been removed in favor of the `pyproject.toml` file.
|
||||||
|
|
|
@ -6,7 +6,7 @@ started.
|
||||||
The first step is to clone the project, you can find more information about that in
|
The first step is to clone the project, you can find more information about that in
|
||||||
the [getting started guide](../install.md). Once that's done, you can:
|
the [getting started guide](../install.md). Once that's done, you can:
|
||||||
|
|
||||||
- choose a task [on the board](https://gitlab.com/la-chariotte/la_chariotte/-/boards) and assign it to
|
- choose a task [on the board](https://framagit.org/la-chariotte/la-chariotte/-/boards) and assign it to
|
||||||
yourself - if you don't know which task to do, feel free to reach to
|
yourself - if you don't know which task to do, feel free to reach to
|
||||||
us.
|
us.
|
||||||
- create a new branch **from develop** naming it to reflect what you want to do
|
- create a new branch **from develop** naming it to reflect what you want to do
|
||||||
|
@ -55,16 +55,6 @@ isort .
|
||||||
black .
|
black .
|
||||||
```
|
```
|
||||||
|
|
||||||
## Working with emails
|
|
||||||
|
|
||||||
To test the appearance of emails, you can use [Sendria](https://github.com/msztolcman/sendria):
|
|
||||||
|
|
||||||
```bash
|
|
||||||
pip install sendria
|
|
||||||
sendria --db mails.sqlite
|
|
||||||
$NAVIGATOR http://127.0.0.1:1080
|
|
||||||
```
|
|
||||||
|
|
||||||
## Docs
|
## Docs
|
||||||
|
|
||||||
We're using [MkDocs](https://www.mkdocs.org/) for this documentation. If you want to
|
We're using [MkDocs](https://www.mkdocs.org/) for this documentation. If you want to
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
|
|
||||||
- **docs.chariotte.fr**, the docs you are reading now. It's handled by [readthedocs.org](https://readthedocs.org).
|
- **docs.chariotte.fr**, the docs you are reading now. It's handled by [readthedocs.org](https://readthedocs.org).
|
||||||
- **chariotte.fr**, the main instance. It's deployed on Alwaysdata
|
- **chariotte.fr**, the main instance. It's deployed on Alwaysdata
|
||||||
- **blog.chariotte.fr**, our blog. It's [a static website](https://gitlab.com/la-chariotte/la-chariotte.gitlab.io) deployed on Gitlab pages.
|
- **blog.chariotte.fr**, our blog. It's [a static website](https://framagit.org/la-chariotte/la-chariotte.frama.io) deployed on Gitlab pages.
|
||||||
|
|
||||||
## The main instance
|
## The main instance
|
||||||
|
|
||||||
|
@ -144,9 +144,12 @@ cd la_chariotte
|
||||||
# Get the code
|
# Get the code
|
||||||
git fetch
|
git fetch
|
||||||
git checkout tag # if we're using a tag, otherwise, just checkout the main branch
|
git checkout tag # if we're using a tag, otherwise, just checkout the main branch
|
||||||
|
pip install -e .
|
||||||
|
npm install
|
||||||
|
|
||||||
python manage.py updatedb
|
python manage.py migrate --settings=prod_settings
|
||||||
python manage.py collectstatic
|
python manage.py compilescss --settings=prod_settings
|
||||||
|
python manage.py collectstatic --settings=prod_settings
|
||||||
```
|
```
|
||||||
|
|
||||||
Then you'll need to restart the server from AD's interface.
|
Then you'll need to restart the server from AD's interface.
|
||||||
|
|
|
@ -22,22 +22,21 @@ classDiagram
|
||||||
name
|
name
|
||||||
deadline : DateTime
|
deadline : DateTime
|
||||||
delivery_date : Date
|
delivery_date : Date
|
||||||
|
delivery_slot
|
||||||
place
|
place
|
||||||
description
|
description
|
||||||
orga : CustomUser
|
orga : CustomUser
|
||||||
total_price
|
|
||||||
}
|
}
|
||||||
class Item{
|
class Item{
|
||||||
name
|
name
|
||||||
grouped_order : GroupedOrder
|
grouped_order : GroupedOrder
|
||||||
ordered_nb
|
ordered_nb
|
||||||
total_price
|
price
|
||||||
max_limit
|
max_limit
|
||||||
}
|
}
|
||||||
class Order{
|
class Order{
|
||||||
grouped_order : GroupedOrder
|
grouped_order : GroupedOrder
|
||||||
author : OrderAuthor
|
author : OrderAuthor
|
||||||
price
|
|
||||||
created_date
|
created_date
|
||||||
note
|
note
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
First, clone the project
|
First, clone the project
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone git@gitlab.com:la-chariotte/la_chariotte.git
|
git clone https://framagit.org/la-chariotte/la-chariotte.git
|
||||||
```
|
```
|
||||||
|
|
||||||
## Virtual environment
|
## Virtual environment
|
||||||
|
@ -73,7 +73,7 @@ CREATE DATABASE chariotte
|
||||||
|
|
||||||
## Create a configuration file
|
## Create a configuration file
|
||||||
|
|
||||||
Create a local configuration file:
|
Create a local configuration file named "local_settings.py" at the project root :
|
||||||
|
|
||||||
```python title="local_settings.py"
|
```python title="local_settings.py"
|
||||||
from la_chariotte.settings import *
|
from la_chariotte.settings import *
|
||||||
|
@ -102,3 +102,13 @@ To create a superuser, who will have access to the admin interface (/admin):
|
||||||
```shell
|
```shell
|
||||||
python manage.py createsuperuser
|
python manage.py createsuperuser
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Working with emails
|
||||||
|
|
||||||
|
To test the appearance of emails, you can use [Sendria](https://github.com/msztolcman/sendria):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pip install sendria
|
||||||
|
sendria --db mails.sqlite
|
||||||
|
$NAVIGATOR http://127.0.0.1:1080
|
||||||
|
```
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
__version__ = "1.1.0"
|
__version__ = "1.3.0"
|
||||||
|
|
|
@ -8,6 +8,6 @@
|
||||||
</p>
|
</p>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<p>Votre mot de passe a été correctement réinitialisé. Vous pouvez maintenant vous connecter.</p>
|
<p>Votre mot de passe a été correctement réinitialisé. Vous pouvez maintenant vous connecter.</p>
|
||||||
<button class="button is-primary" href="{% url 'accounts:login'%}">Se connecter</button>
|
<a class="button is-primary" href="{% url 'accounts:login' %}">Se connecter</a>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
|
@ -9,8 +9,8 @@
|
||||||
{% if validlink %}
|
{% if validlink %}
|
||||||
<form method="POST">
|
<form method="POST">
|
||||||
<p>Veuillez entrer un nouveau mot de passe tout neuf.</p>
|
<p>Veuillez entrer un nouveau mot de passe tout neuf.</p>
|
||||||
|
{{ form | crispy }}
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
{% crispy form %}
|
|
||||||
<input class="button is-primary"
|
<input class="button is-primary"
|
||||||
type="submit"
|
type="submit"
|
||||||
value="Changer le mot de passe">
|
value="Changer le mot de passe">
|
||||||
|
|
|
@ -8,7 +8,8 @@
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<form method="POST">
|
<form method="POST">
|
||||||
<p>Entrez votre mail pour recevoir les instructions pour le réinitialiser.</p>
|
<p>Entrez votre mail pour recevoir les instructions pour le réinitialiser.</p>
|
||||||
{% crispy form %}
|
{{ form | crispy }}
|
||||||
|
{% csrf_token %}
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<input class="button is-primary" type="submit" value="Envoyer">
|
<input class="button is-primary" type="submit" value="Envoyer">
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -4,6 +4,7 @@ import pytest
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from la_chariotte.order.models import GroupedOrder, Item
|
from la_chariotte.order.models import GroupedOrder, Item
|
||||||
|
from la_chariotte.tests.utils import create_grouped_order
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
|
@ -23,6 +24,18 @@ def other_user(django_user_model):
|
||||||
return user
|
return user
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def authenticated_user_with_name(django_user_model):
|
||||||
|
username = "toto@gmail.com"
|
||||||
|
password = "tata"
|
||||||
|
first_name = "boule"
|
||||||
|
last_name = "bill"
|
||||||
|
user = django_user_model.objects.create_user(
|
||||||
|
username=username, password=password, first_name=first_name, last_name=last_name
|
||||||
|
)
|
||||||
|
return user
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def simple_grouped_order(other_user):
|
def simple_grouped_order(other_user):
|
||||||
date = timezone.now().date() + datetime.timedelta(days=30)
|
date = timezone.now().date() + datetime.timedelta(days=30)
|
||||||
|
@ -39,6 +52,19 @@ def simple_grouped_order(other_user):
|
||||||
return grouped_order
|
return grouped_order
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def connected_grouped_order(client, authenticated_user_with_name):
|
||||||
|
client.force_login(authenticated_user_with_name)
|
||||||
|
grouped_order = create_grouped_order(
|
||||||
|
name="Test grouped order",
|
||||||
|
orga_user=authenticated_user_with_name,
|
||||||
|
days_before_delivery_date=30,
|
||||||
|
days_before_deadline=5,
|
||||||
|
)
|
||||||
|
item = Item.objects.create(name="test item", grouped_order=grouped_order, price=2)
|
||||||
|
return grouped_order
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(autouse=True)
|
@pytest.fixture(autouse=True)
|
||||||
def password_hasher_setup(settings):
|
def password_hasher_setup(settings):
|
||||||
# Use a weaker password hasher during tests, for speed
|
# Use a weaker password hasher during tests, for speed
|
||||||
|
|
|
@ -18,6 +18,10 @@ class GroupedOrderForm(forms.ModelForm):
|
||||||
widget=forms.TimeInput(attrs={"type": "time"}),
|
widget=forms.TimeInput(attrs={"type": "time"}),
|
||||||
initial=datetime.time(hour=23, minute=59, second=59),
|
initial=datetime.time(hour=23, minute=59, second=59),
|
||||||
)
|
)
|
||||||
|
is_phone_mandatory = forms.BooleanField(
|
||||||
|
label="Numéro de téléphone obligatoire pour les participants",
|
||||||
|
required=False,
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = GroupedOrder
|
model = GroupedOrder
|
||||||
|
@ -29,6 +33,7 @@ class GroupedOrderForm(forms.ModelForm):
|
||||||
"delivery_slot",
|
"delivery_slot",
|
||||||
"place",
|
"place",
|
||||||
"description",
|
"description",
|
||||||
|
"is_phone_mandatory",
|
||||||
]
|
]
|
||||||
widgets = {
|
widgets = {
|
||||||
"name": forms.TextInput(
|
"name": forms.TextInput(
|
||||||
|
|
|
@ -0,0 +1,36 @@
|
||||||
|
# Generated by Django 4.2.16 on 2024-10-31 21:27
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("order", "0026_groupedorder_delivery_slot"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="groupedorder",
|
||||||
|
name="total_price",
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="item",
|
||||||
|
name="ordered_nb",
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="order",
|
||||||
|
name="articles_nb",
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="order",
|
||||||
|
name="price",
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="order",
|
||||||
|
name="created_date",
|
||||||
|
field=models.DateTimeField(
|
||||||
|
auto_now_add=True, verbose_name="Date et heure de commande"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,20 @@
|
||||||
|
# Generated by Django 4.2.16 on 2024-12-08 15:15
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("order", "0027_remove_groupedorder_total_price_and_more"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="groupedorder",
|
||||||
|
name="is_phone_mandatory",
|
||||||
|
field=models.BooleanField(
|
||||||
|
default=False, verbose_name="Numéro de téléphone obligatoire"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,20 @@
|
||||||
|
# Generated by Django 4.2.16 on 2024-12-08 15:41
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
def set_phone_mandatory(apps, schema_editor):
|
||||||
|
"""For continuity, force mandatory phone for the orders that were created so far."""
|
||||||
|
GroupedOrder = apps.get_model("order", "GroupedOrder")
|
||||||
|
for grouped_order in GroupedOrder.objects.all():
|
||||||
|
grouped_order.is_phone_mandatory = True
|
||||||
|
grouped_order.save()
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("order", "0028_groupedorder_is_phone_mandatory"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RunPython(set_phone_mandatory),
|
||||||
|
]
|
|
@ -25,8 +25,10 @@ class GroupedOrder(models.Model):
|
||||||
max_length=100, null=True, blank=True, verbose_name="Lieu de livraison"
|
max_length=100, null=True, blank=True, verbose_name="Lieu de livraison"
|
||||||
)
|
)
|
||||||
description = models.TextField("Description", null=True, blank=True)
|
description = models.TextField("Description", null=True, blank=True)
|
||||||
total_price = models.DecimalField(max_digits=10, decimal_places=2, default=0)
|
|
||||||
code = models.CharField(auto_created=True)
|
code = models.CharField(auto_created=True)
|
||||||
|
is_phone_mandatory = models.BooleanField(
|
||||||
|
default=False, verbose_name="Numéro de téléphone obligatoire"
|
||||||
|
)
|
||||||
|
|
||||||
def create_code_from_pk(self):
|
def create_code_from_pk(self):
|
||||||
"""When a grouped order is created, a unique code is generated, to be used to
|
"""When a grouped order is created, a unique code is generated, to be used to
|
||||||
|
@ -52,16 +54,18 @@ class GroupedOrder(models.Model):
|
||||||
)
|
)
|
||||||
self.code = f"{base_36_pk}{random_string}"[:code_length]
|
self.code = f"{base_36_pk}{random_string}"[:code_length]
|
||||||
|
|
||||||
def compute_total_price(self):
|
@property
|
||||||
|
def total_price(self):
|
||||||
price = 0
|
price = 0
|
||||||
for order in self.order_set.all():
|
for order in self.order_set.all():
|
||||||
price += order.price
|
price += order.price
|
||||||
self.total_price = price
|
return price
|
||||||
self.save()
|
|
||||||
|
|
||||||
def compute_items_ordered_nb(self):
|
def get_total_ordered_items(self):
|
||||||
|
total_nb = 0
|
||||||
for item in self.item_set.all():
|
for item in self.item_set.all():
|
||||||
item.compute_ordered_nb()
|
total_nb += item.ordered_nb
|
||||||
|
return total_nb
|
||||||
|
|
||||||
def is_open(self):
|
def is_open(self):
|
||||||
return self.deadline >= timezone.now()
|
return self.deadline >= timezone.now()
|
||||||
|
@ -97,8 +101,6 @@ class GroupedOrder(models.Model):
|
||||||
class OrderAuthor(models.Model):
|
class OrderAuthor(models.Model):
|
||||||
"""Used when a user orders (with or without an account)"""
|
"""Used when a user orders (with or without an account)"""
|
||||||
|
|
||||||
# TODO faire le lien avec CustomUser (CustomUser hérite de OrderAuthor),
|
|
||||||
# pour ensuite préremplir quand on est connecté·e
|
|
||||||
first_name = models.CharField(verbose_name="Prénom")
|
first_name = models.CharField(verbose_name="Prénom")
|
||||||
last_name = models.CharField(verbose_name="Nom")
|
last_name = models.CharField(verbose_name="Nom")
|
||||||
phone = models.CharField(
|
phone = models.CharField(
|
||||||
|
@ -119,26 +121,24 @@ class Order(models.Model):
|
||||||
GroupedOrder, on_delete=models.CASCADE, related_name="order_set"
|
GroupedOrder, on_delete=models.CASCADE, related_name="order_set"
|
||||||
)
|
)
|
||||||
author = models.ForeignKey(OrderAuthor, on_delete=models.CASCADE)
|
author = models.ForeignKey(OrderAuthor, on_delete=models.CASCADE)
|
||||||
articles_nb = models.PositiveIntegerField(default=0)
|
created_date = models.DateTimeField("Date et heure de commande", auto_now_add=True)
|
||||||
price = models.DecimalField(max_digits=10, decimal_places=2, default=0)
|
|
||||||
created_date = models.DateTimeField("Date de la commande", auto_now_add=True)
|
|
||||||
note = models.TextField(max_length=200, null=True, blank=True)
|
note = models.TextField(max_length=200, null=True, blank=True)
|
||||||
|
|
||||||
def compute_order_articles_nb(self):
|
@property
|
||||||
|
def articles_nb(self):
|
||||||
"""Computes the number of articles in this order"""
|
"""Computes the number of articles in this order"""
|
||||||
articles_nb = 0
|
articles_nb = 0
|
||||||
for ord_item in self.ordered_items.all():
|
for ord_item in self.ordered_items.all():
|
||||||
articles_nb += ord_item.nb
|
articles_nb += ord_item.nb
|
||||||
self.articles_nb = articles_nb
|
return articles_nb
|
||||||
self.save()
|
|
||||||
|
|
||||||
def compute_order_price(self):
|
@property
|
||||||
|
def price(self):
|
||||||
"""Computes the total price of the order"""
|
"""Computes the total price of the order"""
|
||||||
price = 0
|
price = 0
|
||||||
for ord_item in self.ordered_items.all():
|
for ord_item in self.ordered_items.all():
|
||||||
price += ord_item.get_price()
|
price += ord_item.get_price()
|
||||||
self.price = price
|
return price
|
||||||
self.save()
|
|
||||||
|
|
||||||
def __str__(self): # pragma: no cover
|
def __str__(self): # pragma: no cover
|
||||||
return (
|
return (
|
||||||
|
@ -153,15 +153,13 @@ class Item(models.Model):
|
||||||
price = models.DecimalField(max_digits=10, decimal_places=2)
|
price = models.DecimalField(max_digits=10, decimal_places=2)
|
||||||
max_limit = models.PositiveSmallIntegerField(null=True, blank=True)
|
max_limit = models.PositiveSmallIntegerField(null=True, blank=True)
|
||||||
|
|
||||||
ordered_nb = models.IntegerField(default=0)
|
@property
|
||||||
|
def ordered_nb(self):
|
||||||
def compute_ordered_nb(self):
|
|
||||||
"""Computes the number of times this item has been ordered"""
|
"""Computes the number of times this item has been ordered"""
|
||||||
ordered_nb = 0
|
ordered_nb = 0
|
||||||
for order in self.orders.all():
|
for order in self.orders.all():
|
||||||
ordered_nb += order.nb
|
ordered_nb += order.nb
|
||||||
self.ordered_nb = ordered_nb
|
return ordered_nb
|
||||||
self.save()
|
|
||||||
|
|
||||||
def get_total_price(self):
|
def get_total_price(self):
|
||||||
"""Returns the total price of all orders on this item"""
|
"""Returns the total price of all orders on this item"""
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% load crispy_forms_tags %}
|
||||||
|
|
||||||
{% block title %}Nouvelle commande groupée{% endblock %}
|
{% block title %}Nouvelle commande groupée{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
@ -10,8 +12,9 @@
|
||||||
<p class="title">Nouvelle commande groupée</p>
|
<p class="title">Nouvelle commande groupée</p>
|
||||||
<div class="columns">
|
<div class="columns">
|
||||||
<div class="column is-8">
|
<div class="column is-8">
|
||||||
<form method="post" onsubmit="deadlinePassedCheck(event)">{% csrf_token %}
|
<form method="post" onsubmit="deadlinePassedCheck(event)">
|
||||||
{{ form.as_p }}
|
{% csrf_token %}
|
||||||
|
{{ form | crispy }}
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<a class="button is-light" href="{% url 'order:index' %}">Annuler</a>
|
<a class="button is-light" href="{% url 'order:index' %}">Annuler</a>
|
||||||
<input class="button is-primary" type="submit" value="Suivant">
|
<input class="button is-primary" type="submit" value="Suivant">
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<p><i class="fa fa-calendar-check-o mr-3" aria-label="Date limite de commande"
|
<p><i class="fa fa-calendar-check-o mr-3" aria-label="Date limite de commande"
|
||||||
title="Date limite de commande" aria-hidden="true"></i>
|
title="Date limite de commande" aria-hidden="true"></i>
|
||||||
Commandes avant le {{ grouped_order.deadline }}
|
Commandes avant le {{ grouped_order.deadline|date:'d M Y' }} à {{ grouped_order.deadline|date:'H:i' }}
|
||||||
</p>
|
</p>
|
||||||
<p><i class="fa fa-truck mr-3" aria-label="Date de livraison" title="Date de livraison"
|
<p><i class="fa fa-truck mr-3" aria-label="Date de livraison" title="Date de livraison"
|
||||||
aria-hidden="true"></i>
|
aria-hidden="true"></i>
|
||||||
|
@ -77,8 +77,8 @@
|
||||||
<p>Il n'y a pas de produits disponibles dans cette commande !</p>
|
<p>Il n'y a pas de produits disponibles dans cette commande !</p>
|
||||||
{% else %}
|
{% else %}
|
||||||
<p class="title">Commander</p>
|
<p class="title">Commander</p>
|
||||||
<form id="inputs-parent" method="post" action="{% url 'order:order' grouped_order.code %}">
|
<form id="inputs-parent" method="post" action="{% url 'order:order' grouped_order.code %}" onSubmit="document.getElementById('submit').disabled=true;">
|
||||||
|
{% csrf_token %}
|
||||||
<!-- Tableau de commandes - sur ordi -->
|
<!-- Tableau de commandes - sur ordi -->
|
||||||
<table class="table is-hidden-touch">
|
<table class="table is-hidden-touch">
|
||||||
<thead>
|
<thead>
|
||||||
|
@ -91,7 +91,7 @@
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for item in grouped_order.item_set.all %}
|
{% for item in grouped_order.item_set.all %}
|
||||||
<tr>
|
<tr>
|
||||||
{% csrf_token %}
|
|
||||||
<td>{{ item.name }}</td>
|
<td>{{ item.name }}</td>
|
||||||
<td>{{ item.price }} €</td>
|
<td>{{ item.price }} €</td>
|
||||||
<td><input name="quantity_{{ item.id }}" size="4" type="number" value="0" min="0"
|
<td><input name="quantity_{{ item.id }}" size="4" type="number" value="0" min="0"
|
||||||
|
@ -152,25 +152,27 @@
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<p><label for="first_name">Prénom : </label>
|
<p><label for="first_name">Prénom : </label>
|
||||||
<input id="first_name" type="text" name="first_name" placeholder="Votre prénom"
|
<input id="first_name" type="text" name="first_name" placeholder="Votre prénom"
|
||||||
value="{{ author.first_name }}" required></p>
|
value="{{ order_author.first_name }}" required></p>
|
||||||
<p><label for="first_name">Nom : </label>
|
<p><label for="first_name">Nom : </label>
|
||||||
<input id="last_name" type="text" name="last_name" placeholder="Votre nom"
|
<input id="last_name" type="text" name="last_name" placeholder="Votre nom"
|
||||||
value="{{ author.last_name }}" required></p>
|
value="{{ order_author.last_name }}" required></p>
|
||||||
</div>
|
</div>
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<p><label for="phone">Numéro de téléphone :</label>
|
<p><label for="phone">Numéro de téléphone {% if not is_phone_mandatory %}<em>(facultatif)</em> {% endif %}:</label>
|
||||||
<input id="phone" type="tel" pattern="[0-9]{10}" placeholder="0601020304" name="phone"
|
<input id="phone" type="tel" pattern="[0-9]{10}"
|
||||||
value="{{ author.phone }}" required></p>
|
placeholder="0601020304" name="phone"
|
||||||
|
value="{{ order_author.phone }}"
|
||||||
|
{% if is_phone_mandatory %}required{% endif %}></p>
|
||||||
<p><label for="email">Adresse mail : </label>
|
<p><label for="email">Adresse mail : </label>
|
||||||
<input id="email" type="email" placeholder="exemple@mail.fr" name="email"
|
<input id="email" type="email" placeholder="exemple@mail.fr" name="email"
|
||||||
value="{{ author.email }}" required></p>
|
value="{{ order_author.email }}" required></p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p><label for="note">Note à l'organisateur·ice</label>
|
<p><label for="note">Note à l'organisateur·ice<em> (facultatif)</em> :</label>
|
||||||
<textarea id="note" rows=3 placeholder="(facultatif)" name="note">{{ note }}</textarea></p>
|
<textarea id="note" rows=3 name="note">{{ note }}</textarea></p>
|
||||||
|
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<button type="submit" value="Order" class="button is-primary">
|
<button id="submit" type="submit" value="Order" class="button is-primary">
|
||||||
<i class="fa fa-shopping-basket mr-3" aria-hidden="true"></i>Commander
|
<i class="fa fa-shopping-basket mr-3" aria-hidden="true"></i>Commander
|
||||||
</button>
|
</button>
|
||||||
<p>Total : <span class="total">0</span> €</p>
|
<p>Total : <span class="total">0</span> €</p>
|
||||||
|
@ -186,7 +188,6 @@
|
||||||
// Compute total price whenever a value in input is modified
|
// Compute total price whenever a value in input is modified
|
||||||
document.getElementById("inputs-parent").addEventListener("change", function () {
|
document.getElementById("inputs-parent").addEventListener("change", function () {
|
||||||
inputs = [...document.getElementsByTagName("input")].filter(input => input.getAttribute("name").indexOf("quantity_") === 0); //filter the inputs to get the quantity inputs only
|
inputs = [...document.getElementsByTagName("input")].filter(input => input.getAttribute("name").indexOf("quantity_") === 0); //filter the inputs to get the quantity inputs only
|
||||||
|
|
||||||
prices = {{ prices_dict | safe }};
|
prices = {{ prices_dict | safe }};
|
||||||
let total_price = 0;
|
let total_price = 0;
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,7 @@
|
||||||
<p><i class="fa fa-map-pin mr-3" aria-label="Lieu" title="Lieu" aria-hidden="true"></i>{{ grouped_order.place }}</p>
|
<p><i class="fa fa-map-pin mr-3" aria-label="Lieu" title="Lieu" aria-hidden="true"></i>{{ grouped_order.place }}</p>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<p><i class="fa fa-calendar-check-o mr-3" aria-label="Date limite de commande" title="Date limite de commande" aria-hidden="true"></i>
|
<p><i class="fa fa-calendar-check-o mr-3" aria-label="Date limite de commande" title="Date limite de commande" aria-hidden="true"></i>
|
||||||
Commandes avant le {{ grouped_order.deadline }}
|
Commandes avant le {{ grouped_order.deadline|date:'d M Y' }} à {{ grouped_order.deadline|date:'H:i' }}
|
||||||
</p>
|
</p>
|
||||||
<p><i class="fa fa-truck mr-3" aria-label="Date de livraison" title="Date de livraison" aria-hidden="true"></i>
|
<p><i class="fa fa-truck mr-3" aria-label="Date de livraison" title="Date de livraison" aria-hidden="true"></i>
|
||||||
Livraison le {{ grouped_order.delivery_date }}{% if grouped_order.delivery_slot %}, {{ grouped_order.delivery_slot }}{% endif %}
|
Livraison le {{ grouped_order.delivery_date }}{% if grouped_order.delivery_slot %}, {{ grouped_order.delivery_slot }}{% endif %}
|
||||||
|
@ -90,10 +90,10 @@
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<p>Pour vous aider à distribuer les produits le jour J, vous pouvez télécharger la liste des commandes
|
<p>Pour vous aider à distribuer les produits le jour J, vous pouvez télécharger la liste des commandes
|
||||||
au format PDF pour l'<strong>imprimer</strong>, ou au format CSV pour l'<strong>afficher dans un tableur</strong> :</p>
|
au format PDF pour l'<strong>imprimer</strong>, ou au format CSV pour l'<strong>afficher dans un tableur</strong> :</p>
|
||||||
<a class="button is-info" href="{% url 'order:grouped_order_sheet' grouped_order.code %}" target="_blank">
|
<a class="button is-info" href="{% url 'order:grouped_order_sheet' grouped_order.code %}">
|
||||||
<i class="fa fa-file-pdf-o mr-3" aria-hidden="true"></i>Commandes en PDF
|
<i class="fa fa-file-pdf-o mr-3" aria-hidden="true"></i>Commandes en PDF
|
||||||
</a>
|
</a>
|
||||||
<a class="button is-info" href="{% url 'order:grouped_order_csv_export' grouped_order.code %}" target="_blank">
|
<a class="button is-info" href="{% url 'order:grouped_order_csv_export' grouped_order.code %}">
|
||||||
<i class="fa fa-file-excel-o mr-3" aria-hidden="true"></i>Commandes en CSV
|
<i class="fa fa-file-excel-o mr-3" aria-hidden="true"></i>Commandes en CSV
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
@ -126,7 +126,7 @@
|
||||||
<tfoot>
|
<tfoot>
|
||||||
<th>Total</th>
|
<th>Total</th>
|
||||||
<th></th>
|
<th></th>
|
||||||
<th></th>
|
<th>{{ total_ordered_items }}</th>
|
||||||
<th>{{ grouped_order.total_price }} €</th>
|
<th>{{ grouped_order.total_price }} €</th>
|
||||||
</tfoot>
|
</tfoot>
|
||||||
</table>
|
</table>
|
||||||
|
@ -137,11 +137,11 @@
|
||||||
|
|
||||||
<div id="commandes" class="box tabcontent">
|
<div id="commandes" class="box tabcontent">
|
||||||
<div class="buttons is-pulled-right">
|
<div class="buttons is-pulled-right">
|
||||||
<a class="button is-info" href="{% url 'order:email_list' grouped_order.code %}?format=csv" target="_blank">
|
<a class="button is-info" href="{% url 'order:email_list' grouped_order.code %}?format=csv">
|
||||||
<i class="fa fa-file-excel-o mr-3" aria-hidden="true"></i>Emails en CSV
|
<i class="fa fa-file-excel-o mr-3" aria-hidden="true"></i>Emails en CSV
|
||||||
</a>
|
</a>
|
||||||
<input id="email_list" name="email_list" hidden="true" value="{{ emails_list|join:';' }}" />
|
<input id="email_list" name="email_list" hidden="true" value="{{ emails_list|join:';' }}" />
|
||||||
<a class="button is-info" onclick="copyText('email_list')" target="_blank">
|
<a class="button is-info" onclick="copyText('email_list')">
|
||||||
<i class="fa fa-files-o mr-3" aria-hidden="true"></i>Copier les emails
|
<i class="fa fa-files-o mr-3" aria-hidden="true"></i>Copier les emails
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
@ -161,7 +161,7 @@
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ order.author }}</td>
|
<td>{{ order.author }}</td>
|
||||||
<td>{{ order.price }} €</td>
|
<td>{{ order.price }} €</td>
|
||||||
<td><a href="mailto:{{ order.author.email }}">{{ order.author.email }}</a></td>
|
<td><a href="mailto:{{ order.author.email }}">{{ order.author.email }}</a>{% if order.author.phone %} / {{ order.author.phone }}{% endif %}</td>
|
||||||
<td>
|
<td>
|
||||||
<button class="button is-info is-small js-modal-trigger" data-target="order-detail-{{ order.id }}">
|
<button class="button is-info is-small js-modal-trigger" data-target="order-detail-{{ order.id }}">
|
||||||
Voir
|
Voir
|
||||||
|
@ -180,43 +180,36 @@
|
||||||
<!-- Order detail modal -->
|
<!-- Order detail modal -->
|
||||||
<div id="order-detail-{{ order.id }}" class="modal">
|
<div id="order-detail-{{ order.id }}" class="modal">
|
||||||
<div class="modal-background"></div>
|
<div class="modal-background"></div>
|
||||||
<div class="modal-card">
|
<div class="modal-card ">
|
||||||
<header class="modal-card-head has-background-info">
|
<header class="modal-card-head has-background-info">
|
||||||
<div class="modal-card-title-container">
|
<div class="modal-card-title-container">
|
||||||
<p class="modal-card-title">Commande de {{ order.author }}</p>
|
<p class="modal-card-title mb-2">Commande de {{ order.author }}</p>
|
||||||
|
<p class="has-text-grey-dark">Le {{ order.created_date|date:'d M Y' }} à {{ order.created_date|date:'H:i' }}</p>
|
||||||
</div>
|
</div>
|
||||||
<button class="delete" aria-label="close"></button>
|
<button class="delete" aria-label="close"></button>
|
||||||
</header>
|
</header>
|
||||||
<section class="modal-card-body">
|
<section class="modal-card-body">
|
||||||
<div class="columns is-multiline">
|
<div class="columns">
|
||||||
<div class="column is-full">
|
<div class="column">
|
||||||
{% for item in order.ordered_items.all %}
|
{% for item in order.ordered_items.all %}
|
||||||
{{ item.nb }} × {{ item.item.name }}
|
{{ item.nb }} × {{ item.item.name }}
|
||||||
<hr class="mb-1 mt-1">
|
<hr class="mb-0 mt-1">
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
{% if order.note %}
|
{% if order.note %}
|
||||||
<div class="column is-full">
|
<div class="column is-4">
|
||||||
<p class="mini-title">Note à l'organisateur·ice</p>
|
<p class="mini-title">Note à l'organisateur·ice</p>
|
||||||
<p>{{ order.note }}</p>
|
<p>{{ order.note }}</p>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
|
||||||
<div class="column is-full">
|
|
||||||
<p class="mini-title">Numéro de téléphone</p>
|
|
||||||
<p>{{ order.author.phone }}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
{% endif %}
|
||||||
</section>
|
</section>
|
||||||
<footer class="modal-card-foot">
|
<footer class="modal-card-foot">
|
||||||
<div class="columns">
|
<p class="mini-title">Total à payer</p>
|
||||||
<div class="column is-half">
|
<p>{{ order.price }} €</p>
|
||||||
<p class="mini-title">Total à payer</p>
|
|
||||||
<p>{{ order.price }} €</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</footer>
|
</footer>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Confirm delete order modal -->
|
<!-- Confirm delete order modal -->
|
||||||
<div id="confirm-delete-{{ order.id }}" class="modal">
|
<div id="confirm-delete-{{ order.id }}" class="modal">
|
||||||
|
|
|
@ -21,26 +21,22 @@
|
||||||
}
|
}
|
||||||
{% if items|length > 10 %}
|
{% if items|length > 10 %}
|
||||||
.item_name {
|
.item_name {
|
||||||
text-align:center;
|
|
||||||
white-space:nowrap;
|
|
||||||
-webkit-transform: rotate(-90deg);
|
-webkit-transform: rotate(-90deg);
|
||||||
-moz-transform: rotate(-90deg);
|
-moz-transform: rotate(-90deg);
|
||||||
-ms-transform: rotate(-90deg);
|
-ms-transform: rotate(-90deg);
|
||||||
-o-transform: rotate(-90deg);
|
-o-transform: rotate(-90deg);
|
||||||
transform: rotate(-90deg);
|
transform: rotate(-90deg);
|
||||||
font-size: 8pt;
|
font-size: 8pt;
|
||||||
|
height:220px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.item_name div {
|
.item_name div {
|
||||||
margin:-10px -120% ;
|
width:200px;
|
||||||
display:inline-block;
|
-webkit-transform: translateX(-40%);
|
||||||
}
|
-moz-transform: translateX(-40%);
|
||||||
.item_name div:before{
|
-ms-transform: translateX(-40%);
|
||||||
content:'';
|
-o-transform: translateX(-40%);
|
||||||
width:0;
|
transform: translateX(-40%);
|
||||||
padding-top:110%;
|
|
||||||
display:inline-block;
|
|
||||||
vertical-align:middle;
|
|
||||||
}
|
}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
@ -53,20 +49,19 @@
|
||||||
border: 1px black solid;
|
border: 1px black solid;
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
table-layout: fixed
|
||||||
}
|
}
|
||||||
|
|
||||||
th, td {
|
th, td {
|
||||||
vertical-align: center;
|
|
||||||
padding: 3px 2px 2px 2px;
|
padding: 3px 2px 2px 2px;
|
||||||
border: 1px black solid;
|
border: 1px black solid;
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
word-wrap: break-word;
|
||||||
}
|
}
|
||||||
|
|
||||||
th {
|
th {
|
||||||
page-break-inside: avoid;
|
page-break-inside: avoid;
|
||||||
word-break: break-all;
|
|
||||||
word-wrap: break-word;
|
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -85,7 +80,7 @@
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th style="font-size: 0.5em; width: 2em">OK</th>
|
<th style="font-size: 0.5em; width: 2em">OK</th>
|
||||||
<th style="text-align: center">Nom</th>
|
<th style="text-align: center; width: 20%">Nom</th>
|
||||||
{% for item in items %}
|
{% for item in items %}
|
||||||
<th class="item_name" style="font-weight: normal;">
|
<th class="item_name" style="font-weight: normal;">
|
||||||
<div>{{ item.name }}</div>
|
<div>{{ item.name }}</div>
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% load crispy_forms_tags %}
|
||||||
|
|
||||||
{% block title %}Modifier la commande groupée{% endblock %}
|
{% block title %}Modifier la commande groupée{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
@ -9,13 +11,14 @@
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<p class="title">{{ grouped_order.name }} - modifier</p>
|
<p class="title">{{ grouped_order.name }} - modifier</p>
|
||||||
<form method="post" onsubmit="deadlinePassedCheck(event)">{% csrf_token %}
|
<form method="post" onsubmit="deadlinePassedCheck(event)">{% csrf_token %}
|
||||||
{{ form.as_p }}
|
{{ form | crispy }}
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<a class="button is-light" href="{% url 'order:index' %}">Annuler</a>
|
<a class="button is-light" href="{% url 'order:index' %}">Annuler</a>
|
||||||
<input class="button is-primary" type="submit" value="Suivant">
|
<input class="button is-primary" type="submit" value="Suivant">
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extra_js %}
|
{% block extra_js %}
|
||||||
|
|
|
@ -13,6 +13,7 @@ from icalendar import Calendar, Event, vCalAddress, vText
|
||||||
|
|
||||||
from ..forms import GroupedOrderForm, Item, JoinGroupedOrderForm
|
from ..forms import GroupedOrderForm, Item, JoinGroupedOrderForm
|
||||||
from ..models import GroupedOrder, OrderAuthor
|
from ..models import GroupedOrder, OrderAuthor
|
||||||
|
from .mixins import UserIsOrgaMixin
|
||||||
|
|
||||||
|
|
||||||
class IndexView(LoginRequiredMixin, generic.ListView):
|
class IndexView(LoginRequiredMixin, generic.ListView):
|
||||||
|
@ -111,23 +112,36 @@ class GroupedOrderDetailView(generic.DetailView):
|
||||||
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
items = self.get_object().item_set.all()
|
grouped_order = self.get_object()
|
||||||
|
items = grouped_order.item_set.all()
|
||||||
|
|
||||||
remaining_qty = {item.id: item.get_remaining_nb() for item in items}
|
remaining_qty = {item.id: item.get_remaining_nb() for item in items}
|
||||||
prices_dict = {item.id: item.price for item in items}
|
prices_dict = {item.id: item.price for item in items}
|
||||||
|
|
||||||
|
if self.request.user.is_authenticated:
|
||||||
|
order_author = OrderAuthor(
|
||||||
|
first_name=self.request.user.first_name,
|
||||||
|
last_name=self.request.user.last_name,
|
||||||
|
email=self.request.user.username,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
order_author = None
|
||||||
|
|
||||||
context = super().get_context_data(**kwargs)
|
context = super().get_context_data(**kwargs)
|
||||||
context.update(
|
context.update(
|
||||||
{
|
{
|
||||||
# Used for the js display of total price of an order
|
# Used for the js display of total price of an order
|
||||||
"prices_dict": json.dumps(prices_dict, cls=DjangoJSONEncoder),
|
"prices_dict": json.dumps(prices_dict, cls=DjangoJSONEncoder),
|
||||||
"remaining_qty": remaining_qty,
|
"remaining_qty": remaining_qty,
|
||||||
|
"order_author": order_author,
|
||||||
|
# Used to set if the phone is required in the form
|
||||||
|
"is_phone_mandatory": grouped_order.is_phone_mandatory,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
class GroupedOrderOverview(UserPassesTestMixin, generic.DetailView):
|
class GroupedOrderOverview(UserIsOrgaMixin, generic.DetailView):
|
||||||
model = GroupedOrder
|
model = GroupedOrder
|
||||||
template_name = "order/grouped_order_overview.html"
|
template_name = "order/grouped_order_overview.html"
|
||||||
context_object_name = "grouped_order"
|
context_object_name = "grouped_order"
|
||||||
|
@ -136,14 +150,8 @@ class GroupedOrderOverview(UserPassesTestMixin, generic.DetailView):
|
||||||
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
# Restrict access to the manager
|
# Staff can see but not edit grouped orders
|
||||||
return self.get_object().orga == self.request.user
|
return super().test_func() or self.request.user.is_staff
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
|
||||||
# Compute grouped order total price before display
|
|
||||||
self.get_object().compute_total_price()
|
|
||||||
self.get_object().compute_items_ordered_nb()
|
|
||||||
return super().get(self, request, *args, **kwargs)
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
context = super(GroupedOrderOverview, self).get_context_data(**kwargs)
|
context = super(GroupedOrderOverview, self).get_context_data(**kwargs)
|
||||||
|
@ -155,6 +163,7 @@ class GroupedOrderOverview(UserPassesTestMixin, generic.DetailView):
|
||||||
order__in=self.get_object().order_set.all()
|
order__in=self.get_object().order_set.all()
|
||||||
)
|
)
|
||||||
context["emails_list"] = set(participant.email for participant in participants)
|
context["emails_list"] = set(participant.email for participant in participants)
|
||||||
|
context["total_ordered_items"] = self.get_object().get_total_ordered_items()
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
|
@ -175,7 +184,7 @@ class GroupedOrderCreateView(LoginRequiredMixin, generic.CreateView):
|
||||||
return super().form_valid(form)
|
return super().form_valid(form)
|
||||||
|
|
||||||
|
|
||||||
class GroupedOrderUpdateView(UserPassesTestMixin, generic.UpdateView):
|
class GroupedOrderUpdateView(UserIsOrgaMixin, generic.UpdateView):
|
||||||
model = GroupedOrder
|
model = GroupedOrder
|
||||||
template_name = "order/grouped_order_update.html"
|
template_name = "order/grouped_order_update.html"
|
||||||
context_object_name = "grouped_order"
|
context_object_name = "grouped_order"
|
||||||
|
@ -184,24 +193,16 @@ class GroupedOrderUpdateView(UserPassesTestMixin, generic.UpdateView):
|
||||||
def get_object(self, queryset=None):
|
def get_object(self, queryset=None):
|
||||||
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
|
||||||
def test_func(self):
|
|
||||||
# Restrict access to the manager
|
|
||||||
return self.get_object().orga == self.request.user
|
|
||||||
|
|
||||||
def get_form_kwargs(self):
|
def get_form_kwargs(self):
|
||||||
kwargs = super().get_form_kwargs()
|
kwargs = super().get_form_kwargs()
|
||||||
kwargs["user"] = self.request.user
|
kwargs["user"] = self.request.user
|
||||||
return kwargs
|
return kwargs
|
||||||
|
|
||||||
|
|
||||||
class GroupedOrderDuplicateView(UserPassesTestMixin, generic.RedirectView):
|
class GroupedOrderDuplicateView(UserIsOrgaMixin, generic.RedirectView):
|
||||||
def get_object(self, queryset=None):
|
def get_object(self, queryset=None):
|
||||||
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
|
||||||
def test_func(self):
|
|
||||||
# Restrict access to the manager
|
|
||||||
return self.get_object().orga == self.request.user
|
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
def get(self, request, *args, **kwargs):
|
||||||
# overwrite the ``get`` function to copy the initial grouped order before
|
# overwrite the ``get`` function to copy the initial grouped order before
|
||||||
# redirecting to the update view of the new grouped order
|
# redirecting to the update view of the new grouped order
|
||||||
|
@ -224,7 +225,6 @@ class GroupedOrderDuplicateView(UserPassesTestMixin, generic.RedirectView):
|
||||||
# duplicate each item and add it to new_grouped_order
|
# duplicate each item and add it to new_grouped_order
|
||||||
for item in initial_grouped_order.item_set.all():
|
for item in initial_grouped_order.item_set.all():
|
||||||
item.pk = None
|
item.pk = None
|
||||||
item.ordered_nb = 0
|
|
||||||
item.save()
|
item.save()
|
||||||
new_grouped_order.item_set.add(item)
|
new_grouped_order.item_set.add(item)
|
||||||
|
|
||||||
|
@ -238,7 +238,7 @@ class GroupedOrderDuplicateView(UserPassesTestMixin, generic.RedirectView):
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class GroupedOrderDeleteView(UserPassesTestMixin, generic.DeleteView):
|
class GroupedOrderDeleteView(UserIsOrgaMixin, generic.DeleteView):
|
||||||
model = GroupedOrder
|
model = GroupedOrder
|
||||||
template_name = "order/grouped_order_confirm_delete.html"
|
template_name = "order/grouped_order_confirm_delete.html"
|
||||||
context_object_name = "grouped_order"
|
context_object_name = "grouped_order"
|
||||||
|
@ -249,10 +249,6 @@ class GroupedOrderDeleteView(UserPassesTestMixin, generic.DeleteView):
|
||||||
def get_success_url(self):
|
def get_success_url(self):
|
||||||
return reverse_lazy("order:index")
|
return reverse_lazy("order:index")
|
||||||
|
|
||||||
def test_func(self):
|
|
||||||
# Restrict access to the manager
|
|
||||||
return self.get_object().orga == self.request.user
|
|
||||||
|
|
||||||
def form_valid(self, form):
|
def form_valid(self, form):
|
||||||
# Delete related OrderAuthors
|
# Delete related OrderAuthors
|
||||||
grouped_order = self.get_object()
|
grouped_order = self.get_object()
|
||||||
|
@ -284,23 +280,27 @@ class GroupedOrderAddItemsView(UserPassesTestMixin, generic.ListView):
|
||||||
return grouped_order.orga == self.request.user
|
return grouped_order.orga == self.request.user
|
||||||
|
|
||||||
|
|
||||||
class GroupedOrderExportView(UserPassesTestMixin, generic.DetailView):
|
class GroupedOrderExportView(UserIsOrgaMixin, generic.DetailView):
|
||||||
model = GroupedOrder
|
model = GroupedOrder
|
||||||
template_name = "order/grouped_order_sheet.html"
|
template_name = "order/grouped_order_sheet.html"
|
||||||
context_object_name = "grouped_order"
|
context_object_name = "grouped_order"
|
||||||
|
|
||||||
|
def test_func(self):
|
||||||
|
# Grouped order orga, superuser and staff can get this view
|
||||||
|
return super().test_func() or self.request.user.is_staff
|
||||||
|
|
||||||
def get_object(self, queryset=None):
|
def get_object(self, queryset=None):
|
||||||
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
return get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
|
||||||
def test_func(self):
|
|
||||||
# Restrict access to the manager
|
|
||||||
return self.get_object().orga == self.request.user
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
context = super(GroupedOrderExportView, self).get_context_data(**kwargs)
|
context = super(GroupedOrderExportView, self).get_context_data(**kwargs)
|
||||||
grouped_order = self.get_object()
|
grouped_order = self.get_object()
|
||||||
|
|
||||||
items = grouped_order.item_set.filter(ordered_nb__gt=0).order_by("name")
|
items = [
|
||||||
|
item
|
||||||
|
for item in grouped_order.item_set.all().order_by("name")
|
||||||
|
if item.ordered_nb > 0
|
||||||
|
]
|
||||||
orders = grouped_order.order_set.all().order_by(
|
orders = grouped_order.order_set.all().order_by(
|
||||||
"author__last_name", "author__first_name"
|
"author__last_name", "author__first_name"
|
||||||
)
|
)
|
||||||
|
@ -328,9 +328,9 @@ class DownloadGroupedOrderSheetView(WeasyTemplateResponseMixin, GroupedOrderExpo
|
||||||
|
|
||||||
class ExportGroupOrderEmailAdressesToDownloadView(UserPassesTestMixin, generic.View):
|
class ExportGroupOrderEmailAdressesToDownloadView(UserPassesTestMixin, generic.View):
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
# Restrict access to the manager
|
# Restrict access to the manager, superuser and staff
|
||||||
origin = get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
origin = get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
return origin.orga == self.request.user
|
return origin.orga == self.request.user or self.request.user.is_staff
|
||||||
|
|
||||||
def get(self, request, *args, **kwargs):
|
def get(self, request, *args, **kwargs):
|
||||||
grouped_order = get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
grouped_order = get_object_or_404(GroupedOrder, code=self.kwargs.get("code"))
|
||||||
|
@ -344,10 +344,9 @@ class ExportGroupOrderEmailAdressesToDownloadView(UserPassesTestMixin, generic.V
|
||||||
response = http.HttpResponse(content_type="text/csv")
|
response = http.HttpResponse(content_type="text/csv")
|
||||||
filename = f"commande _{grouped_order.name.replace(' ', '_')}"
|
filename = f"commande _{grouped_order.name.replace(' ', '_')}"
|
||||||
response["Content-Disposition"] = f'attachment; filename="{filename}.csv"'
|
response["Content-Disposition"] = f'attachment; filename="{filename}.csv"'
|
||||||
writer = csv.writer(response)
|
writer = csv.writer(response, delimiter=";")
|
||||||
writer.writerow(["order_name", "email"])
|
|
||||||
for email in email_list:
|
for email in email_list:
|
||||||
writer.writerow([grouped_order.name, email])
|
writer.writerow([email])
|
||||||
return response
|
return response
|
||||||
else:
|
else:
|
||||||
email_list = ";\n".join(email_list)
|
email_list = ";\n".join(email_list)
|
||||||
|
@ -362,10 +361,10 @@ class ExportGroupedOrderToCSVView(GroupedOrderExportView):
|
||||||
response = http.HttpResponse(
|
response = http.HttpResponse(
|
||||||
content_type="text/csv",
|
content_type="text/csv",
|
||||||
headers={
|
headers={
|
||||||
"Content-Disposition": f'attachment; filename="{ context["object"].name }-commandes"'
|
"Content-Disposition": f'attachment; filename="{context["object"].name}-commandes"'
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
writer = csv.writer(response)
|
writer = csv.writer(response, delimiter=";")
|
||||||
|
|
||||||
# write headers rows
|
# write headers rows
|
||||||
row = ["", ""]
|
row = ["", ""]
|
||||||
|
@ -374,6 +373,9 @@ class ExportGroupedOrderToCSVView(GroupedOrderExportView):
|
||||||
row.append("Prix de la commande")
|
row.append("Prix de la commande")
|
||||||
row.append("Mail")
|
row.append("Mail")
|
||||||
row.append("Téléphone")
|
row.append("Téléphone")
|
||||||
|
row.append("Note")
|
||||||
|
row.append("Date")
|
||||||
|
row.append("Heure")
|
||||||
writer.writerow(row)
|
writer.writerow(row)
|
||||||
|
|
||||||
row = ["", "Prix unitaire TTC (€)"]
|
row = ["", "Prix unitaire TTC (€)"]
|
||||||
|
@ -393,6 +395,9 @@ class ExportGroupedOrderToCSVView(GroupedOrderExportView):
|
||||||
row.append(str(order.price).replace(".", ","))
|
row.append(str(order.price).replace(".", ","))
|
||||||
row.append(order.author.email)
|
row.append(order.author.email)
|
||||||
row.append(f"'{order.author.phone}")
|
row.append(f"'{order.author.phone}")
|
||||||
|
row.append(order.note)
|
||||||
|
row.append(order.created_date.strftime("%d/%m/%Y"))
|
||||||
|
row.append(order.created_date.strftime("%H:%M"))
|
||||||
writer.writerow(row)
|
writer.writerow(row)
|
||||||
|
|
||||||
# write total row
|
# write total row
|
||||||
|
|
|
@ -29,5 +29,8 @@ class ItemDeleteView(UserPassesTestMixin, generic.DeleteView):
|
||||||
return reverse_lazy("order:manage_items", args=[self.object.grouped_order.code])
|
return reverse_lazy("order:manage_items", args=[self.object.grouped_order.code])
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
# Restrict access to the manager
|
# Restrict access to the manager or a superuser
|
||||||
return self.get_object().grouped_order.orga == self.request.user
|
return (
|
||||||
|
self.get_object().grouped_order.orga == self.request.user
|
||||||
|
or self.request.user.is_superuser
|
||||||
|
)
|
||||||
|
|
13
la_chariotte/order/views/mixins.py
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
from django.contrib.auth.mixins import UserPassesTestMixin
|
||||||
|
|
||||||
|
|
||||||
|
class UserIsOrgaMixin(UserPassesTestMixin):
|
||||||
|
"""
|
||||||
|
The view is accessible only if the request user is orga or superuser
|
||||||
|
"""
|
||||||
|
|
||||||
|
def test_func(self):
|
||||||
|
return (
|
||||||
|
self.get_object().orga == self.request.user
|
||||||
|
or self.request.user.is_superuser
|
||||||
|
)
|
|
@ -2,6 +2,7 @@ from django import http
|
||||||
from django.contrib.auth.mixins import UserPassesTestMixin
|
from django.contrib.auth.mixins import UserPassesTestMixin
|
||||||
from django.shortcuts import get_object_or_404, render
|
from django.shortcuts import get_object_or_404, render
|
||||||
from django.urls import reverse, reverse_lazy
|
from django.urls import reverse, reverse_lazy
|
||||||
|
from django.utils import timezone
|
||||||
from django.views import generic
|
from django.views import generic
|
||||||
|
|
||||||
from la_chariotte.mail.utils import send_order_confirmation_mail
|
from la_chariotte.mail.utils import send_order_confirmation_mail
|
||||||
|
@ -48,7 +49,12 @@ def place_order(request, code):
|
||||||
author = OrderAuthor.objects.create(
|
author = OrderAuthor.objects.create(
|
||||||
first_name=first_name, last_name=last_name, email=email, phone=phone
|
first_name=first_name, last_name=last_name, email=email, phone=phone
|
||||||
)
|
)
|
||||||
order = Order.objects.create(author=author, grouped_order=grouped_order, note=note)
|
order = Order.objects.create(
|
||||||
|
author=author,
|
||||||
|
grouped_order=grouped_order,
|
||||||
|
note=note,
|
||||||
|
created_date=timezone.now(),
|
||||||
|
)
|
||||||
|
|
||||||
# add items to the order
|
# add items to the order
|
||||||
error_message = None
|
error_message = None
|
||||||
|
@ -66,7 +72,6 @@ def place_order(request, code):
|
||||||
if error_message:
|
if error_message:
|
||||||
order.delete()
|
order.delete()
|
||||||
author.delete()
|
author.delete()
|
||||||
grouped_order.compute_items_ordered_nb()
|
|
||||||
return render(
|
return render(
|
||||||
request,
|
request,
|
||||||
"order/grouped_order_detail.html",
|
"order/grouped_order_detail.html",
|
||||||
|
@ -97,8 +102,6 @@ def place_order(request, code):
|
||||||
)
|
)
|
||||||
|
|
||||||
# Send confirmation mail and redirect to confirmation page
|
# Send confirmation mail and redirect to confirmation page
|
||||||
order.compute_order_price()
|
|
||||||
grouped_order.compute_items_ordered_nb()
|
|
||||||
send_order_confirmation_mail(order)
|
send_order_confirmation_mail(order)
|
||||||
|
|
||||||
# Redirect to prevent data from being posted twice when the user hits the Back
|
# Redirect to prevent data from being posted twice when the user hits the Back
|
||||||
|
@ -117,7 +120,6 @@ def validate_item_ordered_nb(item, ordered_nb):
|
||||||
|
|
||||||
def validate_articles_ordered_nb(order):
|
def validate_articles_ordered_nb(order):
|
||||||
"""Return an error if no items are ordered"""
|
"""Return an error if no items are ordered"""
|
||||||
order.compute_order_articles_nb()
|
|
||||||
if order.articles_nb == 0:
|
if order.articles_nb == 0:
|
||||||
return "Veuillez commander au moins un produit"
|
return "Veuillez commander au moins un produit"
|
||||||
return None
|
return None
|
||||||
|
@ -136,5 +138,8 @@ class OrderDeleteView(UserPassesTestMixin, generic.DeleteView):
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
"""Accessible only if the requesting user is the grouped order organizer"""
|
# Restrict access to the manager or a superuser
|
||||||
return self.get_object().grouped_order.orga == self.request.user
|
return (
|
||||||
|
self.get_object().grouped_order.orga == self.request.user
|
||||||
|
or self.request.user.is_superuser
|
||||||
|
)
|
||||||
|
|
28
la_chariotte/order/views/stats.py
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
import json
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
from django.http import HttpResponse
|
||||||
|
from django.utils.timezone import make_aware
|
||||||
|
|
||||||
|
from la_chariotte.accounts.models import CustomUser
|
||||||
|
from la_chariotte.order.models import GroupedOrder, Order
|
||||||
|
|
||||||
|
|
||||||
|
def stats(request):
|
||||||
|
last_month = make_aware(datetime.today() - timedelta(days=30))
|
||||||
|
|
||||||
|
last_month_grouped_orders = GroupedOrder.objects.filter(
|
||||||
|
deadline__gte=last_month
|
||||||
|
).count()
|
||||||
|
|
||||||
|
return HttpResponse(
|
||||||
|
json.dumps(
|
||||||
|
{
|
||||||
|
"total_users": CustomUser.objects.count(),
|
||||||
|
"total_grouped_orders": GroupedOrder.objects.count(),
|
||||||
|
"total_orders": Order.objects.count(),
|
||||||
|
"last_month_grouped_orders": last_month_grouped_orders,
|
||||||
|
}
|
||||||
|
),
|
||||||
|
content_type="application/json",
|
||||||
|
)
|
|
@ -7,8 +7,10 @@ from sentry_sdk.integrations.django import DjangoIntegration
|
||||||
BASE_DIR = Path(__file__).resolve().parent.parent
|
BASE_DIR = Path(__file__).resolve().parent.parent
|
||||||
BASE_URL = os.getenv("BASE_URL", "http://127.0.0.1:8000")
|
BASE_URL = os.getenv("BASE_URL", "http://127.0.0.1:8000")
|
||||||
PROJECT_NAME = os.getenv("PROJECT_NAME", "La Chariotte")
|
PROJECT_NAME = os.getenv("PROJECT_NAME", "La Chariotte")
|
||||||
GITLAB_LINK = "https://gitlab.com/hashbangfr/la_chariotte"
|
GITLAB_LINK = "https://framagit.org/la-chariotte/la-chariotte"
|
||||||
CONTACT_MAIL = "contact@chariotte.fr"
|
CONTACT_MAIL = "contact@chariotte.fr"
|
||||||
|
HELLOASSO_LINK = "https://www.helloasso.com/associations/la-chariotte/"
|
||||||
|
FEEDBACK_LINK = "https://framaforms.org/votre-avis-sur-la-chariotte-1709742328"
|
||||||
|
|
||||||
# SECURITY WARNING: keep the secret key used in production secret!
|
# SECURITY WARNING: keep the secret key used in production secret!
|
||||||
SECRET_KEY = os.getenv(
|
SECRET_KEY = os.getenv(
|
||||||
|
|
BIN
la_chariotte/static/img/notice/add_item.png
Normal file
After Width: | Height: | Size: 42 KiB |
BIN
la_chariotte/static/img/notice/add_item_2.png
Normal file
After Width: | Height: | Size: 57 KiB |
BIN
la_chariotte/static/img/notice/copy_link.png
Normal file
After Width: | Height: | Size: 80 KiB |
BIN
la_chariotte/static/img/notice/create_account.jpg
Normal file
After Width: | Height: | Size: 45 KiB |
BIN
la_chariotte/static/img/notice/create_account_form.jpg
Normal file
After Width: | Height: | Size: 74 KiB |
BIN
la_chariotte/static/img/notice/create_grouped_order.jpg
Normal file
After Width: | Height: | Size: 38 KiB |
BIN
la_chariotte/static/img/notice/create_grouped_order_form.jpg
Normal file
After Width: | Height: | Size: 48 KiB |
BIN
la_chariotte/static/img/notice/edit_grouped_order.png
Normal file
After Width: | Height: | Size: 298 KiB |
BIN
la_chariotte/static/img/notice/get_spreadsheet.png
Normal file
After Width: | Height: | Size: 42 KiB |
BIN
la_chariotte/static/img/notice/go_to_overview.png
Normal file
After Width: | Height: | Size: 242 KiB |
BIN
la_chariotte/static/img/notice/grouped_order_sum_up.png
Normal file
After Width: | Height: | Size: 208 KiB |
BIN
la_chariotte/static/img/notice/items_overview.png
Normal file
After Width: | Height: | Size: 57 KiB |
BIN
la_chariotte/static/img/notice/join_grouped_order.png
Normal file
After Width: | Height: | Size: 16 KiB |
BIN
la_chariotte/static/img/notice/log_in.jpg
Normal file
After Width: | Height: | Size: 33 KiB |
BIN
la_chariotte/static/img/notice/my_grouped_orders.jpg
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
la_chariotte/static/img/notice/order_form.png
Normal file
After Width: | Height: | Size: 51 KiB |
BIN
la_chariotte/static/img/notice/print_orders_list.png
Normal file
After Width: | Height: | Size: 42 KiB |
BIN
la_chariotte/static/img/notice/share_grouped_order.png
Normal file
After Width: | Height: | Size: 71 KiB |
14
la_chariotte/static/sass/base/_accordion.sass
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
.accordion
|
||||||
|
transition: max-height 0.2s ease-out
|
||||||
|
border: none
|
||||||
|
outline: none
|
||||||
|
transition: 0.4s
|
||||||
|
.message-header
|
||||||
|
cursor: pointer
|
||||||
|
background: $white-ter
|
||||||
|
border: 1px solid $primary
|
||||||
|
color: $primary
|
||||||
|
p
|
||||||
|
margin: 0
|
||||||
|
.message-body
|
||||||
|
display: none
|
|
@ -9,4 +9,8 @@
|
||||||
padding-top: $navbar-height
|
padding-top: $navbar-height
|
||||||
|
|
||||||
.formatted-text
|
.formatted-text
|
||||||
white-space: pre-wrap
|
white-space: pre-wrap
|
||||||
|
|
||||||
|
img.notice-img
|
||||||
|
border: $info 3px solid
|
||||||
|
margin-bottom: 1em
|
|
@ -1,6 +1,18 @@
|
||||||
|
.footer
|
||||||
|
padding-bottom: 10px !important
|
||||||
|
padding: 50px 0
|
||||||
|
padding-left: 2em
|
||||||
|
font-size: 1em
|
||||||
|
|
||||||
|
logo > h2
|
||||||
|
font-size: 2rem
|
||||||
|
|
||||||
footer > .columns
|
footer > .columns
|
||||||
margin-bottom: 1rem !important
|
margin-bottom: 1rem !important
|
||||||
margin-top: 0
|
margin-top: 0
|
||||||
|
|
||||||
footer .margin-auto
|
footer .margin-auto
|
||||||
margin: auto
|
margin: auto
|
||||||
|
|
||||||
|
footer .logo
|
||||||
|
letter-spacing: 1px
|
||||||
|
|
|
@ -17,3 +17,4 @@
|
||||||
@import "./base/titles"
|
@import "./base/titles"
|
||||||
@import "./base/tabs"
|
@import "./base/tabs"
|
||||||
@import "./base/footer"
|
@import "./base/footer"
|
||||||
|
@import "./base/accordion"
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
{% load static %}
|
{% load static %}
|
||||||
{% load settings %}
|
{% load settings %}
|
||||||
{% load sass_tags %}
|
{% load sass_tags %}
|
||||||
|
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="fr">
|
<html lang="fr">
|
||||||
<head>
|
<head>
|
||||||
|
@ -94,6 +93,9 @@
|
||||||
<a class="navbar-item" href="{% url 'notice' %}">
|
<a class="navbar-item" href="{% url 'notice' %}">
|
||||||
<i class="fa fa-cog mr-3" aria-hidden="true"></i>Comment ça marche ?
|
<i class="fa fa-cog mr-3" aria-hidden="true"></i>Comment ça marche ?
|
||||||
</a>
|
</a>
|
||||||
|
<a class="navbar-item" href="{% url 'faq' %}">
|
||||||
|
<i class="fa fa-question mr-3" aria-hidden="true"></i>FAQ
|
||||||
|
</a>
|
||||||
<a class="navbar-item" href="https://blog.chariotte.fr/last">
|
<a class="navbar-item" href="https://blog.chariotte.fr/last">
|
||||||
<i class="fa fa-newspaper-o mr-3" aria-hidden="true"></i>Actualités</a>
|
<i class="fa fa-newspaper-o mr-3" aria-hidden="true"></i>Actualités</a>
|
||||||
</div>
|
</div>
|
||||||
|
@ -121,46 +123,62 @@
|
||||||
{% block content %}{% endblock %}
|
{% block content %}{% endblock %}
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
<footer class="footer">
|
<footer class="footer section">
|
||||||
<div class="content columns has-text-centered">
|
<div class="container">
|
||||||
<div class="column margin-auto">
|
<div class="columns is-multiline is-justify-content-center">
|
||||||
<p>
|
<div class="column is-4-desktop is-4-tablet">
|
||||||
<strong>La Chariotte</strong> | version {{ version }}
|
<div class="widget mb-0">
|
||||||
</p>
|
<div class="logo mb-4">
|
||||||
<p>
|
<p>
|
||||||
<a href="{% url 'legal_notice' %}">Mentions légales</a>
|
{% settings_value "PROJECT_NAME" %}
|
||||||
</p>
|
| <a href='{% settings_value "GITLAB_LINK" %}'>version {{ version }}</a>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<ul class="list-unstyled footer-menu lh-35">
|
||||||
|
<a href="mailto:{% settings_value "CONTACT_MAIL" %}"><i class="fa fa-envelope-square mr-2 text-muted"></i>
|
||||||
|
{% settings_value "CONTACT_MAIL" %}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="{% url 'legal_notice' %}"><i class="fa fa-legal mr-2 text-muted"></i>Mentions légales</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="column is-2-desktop is-4-tablet">
|
||||||
|
<div class="widget mb-0">
|
||||||
|
<h4 class="mb-4">Suivez nous</h4>
|
||||||
|
<ul class="list-unstyled footer-socials">
|
||||||
|
<li>
|
||||||
|
<a href="https://blog.chariotte.fr"><i class="fa fa-rss-square mr-2 text-muted"></i>Chariotte news</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://mastodon.scop.coop/@la_chariotte"><i class="fa fa-mastodon-square mr-2 text-muted"></i>Mastodon</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://www.facebook.com/profile.php?id=100091984176981"><i class="fa fa-facebook-square mr-2 text-muted"></i>Facebook</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://www.linkedin.com/company/la-chariotte-commandes-groupees/"><i class="fa fa-linkedin-square mr-2 text-muted"></i>LinkedIn</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="column is-3-desktop is-4-tablet">
|
||||||
|
<div class="widget mb-0">
|
||||||
|
<h4 class="mb-4">Soutenez-nous</h4>
|
||||||
|
<ul class="list-unstyled">
|
||||||
|
<li>
|
||||||
|
<a href="{% settings_value "FEEDBACK_LINK" %}"><i class="fa fa-comments mr-2 text-muted"></i>Que pensez-vous de la Chariotte ?</a>
|
||||||
|
</li>
|
||||||
|
<li><a href="{% settings_value "HELLOASSO_LINK" %}"><i class="fa fa-heart mr-2 text-muted"></i>Faire un (petit) don</a></li>
|
||||||
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
<div class="column is-size-6 columns">
|
</div>
|
||||||
<div class="column margin-auto">
|
</footer>
|
||||||
<p>Contact : <a href="mailto:{% settings_value "CONTACT_MAIL" %}">{% settings_value "CONTACT_MAIL" %}</a>
|
</body>
|
||||||
<p>Suivez <strong>la Chariotte</strong> sur :</p>
|
</html>
|
||||||
</div>
|
<script>
|
||||||
<div class="column">
|
|
||||||
<a href="https://www.facebook.com/profile.php?id=100091984176981"
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener">
|
|
||||||
<i class="fa fa-facebook-square mr-2" aria-hidden="true"></i>Facebook
|
|
||||||
</a>
|
|
||||||
<br>
|
|
||||||
<a href="https://www.linkedin.com/company/la-chariotte-commandes-groupees/"
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener">
|
|
||||||
<i class="fa fa-linkedin-square mr-3" aria-hidden="true"></i>Linkedin
|
|
||||||
</a>
|
|
||||||
<br>
|
|
||||||
<a href="https://mastodon.scop.coop/@la_chariotte"
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener">
|
|
||||||
<i class="fa fa-mastodon-square mr-3" aria-hidden="true"></i>Mastodon
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</footer>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
<script>
|
|
||||||
// For responsive menu
|
// For responsive menu
|
||||||
document.addEventListener('DOMContentLoaded', () => {
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
|
||||||
|
@ -183,4 +201,4 @@
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
{% block extra_js %} {% endblock %}
|
{% block extra_js %} {% endblock %}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
@ -31,7 +31,7 @@
|
||||||
<p>
|
<p>
|
||||||
La Chariotte est développée sous licence libre Affero GPL.
|
La Chariotte est développée sous licence libre Affero GPL.
|
||||||
Cela signifie que vous pouvez
|
Cela signifie que vous pouvez
|
||||||
<a href="{{ gitlab }}" rel="noopener" target="_blank">aller voir le code</a> si vous êtes curieux, et y contribuer si vous êtes motivé·e !
|
<a href="{{ gitlab }}" rel="noopener">aller voir le code</a> si vous êtes curieux, et y contribuer si vous êtes motivé·e !
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -45,7 +45,7 @@
|
||||||
Nous sommes très, très preneurs de vos suggestions, critiques constructives et demandes, et ouvert·e·s à toute question !
|
Nous sommes très, très preneurs de vos suggestions, critiques constructives et demandes, et ouvert·e·s à toute question !
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
Faites des retours sur la version actuelle <a href="https://framaforms.org/vos-retours-sur-la-chariotte-1692345453" rel="noopener" target="_blank">ici</a>, ou par mail à <a href="mailto:{{ mail }}">{{ mail }}</a>.
|
Faites des retours sur la version actuelle <a href="https://framaforms.org/vos-retours-sur-la-chariotte-1692345453" rel="noopener">ici</a>, ou par mail à <a href="mailto:{{ mail }}">{{ mail }}</a>.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
241
la_chariotte/templates/help/faq.html
Normal file
|
@ -0,0 +1,241 @@
|
||||||
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
|
{% block title %}FAQ{% endblock %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
{% load static %}
|
||||||
|
{% load settings %}
|
||||||
|
|
||||||
|
<p class="desktop-hidden mobile-content-title">
|
||||||
|
{% block content_title %}La Chariotte - Facile A Qomprendre{% endblock %}
|
||||||
|
</p>
|
||||||
|
<div class="box">
|
||||||
|
<div class="accordion">
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>C'est quoi l'outil la Chariotte ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
La Chariotte c'est une application web qui permet de simplifier la gestion et l'organisation de commandes groupées.
|
||||||
|
Voyez ça comme un excel amélioré spécifique aux besoins de l'organisation d'une commande groupée.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>C'est quoi la Chariotte plus largement ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
C'est avant tout un logiciel libre dont vous pouvez utiliser l'instance officielle sur chariotte.fr.
|
||||||
|
C'est aussi le nom de l'association responsable de la maintenance de ce logiciel. Tous bénévoles,
|
||||||
|
les membres de la Chariotte font vivre et améliorent continuellement l'outil pour répondre à vos besoins.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Est-ce que je peux rejoindre une commande commande créée par une personne que je ne connais pas ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
C'est prévu a long terme mais pour l'instant non ! Aujourd'hui la Chariotte n'a pas pour visée de
|
||||||
|
mettre en relation des organisateurs/producteurs et des participants. Il n'est donc pas
|
||||||
|
possible de trouver une liste de commandes groupées existantes que vous pourriez rejoindre à volonté.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Faut-il faire partie de l'asso pour utiliser la chariotte ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
Non !
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Puis-je rejoindre l'association ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
Bien sûr et avec grand plaisir !
|
||||||
|
Si vous êtes intéressés pour rejoindre cette aventure,
|
||||||
|
contactez-nous sur <a href="mailto:{% settings_value "CONTACT_MAIL" %}">contact@chariotte.fr</a>.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>C'est quoi un logiciel libre ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
La Chariotte est développée sous licence libre Affero GPL.
|
||||||
|
Cela signifie que vous pouvez <a href="{% settings_value "GITLAB_LINK" %}">aller voir le code</a>
|
||||||
|
si vous êtes curieux, y contribuer si vous êtes motivé·e, ou même créer votre propre instance si vous le souhaitez !
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>C'est quoi une commande groupée ? Ça s'organise comment ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
C'est la mise en relation ponctuelle d'un producteur, ou plutôt ses produits,
|
||||||
|
avec des particuliers, pour permettre une consommation en circuit court et hors
|
||||||
|
des modes de consommation de masse conventionnels.
|
||||||
|
<br><br>
|
||||||
|
Pour faire simple, imaginons que vous connaissiez l'apicultrice du coin et que plusieurs de
|
||||||
|
vos voisins vous aient dit être prêts à acheter du miel.
|
||||||
|
Vous pouvez alors commander 40 pots de miel à votre amie et les redistribuer à tout le quartier,
|
||||||
|
tout ça en un seul voyage !
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Quelle est la différence avec une AMAP ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
Dans les faits, la plupart des AMAPs fonctionnent sur la base de commandes
|
||||||
|
groupées régulières et à destination d'un public d'adhérents fixe.<br>
|
||||||
|
La Chariotte n'a pas vocation à remplacer le système des AMAPs, qui apporte un revenu régulier aux producteur.ice.s !
|
||||||
|
L'idée est d'apporter une solution complémentaire pour les endroits où les AMAPs ne sont pas établies, ou pour un produit en particulier qui n'est pas distribué régulièrement dans votre AMAP.
|
||||||
|
Il arrive aussi parfois que la Chariotte soit choisie comme outil de gestion pour l'AMAP si elle n'en avait pas encore.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Quel est l'avantage de la Chariotte par rapport au logiciel de mon AMAP ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
Nous avons conçu la Chariotte spécifiquement pour répondre aux besoins des
|
||||||
|
organisateur.ice.s et des participant.e.s à une commande groupée, en particulier des commandes groupées ponctuelles.
|
||||||
|
Il s'agit d'un outil spécialisé dont la seule ambition est de simplifier la vie de ses utilisateurs.
|
||||||
|
Libre à vous d'évaluer vos options et de choisir l'outil qui convient le mieux à votre organisation !
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Je veux organiser une commande groupée, où est-ce que je trouve des producteurs ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
La Chariotte ne fournit pour l'instant pas de liste de producteurs ;
|
||||||
|
c'est une bonne occasion de faire connaissance avec ceux près de chez vous !
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Suis-je obligé de créer un compte pour commander ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
Même pas ! La création d'un compte n'est obligatoire que pour les utilisateurs souhaitant organiser
|
||||||
|
leurs propres commandes groupées. La participation à une commande existante
|
||||||
|
se veut à destination du plus grand nombre et ne nécessite donc pas de compte sur le site.
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
<article class="message">
|
||||||
|
<div class="message-header">
|
||||||
|
<p>Comment puis-je soutenir la Chariotte ?</p>
|
||||||
|
<a>
|
||||||
|
<span class="icon">
|
||||||
|
<i class="fa fa-angle-down" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="message-body">
|
||||||
|
<p>Nous sommes une jeune association, et nous avons besoin de trois choses : de la visibilité,
|
||||||
|
des retours d'utilisateurs et un peu d'argent !</li>
|
||||||
|
<ul><li>Vous pouvez nous rejoindre sur les réseaux (Facebook, Mastodon, LinkedIn) pour aimer
|
||||||
|
et partager nos posts, ou encore parler de la Chariotte à vos amis !</li>
|
||||||
|
<li>Si vous saisissez une occasion d'utiliser la Chariotte, nous avons besoin de vous !
|
||||||
|
N'hésitez pas à nous faire un rapport de votre expérience de la Chariotte en pointant
|
||||||
|
les bons et les mauvais côtés de l'application. Par exemple en remplissant
|
||||||
|
<a href="{% settings_value "FEEDBACK_LINK" %}">ce formulaire</a> !</li>
|
||||||
|
<li>Si le projet vous parle et que vous avez envie de nous soutenir de manière plus significative,
|
||||||
|
nous avons besoin d'un peu d'argent de poche pour soutenir les frais de l'association ;
|
||||||
|
<a href ="{% settings_value "HELLOASSO_LINK" %}">par ici</a> pour en savoir plus !</li>
|
||||||
|
</ul>
|
||||||
|
<p>Et enfin, si vous souhaitez en savoir plus sur l'association en elle-même et que vous
|
||||||
|
souhaitez faire partie intégrante du projet, la Chariotte est à la recherche de bénévoles motivé·e·s !
|
||||||
|
N'hésitez pas à nous écrire un mot sur contact@chariotte.fr pour en savoir plus et rentrer en contact directement avec nous.</p>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
<script>
|
||||||
|
{% block extra_js %}
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
var acc = document.getElementsByClassName("message-header");
|
||||||
|
var i;
|
||||||
|
|
||||||
|
for (i = 0; i < acc.length; i++) {
|
||||||
|
acc[i].addEventListener("click", function() {
|
||||||
|
/* Toggle between adding and removing the "active" class,
|
||||||
|
to highlight the button that controls the panel */
|
||||||
|
this.classList.toggle("active");
|
||||||
|
|
||||||
|
/* Toggle between hiding and showing the active panel */
|
||||||
|
var panel = this.nextElementSibling;
|
||||||
|
if (panel.style.display === "block") {
|
||||||
|
panel.style.display = "none";
|
||||||
|
} else {
|
||||||
|
panel.style.display = "block";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
{% endblock %}
|
||||||
|
</script>
|
|
@ -34,7 +34,7 @@
|
||||||
<p class="title">Licence</p>
|
<p class="title">Licence</p>
|
||||||
<p>
|
<p>
|
||||||
La Chariotte est développée sous licence libre <strong>Affero GPL</strong>. Le code source est
|
La Chariotte est développée sous licence libre <strong>Affero GPL</strong>. Le code source est
|
||||||
consultable <a href="{{ gitlab }}" rel="noopener" target="_blank">sur Gitlab</a> et ouvert à toute contribution ou suggestion.
|
consultable <a href="{{ gitlab }}" rel="noopener">sur Gitlab</a> et ouvert à toute contribution ou suggestion.
|
||||||
</p>
|
</p>
|
||||||
<p class="title">Conditions d'utilisation</p>
|
<p class="title">Conditions d'utilisation</p>
|
||||||
<p>
|
<p>
|
||||||
|
|
|
@ -9,11 +9,126 @@
|
||||||
{% block content_title %}La Chariotte - mode d'emploi{% endblock %}
|
{% block content_title %}La Chariotte - mode d'emploi{% endblock %}
|
||||||
</p>
|
</p>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<p>Voilà un petit dessin pour vous expliquer le déroulé d'une commande groupée avec la Chariotte.</p>
|
<p>Comment marche la Chariotte et qu'est-ce qu'on peut y faire ? Voilà comment vous pouvez :</p>
|
||||||
<figure class="image">
|
<ul>
|
||||||
<img src="{% static 'img/notice_1.jpg' %}"/>
|
<li><a href="#join_grouped_order">Rejoindre une commande et commander des produits</a></li>
|
||||||
<img src="{% static 'img/notice_2.jpg' %}"/>
|
<li><a href="#pay">Payer (pas encore !)</a></li>
|
||||||
<img src="{% static 'img/notice_3.jpg' %}"/>
|
<li><a href="#create_account">Créer un compte (nécessaire seulement pour organiser une commande groupée)</a></li>
|
||||||
</figure>
|
<li><a href="#create_grouped_order">Créer une commande groupée</a></li>
|
||||||
|
<li><a href="#add_item">Ajouter des produits à une commande groupée</a></li>
|
||||||
|
<li><a href="#go_to_overview">Accéder à la page de gestion de ma commande groupée</a></li>
|
||||||
|
<li><a href="#delete_item">Modifier des produits d'une commande après sa création</a></li>
|
||||||
|
<li><a href="#edit_grouped_order">Modifier des informations de la commande</a></li>
|
||||||
|
<li><a href="#share_grouped_order">Partager une commande à des participants</a></li>
|
||||||
|
<li><a href="#print_orders_list">Imprimer ma liste de commandes</a></li>
|
||||||
|
<li><a href="#get_spreadsheet">Voir les commandes sous forme de tableur</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="join_grouped_order">
|
||||||
|
<p class="title">Rejoindre une commande et commander des produits</p>
|
||||||
|
<p>Pour rejoindre une commande groupée, il faut y avoir été invité.e par son organisateur.ice.
|
||||||
|
Si vous avez un code à 6 caractères, vous pouvez l'entrer dans l'encadré "Rejoindre une commande groupée"
|
||||||
|
sur la page d'accueil de la Chariotte.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/join_grouped_order.png' %}"/>
|
||||||
|
<p>Sinon, cliquez sur le lien que l'organisateur.ice vous a envoyé. Vous arriverez sur la page de commande.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/order_form.png' %}"/>
|
||||||
|
<p>Une fois la commande validée, vous recevrez un mail de confirmation.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="pay">
|
||||||
|
<p class="title">Payer (pas encore !)</p>
|
||||||
|
<p>Pour l’instant, le paiement n’est pas pris en compte par le site de La Chariotte,
|
||||||
|
c’est l’organisateur.ice qui doit vous indiquer les modalités de paiement.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="create_account">
|
||||||
|
<p class="title">Créer un compte (nécessaire seulement pour organiser une commande groupée)</p>
|
||||||
|
<p>1. Cliquez sur “Créer un compte”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/create_account.jpg' %}"/>
|
||||||
|
<p>2. Remplissez toutes les informations du formulaire, puis cliquez sur valider.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/create_account_form.jpg' %}"/>
|
||||||
|
<p>3. Vous pouvez maintenant vous connecter.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="create_grouped_order">
|
||||||
|
<p class="title">Créer une commande groupée</p>
|
||||||
|
<p>1. En ayant un compte (nécessaire seulement pour les organisateurs!), vous aurez accès à une page de gestion des commandes groupées en cliquant sur le menu “Mes commandes groupées”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/my_grouped_orders.jpg' %}"/>
|
||||||
|
<p>2. C’est depuis cette page que vous pourrez créer une nouvelle commande en cliquant sur le bouton “Créer une nouvelle commande groupée” : </p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/create_grouped_order.jpg' %}"/>
|
||||||
|
<p>3. Quelques informations vous seront alors demandées :</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/create_grouped_order_form.jpg' %}"/>
|
||||||
|
<p>4. En cliquant sur "Suivant", votre commande groupée sera validée.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="add_item">
|
||||||
|
<p class="title">Ajouter des produits à une commande groupée</p>
|
||||||
|
<p>Directement après avoir validé la commande groupée, vous accédez à la page de gestion des produits.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/add_item.png' %}"/>
|
||||||
|
<p>Ici, vous pouvez renseigner les produits qui constitueront votre commande groupée.
|
||||||
|
Indiquez le nom du produit, son prix et la quantité disponible.
|
||||||
|
Cliquez sur “Ajouter” pour valider l’ajout.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/add_item_2.png' %}"/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="go_to_overview">
|
||||||
|
<p class="title">Accéder à la page de gestion de ma commande groupée</p>
|
||||||
|
<p>Depuis le menu “Mes commandes groupées”, puis en cliquant sur le nom de la commande groupée
|
||||||
|
que vous voulez gérer, vous pourrez cliquer ensuite sur “Gestion de la commande groupee”.
|
||||||
|
</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/go_to_overview.png' %}"/>
|
||||||
|
<p>Sinon depuis la page “Mes commandes groupées” il est aussi possible de cliquer directement sur
|
||||||
|
l’icone de la ligne "liste" de la commande groupée que vous voulez gérer.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="delete_item">
|
||||||
|
<p class="title">Modifier des produits d'une commande après sa création</p>
|
||||||
|
<p>1. Si vous souhaitez modifier une information concernant la commande (titre, date, description, lieu de livraison, …),
|
||||||
|
il faut d’abord <a href="#go_to_overview">accéder à la page de gestion de la commande groupée</a>.</p>
|
||||||
|
<p>2. Ensuite, dans "Résumé de la commande”, cliquez sur “Gérer les produits”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/grouped_order_sum_up.png' %}"/>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/items_overview.png' %}"/>
|
||||||
|
<p>3. Vous êtes maintenant sur la page de gestion des produits depuis laquelle vous pouvez ajouter ou supprimer des produits.
|
||||||
|
Cliquez sur “Valider” pour sauvegarder les modifications.</p>
|
||||||
|
<p>Attention : actuellement, il n'est pas possible de supprimer un produit une fois qu'il a été commandé par un.e participant.e.
|
||||||
|
Si vous voulez vraiment le supprimer, il faudra d'abord supprimer les commandes des participant.e.s concerné.e.s.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="edit_grouped_order">
|
||||||
|
<p class="title">Modifier des informations de la commande</p>
|
||||||
|
<p>1. Si vous souhaitez modifier une information concernant la commande (titre, date, description, lieu de livraison, …),
|
||||||
|
il faut d’abord <a href="#go_to_overview">accéder à la page de gestion de la commande groupée</a>.</p>
|
||||||
|
<p>2. Vous pouvez ensuite cliquer sur le bouton “Modifier la commande”
|
||||||
|
qui vous amenera vers le formulaire pre-rempli des informations de la commande. </p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/edit_grouped_order.png' %}"/>
|
||||||
|
<p>3. Modifiez les informations que vous voulez, puis cliquez sur “Suivant” pour valider.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="share_grouped_order">
|
||||||
|
<p class="title">Partager une commande à des participants</p>
|
||||||
|
<p>Une fois votre commande bien préparée (quantité et prix des produits corrects, informations de la commande justes, …),
|
||||||
|
il faut maintenant la partager à des participants pour qu’ils puissent commander.</p>
|
||||||
|
<p>1. Depuis la page de <a href="#go_to_overview">gestion de la commande</a>,
|
||||||
|
allez dans l’onglet “partager et exporter”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/share_grouped_order.png' %}"/>
|
||||||
|
<p>2. Vous pouvez copier le lien, et l’envoyer à vos connaissances.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/copy_link.png' %}"/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="print_orders_list">
|
||||||
|
<p class="title">Imprimer ma liste de commandes</p>
|
||||||
|
<p>1. Pour cela, il faut d’abord <a href="#go_to_overview">accéder à la page de gestion de la commande groupée</a>.</p>
|
||||||
|
<p>2. Ensuite, dans "Partager et exporter”, cliquez sur “Commandes en PDF”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/print_orders_list.png' %}"/>
|
||||||
|
<p>3. Cela lance le téléchargement de votre liste de commandes en PDF.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="box" id="get_spreadsheet">
|
||||||
|
<p class="title">Voir les commandes sous forme de tableur</p>
|
||||||
|
<p>1. Pour cela, il faut d’abord <a href="#go_to_overview">accéder à la page de gestion de la commande groupée</a>.</p>
|
||||||
|
<p>2. Ensuite, dans "Partager et exporter”, cliquez sur “Commandes en CSV”.</p>
|
||||||
|
<img class="notice-img" src="{% static 'img/notice/get_spreadsheet.png' %}"/>
|
||||||
|
<p>3. Le format CSV vous permet d'ouvrir le fichier avec le tableur de votre choix (libreoffice, excel, ...).</p>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
0
la_chariotte/tests/__init__.py
Normal file
|
@ -19,7 +19,7 @@ def test_send_order_confirmation_mail(mailoutbox, simple_grouped_order, client):
|
||||||
"code": simple_grouped_order.code,
|
"code": simple_grouped_order.code,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
response = client.post(
|
client.post(
|
||||||
order_url,
|
order_url,
|
||||||
{
|
{
|
||||||
f"quantity_{item.pk}": [4, 0],
|
f"quantity_{item.pk}": [4, 0],
|
|
@ -1,7 +1,7 @@
|
||||||
import pytest
|
import pytest
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
|
||||||
from la_chariotte.order.tests.utils import create_grouped_order
|
from .utils import create_grouped_order
|
||||||
|
|
||||||
pytestmark = pytest.mark.django_db
|
pytestmark = pytest.mark.django_db
|
||||||
|
|
|
@ -6,7 +6,8 @@ from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from la_chariotte.order.models import GroupedOrder, Item
|
from la_chariotte.order.models import GroupedOrder, Item
|
||||||
from la_chariotte.order.tests.utils import create_grouped_order
|
|
||||||
|
from .utils import create_grouped_order
|
||||||
|
|
||||||
pytestmark = pytest.mark.django_db
|
pytestmark = pytest.mark.django_db
|
||||||
|
|
|
@ -4,16 +4,14 @@ from io import StringIO
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from django.contrib import auth
|
from django.contrib import auth
|
||||||
|
from django.contrib.auth import get_user
|
||||||
from django.forms.utils import to_current_timezone
|
from django.forms.utils import to_current_timezone
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from icalendar import Calendar, vText
|
from icalendar import Calendar, vText
|
||||||
|
|
||||||
from la_chariotte.order import models
|
from la_chariotte.order import models
|
||||||
from la_chariotte.order.tests.utils import (
|
from la_chariotte.tests.utils import create_grouped_order, order_items_in_grouped_order
|
||||||
create_grouped_order,
|
|
||||||
order_items_in_grouped_order,
|
|
||||||
)
|
|
||||||
|
|
||||||
pytestmark = pytest.mark.django_db
|
pytestmark = pytest.mark.django_db
|
||||||
|
|
||||||
|
@ -217,11 +215,32 @@ class TestJoinGroupedOrderView:
|
||||||
response = client.post(join_url, {"code": "123456"})
|
response = client.post(join_url, {"code": "123456"})
|
||||||
|
|
||||||
assert (
|
assert (
|
||||||
"nous ne trouvons aucune commande avec ce code" in response.content.decode()
|
"Désolé, nous ne trouvons aucune commande avec ce code"
|
||||||
|
in response.content.decode()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class TestGroupedOrderDetailView:
|
class TestGroupedOrderDetailView:
|
||||||
|
def test_order_item_with_authenticated_user(self, client, connected_grouped_order):
|
||||||
|
detail_url = reverse(
|
||||||
|
"order:grouped_order_detail",
|
||||||
|
kwargs={
|
||||||
|
"code": connected_grouped_order.code,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
response = client.get(detail_url)
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
order_author = response.context[0]["order_author"]
|
||||||
|
current_user = get_user(client)
|
||||||
|
|
||||||
|
assert order_author is not None
|
||||||
|
|
||||||
|
assert order_author.first_name == current_user.first_name
|
||||||
|
assert order_author.email == current_user.username
|
||||||
|
assert order_author.last_name == current_user.last_name
|
||||||
|
|
||||||
def test_order_item(self, client, other_user):
|
def test_order_item(self, client, other_user):
|
||||||
"""
|
"""
|
||||||
From the OrderDetailView, we order an item using the order form, and it creates an models.Order with an Ordered_item inside
|
From the OrderDetailView, we order an item using the order form, and it creates an models.Order with an Ordered_item inside
|
||||||
|
@ -302,7 +321,9 @@ class TestGroupedOrderDetailView:
|
||||||
# OrderedItems are not created when the ordered quantity is 0
|
# OrderedItems are not created when the ordered quantity is 0
|
||||||
assert models.OrderedItem.objects.count() == 2
|
assert models.OrderedItem.objects.count() == 2
|
||||||
|
|
||||||
def test_order_item__no_articles_ordered(self, client, other_user):
|
def test_order_item__no_articles_ordered(
|
||||||
|
self, client, authenticated_user_with_name
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
From the OrderDetailView, we order without having changed any item quantity.
|
From the OrderDetailView, we order without having changed any item quantity.
|
||||||
An error is raised.
|
An error is raised.
|
||||||
|
@ -312,7 +333,7 @@ class TestGroupedOrderDetailView:
|
||||||
days_before_delivery_date=5,
|
days_before_delivery_date=5,
|
||||||
days_before_deadline=2,
|
days_before_deadline=2,
|
||||||
name="gr order test",
|
name="gr order test",
|
||||||
orga_user=other_user,
|
orga_user=authenticated_user_with_name,
|
||||||
)
|
)
|
||||||
item = models.Item.objects.create(
|
item = models.Item.objects.create(
|
||||||
name="test item 1", grouped_order=grouped_order, price=1
|
name="test item 1", grouped_order=grouped_order, price=1
|
||||||
|
@ -338,10 +359,10 @@ class TestGroupedOrderDetailView:
|
||||||
order_url,
|
order_url,
|
||||||
{
|
{
|
||||||
f"quantity_{item.pk}": [0, 0],
|
f"quantity_{item.pk}": [0, 0],
|
||||||
"first_name": "Prénom test",
|
"first_name": {authenticated_user_with_name.first_name},
|
||||||
"last_name": "Nom test",
|
"last_name": {authenticated_user_with_name.last_name},
|
||||||
"phone": "0645632569",
|
"phone": "0645632569",
|
||||||
"email": "test@mail.fr",
|
"email": {authenticated_user_with_name.email},
|
||||||
"note": "test note",
|
"note": "test note",
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
@ -350,12 +371,14 @@ class TestGroupedOrderDetailView:
|
||||||
response.context["error_message"]
|
response.context["error_message"]
|
||||||
== "Veuillez commander au moins un produit"
|
== "Veuillez commander au moins un produit"
|
||||||
)
|
)
|
||||||
assert response.context["author"].first_name == "Prénom test"
|
assert (
|
||||||
|
response.context["author"].first_name
|
||||||
|
== authenticated_user_with_name.first_name
|
||||||
|
)
|
||||||
content = response.content.decode()
|
content = response.content.decode()
|
||||||
assert "Prénom test" in content
|
assert authenticated_user_with_name.first_name in content
|
||||||
assert "Nom test" in content
|
assert authenticated_user_with_name.last_name in content
|
||||||
assert "0645632569" in content
|
assert authenticated_user_with_name.email in content
|
||||||
assert "test@mail.fr" in content
|
|
||||||
assert "test note" in content
|
assert "test note" in content
|
||||||
assert not models.Order.objects.first()
|
assert not models.Order.objects.first()
|
||||||
assert not models.OrderAuthor.objects.first()
|
assert not models.OrderAuthor.objects.first()
|
||||||
|
@ -639,6 +662,36 @@ class TestGroupedOrderDetailView:
|
||||||
assert order.price == 4
|
assert order.price == 4
|
||||||
assert item.get_remaining_nb() == 16
|
assert item.get_remaining_nb() == 16
|
||||||
|
|
||||||
|
def test_phone_not_required_display(self, client, other_user):
|
||||||
|
"""a user orders something without entering phone when it is required"""
|
||||||
|
grouped_order = create_grouped_order(
|
||||||
|
days_before_delivery_date=5,
|
||||||
|
days_before_deadline=2,
|
||||||
|
name="gr order test",
|
||||||
|
orga_user=other_user,
|
||||||
|
)
|
||||||
|
assert grouped_order.is_phone_mandatory == True
|
||||||
|
item = models.Item.objects.create(
|
||||||
|
name="test item 1", grouped_order=grouped_order, price=1, max_limit=2
|
||||||
|
)
|
||||||
|
detail_url = reverse(
|
||||||
|
"order:grouped_order_detail",
|
||||||
|
kwargs={
|
||||||
|
"code": grouped_order.code,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
response = client.get(detail_url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert "gr order test" in response.content.decode()
|
||||||
|
assert (
|
||||||
|
"Numéro de téléphone <em>(facultatif)</em>" not in response.content.decode()
|
||||||
|
)
|
||||||
|
grouped_order.is_phone_mandatory = False
|
||||||
|
grouped_order.save()
|
||||||
|
response = client.get(detail_url)
|
||||||
|
assert "gr order test" in response.content.decode()
|
||||||
|
assert "Numéro de téléphone <em>(facultatif)</em>" in response.content.decode()
|
||||||
|
|
||||||
|
|
||||||
class TestGroupedOrderOverview:
|
class TestGroupedOrderOverview:
|
||||||
def test_get_overview(self, client_log):
|
def test_get_overview(self, client_log):
|
||||||
|
@ -683,6 +736,7 @@ class TestGroupedOrderOverview:
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert "test item" in response.content.decode()
|
assert "test item" in response.content.decode()
|
||||||
assert {"test@mail.fr"} == response.context["emails_list"]
|
assert {"test@mail.fr"} == response.context["emails_list"]
|
||||||
|
assert 4 == response.context["total_ordered_items"]
|
||||||
assert "gr order test" in response.content.decode()
|
assert "gr order test" in response.content.decode()
|
||||||
item.refresh_from_db()
|
item.refresh_from_db()
|
||||||
assert item.get_total_price() == 8
|
assert item.get_total_price() == 8
|
||||||
|
@ -737,6 +791,25 @@ class TestGroupedOrderOverview:
|
||||||
response = client_log.get(orga_view_url)
|
response = client_log.get(orga_view_url)
|
||||||
assert response.status_code == 403
|
assert response.status_code == 403
|
||||||
|
|
||||||
|
def test_superuser_can_access_overview(self, other_user, admin_client):
|
||||||
|
"""
|
||||||
|
A superuser that is not orga can get the GroupedOrderOverview
|
||||||
|
"""
|
||||||
|
grouped_order = create_grouped_order(
|
||||||
|
days_before_delivery_date=5,
|
||||||
|
days_before_deadline=2,
|
||||||
|
name="gr order test",
|
||||||
|
orga_user=other_user,
|
||||||
|
)
|
||||||
|
orga_view_url = reverse(
|
||||||
|
"order:grouped_order_overview",
|
||||||
|
kwargs={
|
||||||
|
"code": grouped_order.code,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
response = admin_client.get(orga_view_url)
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
def test_deadline_passed(self, client_log):
|
def test_deadline_passed(self, client_log):
|
||||||
"""
|
"""
|
||||||
If the deadline is passed, the user sees a message but cannot share link
|
If the deadline is passed, the user sees a message but cannot share link
|
||||||
|
@ -1353,7 +1426,7 @@ class TestGroupedOrderSheetView:
|
||||||
response = client_log.get(generate_sheet_url)
|
response = client_log.get(generate_sheet_url)
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response.context["grouped_order"] == grouped_order
|
assert response.context["grouped_order"] == grouped_order
|
||||||
assert response.context["items"].count() == 0
|
assert len(response.context["items"]) == 0
|
||||||
assert len(response.context["orders_dict"]) == 0
|
assert len(response.context["orders_dict"]) == 0
|
||||||
|
|
||||||
# we order some items in the grouped order
|
# we order some items in the grouped order
|
||||||
|
@ -1361,7 +1434,7 @@ class TestGroupedOrderSheetView:
|
||||||
response = client_log.get(generate_sheet_url)
|
response = client_log.get(generate_sheet_url)
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response.context["grouped_order"] == grouped_order
|
assert response.context["grouped_order"] == grouped_order
|
||||||
assert response.context["items"].count() == 2
|
assert len(response.context["items"]) == 2
|
||||||
assert response.context["orders_dict"][order] == [3, 2]
|
assert response.context["orders_dict"][order] == [3, 2]
|
||||||
assert response.context["grouped_order"].total_price == 35
|
assert response.context["grouped_order"].total_price == 35
|
||||||
|
|
||||||
|
@ -1479,8 +1552,7 @@ class TestExportGroupOrderEmailAdressesToDownloadView:
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response["Content-Type"] == "text/csv"
|
assert response["Content-Type"] == "text/csv"
|
||||||
content = response.content.decode()
|
content = response.content.decode()
|
||||||
assert "order_name,email" in content
|
assert "test@mail.fr\r\n" in content
|
||||||
assert "order_name,email\r\ngr order test,test@mail.fr\r\n" in content
|
|
||||||
|
|
||||||
def test_export_format_default(self, client_log):
|
def test_export_format_default(self, client_log):
|
||||||
grouped_order = create_grouped_order(
|
grouped_order = create_grouped_order(
|
||||||
|
@ -1587,8 +1659,10 @@ class TestExportGroupedOrderToCSVView:
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
|
|
||||||
content = response.content.decode("utf-8")
|
content = response.content.decode("utf-8")
|
||||||
csv_reader = csv.reader(StringIO(content))
|
csv_reader = csv.reader(StringIO(content), delimiter=";")
|
||||||
body = list(csv_reader)
|
body = list(csv_reader)
|
||||||
|
created_date = f"{timezone.now().strftime('%d/%m/%Y')}"
|
||||||
|
created_time = f"{timezone.now().strftime('%H:%M')}"
|
||||||
assert body == [
|
assert body == [
|
||||||
[
|
[
|
||||||
"",
|
"",
|
||||||
|
@ -1598,11 +1672,47 @@ class TestExportGroupedOrderToCSVView:
|
||||||
"Prix de la commande",
|
"Prix de la commande",
|
||||||
"Mail",
|
"Mail",
|
||||||
"Téléphone",
|
"Téléphone",
|
||||||
|
"Note",
|
||||||
|
"Date",
|
||||||
|
"Heure",
|
||||||
],
|
],
|
||||||
["", "Prix unitaire TTC (€)", "2,00", "9,00"],
|
["", "Prix unitaire TTC (€)", "2,00", "9,00"],
|
||||||
["Nom", "Prénom"],
|
["Nom", "Prénom"],
|
||||||
["alescargot", "bob", "1", "0", "2,00", "bob2@escargot.fr", "'000"],
|
[
|
||||||
["alescargot", "bobby", "0", "1", "9,00", "bob3@escargot.fr", "'000"],
|
"alescargot",
|
||||||
["lescargot", "bob", "3", "2", "24,00", "bob@escargot.fr", "'000"],
|
"bob",
|
||||||
|
"1",
|
||||||
|
"0",
|
||||||
|
"2,00",
|
||||||
|
"bob2@escargot.fr",
|
||||||
|
"'000",
|
||||||
|
"",
|
||||||
|
created_date,
|
||||||
|
created_time,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"alescargot",
|
||||||
|
"bobby",
|
||||||
|
"0",
|
||||||
|
"1",
|
||||||
|
"9,00",
|
||||||
|
"bob3@escargot.fr",
|
||||||
|
"'000",
|
||||||
|
"",
|
||||||
|
created_date,
|
||||||
|
created_time,
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"lescargot",
|
||||||
|
"bob",
|
||||||
|
"3",
|
||||||
|
"2",
|
||||||
|
"24,00",
|
||||||
|
"bob@escargot.fr",
|
||||||
|
"'000",
|
||||||
|
"",
|
||||||
|
created_date,
|
||||||
|
created_time,
|
||||||
|
],
|
||||||
["", "TOTAL", "4", "3", "35,00"],
|
["", "TOTAL", "4", "3", "35,00"],
|
||||||
]
|
]
|
|
@ -3,7 +3,8 @@ from django.contrib import auth
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
|
||||||
from la_chariotte.order import models
|
from la_chariotte.order import models
|
||||||
from la_chariotte.order.tests.utils import create_grouped_order
|
|
||||||
|
from .utils import create_grouped_order
|
||||||
|
|
||||||
pytestmark = pytest.mark.django_db
|
pytestmark = pytest.mark.django_db
|
||||||
|
|
|
@ -3,10 +3,8 @@ from django.contrib import auth
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
|
||||||
from la_chariotte.order import models
|
from la_chariotte.order import models
|
||||||
from la_chariotte.order.tests.utils import (
|
|
||||||
create_grouped_order,
|
from .utils import create_grouped_order, order_items_in_grouped_order
|
||||||
order_items_in_grouped_order,
|
|
||||||
)
|
|
||||||
|
|
||||||
pytestmark = pytest.mark.django_db
|
pytestmark = pytest.mark.django_db
|
||||||
|
|
38
la_chariotte/tests/test_views_stats.py
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
from django.contrib import auth
|
||||||
|
from django.urls import reverse
|
||||||
|
|
||||||
|
from .utils import create_grouped_order, order_items_in_grouped_order
|
||||||
|
|
||||||
|
|
||||||
|
def test_stats(client_log):
|
||||||
|
response = client_log.get(reverse("stats"))
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert response.json() == {
|
||||||
|
"total_grouped_orders": 0,
|
||||||
|
"total_orders": 0,
|
||||||
|
"total_users": 1,
|
||||||
|
"last_month_grouped_orders": 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
# Create a new grouped-order, place an order and check that the stats changed.
|
||||||
|
grouped_order = create_grouped_order(
|
||||||
|
days_before_delivery_date=5,
|
||||||
|
days_before_deadline=2,
|
||||||
|
name="gr order test",
|
||||||
|
orga_user=auth.get_user(client_log),
|
||||||
|
)
|
||||||
|
order_items_in_grouped_order(grouped_order)
|
||||||
|
|
||||||
|
grouped_order.deadline = datetime.now() - timedelta(days=10)
|
||||||
|
grouped_order.save()
|
||||||
|
|
||||||
|
response = client_log.get(reverse("stats"))
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert response.json() == {
|
||||||
|
"total_grouped_orders": 1,
|
||||||
|
"total_orders": 3,
|
||||||
|
"total_users": 1,
|
||||||
|
"last_month_grouped_orders": 1,
|
||||||
|
}
|
|
@ -9,12 +9,20 @@ pytestmark = pytest.mark.django_db
|
||||||
|
|
||||||
|
|
||||||
def create_grouped_order(
|
def create_grouped_order(
|
||||||
days_before_delivery_date, days_before_deadline, name, orga_user
|
days_before_delivery_date,
|
||||||
|
days_before_deadline,
|
||||||
|
name,
|
||||||
|
orga_user,
|
||||||
|
is_phone_mandatory=True,
|
||||||
):
|
):
|
||||||
date = timezone.now().date() + datetime.timedelta(days=days_before_delivery_date)
|
date = timezone.now().date() + datetime.timedelta(days=days_before_delivery_date)
|
||||||
deadline = timezone.now() + datetime.timedelta(days=days_before_deadline)
|
deadline = timezone.now() + datetime.timedelta(days=days_before_deadline)
|
||||||
grouped_order = models.GroupedOrder.objects.create(
|
grouped_order = models.GroupedOrder.objects.create(
|
||||||
name=name, orga=orga_user, delivery_date=date, deadline=deadline
|
name=name,
|
||||||
|
orga=orga_user,
|
||||||
|
delivery_date=date,
|
||||||
|
deadline=deadline,
|
||||||
|
is_phone_mandatory=is_phone_mandatory,
|
||||||
)
|
)
|
||||||
grouped_order.create_code_from_pk()
|
grouped_order.create_code_from_pk()
|
||||||
grouped_order.save()
|
grouped_order.save()
|
||||||
|
@ -44,11 +52,4 @@ def order_items_in_grouped_order(grouped_order):
|
||||||
models.OrderedItem.objects.create(order=order, item=item_2, nb=2)
|
models.OrderedItem.objects.create(order=order, item=item_2, nb=2)
|
||||||
models.OrderedItem.objects.create(order=order_2, item=item_1, nb=1)
|
models.OrderedItem.objects.create(order=order_2, item=item_1, nb=1)
|
||||||
models.OrderedItem.objects.create(order=order_3, item=item_2, nb=1)
|
models.OrderedItem.objects.create(order=order_3, item=item_2, nb=1)
|
||||||
item_1.compute_ordered_nb()
|
|
||||||
item_2.compute_ordered_nb()
|
|
||||||
order.compute_order_price()
|
|
||||||
order_2.compute_order_price()
|
|
||||||
order_3.compute_order_price()
|
|
||||||
grouped_order.compute_total_price()
|
|
||||||
grouped_order.save()
|
|
||||||
return order
|
return order
|
|
@ -25,6 +25,7 @@ from django.views.generic.base import TemplateView
|
||||||
|
|
||||||
from la_chariotte import settings
|
from la_chariotte import settings
|
||||||
from la_chariotte.order.views import JoinGroupedOrderView
|
from la_chariotte.order.views import JoinGroupedOrderView
|
||||||
|
from la_chariotte.order.views.stats import stats
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path("admin/", admin.site.urls),
|
path("admin/", admin.site.urls),
|
||||||
|
@ -61,6 +62,11 @@ urlpatterns = [
|
||||||
TemplateView.as_view(template_name="help/notice.html"),
|
TemplateView.as_view(template_name="help/notice.html"),
|
||||||
name="notice",
|
name="notice",
|
||||||
),
|
),
|
||||||
|
path(
|
||||||
|
"faq",
|
||||||
|
TemplateView.as_view(template_name="help/faq.html"),
|
||||||
|
name="faq",
|
||||||
|
),
|
||||||
path(
|
path(
|
||||||
"mentions-legales",
|
"mentions-legales",
|
||||||
TemplateView.as_view(
|
TemplateView.as_view(
|
||||||
|
@ -83,4 +89,5 @@ urlpatterns = [
|
||||||
),
|
),
|
||||||
name="about_chariotte",
|
name="about_chariotte",
|
||||||
),
|
),
|
||||||
|
path("stats/", stats, name="stats"),
|
||||||
]
|
]
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
site_name: La chariotte
|
site_name: La chariotte
|
||||||
site_description: An application for grouped-orders
|
site_description: An application for grouped-orders
|
||||||
repo_name: la-chariotte/la_chariotte
|
repo_name: la-chariotte/la_chariotte
|
||||||
repo_url: https://gitlab.com/la-chariotte/la_chariotte
|
repo_url: https://framagit.org/la-chariotte/la-chariotte
|
||||||
nav:
|
nav:
|
||||||
- How-tos:
|
- How-tos:
|
||||||
- Getting started: install.md
|
- Getting started: install.md
|
||||||
|
|
|
@ -15,7 +15,7 @@ dependencies = [
|
||||||
"icalendar>=5.0.7,<6",
|
"icalendar>=5.0.7,<6",
|
||||||
"base36>=0.1.1,<1",
|
"base36>=0.1.1,<1",
|
||||||
"django-weasyprint==1.1.0.post1", # see below
|
"django-weasyprint==1.1.0.post1", # see below
|
||||||
"weasyprint==52.5", # pin weasyprint to not depend on latest libpango
|
"weasyprint==52.5", # pin weasyprint to not depend on latest libpango
|
||||||
"html2text>=2020.1.16",
|
"html2text>=2020.1.16",
|
||||||
"django-crispy-forms>=2.0,<3",
|
"django-crispy-forms>=2.0,<3",
|
||||||
"crispy-bulma>=0.11.0,<1",
|
"crispy-bulma>=0.11.0,<1",
|
||||||
|
@ -50,7 +50,7 @@ dev = [
|
||||||
]
|
]
|
||||||
|
|
||||||
[tool.pytest.ini_options]
|
[tool.pytest.ini_options]
|
||||||
DJANGO_SETTINGS_MODULE = "la_chariotte.settings"
|
DJANGO_SETTINGS_MODULE = "local_settings"
|
||||||
addopts = "-x --ff --isort --black --reuse-db --cov-report xml --cov-report term-missing --cov=la_chariotte -p no:warnings"
|
addopts = "-x --ff --isort --black --reuse-db --cov-report xml --cov-report term-missing --cov=la_chariotte -p no:warnings"
|
||||||
isort_ignore = ["*migrations/*.py"]
|
isort_ignore = ["*migrations/*.py"]
|
||||||
|
|
||||||
|
|
|
@ -1,218 +0,0 @@
|
||||||
#
|
|
||||||
# This file is autogenerated by pip-compile with Python 3.11
|
|
||||||
# by the following command:
|
|
||||||
#
|
|
||||||
# pip-compile --extra=dev --output-file=requirements-dev.txt
|
|
||||||
#
|
|
||||||
arabic-reshaper==3.0.0
|
|
||||||
# via xhtml2pdf
|
|
||||||
asgiref==3.6.0
|
|
||||||
# via django
|
|
||||||
asn1crypto==1.5.1
|
|
||||||
# via
|
|
||||||
# oscrypto
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
attrs==22.2.0
|
|
||||||
# via pytest
|
|
||||||
base36==0.1.1
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
black==23.3.0
|
|
||||||
# via pytest-black
|
|
||||||
brotli==1.0.9
|
|
||||||
# via fonttools
|
|
||||||
build==0.10.0
|
|
||||||
# via pip-tools
|
|
||||||
certifi==2023.7.22
|
|
||||||
# via
|
|
||||||
# requests
|
|
||||||
# sentry-sdk
|
|
||||||
cffi==1.15.1
|
|
||||||
# via
|
|
||||||
# cryptography
|
|
||||||
# weasyprint
|
|
||||||
chardet==4.0.0
|
|
||||||
# via diff-cover
|
|
||||||
charset-normalizer==3.2.0
|
|
||||||
# via requests
|
|
||||||
click==8.1.3
|
|
||||||
# via
|
|
||||||
# black
|
|
||||||
# pip-tools
|
|
||||||
# pyhanko
|
|
||||||
coverage[toml]==7.2.2
|
|
||||||
# via pytest-cov
|
|
||||||
cryptography==41.0.3
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
cssselect2==0.7.0
|
|
||||||
# via
|
|
||||||
# svglib
|
|
||||||
# weasyprint
|
|
||||||
diff-cover==4.2.3
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
django==4.2
|
|
||||||
# via
|
|
||||||
# django-weasyprint
|
|
||||||
# la-chariotte (pyproject.toml)
|
|
||||||
django-weasyprint==2.2.1
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
fonttools[woff]==4.42.1
|
|
||||||
# via weasyprint
|
|
||||||
html2text==2020.1.16
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
html5lib==1.1
|
|
||||||
# via
|
|
||||||
# weasyprint
|
|
||||||
# xhtml2pdf
|
|
||||||
icalendar==5.0.7
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
idna==3.4
|
|
||||||
# via requests
|
|
||||||
importlib-metadata==6.1.0
|
|
||||||
# via inflect
|
|
||||||
inflect==3.0.2
|
|
||||||
# via
|
|
||||||
# diff-cover
|
|
||||||
# jinja2-pluralize
|
|
||||||
iniconfig==2.0.0
|
|
||||||
# via pytest
|
|
||||||
isort==5.12.0
|
|
||||||
# via pytest-isort
|
|
||||||
jinja2==3.1.2
|
|
||||||
# via
|
|
||||||
# diff-cover
|
|
||||||
# jinja2-pluralize
|
|
||||||
jinja2-pluralize==0.3.0
|
|
||||||
# via diff-cover
|
|
||||||
lxml==4.9.3
|
|
||||||
# via svglib
|
|
||||||
markupsafe==2.1.2
|
|
||||||
# via jinja2
|
|
||||||
mypy-extensions==1.0.0
|
|
||||||
# via black
|
|
||||||
oscrypto==1.3.0
|
|
||||||
# via pyhanko-certvalidator
|
|
||||||
packaging==23.0
|
|
||||||
# via
|
|
||||||
# black
|
|
||||||
# build
|
|
||||||
# pytest
|
|
||||||
pathspec==0.11.1
|
|
||||||
# via black
|
|
||||||
pillow==10.0.0
|
|
||||||
# via
|
|
||||||
# reportlab
|
|
||||||
# weasyprint
|
|
||||||
# xhtml2pdf
|
|
||||||
pip-tools==6.12.3
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
platformdirs==3.2.0
|
|
||||||
# via black
|
|
||||||
pluggy==1.0.0
|
|
||||||
# via
|
|
||||||
# diff-cover
|
|
||||||
# pytest
|
|
||||||
psycopg2-binary==2.9.6
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
pycparser==2.21
|
|
||||||
# via cffi
|
|
||||||
pydyf==0.7.0
|
|
||||||
# via weasyprint
|
|
||||||
pygments==2.14.0
|
|
||||||
# via diff-cover
|
|
||||||
pyhanko==0.20.0
|
|
||||||
# via xhtml2pdf
|
|
||||||
pyhanko-certvalidator==0.23.0
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# xhtml2pdf
|
|
||||||
pypdf==3.15.4
|
|
||||||
# via xhtml2pdf
|
|
||||||
pyphen==0.14.0
|
|
||||||
# via weasyprint
|
|
||||||
pypng==0.20220715.0
|
|
||||||
# via qrcode
|
|
||||||
pyproject-hooks==1.0.0
|
|
||||||
# via build
|
|
||||||
pytest==7.2.2
|
|
||||||
# via
|
|
||||||
# la-chariotte (pyproject.toml)
|
|
||||||
# pytest-black
|
|
||||||
# pytest-cov
|
|
||||||
# pytest-django
|
|
||||||
# pytest-isort
|
|
||||||
pytest-black==0.3.12
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
pytest-cov==4.0.0
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
pytest-django==4.5.2
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
pytest-isort==3.1.0
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
python-bidi==0.4.2
|
|
||||||
# via xhtml2pdf
|
|
||||||
python-dateutil==2.8.2
|
|
||||||
# via icalendar
|
|
||||||
pytz==2023.3
|
|
||||||
# via icalendar
|
|
||||||
pyyaml==6.0.1
|
|
||||||
# via pyhanko
|
|
||||||
qrcode==7.4.2
|
|
||||||
# via pyhanko
|
|
||||||
reportlab==3.6.13
|
|
||||||
# via
|
|
||||||
# svglib
|
|
||||||
# xhtml2pdf
|
|
||||||
requests==2.31.0
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
sentry-sdk==0.20.3
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
six==1.16.0
|
|
||||||
# via
|
|
||||||
# html5lib
|
|
||||||
# python-bidi
|
|
||||||
# python-dateutil
|
|
||||||
sqlparse==0.4.3
|
|
||||||
# via django
|
|
||||||
svglib==1.5.1
|
|
||||||
# via xhtml2pdf
|
|
||||||
tinycss2==1.2.1
|
|
||||||
# via
|
|
||||||
# cssselect2
|
|
||||||
# svglib
|
|
||||||
# weasyprint
|
|
||||||
toml==0.10.2
|
|
||||||
# via pytest-black
|
|
||||||
typing-extensions==4.7.1
|
|
||||||
# via qrcode
|
|
||||||
tzlocal==5.0.1
|
|
||||||
# via pyhanko
|
|
||||||
uritools==4.0.1
|
|
||||||
# via pyhanko-certvalidator
|
|
||||||
urllib3==2.0.4
|
|
||||||
# via
|
|
||||||
# requests
|
|
||||||
# sentry-sdk
|
|
||||||
weasyprint==59.0
|
|
||||||
# via django-weasyprint
|
|
||||||
webencodings==0.5.1
|
|
||||||
# via
|
|
||||||
# cssselect2
|
|
||||||
# html5lib
|
|
||||||
# tinycss2
|
|
||||||
wheel==0.40.0
|
|
||||||
# via pip-tools
|
|
||||||
xhtml2pdf==0.2.11
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
zipp==3.15.0
|
|
||||||
# via importlib-metadata
|
|
||||||
zopfli==0.2.2
|
|
||||||
# via fonttools
|
|
||||||
|
|
||||||
# The following packages are considered to be unsafe in a requirements file:
|
|
||||||
# pip
|
|
||||||
# setuptools
|
|
139
requirements.txt
|
@ -1,139 +0,0 @@
|
||||||
#
|
|
||||||
# This file is autogenerated by pip-compile with Python 3.11
|
|
||||||
# by the following command:
|
|
||||||
#
|
|
||||||
# pip-compile --output-file=requirements.txt
|
|
||||||
#
|
|
||||||
arabic-reshaper==3.0.0
|
|
||||||
# via xhtml2pdf
|
|
||||||
asgiref==3.7.2
|
|
||||||
# via django
|
|
||||||
asn1crypto==1.5.1
|
|
||||||
# via
|
|
||||||
# oscrypto
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
base36==0.1.1
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
brotli==1.0.9
|
|
||||||
# via fonttools
|
|
||||||
certifi==2023.5.7
|
|
||||||
# via
|
|
||||||
# requests
|
|
||||||
# sentry-sdk
|
|
||||||
cffi==1.15.1
|
|
||||||
# via
|
|
||||||
# cryptography
|
|
||||||
# weasyprint
|
|
||||||
charset-normalizer==3.1.0
|
|
||||||
# via requests
|
|
||||||
click==8.1.3
|
|
||||||
# via pyhanko
|
|
||||||
cryptography==40.0.2
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
cssselect2==0.7.0
|
|
||||||
# via
|
|
||||||
# svglib
|
|
||||||
# weasyprint
|
|
||||||
django==4.2.1
|
|
||||||
# via
|
|
||||||
# django-weasyprint
|
|
||||||
# la-chariotte (pyproject.toml)
|
|
||||||
django-weasyprint==1.1.0.post1
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
fonttools[woff]==4.42.1
|
|
||||||
# via weasyprint
|
|
||||||
html2text==2020.1.16
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
html5lib==1.1
|
|
||||||
# via
|
|
||||||
# weasyprint
|
|
||||||
# xhtml2pdf
|
|
||||||
icalendar==5.0.7
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
idna==3.4
|
|
||||||
# via requests
|
|
||||||
lxml==4.9.2
|
|
||||||
# via svglib
|
|
||||||
oscrypto==1.3.0
|
|
||||||
# via pyhanko-certvalidator
|
|
||||||
pillow==9.5.0
|
|
||||||
# via
|
|
||||||
# reportlab
|
|
||||||
# weasyprint
|
|
||||||
# xhtml2pdf
|
|
||||||
psycopg2-binary==2.9.6
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
pycparser==2.21
|
|
||||||
# via cffi
|
|
||||||
pydyf==0.7.0
|
|
||||||
# via weasyprint
|
|
||||||
pyhanko==0.18.1
|
|
||||||
# via xhtml2pdf
|
|
||||||
pyhanko-certvalidator==0.22.0
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# xhtml2pdf
|
|
||||||
pypdf==3.9.0
|
|
||||||
# via xhtml2pdf
|
|
||||||
pyphen==0.14.0
|
|
||||||
# via weasyprint
|
|
||||||
pypng==0.20220715.0
|
|
||||||
# via qrcode
|
|
||||||
python-bidi==0.4.2
|
|
||||||
# via xhtml2pdf
|
|
||||||
python-dateutil==2.8.2
|
|
||||||
# via icalendar
|
|
||||||
pytz==2023.3
|
|
||||||
# via icalendar
|
|
||||||
pyyaml==6.0
|
|
||||||
# via pyhanko
|
|
||||||
qrcode==7.4.2
|
|
||||||
# via pyhanko
|
|
||||||
reportlab==3.6.13
|
|
||||||
# via
|
|
||||||
# svglib
|
|
||||||
# xhtml2pdf
|
|
||||||
requests==2.31.0
|
|
||||||
# via
|
|
||||||
# pyhanko
|
|
||||||
# pyhanko-certvalidator
|
|
||||||
sentry-sdk==0.20.3
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
six==1.16.0
|
|
||||||
# via
|
|
||||||
# html5lib
|
|
||||||
# python-bidi
|
|
||||||
# python-dateutil
|
|
||||||
sqlparse==0.4.4
|
|
||||||
# via django
|
|
||||||
svglib==1.5.1
|
|
||||||
# via xhtml2pdf
|
|
||||||
tinycss2==1.2.1
|
|
||||||
# via
|
|
||||||
# cssselect2
|
|
||||||
# svglib
|
|
||||||
# weasyprint
|
|
||||||
typing-extensions==4.6.2
|
|
||||||
# via qrcode
|
|
||||||
tzlocal==5.0.1
|
|
||||||
# via pyhanko
|
|
||||||
uritools==4.0.1
|
|
||||||
# via pyhanko-certvalidator
|
|
||||||
urllib3==2.0.2
|
|
||||||
# via
|
|
||||||
# requests
|
|
||||||
# sentry-sdk
|
|
||||||
weasyprint==52.5
|
|
||||||
# via django-weasyprint
|
|
||||||
webencodings==0.5.1
|
|
||||||
# via
|
|
||||||
# cssselect2
|
|
||||||
# html5lib
|
|
||||||
# tinycss2
|
|
||||||
xhtml2pdf==0.2.11
|
|
||||||
# via la-chariotte (pyproject.toml)
|
|
||||||
zopfli==0.2.2
|
|
||||||
# via fonttools
|
|