Compare commits

...

3 Commits

Author SHA1 Message Date
VARUN 1d295837df settings.py
2 months ago
VARUN 0e7dc8327c Merge branch 'master' of http://35.207.205.18:3000/Arminta-Careers/arminta-web
2 months ago
VARUN 9205554211 settings.py
2 months ago

@ -0,0 +1,73 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AutoImportSettings">
<option name="autoReloadType" value="SELECTIVE" />
</component>
<component name="ChangeListManager">
<list default="true" id="1cf71b98-1570-4a2e-b272-bc9b74135085" name="Changes" comment="" />
<option name="SHOW_DIALOG" value="false" />
<option name="HIGHLIGHT_CONFLICTS" value="true" />
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
<option name="LAST_RESOLUTION" value="IGNORE" />
</component>
<component name="ProjectColorInfo">{
&quot;associatedIndex&quot;: 6
}</component>
<component name="ProjectId" id="2iV2x0vEEAZI3FGpJzgs4iS87pf" />
<component name="ProjectViewState">
<option name="hideEmptyMiddlePackages" value="true" />
<option name="showLibraryContents" value="true" />
</component>
<component name="PropertiesComponent">{
&quot;keyToString&quot;: {
&quot;RunOnceActivity.OpenProjectViewOnStart&quot;: &quot;true&quot;,
&quot;RunOnceActivity.ShowReadmeOnStart&quot;: &quot;true&quot;,
&quot;last_opened_file_path&quot;: &quot;C:/Users/laxmib/pythonProject7&quot;
}
}</component>
<component name="RunManager">
<configuration name="main" type="PythonConfigurationType" factoryName="Python" nameIsGenerated="true">
<module name="pythonProject11" />
<option name="INTERPRETER_OPTIONS" value="" />
<option name="PARENT_ENVS" value="true" />
<envs>
<env name="PYTHONUNBUFFERED" value="1" />
</envs>
<option name="SDK_HOME" value="" />
<option name="WORKING_DIRECTORY" value="$PROJECT_DIR$" />
<option name="IS_MODULE_SDK" value="true" />
<option name="ADD_CONTENT_ROOTS" value="true" />
<option name="ADD_SOURCE_ROOTS" value="true" />
<option name="SCRIPT_NAME" value="$PROJECT_DIR$/main.py" />
<option name="PARAMETERS" value="" />
<option name="SHOW_COMMAND_LINE" value="false" />
<option name="EMULATE_TERMINAL" value="false" />
<option name="MODULE_MODE" value="false" />
<option name="REDIRECT_INPUT" value="false" />
<option name="INPUT_FILE" value="" />
<method v="2" />
</configuration>
</component>
<component name="SpellCheckerSettings" RuntimeDictionaries="0" Folders="0" CustomDictionaries="0" DefaultDictionary="application-level" UseSingleDictionary="true" transferred="true" />
<component name="TaskManager">
<task active="true" id="Default" summary="Default task">
<changelist id="1cf71b98-1570-4a2e-b272-bc9b74135085" name="Changes" comment="" />
<created>1719561666463</created>
<option name="number" value="Default" />
<option name="presentableId" value="Default" />
<updated>1719561666463</updated>
</task>
<servers />
</component>
<component name="XDebuggerManager">
<breakpoint-manager>
<breakpoints>
<line-breakpoint enabled="true" suspend="THREAD" type="python-line">
<url>file://$PROJECT_DIR$/main.py</url>
<line>8</line>
<option name="timeStamp" value="1" />
</line-breakpoint>
</breakpoints>
</breakpoint-manager>
</component>
</project>

Binary file not shown.

Binary file not shown.

Binary file not shown.

File diff suppressed because one or more lines are too long

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

File diff suppressed because one or more lines are too long

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

@ -20,12 +20,12 @@ class Migration(migrations.Migration):
migrations.AlterField(
model_name='jobposting',
name='BudgetMax',
field=models.DecimalField(decimal_places=2, max_digits=20),
field=models.DecimalField(decimal_places=2, max_digits=100),
),
migrations.AlterField(
model_name='jobposting',
name='BudgetMin',
field=models.DecimalField(decimal_places=2, max_digits=20),
field=models.DecimalField(decimal_places=2, max_digits=100),
),
migrations.AlterField(
model_name='jobposting',

@ -0,0 +1,18 @@
# Generated by Django 4.2.11 on 2025-01-08 06:29
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('skyonnadmin', '0063_remove_internalteam_band_internalteam_level'),
]
operations = [
migrations.AlterField(
model_name='internalteam',
name='Email1',
field=models.EmailField(max_length=254, null=True, unique=True),
),
]

@ -0,0 +1,18 @@
# Generated by Django 4.2.11 on 2025-01-08 06:37
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('skyonnadmin', '0064_alter_internalteam_email1'),
]
operations = [
migrations.AddField(
model_name='internalteam',
name='FullName',
field=models.CharField(max_length=100, null=True),
),
]

@ -173,14 +173,16 @@ class InternalTeam(models.Model):
Country = models.CharField(max_length=100, null=True)
PhoneNo1 = models.CharField(max_length=15, null=True)
LandNumber = models.CharField(max_length=25, blank=True, null=True)
Email1 = models.EmailField(null=True)
Email1 = models.EmailField(null=True,unique=True)
Email2 = models.EmailField(max_length=50,null=True)
Level = models.CharField(max_length=50, null=True)
FullName = models.CharField(max_length=100, null=True)
Login = models.EmailField(null=True)
Password = models.CharField(max_length=50, null=True)
user_id = models.CharField(max_length=50, null=True)
def __str__(self):
return f"{self.FirstName} {self.LastName}"
return f"{self.Email1}"

@ -1073,6 +1073,8 @@ def save_Internal_team_Details(request):
Email2 = request.POST.get('Email2')
Level = request.POST.get('Level')
fullName = FirstName + '' + LastName
Login = Email1
Password = FirstName + '@123'
user_id = FirstName.upper()[:3] + PhoneNo1[:4]
@ -1095,7 +1097,8 @@ def save_Internal_team_Details(request):
Level=Level,
Login=Login,
Password=Password,
user_id=user_id
user_id=user_id,
FullName = fullName
)
internal_team.save()

