ku-polls/polls/models.py
sosokker 01ea565859 Add User+Authenticate(login,logout,reset password)
Also add view for signup page + custom form for signup
2023-09-11 23:38:13 +07:00

204 lines
6.4 KiB
Python

"""
This module defines the models for the polls app.
It includes the Question and Choice models, which represent poll questions
and the choices associated with them. These models are used to store and
get poll data in the database.
Attributes:
None
"""
from django.db import models
from django.utils import timezone
from django.contrib import admin
from django.core.validators import MaxValueValidator, MinValueValidator
from django.db.models import Sum
from django.contrib.auth.models import User
class Question(models.Model):
"""
Represents a poll question.
Attributes:
question_text (str): The text of the poll question.
pub_date (datetime): The date and time when the question was published.
end_date (datetime): The date and time when the question will end.
long_description (str): The long description of the poll question.
short_description (str): The short description of the poll question.
up_vote_count (int): The number of up votes the question has received.
down_vote_count (int): The number of down votes the question has received.
participant_count (int): The number of participants in the poll.
"""
question_text = models.CharField(max_length=100)
short_description = models.CharField(max_length=200, default="Cool kids have polls")
long_description = models.TextField(
max_length=2000, default="No description provide for this poll."
)
pub_date = models.DateTimeField(
"date published", default=timezone.now, editable=True
)
end_date = models.DateTimeField("date ended", null=True)
up_vote_count = models.PositiveIntegerField(
default=0, validators=[MinValueValidator(0), MaxValueValidator(2147483647)]
)
down_vote_count = models.PositiveIntegerField(
default=0, validators=[MinValueValidator(0), MaxValueValidator(2147483647)]
)
participant_count = models.PositiveIntegerField(
default=0, validators=[MinValueValidator(0), MaxValueValidator(2147483647)]
)
def was_published_recently(self):
"""
Checks if the question was published recently or not.
Returns:
bool: True if the question was published within the last day, else False.
"""
now = timezone.now()
return now - timezone.timedelta(days=1) <= self.pub_date <= now
@admin.display(
boolean=True,
ordering="pub_date",
description="Published recently?",
)
def was_published_recently(self):
now = timezone.now()
return now - timezone.timedelta(days=1) <= self.pub_date <= now
def __str__(self):
"""
Returns a string representation of the question.
"""
return self.question_text
def is_published(self):
"""
Checks if the question is published or not.
Returns:
bool: True if the question is published, else False.
"""
now = timezone.now()
return now >= self.pub_date
def can_vote(self):
"""
Checks if the question can be voted on or not.
Returns:
bool: True if the question is published and not ended, else False.
"""
now = timezone.now()
if self.end_date is None:
return self.pub_date <= now
else:
return self.pub_date <= now <= self.end_date
def calculate_time_left(self):
"""
Calculate the time left until the end date.
Returns:
str: A formatted string representing the time left.
"""
if self.end_date is None:
return "No end date"
now = timezone.now()
time_left = self.end_date - now
days, seconds = divmod(time_left.total_seconds(), 86400)
hours, seconds = divmod(seconds, 3600)
minutes, seconds = divmod(seconds, 60)
time_left_str = ""
if days > 0:
time_left_str += f"{int(days)} Days "
elif hours > 0:
time_left_str += f"{int(hours)} Hours "
elif minutes > 0:
time_left_str += f"{int(minutes)} Mins "
elif seconds > 0:
time_left_str += f"{int(seconds)} Sec "
return time_left_str.strip()
@property
def time_left(self):
return self.calculate_time_left()
def calculate_vote_percentage(self):
"""Calculate the percentage of up votes and down votes."""
total_vote = self.up_vote_count + self.down_vote_count
if total_vote == 0:
return (0, 0)
up_vote_percentage = self.up_vote_count / total_vote * 100
down_vote_percentage = self.down_vote_count / total_vote * 100
return (int(up_vote_percentage), int(down_vote_percentage))
@property
def up_vote_percentage(self):
return self.calculate_vote_percentage()[0]
@property
def down_vote_percentage(self):
return self.calculate_vote_percentage()[1]
class Choice(models.Model):
"""
Represents a choice for a poll question.
Attributes:
question (Question): The poll question to which the choice belongs.
choice_text (str): The text of the choice.
votes (int): The number of votes the choice has received.
"""
question = models.ForeignKey(Question, on_delete=models.CASCADE)
choice_text = models.CharField(max_length=200)
votes = models.PositiveIntegerField(
default=0, validators=[MinValueValidator(0), MaxValueValidator(2147483647)]
)
def tailwind_width_class(self):
"""
Calculate and return the Tailwind CSS width class based on the 'votes' percentage.
"""
total_votes = self.question.choice_set.aggregate(Sum("votes")).get(
"votes__sum", 0
)
#! Tailwind w-0 to w-48
if total_votes == 0:
return "w-0"
ratio = self.votes / total_votes
scaled_value = ratio * 48
return f"w-{int(round(scaled_value))}"
def calculate_percentage(self):
"""Calculate percentage of votes for all choices."""
total_votes_for_question = (
self.question.choice_set.aggregate(Sum("votes"))["votes__sum"] or 0
)
if total_votes_for_question == 0:
return 0
else:
return round((self.votes / total_votes_for_question) * 100, 2)
def __str__(self):
"""
Returns a string representation of the choice.
"""
return self.choice_text