feat: add account page where user can see bought tickets
This commit is contained in:
parent
acb4786f48
commit
3bf86ff66e
@ -1,11 +1,152 @@
|
||||
@page
|
||||
@using System.Globalization
|
||||
@model TicketOffice.Pages.Account.IndexModel
|
||||
|
||||
@{
|
||||
Layout = "~/Pages/Shared/_Layout.cshtml";
|
||||
ViewData["Title"] = "Аккаунт";
|
||||
CultureInfo.CurrentCulture = CultureInfo.GetCultureInfo("uk-UA");
|
||||
}
|
||||
|
||||
<link rel="stylesheet" href="~/css/Account.css"/>
|
||||
<link rel="stylesheet" href="~/css/Popup.css"/>
|
||||
|
||||
<div class="section">
|
||||
<div class="section-header">Придбані квитки</div>
|
||||
@if (Model.Tickets is {Count: 0})
|
||||
{
|
||||
<div class="section-text">Ви не придбали жодного квитка. <a href="/Routes" class="link">Пошук маршрутів</a></div>
|
||||
}
|
||||
else
|
||||
{
|
||||
<div class="tickets">
|
||||
@foreach (var ticket in Model.Tickets)
|
||||
{
|
||||
<div class="tickets">
|
||||
<div class="ticket">
|
||||
<div class="ticket-header">Квиток № @ticket.Id</div>
|
||||
<div class="ticket-body">
|
||||
<div class="ticket-info">
|
||||
<div class="ticket-info-line">
|
||||
<span>Пасажир, місце:</span>
|
||||
<span>@ticket.PassengerLastName @ticket.PassengerFirstName, @ticket.PassengerPlace</span>
|
||||
</div>
|
||||
<div class="ticket-info-line">
|
||||
<span>Номер рейсу:</span>
|
||||
<span> № @ticket.Route.Number</span>
|
||||
</div>
|
||||
<div class="ticket-info-line date">
|
||||
<span>Дата відправлення:</span>
|
||||
<span>
|
||||
@ticket.Route.Cities.First().DepartureTime?.ToString("f").Split(",")[0].ToLower(),
|
||||
@ticket.Route.Cities.First().DepartureTime?.ToString("dd.MM.yyyy"),
|
||||
@ticket.Route.Cities.First().DepartureTime?.ToString("HH:mm")
|
||||
</span>
|
||||
</div>
|
||||
<div class="ticket-info-line date">
|
||||
<span>Дата прибуття:</span>
|
||||
<span>
|
||||
@ticket.Route.Cities.Last().ArrivalTime?.ToString("f").Split(",")[0].ToLower(),
|
||||
@ticket.Route.Cities.Last().ArrivalTime?.ToString("dd.MM.yyyy"),
|
||||
@ticket.Route.Cities.Last().ArrivalTime?.ToString("HH:mm")
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="ticket-footer">
|
||||
<a class="ticket-link-btn" onclick="document.getElementById('popup-city-list-@ticket.Id').style.display = 'inherit'">Маршрут</a>
|
||||
<a class="ticket-link-btn" asp-page-handler="ReturnTicket" asp-route-ReturnTicketId="@ticket.Id">Повернути</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="popup-container" id="popup-city-list-@ticket.Id">
|
||||
<div class="popup">
|
||||
<div class="popup-header">
|
||||
Автобус №@ticket.Route.Number
|
||||
</div>
|
||||
<div class="popup-body">
|
||||
<div class="popup-body-main">
|
||||
<table>
|
||||
<thead>
|
||||
<tr class="tr-intermediate">
|
||||
<th class="th-route">
|
||||
Інформація
|
||||
</th>
|
||||
<th class="th-route">
|
||||
Місто
|
||||
</th>
|
||||
<th class="th-route">
|
||||
Час прибуття
|
||||
</th>
|
||||
<th class="th-route">
|
||||
Час відправлення
|
||||
</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr class="tr-departure">
|
||||
<td class="td-route">
|
||||
Відправлення
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.First().Name
|
||||
</td>
|
||||
<td class="td-route">
|
||||
-
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.First().DepartureTime?.ToString("HH:mm")
|
||||
</td>
|
||||
</tr>
|
||||
|
||||
@for (int i = 1; i < ticket.Route.Cities.Count - 1; i++)
|
||||
{
|
||||
<tr class="tr-intermediate">
|
||||
<td class="td-route">
|
||||
Проміжна станція
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.ToList()[i].Name
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.ToList()[i].ArrivalTime?.ToString("HH:mm")
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.ToList()[i].DepartureTime?.ToString("HH:mm")
|
||||
</td>
|
||||
</tr>
|
||||
}
|
||||
|
||||
<tr class="tr-arrival">
|
||||
<td class="td-route">
|
||||
Прибуття
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.Last().Name
|
||||
</td>
|
||||
<td class="td-route">
|
||||
@ticket.Route.Cities.Last().ArrivalTime?.ToString("HH:mm")
|
||||
</td>
|
||||
<td class="td-route">
|
||||
-
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
<div class="popup-body-footer">
|
||||
<a class="popup-footer-button" onclick="document.getElementById('popup-city-list-@ticket.Id').style.display = 'none'">Закрити</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
</div>
|
||||
}
|
||||
</div>
|
||||
|
||||
<div class="divider"></div>
|
||||
|
||||
<div class="section">
|
||||
<div class="section-header">Керування аккаунтом</div>
|
||||
</div>
|
||||
|
@ -8,5 +8,49 @@ namespace TicketOffice.Pages.Account;
|
||||
|
||||
public class IndexModel : PageModel
|
||||
{
|
||||
public List<Ticket> Tickets { get; set; }
|
||||
[BindProperty(SupportsGet = true)] public int ReturnTicketId { get; set; }
|
||||
|
||||
private readonly TicketOfficeContext _context;
|
||||
private readonly ILogger<IndexModel> _logger;
|
||||
|
||||
public IndexModel(TicketOfficeContext context, ILogger<IndexModel> logger)
|
||||
{
|
||||
_context = context;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public IActionResult OnGet()
|
||||
{
|
||||
if (!ValidateSession())
|
||||
return RedirectToPage("/Auth/Login");
|
||||
|
||||
Tickets = _context.Ticket
|
||||
.Where(t => t.UserId == HttpContext.Session.GetInt32("UserId"))
|
||||
.Include(t => t.Route)
|
||||
.Include(t => t.Route.Cities)
|
||||
.ToList();
|
||||
|
||||
return Page();
|
||||
}
|
||||
|
||||
public IActionResult OnGetReturnTicket()
|
||||
{
|
||||
OnGet();
|
||||
|
||||
_logger.Log(LogLevel.Information, $"\n\n\n\n {ReturnTicketId} \n\n\n\n");
|
||||
|
||||
Ticket returnTicket = _context.Ticket.Find(ReturnTicketId);
|
||||
|
||||
if (returnTicket is not null)
|
||||
{
|
||||
_context.Remove(returnTicket);
|
||||
_context.SaveChanges();
|
||||
return RedirectToPage("./Account");
|
||||
}
|
||||
|
||||
return NotFound();
|
||||
}
|
||||
|
||||
private bool ValidateSession() => HttpContext.Session.GetInt32("UserId") is not null;
|
||||
}
|
@ -27,6 +27,79 @@ body {
|
||||
margin: 2rem 0;
|
||||
}
|
||||
|
||||
.tickets {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: space-evenly;
|
||||
}
|
||||
|
||||
.ticket {
|
||||
background: #eaeef1;
|
||||
box-shadow: 0 5px 10px 1px #aeb1b5;
|
||||
width: 30rem;
|
||||
height: 12rem;
|
||||
border-radius: 0.5rem;
|
||||
margin: 2rem 0 0 0;
|
||||
}
|
||||
|
||||
.ticket-header {
|
||||
background: #a1b0b9;
|
||||
border-radius: 0.5rem 0.5rem 0 0;
|
||||
height: 2.5rem;
|
||||
line-height: 2.5rem;
|
||||
font-size: 1rem;
|
||||
font-weight: 700;
|
||||
color: white;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.ticket-body {
|
||||
height: calc(100% - 5.7rem);
|
||||
}
|
||||
|
||||
.ticket-info {
|
||||
margin: 0.25rem 0.5rem;
|
||||
font-weight: 700;
|
||||
font-size: 0.8rem;
|
||||
}
|
||||
|
||||
.ticket-info-line {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
height: 1.4rem;
|
||||
line-height: 1rem;
|
||||
align-items: end;
|
||||
border-bottom: 1px solid #c5c7cc;
|
||||
}
|
||||
|
||||
.date {
|
||||
font-size: 0.8rem;
|
||||
}
|
||||
|
||||
.ticket-footer {
|
||||
height: 3rem;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.ticket-link-btn {
|
||||
line-height: 2rem;
|
||||
padding: 0 0.8rem;
|
||||
margin: 0.5rem 0.5rem;
|
||||
display: inline-block;
|
||||
color: #1d4965;
|
||||
font-size: 0.8rem;
|
||||
font-weight: 700;
|
||||
background: linear-gradient(0deg,#79b6db,#b3dbf2);
|
||||
border: none;
|
||||
border-radius: .3rem;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.ticket-link-btn:hover {
|
||||
opacity: 0.8;
|
||||
}
|
||||
|
||||
.link-btn {
|
||||
line-height: 2.5rem;
|
||||
padding: 0 1rem;
|
||||
@ -40,6 +113,25 @@ body {
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.link-btn:hover {
|
||||
opacity: 0.8;
|
||||
}
|
||||
|
||||
input.btn {
|
||||
line-height: 2.5rem;
|
||||
padding: 0 1rem;
|
||||
margin: 0 0.5rem;
|
||||
display: inline-block;
|
||||
color: #1d4965;
|
||||
font-weight: 700;
|
||||
font-size: 1rem;
|
||||
background: linear-gradient(0deg,#79b6db,#b3dbf2);
|
||||
border: none;
|
||||
border-radius: .3rem;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.link {
|
||||
color: #245c78;
|
||||
text-decoration: none;
|
||||
|
138
TicketOffice/wwwroot/css/Popup.css
Normal file
138
TicketOffice/wwwroot/css/Popup.css
Normal file
@ -0,0 +1,138 @@
|
||||
.popup-container {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 100;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background: rgba(0, 0, 0, 0.25);
|
||||
display: none;
|
||||
}
|
||||
|
||||
.popup {
|
||||
width: 40rem;
|
||||
height: 30rem;
|
||||
background: #eaeef1;
|
||||
position: fixed;
|
||||
top: calc(50% - 15rem);
|
||||
left: calc(50% - 20rem);
|
||||
border-radius: 0.5rem;
|
||||
box-shadow: 0 10px 15px 5px #6c6e6f;
|
||||
}
|
||||
|
||||
.popup-header {
|
||||
border-radius: 0.5rem 0.5rem 0 0;
|
||||
height: 3rem;
|
||||
background: #a1b0b9;
|
||||
color: white;
|
||||
font-weight: 700;
|
||||
font-size: 1.1rem;
|
||||
text-align: center;
|
||||
line-height: 3rem;
|
||||
}
|
||||
|
||||
.popup-body {
|
||||
width: calc(100% - 2rem);
|
||||
height: calc(100% - 5rem);
|
||||
padding: 1rem 1rem;
|
||||
}
|
||||
|
||||
.popup-body-main {
|
||||
height: calc(100% - 4rem);
|
||||
overflow-y: auto;
|
||||
overflow-x: hidden;
|
||||
}
|
||||
|
||||
tr.tr-intermediate {
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
th.th-route, td.td-route {
|
||||
height: 4rem;
|
||||
line-height: 1.25rem;
|
||||
text-align: center;
|
||||
padding: 0 0.5rem;
|
||||
}
|
||||
|
||||
td.address {
|
||||
font-size: 0.8rem;
|
||||
text-align: justify;
|
||||
line-height: 1rem;
|
||||
}
|
||||
|
||||
tr.tr-departure, tr.tr-arrival {
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.popup-body-footer {
|
||||
text-align: center;
|
||||
margin-top: 1rem;
|
||||
}
|
||||
|
||||
.popup-footer-link-button {
|
||||
line-height: 2.5rem;
|
||||
padding: 0 1rem;
|
||||
margin: 0 0.5rem;
|
||||
display: inline-block;
|
||||
color: #1d4965;
|
||||
font-weight: 500;
|
||||
background: linear-gradient(0deg,#79b6db,#b3dbf2);
|
||||
border: none;
|
||||
border-radius: .3rem;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.popup-footer-button {
|
||||
line-height: 2.5rem;
|
||||
padding: 0 1rem;
|
||||
margin: 0 0.5rem;
|
||||
display: inline-block;
|
||||
color: #1d4965;
|
||||
font-weight: 700;
|
||||
font-size: 1rem;
|
||||
background: linear-gradient(0deg,#79b6db,#b3dbf2);
|
||||
border: none;
|
||||
border-radius: .3rem;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.popup-footer-button:hover, .popup-footer-link-button:hover {
|
||||
opacity: 0.8;
|
||||
}
|
||||
|
||||
/* table */
|
||||
|
||||
table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
border: 1px solid #d7dce1;
|
||||
}
|
||||
|
||||
th {
|
||||
height: 6rem;
|
||||
line-height: 1.6rem;
|
||||
background: #e6e9ed;
|
||||
border: 1px solid #d7dce1;
|
||||
padding: 0 1rem;
|
||||
font-size: 1rem;
|
||||
text-align: start;
|
||||
font-weight: 700;
|
||||
color: #777a7e;
|
||||
}
|
||||
|
||||
tr {
|
||||
line-height: 5rem;
|
||||
background-color: white;
|
||||
}
|
||||
|
||||
tr:hover {
|
||||
background-color: #dee9f4;
|
||||
}
|
||||
|
||||
td {
|
||||
padding: 0 1rem;
|
||||
border: 1px solid #d7dce1;
|
||||
text-align: center;
|
||||
}
|
Loading…
Reference in New Issue
Block a user