@ -1,5 +1,5 @@
"""
Django settings for skyonnweb project ishu.
Django settings for skyonnweb project.
Generated by 'django-admin startproject' using Django 4.2.11.
@ -82,7 +82,7 @@ DATABASES = {
'NAME': 'skyonn4',
'USER': 'root',
'PASSWORD': '',
'HOST': 'localhost',
'HOST': '34.93.75.170',
'PORT': '3306',
'OPTIONS': {
'init_command': "SET sql_mode='STRICT_TRANS_TABLES'",
@ -153,27 +153,4 @@ EMAIL_HOST_USER = 'subramanyajithendra@gmail.com'
EMAIL_HOST_PASSWORD = 'grdwevagtfmfqosb'
TIME_ZONE = 'Asia/Kolkata'
USE_TZ = True
LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'handlers': {
'file': {
'level': 'ERROR',
'class': 'logging.FileHandler',
'filename': '/home/armintacareers/arminta-web/skyonnweb/logs/django.log',
},
},
'loggers': {
'django': {
'handlers': ['file'],
'level': 'ERROR',
'propagate': True,
},
},
}
USE_TZ = True

@ -2,223 +2,596 @@
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link href="https://cdn.jsdelivr.net/npm/remixicon@2.5.0/fonts/remixicon.css" rel="stylesheet">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.2/css/all.min.css" integrity="sha512-z3gLpd7yknf1YoNbCzqRKc4qyor8gaKU1qmn+CShxbuBusANI9QpRohGBreCFkKxLhei6S9CQXFEbbKuqLg0DA==" crossorigin="anonymous" referrerpolicy="no-referrer" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet" >
<link rel="stylesheet" />
<title>Document</title>
<style>
@import url('https://fonts.googleapis.com/css2?family=Roboto:wght@400;500;700&display=swap');
* {
margin: 0;
padding: 0;
box-sizing: border-box;
font-family: 'Roboto', sans-serif;
.Messages{
.header {
display: flex;
align-items: center;
justify-content: space-between;
border-bottom: 1px solid whitesmoke;
position:fixed;
}
.header__left {
display: flex;
align-items: center;
}
.header__left span {
padding: 10px;
cursor: pointer;
}
.header__middle {
display: flex;
flex: 0.7;
align-items: center;
background-color: whitesmoke;
padding: 10px;
border-radius: 5px;
}
.material-icons {
color: gray;
}
.header__middle input {
border: none;
width: 100%;
padding: 10px;
outline: none;
font-size: medium;
background-color: transparent;
}
.header__right {
display: flex;
padding-right: 20px;
}
.header__right span {
padding: 10px;
cursor: pointer;
}
.sidebarOption {
display: flex;
align-items: center;
border-top-right-radius: 20px;
border-bottom-right-radius: 20px;
cursor: pointer;
color: #818181;
margin-top:100px;
}
.sidebarOption .material-icons {
padding: 5px;
}
.sidebarOption h3 {
flex: 1;
margin-left: 10px;
font-size: 16px;
font-weight: 400;
color:#000000;
}
.sidebarOption:hover,
.sidebarOption:hover h3{
width:100px;
}
<!--.sidebarOption:hover,-->
<!--.sidebarOption:hover h3,-->
.sidebarOption:hover .material-icons,
<!--.sidebarOption__active,-->
.sidebarOption__active h3,
.sidebarOption__active .material-icons {
background-color: #fcecec;
color: #c04b37;
}
.material-icons {
color: gray;
border-radius:50%;
}
.header__middle input {
border: none;
width: 10%;
padding: 10px;
outline: none;
font-size: medium;
border-radius:50%;
background-color: transparent;
}
.compose {
margin-top: 15px;
margin-left: 10px;
margin-bottom: 15px;
text-transform: capitalize;
color: gray;
padding: 8px;
border-radius: 30px;
background-color: transparent;
display: flex;
align-items: center;
cursor: pointer;
outline: none;
border: none;
transition: background-color 0.3s, color 0.3s;
box-shadow: 0px 2px 5px -2px rgba(0, 0, 0, 0.75);
}
.compose span {
margin-right: 5px;
}
.compose.active {
background-color: #43A6C6;
color: black; /* Highlighted text color */
}
.compose:hover {
opacity: 0.9; /* Optional hover effect */
}
.emailRow__options {
align-items: center;
}
.emailRow__options .material-icons,
input {
margin: 5px;
}
.emailRow {
display: flex;
align-items: center; /* Ensure proper spacing between elements */
height: 70px; /* Increased height for better visibility */
margin-left: -20px;
/* Add padding inside the row for spacing */
border: 2px solid whitesmoke;
background-color: white;
cursor: pointer;
margin-top: 20px; /* Adjusted margin for spacing */
margin-bottom: 10px; /* Spaced out between rows */
border-radius: 5px; /* Slight rounding for a modern look */
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* Add subtle shadow for better contrast */
width: calc(100% - 40px); /* Make room for consistent padding */
transition: box-shadow 0.3s, transform 0.3s; /* Smooth hover transition */
}
/* Hover effect for the email row */
.emailRow:hover {
border: 2px solid lightgray;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2); /* Enhanced shadow on hover */
transform: translateY(-2px); /* Slight lift effect on hover */
}
/* Style for the button inside the row */
.emailRow__button {
border: 1px solid lightgray;
box-shadow: 0px 2px 5px -2px rgba(0, 0, 0, 0.75);
border-radius: 5px;
cursor: pointer;
background-color: #f9f9f9;
transition: background-color 0.3s, box-shadow 0.3s;
}
/* Hover effect for the button */
.emailRow__button:hover {
background-color: #e6f7ff; /* Light blue for hover */
box-shadow: 0px 4px 6px -2px rgba(0, 0, 0, 0.3); /* Stronger shadow */
border-color: #0095f6;
}
/* Additional child elements remain unaffected */
.emailRow__message {
display: flex;
flex: 0.8;
align-items: center;
font-size: 13px;
}
.emailRow__message h4 {
width: 400px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
padding-left: 5px;
padding-right: 5px;
}
.emailRow {
display: flex;
align-items: center;
justify-content: space-between; /* Ensure proper spacing between elements */
height: 50px;
padding: 10px 15px; /* Small padding for spacing within the row */
border: 1px solid lightgray;
background-color: white;
cursor: pointer;
margin: 0;
margin-top:30px;
}
/* Hover effect for the email row */
.emailRow:hover {
border: 1px solid lightblue; /* Subtle border change on hover */
background-color: #f9f9f9; /* Light background change */
}
/* Button inside the row */
.emailRow__button {
border: 1px solid lightgray;
border-radius: 3px; /* Subtle rounding for the button */
cursor: pointer;
padding: 5px 10px;
background-color: white;
}
/* Button hover effect */
.emailRow__button:hover {
background-color: #e6f7ff;
border-color: #0078d4;
}
/* Ensure child elements fit within the row */
.emailRow__message h4 {
width: 350px; /* Adjust as needed to fit content */
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
/* Titles and text are aligned with appropriate sizing */
.emailRow__title {
font-size: 14px;
}
.emailRow__time {
font-size: 12px;
color: gray;
}
.emailRow__description {
font-size: 13px;
color: #666;
}
/* Remove spacing between rows */
.emailRow + .emailRow {
margin-top: 0; /* No extra space between rows */
}
/* Header */
.compose-header {
background-color: #f5f5f5;
padding: 12px 16px;
display: flex;
justify-content: space-between;
align-items: center;
cursor: grab; /* For drag-and-drop effect */
border-bottom: 1px solid #ddd;
margin-top:500px;
}
.compose-header:hover {
background-color: #eaeaea;
}
.compose-title {
font-size: 16px;
font-weight: bold;
color: #333;
margin: 0;
}
.compose-close {
font-size: 18px;
color: #666;
cursor: pointer;
transition: color 0.2s;
}
.compose-close:hover {
color: #000;
}
/* Body */
.compose-body {
padding: 12px 16px;
flex-grow: 1;
display: flex;
flex-direction: column;
gap: 12px;
}
/* Input Fields */
.compose-input {
width: 100%;
border: 1px solid #ddd;
padding: 8px;
border-radius: 4px;
font-size: 14px;
color: #333;
}
.compose-input:focus {
border-color: #4285f4;
outline: none;
}
/* Buttons */
.compose-button {
padding: 8px 16px;
font-size: 14px;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background-color 0.3s;
}
.compose-button.send {
background-color: #4285f4;
}
.compose-button.send:hover {
background-color: #357ae8;
}
.compose-button.discard {
background-color: #f5f5f5;
color: #333;
}
.compose-button.discard:hover {
background-color: #eaeaea;
}
.sidebar__compose {
margin-top: 15px;
margin-left: 10px;
margin-bottom: 15px;
text-transform: capitalize;
color: gray;
padding: 15px;
border-radius: 30px;
background-color: white;
display: flex;
align-items: center;
cursor: pointer;
outline: none;
border: none;
box-shadow: 0px 2px 5px -2px rgba(0, 0, 0, 0.75);
}
.sidebar__compose span {
margin-right: 5px;
}
.sidebar__compose:hover{
background-color:#87CEFA;
transition-duration: 0.4s;}
.sidebarOption:hover {
background-color: #6699CC;
color:black;
}
body {
position: relative;
width: 100%;
overflow: hidden;
}
.header {
height: 60px;
width: 100%;
display: flex;
align-items: center;
border-bottom: 1px solid black;
}
h2{
color: red;
margin-left: 400px ;
white-space: nowrap;
margin-top: 10px;
}
.search--notification--profile {
display: flex;
align-items: center;
justify-content: space-between;
width: calc(100% - 100px);
padding: 0 40px;
margin-left: -10px;
}
.search {
width: 300px;
padding: 5px;
display: flex;
align-items: center;
justify-content: space-between;
}
.search-container {
display: flex;
align-items: center;
border: 1px solid black;
.sidebarOption__active {
background-color: #6699CC;
color: black;
border-radius: 5px;
padding: 0 5px;
}
.search input {
outline: none;
border: none;
text-indent: 15px;
width: 100%;
height: 40px;
}
.search button {
outline: none;
border: none;
padding: 0;
display: flex;
align-items: center;
justify-content: center;
}
.main {
position: relative;
width: 100%;
min-height: calc(100vh - 60px);
}
.sidebar {
position: absolute;
top: 0;
left: 0;
height: 100%;
width: 250px;
background-color: #fff;
padding: 30px;
display: flex;
flex-direction: column;
justify-content: space-between;
border-right: 1px solid black;
}
.sidebar.active {
width: 103px;
overflow: hidden;
}
.sidebar.active .sidebar--item {
display: none;
}
li {
list-style: none;
}
a {
text-decoration: none;
}
.sidebar--items a,
.sidebar--bottom-items a {
display: flex;
align-items: center;
margin-bottom: 10px;
font-size: 1.1rem;
color: blue;
padding: 10px;
border-radius: 10px;
}
.sidebar--items a:hover,
.sidebar--bottom-items a:hover {
color: red;
width:100px;
}
.material-icons-black span {
color: black;
}
.search {
margin-left: 100px;
margin-top: -30px;
background-color: #F1F4F8;
border-radius: 50px;
width: 400px;
padding: 5px;
display: flex;
align-items: center;
justify-content: space-between;
}
.search input {
background-color: transparent;
outline: none;
border: none;
text-indent: 15px;
width: 85%;
}
.search button {
outline: none;
border: none;
border-radius: 50%;
background-color: #fff;
padding: 8px;
display: flex;
align-items: center;
justify-content: center;
}
.header__right span{
padding:20px;
cursor:pointer;
}
.email-checkbox + label {
margin-right: 15px;
}
#emailCheckbox {
display: none; /* Hidden by default */
position: absolute; /* Position it relative to its container */
top: 100%; /* Position it just below the input field */
left: 0; /* Align it with the left of the input field */
z-index: 10; /* Ensure it appears above other elements */
background-color: white; /* Background color for better visibility */
border: 1px solid #ccc; /* Add a border for separation */
padding: 10px; /* Add some spacing inside the dropdown */
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* Add a subtle shadow */
width: auto; /* Adjust width as needed */
height:auto;
}
#dropdownIcon {
position: relative; /* Position it relative to its container */
cursor: pointer;
}
.email-input-container {
position: relative; /* The dropdown will be positioned relative to this container */
}
}
</style>
<title>Admin Dashboard</title>
</head>
<body>
<section class="header">
<h1 style="color: lightskyblue; margin-top: 10px; margin-left: 50px; text-decoration: underline;">Home</h1>
<h2>Admin <span>Dashboard</span></h2>
<div class="search--notification--profile">
<div class="Messages ">
<form id="messageForm " style="" >
{% csrf_token %}
<input type="hidden" id="user_id" name="user_id" value="{{ user_id }}">
<input type="hidden" id="current_user_id" name="current_user_id" value="{{ current_user_id }}">
<div class="header" style="margin-top:80px;width:75.5%; margin-left:300px;">
<h2 style="margin-left:60px;">Messages</h2>
<div class="search">
<div class="search-container">
<input type="text" placeholder="Search">
<button><i class="ri-search-2-line"></i></button>
</div>
<input type="text" placeholder="Search">
<button><i class="ri-search-2-line"></i></button>
</div>
<div class="header__right" style="margin-right:400px;">
<span class="material-icons">apps</span>
<span class="material-icons">notifications</span>
<span class="material-icons">account_circle</span>
</div>
<div class="bell" style="margin-right: 40px;">
<i class="fa-solid fa-bell"></i>
</div>
<div class="profile" style="margin-right: 40px;">
<i class="fa-solid fa-user"></i>
</div>
<h3 style="margin-right: 40px;">Laxmi</h3>
<a href="#" style="margin-right: 350px; color: blue; font-size: larger; text-decoration: underline;">LogOut</a>
</section>
<section class="main">
<div class="sidebar">
<ul class="sidebar--items">
<li>
<a href="{% url 'client-details' %}" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Client Details</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Team Details</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">New Job Postings</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">All Jobs Postings</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Messages</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Pending Process</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Submitted Resumes</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Add a New SPOC</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Client Details</span>
</a>
</li>
<li>
<a href="#" class="sidebar-link" onclick="activateLink(this)">
<span class="sidebar--item">Activity Report</span>
</a>
</li>
</ul>
</div>
<div class="main__body" style="margin-left: 300px; margin-top: 120px; position: sticky; top: 0; z-index: 10; background-color: white;">
<div class="" style="margin-top:50px;">
<button class="sidebar__compose" id="compose" onclick="showCompose(event)" style="display: block; margin-bottom: 20px;">
<span class="material-icons"> add </span>Compose
</button>
<div class="sidebarOption-container" style="position: sticky; top: 0;">
<div class="sidebarOption sidebarOption__active" style="width:100px;" id="inboxButton" onclick="setActiveBox(this); renderInbox(); HideMessageDetails()">
<span class="material-icons" style="color:black;"> inbox </span>
<h3>Inbox</h3>
</div>
<div class="sidebarOption" id="sentButton" style="margin-top:-32px; margin-left:130px;" onclick="setActiveBox(this); renderSentMessages(); HideMessageDetails()">
<span class="material-icons" style="color:black"> near_me </span>
<h3>Sent</h3>
</div>
<div class="sidebarOption" style="margin-top:-37px; margin-left:250px;" onclick="setActiveBox(this)">
<span class="material-icons" style="color:black">delete</span>
<h3>Trash</h3>
</div>
</div>
</div>
</div>
<div id="InboxContainer" style="display:block;margin-left: 40px; "></div>
<div id="MessageDetailsContainer" style="display: none; height:auto; padding: 20px; border: 1px solid #ddd; margin-left: 0px; margin-top:20px; border:none; width:100%; box-shadow: rgba(100, 100, 111, 0.2) 0px 7px 29px 0px; border-radius:20px;">
<button id="BackToInboxButton" onclick="backToInbox(event)" style="margin-bottom: 10px;">Back to Inbox</button>
<div id="MessageDetails" style="height: auto;"></div>
<div id="ConversationThread" style="height:auto;"></div>
</div>
</div>
</div>
</div>
</form>
<div class="compose-container" id="ComposeContainer"
style="display: none; position: fixed; top:230px; bottom: 10%; right: 5%; width: 30%; height:58%; background-color: #fff; border-radius: 30px; box-shadow: 3px 3px 3px 3px rgba(0.5, 0.5, 0.5, 0.5); padding: 30px; z-index: 1000; margin-top: 50px;">
<div class="container" style="padding: 20px;width:100%">
<form id="ComposeMessage" method="post" action="{% url 'send_message' %}?user_id={{ user_id }}">
{% csrf_token %}
<div class="fixed" style="margin-buttom:10px; width:100%">
<div class="compose-header" style="display: flex; margin-left:-50px; space-between; align-items: center; background-color:#F0F8FF; width:128%; margin-top:-50px;">
<span style="font-size: 18px; font-weight: bold; margin-left:20px;" >New Message</span>
<div class="compose-close" id="Closebutton" onclick="CloseCompose(event)" style="font-size: 30px; margin-left:290px; color: red; cursor: pointer;">&times;</div>
</div>
<div class="compose-body" >
<div class="" style="position: relative;">
<div id="emailBarsContainer" style="margin-top: 10px; display: flex; flex-wrap: wrap;"></div>
<input type="text" id="toField" placeholder="Select Emails" readonly style="width: 128%; text-indent: 10px; border: none; border-bottom: 1px solid #d3d3d3; height:40px;margin-left:-50px; padding-right: 30px; cursor: pointer;">
<span id="dropdownIcon" class="dropdown-icon" style="position: absolute; top: 50%; left: 370px; transform: translateY(-50%); cursor: pointer;">&#9660;</span>
</div>
<div id="clientDetailsContainer" style="display: none;">
<ul>
<div class="nav">
<li><a href="#" class="btn" onclick="loadPage('Admin_client2.html')">Client Details:</a></li>
<li><a href="#" class="btn" onclick="loadPage('Existing_client.html')">Existing Client</a></li>
<li><a href="#" class="btn" onclick="loadPage('Admin_client.html')">Add New Client</a></li>
<li><a href="#" class="btn" onclick="loadPage('page4.html')">Add Sub Clients/Locations</a></li>
</div>
</ul>
<div id="emailCheckbox" style="display: none; margin-top:10px;position: absolute;width: 80%;z-index: 100; background: #ffffff; border: 1px solid #d3d3d3;border-radius: 5px;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);padding: 10px; margin-top: 5px;overflow-y: auto; max-height: 150px;
">
{% for team in teams %}
<label style="display: block; margin-bottom: 10px;">
<input type="checkbox" value="{{ team }}" class="email-checkbox"> {{ team }}
</label>
{% endfor %}
</div>
</div>
<input type="text" id="subject" style="width: 128%; height:40px; margin-top: 0px; border: none; border-bottom: 1px solid #d3d3d3;margin-left:-50px;text-indent: 10px;" class="compose-input" placeholder="Subject">
<textarea id="Message" style="width: 128%; height: 220px; margin-top: 1px; text-indent: 10px; border: none;padding-top: 10px; border-bottom: 1px solid #d3d3d3;margin-left:-50px;border-bottom:1px solid solid black"
class="compose-input" placeholder="Message"></textarea>
<!-- Footer -->
<div class="" style="display: flex; justify-content: space-between; align-items: center; margin-top:20px; margin-right:100px;">
<button type="button" class="compose-button send" id="sendButton"
style="margin-left:-30px; border-radius: 10px; color:white; background:#1F51FF;font-size:17px; border: none; width:70px; height:30px; font-weight:bold; letter-spacing:1px;"
onclick="sendMessage(event)">Send</button>
<button class="compose-button discard" id="Discardbutton" style="margin-left:50px; border-radius: 10px; border: none; position:fixed; color:white; background:#1F51FF;font-size:17px; width:80px; height:30px; font-weight:bold; letter-spacing:1px;" onclick="CloseCompose(event)">Discard</button>
<i class="fa-solid fa-paperclip" onclick="document.getElementById('attachment').click();" style="color: #1F51FF; margin-right:90px; font-size:23px; cursor: pointer;"></i>
<input type="file" id="attachment" name="attachment" style="display: none;" accept=".pdf,.doc,.docx,.txt">
</div>
</div>
</form>
</div>
</div>
<form id="ComposeReply" method="post" action="{% url 'send_reply' %}?user_id={{ user_id }}">
<div id="ComposeBox" style="display: none;padding: 20px; margin-top: 20px; border: 1px solid #ccc;">
<h3>Reply to: ${message.sender_id || message.recipient_id}</h3>
<input type="text" id="toField" name="toField" value="${message.sender_id || message.recipient_id}" readonly style="width: 100%; margin-bottom: 10px;" />
<textarea id="subject" name="subject" rows="2" placeholder="Subject" style="width: 100%; margin-bottom: 10px;"></textarea>
<textarea id="message" name="message" rows="4" placeholder="Message" style="width: 100%; margin-bottom: 10px;"></textarea>
<input type="file" id="attachment" name="attachment" accept=".pdf,.doc,.docx"><br>
<button type="button" onclick="sendReply(event, '${message.message_id}')">Send</button>
</div>
<iframe id="contentFrame" style="width: 100%; height: 80vh; border: none; display: none;"></iframe>
<script>
function loadPage(pageName) {
var iframe = document.getElementById('contentFrame');
iframe.src = pageName;
iframe.style.display = 'block'; // Show the iframe
}
function activateLink(link) {
var links = document.querySelectorAll('.sidebar-link');
links.forEach(function(item) {
item.classList.remove('active');
});
link.classList.add('active');
}
</script>
</section>
</form>
</body>
</html>

@ -2,18 +2,22 @@
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=\, initial-scale=1.0">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet" >
<link href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0-beta3/css/all.min.css" rel="stylesheet">
<link rel="stylesheet" />
<title>Document</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet" />
<style>
.Messages{
.Messages{
.header {
display: flex;
align-items: center;
justify-content: space-between;
border-bottom: 1px solid whitesmoke;
position:fixed;
}
.sidebarOption {
@ -33,17 +37,18 @@
.sidebarOption h3 {
flex: 1;
margin-left: 10px;
font-size: 14px;
font-size: 16px;
font-weight: 400;
color:#000000;
}
.sidebarOption:hover,
.sidebarOption:hover h3{
width:100px;
}
.sidebarOption:hover,
.sidebarOption:hover h3,
<!--.sidebarOption:hover,-->
<!--.sidebarOption:hover h3,-->
.sidebarOption:hover .material-icons,
.sidebarOption__active,
<!--.sidebarOption__active,-->
.sidebarOption__active h3,
.sidebarOption__active .material-icons {
background-color: #fcecec;
@ -51,23 +56,7 @@ width:100px;
}
.sidebar__footer {
display: flex;
justify-content: center;
}
.sidebar__footerIcons .material-icons {
margin: 7px;
cursor: pointer;
}
.header__middle {
display: flex;
flex: 0.7;
align-items: center;
background-color: whitesmoke;
padding: 10px;
border-radius: 5px;
}
.material-icons {
color: gray;
@ -76,23 +65,13 @@ width:100px;
.header__middle input {
border: none;
width: 100%;
width: 10%;
padding: 10px;
outline: none;
font-size: medium;
border-radius:50%;
background-color: transparent;
}
.header__right {
display: flex;
padding-right: 20px;
}
.header__right span {
padding: 10px;
cursor: pointer;
}
.compose {
margin-top: 15px;
margin-left: 10px;
@ -122,46 +101,465 @@ width:100px;
.compose:hover {
opacity: 0.9; /* Optional hover effect */
}
.emailRow__options {
display: flex;
align-items: center;
}
.emailRow__options .material-icons,
input {
margin: 5px;
}
.emailRow {
display: flex;
align-items: center;
height: 50px;
border-bottom: 1px solid whitesmoke;
cursor: pointer;
z-index: 999;
}
.emailRow:hover {
border-top: 1px solid whitesmoke;
box-shadow: 0 4px 4px -2px rgba(0, 0, 0, 0.24);
}
.emailRow__message {
display: flex;
flex: 0.8;
align-items: center;
font-size: 13px;
}
.emailRow__message h4 {
width: 400px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
padding-left: 5px;
padding-right: 5px;
}
.emailRow__title {
font-size: 13px;
flex: 0.3;
}
.emailRow__time {
padding-right: 15px;
font-size: 10px;
font-weight: bold;
}
.emailRow__description {
font-weight: 400;
color: gray;
}
/* Header */
.compose-header {
background-color: #f5f5f5;
padding: 12px 16px;
display: flex;
justify-content: space-between;
align-items: center;
cursor: grab; /* For drag-and-drop effect */
border-bottom: 1px solid #ddd;
margin-top:500px;
}
.compose-header:hover {
background-color: #eaeaea;
}
.compose-title {
font-size: 16px;
font-weight: bold;
color: #333;
margin: 0;
}
.compose-close {
font-size: 18px;
color: #666;
cursor: pointer;
transition: color 0.2s;
}
.compose-close:hover {
color: #000;
}
/* Body */
.compose-body {
padding: 12px 16px;
flex-grow: 1;
display: flex;
flex-direction: column;
gap: 12px;
}
/* Input Fields */
.compose-input {
width: 100%;
border: 1px solid #ddd;
padding: 8px;
border-radius: 4px;
font-size: 14px;
color: #333;
}
.compose-input:focus {
border-color: #4285f4;
outline: none;
}
<!--/* Buttons */-->
<!--.compose-button {-->
<!-- padding: 8px 16px;-->
<!-- font-size: 14px;-->
<!-- border: none;-->
<!-- width:10px;-->
<!-- border-radius: 4px;-->
<!-- cursor: pointer;-->
<!-- transition: background-color 0.3s;-->
<!--}-->
<!--.compose-button.send {-->
<!-- background-color: #4285f4;-->
<!--}-->
<!--.compose-button.send:hover {-->
<!-- background-color: #357ae8;-->
<!--}-->
<!--.compose-button.discard {-->
<!-- background-color: #f5f5f5;-->
<!-- color: #333;-->
<!--}-->
.compose-button.discard:hover {
background-color: red;
}
/* Style for the whole button */
.sidebar__compose {
text-transform: capitalize;
color: gray;
padding: 15px;
border-radius: 20px;
background-color: white;
display: flex; /* Flexbox ensures horizontal alignment */
align-items: center; /* Vertically aligns items */
cursor: pointer;
outline: none;
border: none;
box-shadow: 0px 2px 5px -2px rgba(0, 0, 0, 0.75);
}
/* Style for the 'add' icon */
.sidebar__compose .material-icons {
margin-right: 8px; /* Add space between the icon and text */
vertical-align: middle; /* Ensures the icon is vertically aligned with text */
font-size: 20px; /* You can adjust the size of the icon here */
}
/* Style for the hover effect */
.sidebar__compose:hover {
background-color: #87CEFA;
transition-duration: 0.4s;
}
.sidebarOption:hover {
background-color: #6699CC;
color:black;
}
.sidebarOption__active {
background-color: #6699CC;
color: black;
border-radius: 5px;
width:100px;
}
.material-icons-black span {
color: black;
}
.search {
margin-left: 200px;
margin-top: -30px;
background-color: #F1F4F8;
border-radius: 50px;
width: 400px;
padding: 5px;
display: flex;
align-items: center;
justify-content: space-between;
}
.search input {
background-color: transparent;
outline: none;
border: none;
text-indent: 15px;
width: 85%;
}
.search button {
outline: none;
border: none;
border-radius: 50%;
background-color: #fff;
padding: 8px;
display: flex;
align-items: center;
justify-content: center;
}
.header__right span{
padding:20px;
cursor:pointer;
}
.email-checkbox + label {
margin-right: 15px;
}
#emailCheckbox {
display: none; /* Hidden by default */
position: absolute; /* Position it relative to its container */
top: 100%; /* Position it just below the input field */
left: 0; /* Align it with the left of the input field */
z-index: 10; /* Ensure it appears above other elements */
background-color: white; /* Background color for better visibility */
border: 1px solid #ccc; /* Add a border for separation */
padding: 10px; /* Add some spacing inside the dropdown */
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* Add a subtle shadow */
width: auto; /* Adjust width as needed */
height:auto;
}
#dropdownIcon {
position: relative; /* Position it relative to its container */
cursor: pointer;
}
.email-input-container {
position: relative; /* The dropdown will be positioned relative to this container */
}
#BackToInboxButton {
margin-bottom: 10px;
background: none;
border: none;
cursor: pointer;
display: inline-flex;
align-items: center;
padding: 5px;
}
/* Icon style with the hover background color */
#BackToInboxButton i {
font-size: 20px;
}
/* Initially hide the text and make it transparent */
#BackToInboxButton span {
opacity: 0;
margin-left: 8px;
font-size: 16px;
background-color: black;
color: white;
padding: 2px;
transition: opacity 0.3s ease; /* Smooth opacity transition */
}
/* On hover, show the text and add background color to the arrow */
#BackToInboxButton:hover i {
background-color: #d3d3d3; /* Gray background color on hover */
border-radius: 5%; /* Round the corners of the background */
}
#BackToInboxButton:hover span {
opacity: 1; /* Make text fully visible on hover */
}
html, body {
height: 100%;
margin: 0;
overflow: hidden;
}
.Messages {
height: 100%;
overflow: hidden; /* This ensures the whole page doesn't scroll */
}
.main__body {
margin-left: 300px;
margin-top: 120px;
position: sticky;
top: 0;
z-index: 10;
background-color: white;
max-height: calc(100vh - 160px);
padding-right: 20px; /* Optional: to ensure right-side content doesn't hide under scrollbar */
}
body {
margin: 0;
padding: 0;
font-family: Arial, sans-serif;
overflow-y: auto;
height: 100vh;
}
.sticky-container {
position: sticky;
top: 0;
background-color: white;
z-index: 100;
padding: 20px;
}
}
</style>
</head>
<div class="Messages">
<body style="width:100px;">
<div class="header" style="margin-top:100px;width:80.5%; margin-left:300px;">
<div class="header__middle">
<span class="material-icons">search</span>
<input type="text" style="border-radius: 50%;" placeholder="Search mail">
</div>
<div class="header__right">
<body>
<div class="Messages ">
<form id="messageForm " style="" >
{% csrf_token %}
<input type="hidden" id="user_id" name="user_id" value="{{ user_id }}">
<input type="hidden" id="current_user_id" name="current_user_id" value="{{ current_user_id }}">
<div class="" style="margin-top:0px; width:70.5%; margin-left:300px; ">
</div>
<div class="main__body" style="margin-left: 300px; margin-top: -10px; ">
<div class="" style="margin-top: 10px;">
<div class="sticky-container" style="position: sticky; top: 0px">
<!-- Compose Button -->
<h2 style="margin-left:30px; margin-top:10px; ">Messages</h2>
<div class="search">
<input type="text" placeholder="Search">
<button><i class="ri-search-2-line"></i></button>
</div>
<div class="header__right" style="margin-left:650px; margin-top:-50px">
<span class="material-icons">apps</span>
<span class="material-icons">notifications</span>
<span class="material-icons">account_circle</span>
</div>
<button class="sidebar__compose" id="compose" onclick="showCompose(event)" style="display: block; margin-bottom: 20px;">
<span class="material-icons">add</span> Compose
</button>
<!-- Sidebar Options -->
<div class="sidebarOption-container" style="position: sticky; top: 20px; margin-top: -140px; margin-left: 150px;">
<div class="sidebarOption sidebarOption__active" style="width: 100px;" id="inboxButton" onclick="setActiveBox(this); renderInbox(); HideMessageDetails()">
<span class="material-icons" style="color:black;">inbox</span>
<h3>Inbox</h3>
</div>
<div class="sidebarOption" id="sentButton" style="margin-top: -32px; margin-left: 130px;" onclick="setActiveBox(this); renderSentMessages(); HideMessageDetails()">
<span class="material-icons" style="color:black;">near_me</span>
<h3>Sent</h3>
</div>
<!-- <div class="sidebarOption" style="margin-top: -37px; margin-left: 250px; margin-bottom: 20px;" onclick="setActiveBox(this)">-->
<!-- <span class="material-icons" style="color:black;">delete</span>-->
<!-- <h3>Trash</h3>-->
<!-- </div>-->
</div>
</div>
<div id="InboxContainer" style="display:block;margin-left: 0px; top:100px; "></div>
<div class="main__body" style="margin-left: 300px; margin-top: 30px;">
<div id="MessageDetailsContainer" style="display: none; height:auto; padding: 20px; border: 1px solid #ddd; margin-left: 0px; margin-top:10px; border:none; width:100%; box-shadow: rgba(100, 100, 111, 0.2) 0px 7px 29px 0px; border-radius:20px;">
<button id="BackToInboxButton" onclick="backToInbox(event)">
<i class="fa fa-arrow-left"></i> <!-- Back arrow icon -->
<span id="BackToInboxText">Back to Inbox</span> <!-- Text to show on hover -->
</button>
<div class="" style="margin-top:200px;">
<button class="compose active"><span class="material-icons" > add </span>Compose</button>
<div id="MessageDetails" style="height: auto;"></div>
<div id="ConversationThread" style="height:auto;"></div>
</div>
</div>
</div>
</div>
</form>
<div class="compose-container" id="ComposeContainer"
style="display: none; position: fixed; top:230px; bottom: 10%; right: 5%; width: 30%; height:58%; background-color: #fff; border-radius: 30px; box-shadow: 3px 3px 3px 3px rgba(0.5, 0.5, 0.5, 0.5); padding: 30px; z-index: 1000; margin-top: 50px;">
<div class="container" style="padding: 20px;width:100%">
<form id="ComposeMessage" method="post" action="{% url 'send_message' %}?user_id={{ user_id }}">
{% csrf_token %}
<div class="fixed" style="margin-buttom:10px; width:100%">
<div class="compose-header" style="display: flex; margin-left:-50px; space-between; align-items: center; background-color:#F0F8FF; width:128%; margin-top:-50px;">
<span style="font-size: 18px; font-weight: bold; margin-left:20px;" >New Message</span>
<div class="compose-close" id="Closebutton" onclick="CloseCompose(event)" style="font-size: 30px; margin-left:290px; color: red; cursor: pointer;">&times;</div>
</div>
<div class="compose-body">
<div style="position: relative;">
<div id="emailBarsContainer" style="margin-top: 10px; display: flex; flex-wrap: wrap;"></div>
<!-- Searchable Input Field -->
<input type="text" id="toField" placeholder="Search Emails..." style="width: 128%; text-indent: 10px; border: none; border-bottom: 1px solid #d3d3d3; height: 40px; margin-left: -50px; padding-right: 30px; cursor: pointer;">
<!-- Dropdown Icon -->
<span id="dropdownIcon" class="dropdown-icon" style="position: absolute; top: 50%; left: 370px; transform: translateY(-50%); cursor: pointer;">&#9660;</span>
</div>
<div class="inbox" style="margin-top:-135px; margin-left:150px;">
<div class="sidebarOption sidebarOption__active" style="width:100px;">
<span class="material-icons"> inbox </span>
<h3>Inbox</h3>
</div>
<div class="sidebarOption" style=" margin-top:-32px; margin-left:130px;">
<span class="material-icons" > near_me </span>
<h3>Sent</h3>
</div>
<!-- Searchable Email List -->
<div id="emailCheckbox" style="display: none; position: absolute; width: 80%; z-index: 100; background: #ffffff; border: 1px solid #d3d3d3; border-radius: 5px;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); padding: 10px; overflow-y: auto; max-height: 150px; margin-top: 5px;">
<div class="sidebarOption" style=" margin-top:-37px; margin-left:250px;">
<span class="material-icons">delete</span>
<h3>Trash</h3>
</div>
<!-- Loop through emails -->
{% for team in teams %}
<label class="email-item" style="display: block; margin-bottom: 10px;">
<input type="checkbox" value="{{ team }}" class="email-checkbox"> {{ team }}
</label>
{% endfor %}
</div>
</div>
</div>
<input type="text" id="subject" style="width: 128%; height:40px; margin-top: 0px; border: none; border-bottom: 1px solid #d3d3d3;margin-left:-50px;text-indent: 10px;" class="compose-input" placeholder="Subject">
<textarea id="Message" style="width: 128%; height: 220px; margin-top: 1px; text-indent: 10px; border: none;padding-top: 10px; border-bottom: 1px solid #d3d3d3;margin-left:-50px;border-bottom:1px solid solid black"
class="compose-input" placeholder="Message"></textarea>
<!-- Footer -->
<div class="" style="display: flex; justify-content: space-between; align-items: center; margin-top:20px; margin-right:100px;">
<button type="button" class="compose-button send" id="sendButton"
style="margin-left:-20px; border-radius: 10px; border: none; position:fixed; color:white; background:#1F51FF;font-size:17px; width:60px; height:30px; font-weight:bold; letter-spacing:1px;" onclick="sendMessage(event)">Send</button>
<button class="compose-button discard" id="Discardbutton" style="margin-left:50px; border-radius: 10px; border: none; position:fixed; color:white; background:#1F51FF;font-size:17px; width:80px; height:30px; font-weight:bold; letter-spacing:1px;" onclick="CloseCompose(event)">Discard</button>
<!-- <i class="fa-solid fa-paperclip" onclick="document.getElementById('attachment').click();" style="color: #1F51FF; margin-right:90px; font-size:23px; cursor: pointer;"></i>-->
<input type="file" id="attachment" name="attachment" style="margin-left:150px;" accept=".pdf,.doc,.docx,.txt">
</div>
</div>
</form>
</div>
</div>
<form id="ComposeReply" method="post" action="{% url 'send_reply' %}?user_id={{ user_id }}">
<div id="ComposeBox" style="display: none;padding: 20px; margin-top: 20px; border: 1px solid #ccc;">
<h3>Reply to: ${message.sender_id || message.recipient_id}</h3>
<input type="text" id="toField" name="toField" value="${message.sender_id || message.recipient_id}" readonly style="width: 100%; margin-bottom: 10px;" />
<textarea id="subject" name="subject" rows="2" placeholder="Subject" style="width: 100%; margin-bottom: 10px;"></textarea>
<textarea id="message" name="message" rows="4" placeholder="Message" style="width: 100%; margin-bottom: 10px;"></textarea>
<input type="file" id="attachment" name="attachment" accept=".pdf,.doc,.docx"><br>
<button type="button" onclick="sendReply(event, '${message.message_id}')">Send</button>
</div>
</div></div>
</div></body>
</html>
</form>
</body>
</html>
</html>

File diff suppressed because it is too large Load Diff

@ -0,0 +1,38 @@
# Generated by Django 4.2.11 on 2025-01-08 06:29
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('skyonnadmin', '0064_alter_internalteam_email1'),
('user', '0027_jobposting_maxamount_jobposting_minamount_and_more'),
]
operations = [
migrations.AlterField(
model_name='jobposting',
name='timePeriod',
field=models.CharField(blank=True, max_length=20, null=True),
),
migrations.CreateModel(
name='Message',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('message_id', models.CharField(max_length=255, null=True, unique=True)),
('subject', models.CharField(max_length=255)),
('body', models.TextField()),
('is_read', models.BooleanField(default=False)),
('sent_at', models.DateTimeField(auto_now_add=True)),
('sender_fullname', models.CharField(max_length=255, null=True)),
('recipient_fullname', models.CharField(max_length=255, null=True)),
('recipient', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='received_messages', to='skyonnadmin.internalteam', to_field='Email1')),
('sender', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='sent_messages', to='skyonnadmin.internalteam', to_field='Email1')),
],
options={
'ordering': ['-sent_at'],
},
),
]

@ -0,0 +1,22 @@
# Generated by Django 4.2.11 on 2025-01-09 06:18
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('user', '0028_alter_jobposting_timeperiod_message'),
]
operations = [
migrations.CreateModel(
name='Attachment',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('file', models.FileField(upload_to='attachments/')),
('message', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='attachments', to='user.message')),
],
),
]

@ -0,0 +1,16 @@
# Generated by Django 4.2.11 on 2025-01-09 06:26
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('user', '0029_attachment'),
]
operations = [
migrations.DeleteModel(
name='Attachment',
),
]

@ -0,0 +1,18 @@
# Generated by Django 4.2.11 on 2025-01-09 06:35
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('user', '0030_delete_attachment'),
]
operations = [
migrations.AddField(
model_name='message',
name='attachment',
field=models.FileField(blank=True, null=True, upload_to='attachments/'),
),
]

@ -0,0 +1,54 @@
# Generated by Django 4.2.11 on 2025-01-24 04:26
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('skyonnadmin', '0065_internalteam_fullname'),
('user', '0031_message_attachment'),
]
operations = [
migrations.AddField(
model_name='message',
name='is_trashed',
field=models.BooleanField(default=False),
),
migrations.AddField(
model_name='message',
name='reply_to',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='replies', to='user.message', to_field='message_id'),
),
migrations.AddField(
model_name='message',
name='seen_time',
field=models.DateTimeField(blank=True, null=True),
),
migrations.AlterField(
model_name='message',
name='attachment',
field=models.FileField(blank=True, null=True, upload_to='attachment/'),
),
migrations.RemoveField(
model_name='message',
name='recipient',
),
migrations.CreateModel(
name='FileDownload',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('first_downloaded', models.DateTimeField(auto_now_add=True)),
('last_downloaded', models.DateTimeField(auto_now=True)),
('message', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='downloads', to='user.message', to_field='message_id')),
('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='skyonnadmin.internalteam', to_field='Email1')),
],
),
migrations.AddField(
model_name='message',
name='recipient',
field=models.ManyToManyField(related_name='received_messages', to='skyonnadmin.internalteam'),
),
]

@ -0,0 +1,25 @@
# Generated by Django 4.2.11 on 2025-01-31 06:06
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('skyonnadmin', '0065_internalteam_fullname'),
('user', '0032_message_is_trashed_message_reply_to_and_more'),
]
operations = [
migrations.CreateModel(
name='MessageStatus',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('is_read', models.BooleanField(default=False)),
('seen_time', models.DateTimeField(blank=True, null=True)),
('message', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='status', to='user.message', to_field='message_id')),
('user', models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='skyonnadmin.internalteam', to_field='Email1')),
],
),
]

@ -1,6 +1,7 @@
from django.db import models
from django.contrib.auth.models import AbstractBaseUser, BaseUserManager
from django.contrib.auth.models import User
from skyonnadmin.models import InternalTeam
class CustomUserManager(BaseUserManager):
def create_user(self, phone, password=None, **extra_fields):
@ -96,5 +97,39 @@ class JobPosting(models.Model):
# def __str__(self):
# return self.name
class Message(models.Model):
message_id = models.CharField(unique=True,max_length=255, null=True)
sender = models.ForeignKey(InternalTeam, related_name='sent_messages',to_field='Email1', on_delete=models.CASCADE)
recipient = models.ManyToManyField(InternalTeam, related_name='received_messages')
subject = models.CharField(max_length=255)
body = models.TextField()
is_read = models.BooleanField(default=False)
sent_at = models.DateTimeField(auto_now_add=True)
sender_fullname = models.CharField(max_length=255,null=True)
recipient_fullname = models.CharField(max_length=255,null=True)
reply_to = models.ForeignKey('self',null=True,blank=True,on_delete=models.CASCADE,related_name='replies', to_field='message_id')
is_trashed = models.BooleanField(default=False)
attachment = models.FileField(upload_to='attachment/',null=True,blank=True)
seen_time = models.DateTimeField(null=True, blank=True)
def __str__(self):
return f'{self.message_id}'
class Meta:
ordering = ['-sent_at']
class MessageStatus(models.Model):
message = models.ForeignKey(Message, on_delete=models.CASCADE, to_field='message_id', related_name='status')
user = models.ForeignKey(InternalTeam, on_delete=models.CASCADE, to_field='Email1', null=True)
is_read = models.BooleanField(default=False)
seen_time = models.DateTimeField(null=True, blank=True)
class FileDownload(models.Model):
# Reference to the message containing the attachment
message = models.ForeignKey(Message, on_delete=models.CASCADE,to_field='message_id', related_name='downloads')
user = models.ForeignKey(InternalTeam, on_delete=models.CASCADE,to_field='Email1',null=True)
first_downloaded = models.DateTimeField(auto_now_add=True)
last_downloaded = models.DateTimeField(auto_now=True)
def __str__(self):
return f"Download details for message_id: {self.message.message_id}"

@ -26,4 +26,16 @@ urlpatterns = [
# path('messages/new/', views.new_message, name='new_message'),
path('messages/', views.messages, name='messages'),
path('send_message/', views.send_message, name='send_message'),
path('update_message_read_status/<str:message_id>/', views.update_message_read_status,name='update_message_read_status'),
path('unread_message_count/', views.unread_message_count, name='unread_message_count'),
path('render_messages/', views.render_messages, name='render_messages'),
path('render_sent_messages/', views.render_sent_messages, name='render_sent_messages'),
path('send_reply/', views.send_reply, name='send_reply'),
path('fetch_replies/<str:message_id>/', views.fetch_replies, name='fetch_replies'),
path('render_trash_messages/', views.render_trash_messages, name='render_trash_messages'),
path('track_download/<str:message_id>/', views.track_download, name='track_download'),
path('fetch_download_timestamps/<str:message_id>/', views.fetch_download_timestamps,name='fetch_download_timestamps'),
path('get_message_status/<str:message_id>/', views.get_message_status, name='get_message_status'),
path('get_main_message_id/<str:message_id>/', views.get_main_message_id, name='get_main_message_id'),
]

@ -1,6 +1,6 @@
from django.shortcuts import render, redirect, get_object_or_404
from django.contrib.auth import authenticate, login as auth_login, logout
from .models import CustomUser, JobPosting
from .models import CustomUser, JobPosting,Message,FileDownload,MessageStatus
from skyonnadmin.models import ClientDetails,SubcompanyDetails,Locations,AddContact,InternalTeam
from django.contrib import messages
from django.http import JsonResponse
@ -12,6 +12,11 @@ import os
from django.core.files.storage import FileSystemStorage
import json
from django.core.serializers.json import DjangoJSONEncoder
import uuid
from django.utils.timezone import now,localtime
import pytz
ist_timezone = pytz.timezone('Asia/Kolkata')
def home(request):
return render(request, 'user/home.html')
@ -313,15 +318,44 @@ def create_recruiter(request):
return render(request, 'user/add_recruiter.html')
def send_message(request, recipient_id):
recipient = get_object_or_404(User, pk=recipient_id)
if request.method == 'POST':
subject = request.POST.get('subject')
body = request.POST.get('body')
message = Message.objects.create(sender=request.user, recipient=recipient, subject=subject, body=body)
return redirect('messages')
return render(request, 'send_message.html', {'recipient': recipient})
# def send_message(request, recipient_id):
# recipient = get_object_or_404(User, pk=recipient_id)
#
# if request.method == 'POST':
# # Retrieve form data
# subject = request.POST.get('subject')
# body = request.POST.get('body')
# # attachment = request.FILES.get('attachment') # Retrieve file if uploaded
#
# attachment = None
# if request.FILES.get('attachment'):
# myfile = request.FILES['attachment']
# file_name = os.path.splitext(myfile.name)[0] # Get the file extension
# filename = file_name # Create the new filename using job_id
# fs = FileSystemStorage()
# filename = fs.save(filename, myfile)
# attachment = fs.url(filename) # Get the URL to the file
# print(attachment)
#
#
# # Create a new message object
# message = Message.objects.create(
# sender=request.user, # Current logged-in user
# recipient=recipient, # Recipient
# subject=subject, # Subject
# body=body, # Body of the message
# attachment=attachment
# )
#
# return JsonResponse({
# 'status': 'success',
# 'message': 'Message sent successfully.',
# 'message_id': message.id, # Optional: send back message ID or any other info
# 'attachment_url': attachment, # Send file URL if uploaded
# })
#
# # If GET request, show the message composition page
# return render(request, 'send_message.html', {'recipient': recipient})
def mark_as_read(request, message_id):
message = get_object_or_404(Message, pk=message_id, recipient=request.user)
@ -483,6 +517,676 @@ def get_jobPosting(request):
return JsonResponse({'error': 'Invalid request method'}, status=400)
def messages(request):
user_id= request.GET.get('úser_id')
return render(request, 'user/Messages.html',{'user_id': user_id})
user_id = request.GET.get('user_id','')
current_user_id = InternalTeam.objects.filter(user_id=user_id).values_list('Email1', flat=True).first()
teams = InternalTeam.objects.exclude(Email1=current_user_id)
return render(request,'user/messages.html',{'user_id': user_id,'current_user_id': current_user_id, 'teams':teams})
def generate_unique_message_id(firstname):
"""Generates a unique message_id by checking its existence in the database."""
prefix = firstname[:3].upper()
while True:
unique_id = str(uuid.uuid4().int)[:4] # Generate a random 4-digit number
message_id = prefix + unique_id
if not Message.objects.filter(message_id=message_id).exists():
print(message_id,'created message_id')
return message_id
def send_message(request):
if request.method == 'POST':
try:
current_user = request.GET.get('user_id', None)
current_user_id = InternalTeam.objects.filter(user_id=current_user).values_list('Email1', flat=True).first()
recipients = request.POST.get('toField').split(',') # Multiple emails separated by commas
print(recipients)
subject = request.POST.get('subject')
body = request.POST.get('message')
attachment = request.FILES.get('attachment') # Using .get() to avoid KeyError
attachment_url = None
if attachment:
myfile = request.FILES['attachment']
file_name, file_extension = os.path.splitext(myfile.name)
fs = FileSystemStorage()
# Save the file with a unique filename
filename = fs.save(file_name, attachment)
attachment_url = fs.url(filename)
sender = current_user_id
sender_id = get_object_or_404(InternalTeam, Email1=sender)
sender_fullname = sender_id.FullName
firstname = InternalTeam.objects.filter(user_id=current_user).values_list('FirstName', flat=True).first()
message_id = generate_unique_message_id(firstname)
recipient_fullnames = []
recipient_emails = []
for recipient_email in recipients: # Split by comma
recipient_id = get_object_or_404(InternalTeam, Email1=recipient_email)
print(recipient_id)
recipient_emails.append(recipient_id)
recipient_fullnames.append(recipient_id.FullName)
recipient_details = [{'email': r.Email1, 'fullname': r.FullName} for r in recipient_emails]
message = Message(
sender=sender_id,
subject=subject,
body=body,
is_read=False,
sender_fullname=sender_fullname,
recipient_fullname=", ".join(recipient_fullnames),
message_id=message_id,
attachment=attachment
)
message.save()
message.recipient.set(recipient_emails)
return JsonResponse({
'status': 'success',
'message': 'Message sent successfully.',
'details': {
'subject': subject,
'recipients': recipient_details,
'sender_fullname': sender_fullname,
'recipient_fullname': ",".join(recipient_fullnames),
'body': body,
'sent_at': message.sent_at.strftime('%Y-%m-%d %H:%M:%S'),
'message_id': message.message_id,
'attachment': attachment_url
}
})
except Exception as e:
print("Error occurred:", str(e)) # Print error to console for debugging
return JsonResponse({'status': 'error', 'message': 'Failed to send message'}, status=500)
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
def unread_message_count(request):
user_id = request.GET.get('user_id')
print(user_id)
user = get_object_or_404(InternalTeam, user_id=user_id)
unread_count = Message.objects.filter(recipient=user, is_read=False).count()
return JsonResponse({'unread_count': unread_count})
def render_messages(request):
current_user = request.GET.get('user_id', None)
current_user_id = get_object_or_404(InternalTeam, user_id=current_user)
# Fetch all messages for the user
inbox_messages = Message.objects.filter(
Q(recipient=current_user_id),
is_trashed=False
).order_by('-sent_at')
# Dictionary to group messages by the main message
grouped_messages = {}
for message in inbox_messages:
# Find the main message for grouping
main_message = message
while main_message.reply_to:
main_message = main_message.reply_to
# Check if the main message already has a group
if main_message.id not in grouped_messages:
grouped_messages[main_message.id] = {
'main_message': main_message,
'replies': [],
}
# Add the current message to the main message's replies
if message != main_message:
grouped_messages[main_message.id]['replies'].append(message)
# Prepare the response
messages_data = []
for group in grouped_messages.values():
main_message = group['main_message']
# Sort replies by sent_at
sorted_replies = sorted(group['replies'], key=lambda x: x.sent_at)
# Function to get attachment details
def get_attachment_data(attachment):
if not attachment:
return []
try:
fs = FileSystemStorage()
attachment_url = fs.url(attachment.name)
file_path = fs.path(attachment.name)
file_size = os.path.getsize(file_path)
return [{
'filename': attachment.name.split('/')[-1],
'url': attachment_url,
'size': file_size
}]
except Exception as e:
print(f"Error retrieving attachment: {e}")
return []
def get_first_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.first_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
def get_last_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.last_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
def get_message_status(message, user):
status = MessageStatus.objects.filter(message=message, user=user).first()
if status:
return status.is_read, localtime(status.seen_time).strftime('%m/%d/%Y, %I:%M:%S %p') if status.seen_time else "00:00:00"
return False, "00:00:00"
replies_data = [{
'sender_fullname': reply.sender.FullName,
'body': reply.body,
'sent_at': localtime(reply.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': reply.message_id,
'parent_message_id': reply.reply_to.message_id if reply.reply_to else None,
'main_message_id': main_message.message_id, # Ensures main message ID is always available
'is_read': get_message_status(reply, current_user_id)[0],
'seen_time': get_message_status(reply, current_user_id)[1],
'attachment': get_attachment_data(reply.attachment),
'first_downloaded': get_first_downloaded(reply.message_id),
'last_downloaded': get_last_downloaded(reply.message_id),
} for reply in sorted_replies]
latest_message = sorted_replies[-1] if sorted_replies else main_message
messages_data.append({
'sender_fullname': main_message.sender.FullName,
'sender_id': main_message.sender_id,
'recipient_fullnames': [recipient.FullName for recipient in main_message.recipient.all()],
'recipient_emails': [recipient.Email1 for recipient in main_message.recipient.all()],
'subject': main_message.subject,
'body': main_message.body,
'sent_at': localtime(main_message.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': main_message.message_id,
'parent_message_id': main_message.reply_to.message_id if main_message.reply_to else None,
'main_message_id': main_message.message_id, # Ensures the root message ID is stored
'is_read': get_message_status(main_message, current_user_id)[0],
'seen_time': get_message_status(main_message, current_user_id)[1],
'attachment': get_attachment_data(main_message.attachment),
'replies': replies_data,
'latest_message': {
'sender_fullname': latest_message.sender.FullName,
'subject': latest_message.subject,
'body': latest_message.body,
'sent_at': localtime(latest_message.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': latest_message.message_id,
'is_read': get_message_status(latest_message, current_user_id)[0],
'seen_time': get_message_status(latest_message, current_user_id)[1],
'attachment': get_attachment_data(latest_message.attachment),
'first_downloaded': get_first_downloaded(latest_message.message_id),
'last_downloaded': get_last_downloaded(latest_message.message_id),
}
})
return JsonResponse({'inbox_messages': messages_data})
def render_sent_messages(request):
current_user = request.GET.get('user_id', None)
current_user_id = get_object_or_404(InternalTeam, user_id=current_user)
# Fetch all messages for the user
sent_messages = Message.objects.filter(
Q(sender=current_user_id),
is_trashed=False
).order_by('-sent_at')
# Dictionary to group messages by the main message
grouped_messages = {}
for message in sent_messages:
# Find the main message for grouping
main_message = message
while main_message.reply_to:
main_message = main_message.reply_to
# Check if the main message already has a group
if main_message.id not in grouped_messages:
grouped_messages[main_message.id] = {
'main_message': main_message,
'replies': [],
}
# Add the current message to the main message's replies
if message != main_message:
grouped_messages[main_message.id]['replies'].append(message)
# Prepare the response
messages_data = []
for group in grouped_messages.values():
main_message = group['main_message']
# Sort replies by sent_at
sorted_replies = sorted(group['replies'], key=lambda x: x.sent_at)
# Integrate get_attachment_data
def get_attachment_data(attachment):
if not attachment:
return []
try:
fs = FileSystemStorage()
attachment_url = fs.url(attachment.name)
file_path = fs.path(attachment.name)
file_size = os.path.getsize(file_path)
return [{
'filename': attachment.name.split('/')[-1],
'url': attachment_url,
'size': file_size
}]
except Exception as e:
print(f"Error retrieving attachment: {e}")
return []
def get_first_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.first_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
def get_last_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.last_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
replies_data = [{
'sender_fullname': reply.sender.FullName,
'body': reply.body,
'sent_at': localtime(reply.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': reply.message_id,
'is_read': reply.is_read,
'attachment': get_attachment_data(reply.attachment),
'first_downloaded': get_first_downloaded(reply.message_id),
'last_downloaded': get_last_downloaded(reply.message_id),
} for reply in sorted_replies]
latest_message = sorted_replies[-1] if sorted_replies else main_message
messages_data.append({
'sender_fullname': main_message.sender.FullName,
'sender_id': main_message.sender_id,
'recipient_fullnames': [recipient.FullName for recipient in main_message.recipient.all()],
'recipient_emails': [recipient.Email1 for recipient in main_message.recipient.all()],
'subject': main_message.subject,
'body': main_message.body,
'sent_at': localtime(main_message.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': main_message.message_id,
'is_read': main_message.is_read,
'attachment': get_attachment_data(main_message.attachment),
'replies': replies_data,
'latest_message': {
'sender_fullname': latest_message.sender.FullName,
'recipient_fullnames': [recipient.FullName for recipient in latest_message.recipient.all()],
'subject': latest_message.subject,
'body': latest_message.body,
'sent_at': localtime(latest_message.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'),
'message_id': latest_message.message_id,
'is_read': latest_message.is_read,
'attachment': get_attachment_data(latest_message.attachment),
'first_downloaded': get_first_downloaded(latest_message.message_id),
'last_downloaded': get_last_downloaded(latest_message.message_id),
}
})
return JsonResponse({'sent_messages': messages_data})
def update_message_read_status(request, message_id):
current_user_id = request.GET.get('user_id', None)
current_user = get_object_or_404(InternalTeam, user_id=current_user_id)
if request.method == 'POST':
message = get_object_or_404(Message, message_id=message_id)
# Ensure the current user is a recipient of the message
if current_user not in message.recipient.all():
return JsonResponse({'success': False, 'error': 'User is not a recipient of this message.'}, status=400)
# Fetch or create the read status for the current user
status, created = MessageStatus.objects.get_or_create(
message=message,
user=current_user,
defaults={
'is_read': False, # Initially set is_read as False
'seen_time': None # Initially set seen_time as None
}
)
# If the message is being read for the first time, update the read status
if not status.is_read:
status.is_read = True
status.seen_time = now()
status.save()
return JsonResponse({
'success': True,
'message': f'Message {message_id} updated successfully',
'is_read': status.is_read,
'seen_time': localtime(status.seen_time).strftime('%m/%d/%Y, %I:%M:%S %p') if status.seen_time else "00:00:00"
})
return JsonResponse({'success': False, 'error': 'Invalid request method'}, status=400)
def get_message_status(request, message_id):
message = get_object_or_404(Message, message_id=message_id)
current_user = request.GET.get('user_id')
current_user_id = get_object_or_404(InternalTeam,user_id=current_user)
recipient_status = []
for recipient in message.recipient.all():
seen_entry = MessageStatus.objects.filter(message=message, user=recipient).first()
seen_time = localtime(seen_entry.seen_time).strftime('%m/%d/%Y, %I:%M %p') if seen_entry else "not seen"
recipient_status.append({
'email': recipient.Email1,
'seen_time': seen_time
})
print(recipient_status)
return JsonResponse({'status': recipient_status})
def render_trash_messages(request):
current_user = request.GET.get('user_id', None)
current_user_id = get_object_or_404(InternalTeam, user_id=current_user)
# Filter trash messages for the current user
trash_messages = Message.objects.filter(Q(sender=current_user_id),is_trashed=True).order_by('-sent_at')
print(trash_messages,'trash_messages')
# Prepare message data for response
messages_data = [
{
'sender_fullname': message.sender.FullName, # Adjust to your model field
'sender_id': message.sender_id,
'recipient_fullname': message.recipient.FullName, # Adjust to your model field
'recipient_id': message.recipient_id,
'subject': message.subject,
'body': message.body,
'sent_at': localtime(message.sent_at).strftime('%m/%d/%Y, %I:%M:%S %p'), # Adjust as needed
'message_id': message.message_id,
'message_type': 'inbox' if message.recipient == current_user_id else 'sent',
'is_trashed': message.is_trashed
}
for message in trash_messages
]
return JsonResponse({'trash_messages': messages_data})
def send_reply(request):
if request.method == 'POST':
current_user = request.GET.get('user_id')
parent_message_id = request.POST.get('message_id') # The ID of the message being replied to
recipients = request.POST.get('toField', '').split(',')
subject = request.POST.get('subject', '')
body = request.POST.get('message', '')
attachment = request.FILES.get('attachment')
# Get sender info
sender = InternalTeam.objects.filter(user_id=current_user).values_list('Email1', flat=True).first()
sender_id = get_object_or_404(InternalTeam, Email1=sender)
sender_fullname = sender_id.FullName
# Fetch the parent message and get the sender of the parent message
parent_message = get_object_or_404(Message, message_id=parent_message_id)
parent_recipients = parent_message.recipient.exclude(user_id=current_user)
attachment_url = None
if attachment:
myfile = request.FILES['attachment']
file_name, file_extension = os.path.splitext(myfile.name)
fs = FileSystemStorage()
# Save the file with a unique filename
filename = fs.save(file_name, attachment)
attachment_url = fs.url(filename)
# Generate a random UUID and use its first 4 digits
firstname = InternalTeam.objects.filter(user_id=current_user).values_list('FirstName', flat=True).first()
message_id = generate_unique_message_id(firstname)
recipient_fullnames = []
recipient_emails = []
for recipient_email in recipients: # Split by comma
recipient_id = get_object_or_404(InternalTeam, Email1=recipient_email)
recipient_emails.append(recipient_id)
recipient_fullnames.append(recipient_id.FullName)
recipient_details = [{'email': r.Email1, 'fullname': r.FullName} for r in recipient_emails]
# Create the reply message to the sender of the parent message
reply = Message(
sender=sender_id,
# recipient=None, # Initially set to None, since we're not assigning directly
subject=subject,
body=body,
is_read=False,
sender_fullname=sender_fullname,
recipient_fullname=", ".join(recipient_fullnames),
message_id=message_id,
reply_to_id=parent_message,
attachment=attachment
)
reply.save()
print(reply)
# # Use .set() to assign the recipient to the reply
# all_recipients = list(parent_recipients) + [parent_message.sender]
# print(all_recipients,'all_recipients')
reply.recipient.set(recipient_emails)
# Return a success response with the reply message ID
return JsonResponse({'status': 'success','reply_message_id': reply.message_id})
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
def fetch_replies(request, message_id):
if request.method == 'GET':
try:
original_message = Message.objects.get(message_id=message_id)
replies = Message.objects.filter(reply_to=original_message).prefetch_related('recipient').order_by('sent_at')
def get_attachment_data(reply):
if not reply.attachment:
return None
try:
fs = FileSystemStorage()
attachment_url = fs.url(reply.attachment.name)
file_path = fs.path(reply.attachment.name)
file_size = os.path.getsize(file_path)
return {
'filename': reply.attachment.name.split('/')[-1],
'url': attachment_url,
'size': file_size
}
except Exception as e:
print(f"Error retrieving attachment for message {reply.message_id}: {e}")
return None
def get_first_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.first_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
def get_last_downloaded(message_id):
file_download = FileDownload.objects.filter(message_id=message_id).first()
if file_download:
return localtime(file_download.last_downloaded).strftime('%m/%d/%Y, %I:%M:%S %p')
return None
def serialize_reply(reply):
return {
'message_id': reply.message_id,
'subject': reply.subject,
'sender_id': reply.sender_id,
'sender_fullname': reply.sender_fullname,
'recipient_id': [recipient.Email1 for recipient in reply.recipient.all()],
'recipient_fullname': reply.recipient_fullname,
'body': reply.body,
'sent_at': reply.sent_at.isoformat(),
'parent_message_id': reply.reply_to_id,
'attachment': get_attachment_data(reply),
'first_downloaded': get_first_downloaded(reply.message_id),
'last_downloaded': get_last_downloaded(reply.message_id),
'replies': [serialize_reply(r) for r in Message.objects.filter(reply_to=reply).order_by('sent_at')]
}
replies_data = [serialize_reply(reply) for reply in replies]
return JsonResponse({'status': 'success', 'replies': replies_data})
except Message.DoesNotExist:
return JsonResponse({'status': 'error', 'message': 'Original message not found'}, status=404)
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
def track_download(request, message_id):
if request.method == 'GET':
current_user = request.GET.get('user_id', None)
current_user_id = get_object_or_404(InternalTeam, user_id=current_user)
ist_timezone = pytz.timezone('Asia/Kolkata')
message = get_object_or_404(Message, message_id=message_id)
# Prevent the sender from downloading their own message
if current_user_id == message.sender:
return JsonResponse({'status': 'error', 'message': 'Sender cannot download the message attachment.'}, status=400)
# Ensure the current user is a recipient of the message
if current_user_id not in message.recipient.all():
return JsonResponse({'status': 'error', 'message': 'User is not a recipient of this message.'}, status=400)
recipient = current_user_id
# Track downloads for the original message and its replies
download_records = []
for msg in [message] + list(Message.objects.filter(reply_to=message)):
download, created = FileDownload.objects.get_or_create(
message=msg,
user=recipient,
defaults={
'first_downloaded': localtime(now(), ist_timezone),
'last_downloaded': localtime(now(), ist_timezone)
}
)
# Update the last_downloaded timestamp
if not created:
download.last_downloaded = now()
download.save()
download_records.append({
'message_id': msg.message_id,
'first_downloaded': localtime(download.first_downloaded, ist_timezone).strftime('%m/%d/%Y, %I:%M:%S %p'),
'last_downloaded': localtime(download.last_downloaded, ist_timezone).strftime('%m/%d/%Y, %I:%M:%S %p'),
})
return JsonResponse({'status': 'success', 'downloads': download_records})
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
def fetch_download_timestamps(request, message_id):
if request.method == 'GET':
current_user = request.GET.get('user_id', None)
current_user_id = get_object_or_404(InternalTeam, user_id=current_user).Email1
message = get_object_or_404(Message, message_id=message_id)
ist_timezone = pytz.timezone('Asia/Kolkata')
# Separate main message and reply messages
main_message_downloads = []
reply_message_downloads = {}
# Fetch main message download timestamps
main_message_records = FileDownload.objects.filter(message=message)
for record in main_message_records:
recipient_id = record.user.Email1
main_message_downloads.append({
'message_id': message_id,
'sender_id': message.sender_id,
'recipient_id': recipient_id,
'first_downloaded': localtime(record.first_downloaded, ist_timezone).strftime('%m/%d/%Y, %I:%M:%S %p') if record.first_downloaded else None,
'last_downloaded': localtime(record.last_downloaded, ist_timezone).strftime('%m/%d/%Y, %I:%M:%S %p') if record.last_downloaded else None
})
# Fetch reply message download timestamps
reply_messages = Message.objects.filter(reply_to=message)
for reply in reply_messages:
download_records = FileDownload.objects.filter(message=reply)
print(download_records)
reply_message_downloads[reply.message_id] = []
for record in download_records:
recipient_id = record.user.Email1
reply_message_downloads[reply.message_id].append({
'message_id': reply.message_id,
'sender_id': reply.sender_id,
'recipient_id': recipient_id,
'first_downloaded': localtime(record.first_downloaded, ist_timezone).strftime(
'%m/%d/%Y, %I:%M:%S %p') if record.first_downloaded else None,
'last_downloaded': localtime(record.last_downloaded, ist_timezone).strftime(
'%m/%d/%Y, %I:%M:%S %p') if record.last_downloaded else None
})
print(reply_message_downloads)
return JsonResponse({
'main_message_downloads': main_message_downloads,
'reply_message_downloads': reply_message_downloads,
'current_user_id': current_user_id
})
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
def get_main_message_id(request,message_id):
if request.method == 'GET':
current_message = get_object_or_404(Message, message_id=message_id)
print(current_message, 'current_message')
# Traverse the chain of parent messages to find the main message
while current_message.reply_to:
current_message = get_object_or_404(Message, message_id=current_message.reply_to)
return JsonResponse({'main_message_id': current_message.message_id})
else:
return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
# def unread_message_count(request):
# if request.method == 'GET':
# user_id = request.GET.get('user_id')
# print(user_id)
#
# user = get_object_or_404(InternalTeam, user_id=user_id)
#
# # Get all messages that are unread in Message model
# unread = Message.objects.filter(recipient=user, is_read=False).values_list('message_id', flat=True)
# messages_in_status = MessageStatus.objects.filter(message_id__in=unread, user=user).values_list('message_id', flat=True)
#
# unread_not_in_status = set(unread) - set(messages_in_status)
# print(unread_not_in_status, 'Unread messages')
# unread_count = len(unread_not_in_status)
#
# return JsonResponse({
# 'unread_count': unread_count,
# 'unread_messages': list(unread_not_in_status) # Convert set to list for JSON response
# })
# return JsonResponse({'status': 'error', 'message': 'Invalid request method'}, status=400)
Loading…
Cancel
Save