You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

208 lines
7.4 KiB

from django.db import models
from django.utils.translation import gettext_lazy as _
from filer.fields.image import FilerImageField
class NarratorLayer(models.Model):
"""
Model for narrator layers/classes (Tabaqat)
Represents the classification level of narrators in hadis chains
"""
name = models.CharField(max_length=255, verbose_name=_('name'))
number = models.PositiveIntegerField(verbose_name=_('layer number'), unique=True)
description = models.TextField(verbose_name=_('description'), blank=True, null=True)
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
verbose_name = _('Narrator Layer')
verbose_name_plural = _('Narrator Layers')
ordering = ['number']
def __str__(self):
return f"{_('Layer')} {self.number} - {self.name}"
class Transmitters(models.Model):
class ReliabilityLevel(models.TextChoices):
VERY_RELIABLE = 'very_reliable', _('Very Reliable')
RELIABLE = 'reliable', _('Reliable')
ACCEPTABLE = 'acceptable', _('Acceptable')
WEAK = 'weak', _('Weak')
VERY_WEAK = 'very_weak', _('Very Weak')
UNKNOWN = 'unknown', _('Unknown')
class MadhhabChoices(models.TextChoices):
SHIA = 'shia', _('Shia')
SUNNI = 'sunni', _('Sunni')
HANAFI = 'hanafi', _('Hanafi')
MALIKI = 'maliki', _('Maliki')
SHAFII = 'shafii', _('Shafi\'i')
HANBALI = 'hanbali', _('Hanbali')
OTHER = 'other', _('Other')
UNKNOWN = 'unknown', _('Unknown')
# Basic Information
full_name = models.CharField(max_length=255, verbose_name=_('full name'))
kunya = models.CharField(max_length=255, verbose_name=_('Kunya'), blank=True, null=True, help_text=_('e.g., Abu Abdullah'))
known_as = models.CharField(max_length=255, verbose_name=_('Known As'), blank=True, null=True)
nickname = models.CharField(max_length=255, verbose_name=_('Nickname/Laqab'), blank=True, null=True)
# Geographic Information
origin = models.CharField(max_length=255, verbose_name=_('Origin'), blank=True, null=True, help_text=_('Place of origin'))
lived_in = models.CharField(max_length=255, verbose_name=_('Lived In'), blank=True, null=True, help_text=_('Places where they lived'))
died_in = models.CharField(max_length=255, verbose_name=_('Died In'), blank=True, null=True, help_text=_('Place of death'))
# Date Information
birth_year_hijri = models.IntegerField(verbose_name=_("Birth Year (Hijri)"), null=True, blank=True)
death_year_hijri = models.IntegerField(verbose_name=_("Death Year (Hijri)"), null=True, blank=True)
age_at_death = models.PositiveIntegerField(verbose_name=_('Age at Death'), blank=True, null=True)
# Religious & Academic Information
reliability = models.CharField(
max_length=20,
choices=ReliabilityLevel.choices,
default=ReliabilityLevel.UNKNOWN,
verbose_name=_('Reliability Level')
)
madhhab = models.CharField(
max_length=20,
choices=MadhhabChoices.choices,
default=MadhhabChoices.UNKNOWN,
verbose_name=_('Madhhab/School of Thought')
)
# Presence in Famous Collections
in_sahih_muslim = models.BooleanField(
default=False,
verbose_name=_('In Sahih Muslim'),
help_text=_('Is this narrator present in Sahih Muslim?')
)
in_sahih_bukhari = models.BooleanField(
default=False,
verbose_name=_('In Sahih Bukhari'),
help_text=_('Is this narrator present in Sahih Bukhari?')
)
# Additional Information
description = models.TextField(blank=True, null=True, verbose_name=_("Description"))
thumbnail = FilerImageField(
related_name="+",
on_delete=models.CASCADE,
help_text=_('image allowed'),
null=True,
blank=True
)
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
verbose_name = _('Transmitter')
verbose_name_plural = _('Transmitters')
ordering = ('full_name',)
def __str__(self):
return self.full_name
class HadisTransmitter(models.Model):
class ReliabilityStatus(models.TextChoices):
RELIABLE = 'reliable', _('Reliable')
WEAK = 'weak', _('Weak')
UNKNOWN = 'unknown', _('Unknown')
hadis = models.ForeignKey(
"hadis.Hadis",
on_delete=models.CASCADE,
verbose_name=_('hadis'),
related_name='transmitters'
)
transmitter = models.ForeignKey(
Transmitters,
on_delete=models.CASCADE,
verbose_name=_('transmitter'),
related_name='hadises'
)
narrator_layer = models.ForeignKey(
NarratorLayer,
on_delete=models.SET_NULL,
verbose_name=_('narrator layer'),
related_name='transmitters',
null=True,
blank=True,
help_text=_('The layer/class (Tabaqah) this narrator belongs to')
)
status = models.CharField(
max_length=20,
choices=ReliabilityStatus.choices,
default=ReliabilityStatus.UNKNOWN,
verbose_name=_('reliability status'),
help_text=_('Reliability status of the narrator')
)
order = models.PositiveIntegerField(
default=0,
verbose_name=_('Order'),
help_text=_('Order in the chain of transmission')
)
is_gap = models.BooleanField(default=False, verbose_name=_('is gap'))
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
class Meta:
verbose_name = _('Hadis Transmitter')
verbose_name_plural = _('Hadis Transmitters')
ordering = ('hadis', 'order')
unique_together = ('hadis', 'transmitter', 'order')
def __str__(self):
layer_info = f" - {self.narrator_layer}" if self.narrator_layer else ""
return f'{self.hadis.number} - {self.transmitter.full_name} ({self.order}){layer_info}'
class TransmitterOpinion(models.Model):
"""
Model for scholarly opinions about transmitters
"""
class OpinionStatus(models.TextChoices):
CONFIRMED = 'confirmed', _('Confirmed')
MIXED = 'mixed', _('Mixed')
REJECTED = 'rejected', _('Rejected')
transmitter = models.ForeignKey(
Transmitters,
on_delete=models.CASCADE,
verbose_name=_('transmitter'),
related_name='opinions'
)
scholar_name = models.CharField(
max_length=255,
verbose_name=_('Scholar Name'),
help_text=_('Name of the scholar who gave this opinion')
)
opinion_text = models.TextField(
verbose_name=_('Opinion Text'),
help_text=_('The scholar\'s opinion about this transmitter')
)
status = models.CharField(
max_length=20,
choices=OpinionStatus.choices,
default=OpinionStatus.CONFIRMED,
verbose_name=_('Opinion Status'),
help_text=_('Status of the opinion')
)
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('created at'))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('updated at'))
class Meta:
verbose_name = _('Transmitter Opinion')
verbose_name_plural = _('Transmitter Opinions')
ordering = ('-created_at',)
def __str__(self):
return f"{self.scholar_name}'s opinion on {self.transmitter.full_name} ({self.status})"