feat: improve route list design and functionallity
This commit is contained in:
parent
1b3b598539
commit
1146093913
@ -1,13 +1,16 @@
|
|||||||
@page
|
@page
|
||||||
|
@using System.Globalization
|
||||||
|
@using Newtonsoft.Json
|
||||||
@model TicketOffice.Pages.IndexModel
|
@model TicketOffice.Pages.IndexModel
|
||||||
@{
|
@{
|
||||||
ViewData["Title"] = "Home page";
|
ViewData["Title"] = "Home page";
|
||||||
|
CultureInfo.CurrentCulture = CultureInfo.GetCultureInfo("uk-UA");
|
||||||
}
|
}
|
||||||
|
|
||||||
<link rel="stylesheet" href="~/css/Routes.css" asp-append-version="true"/>
|
<link rel="stylesheet" href="~/css/Routes.css" asp-append-version="true"/>
|
||||||
|
|
||||||
<form class="search-block">
|
<form class="search-block">
|
||||||
<div class="stations">
|
<div class="opt">
|
||||||
<div class="station">
|
<div class="station">
|
||||||
<div class="title">
|
<div class="title">
|
||||||
Звідки
|
Звідки
|
||||||
@ -21,20 +24,22 @@
|
|||||||
</div>
|
</div>
|
||||||
<input type="text" autocomplete="off" asp-for="to">
|
<input type="text" autocomplete="off" asp-for="to">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
<div class="opt">
|
|
||||||
<div class="date">
|
<div class="date">
|
||||||
<div class="title">
|
<div class="title">
|
||||||
Дата відправлення
|
Дата відправлення
|
||||||
</div>
|
</div>
|
||||||
<input type="date" value="@Model.date.ToString("yyyy-MM-dd")" asp-for="date">
|
<input type="date" value="@Model.date.ToString("yyyy-MM-dd")" asp-for="date">
|
||||||
</div>
|
</div>
|
||||||
|
<div class="search-btn">
|
||||||
|
<input type="submit" class="search-btn" value="Пошук"/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<input type="submit" value="Пошук"/>
|
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
<!--
|
||||||
<a asp-page-handler="SortByDuration" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd") asp-route-isDescending="false">За зростанням</a>
|
<a asp-page-handler="SortByDuration" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd") asp-route-isDescending="false">За зростанням</a>
|
||||||
<a asp-page-handler="SortByDuration" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd") asp-route-isDescending="true">За спаданням</a>
|
<a asp-page-handler="SortByDuration" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd") asp-route-isDescending="true">За спаданням</a>
|
||||||
|
-->
|
||||||
|
|
||||||
@if (Model.Route.Count > 0)
|
@if (Model.Route.Count > 0)
|
||||||
{
|
{
|
||||||
@ -44,6 +49,7 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<th>
|
<th>
|
||||||
№ автобуса
|
№ автобуса
|
||||||
|
<a class="link-btn-sort" asp-page-handler="SortByNumber" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd")>🠕</a>
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
Звідки / Куди
|
Звідки / Куди
|
||||||
@ -52,11 +58,18 @@
|
|||||||
Дата
|
Дата
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
<div class="departure">Відправлення</div>
|
<div class="departure">
|
||||||
<div class="arrival">Прибуття</div>
|
Відправлення
|
||||||
|
<a class="link-btn-sort" asp-page-handler="SortByDeparture" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd")>🠕</a>
|
||||||
|
</div>
|
||||||
|
<div class="arrival">
|
||||||
|
Прибуття
|
||||||
|
<a class="link-btn-sort" asp-page-handler="SortByArrival" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd")>🠕</a>
|
||||||
|
</div>
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
Тривалість
|
Тривалість
|
||||||
|
<a class="link-btn-sort" asp-page-handler="SortByDuration" asp-route-from=@Model.from a in a asp-route-to=@Model.to asp-route-date=@Model.date.ToString("yyyy-MM-dd")>🠕</a>
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
Вільніих місць
|
Вільніих місць
|
||||||
@ -71,7 +84,8 @@
|
|||||||
{
|
{
|
||||||
<tr class="table-row">
|
<tr class="table-row">
|
||||||
<td class="num">
|
<td class="num">
|
||||||
@route.Number
|
<div>@route.Number</div>
|
||||||
|
<div class="city-list-btn"><a class="city-list-btn">Маршрут</a></div>
|
||||||
</td>
|
</td>
|
||||||
<td class="city">
|
<td class="city">
|
||||||
<div>@route.Cities.First().Name</div>
|
<div>@route.Cities.First().Name</div>
|
||||||
@ -79,12 +93,20 @@
|
|||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<div class="route-date">
|
<div class="route-date">
|
||||||
<span>Відправлення</span>
|
<span>
|
||||||
<span>@route.Cities.First().DepartureTime?.ToString("dd.MM.yyy")</span>
|
Відправлення
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
@route.Cities.First().ArrivalTime?.ToString("f").Split(",")[0].ToLower(),
|
||||||
|
@route.Cities.First().DepartureTime?.ToString("dd.MM.yyyy")
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="route-date">
|
<div class="route-date">
|
||||||
<span>Прибуття</span>
|
<span>Прибуття</span>
|
||||||
<span>@route.Cities.Last().ArrivalTime?.ToString("dd.MM.yyy")</span>
|
<span>
|
||||||
|
@route.Cities.Last().ArrivalTime?.ToString("f").Split(",")[0].ToLower(),
|
||||||
|
@route.Cities.Last().ArrivalTime?.ToString("dd.MM.yyyy")
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td class="time">
|
<td class="time">
|
||||||
@ -93,13 +115,13 @@
|
|||||||
</td>
|
</td>
|
||||||
<td class="duration">
|
<td class="duration">
|
||||||
@{ TimeSpan? duration = route.Cities.Last().ArrivalTime - route.Cities.First().DepartureTime; }
|
@{ TimeSpan? duration = route.Cities.Last().ArrivalTime - route.Cities.First().DepartureTime; }
|
||||||
@($"{duration?.TotalHours.ToString().Split(".")[0]}:{duration?.Minutes}")
|
@($"{duration?.TotalHours.ToString().Split(",")[0]}:{duration?.Minutes}")
|
||||||
</td>
|
</td>
|
||||||
<td class="capacity">
|
<td class="capacity">
|
||||||
@(route.Capacity - route.Tickets.Count)
|
@(route.Capacity - route.Tickets.Count)
|
||||||
</td>
|
</td>
|
||||||
<td class="action">
|
<td class="action">
|
||||||
<a asp-page=".">Вибрати</a>
|
<a class="link-btn-choose">Вибрати</a>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,8 @@
|
|||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using System.Globalization;
|
||||||
|
using Newtonsoft.Json;
|
||||||
using TicketOffice.Data;
|
using TicketOffice.Data;
|
||||||
using TicketOffice.Models;
|
using TicketOffice.Models;
|
||||||
using Route = TicketOffice.Models.Route;
|
using Route = TicketOffice.Models.Route;
|
||||||
@ -22,38 +24,95 @@ public class IndexModel : PageModel
|
|||||||
[BindProperty(SupportsGet = true)] public string from { get; set; }
|
[BindProperty(SupportsGet = true)] public string from { get; set; }
|
||||||
[BindProperty(SupportsGet = true)] public string to { get; set; }
|
[BindProperty(SupportsGet = true)] public string to { get; set; }
|
||||||
[BindProperty(SupportsGet = true)] public DateTime date { get; set; } = new DateTime(2022, 03, 28, 0, 0, 0).Date;
|
[BindProperty(SupportsGet = true)] public DateTime date { get; set; } = new DateTime(2022, 03, 28, 0, 0, 0).Date;
|
||||||
|
[BindProperty(SupportsGet = true)] public string SortString { get; set; }
|
||||||
|
|
||||||
public void OnGet()
|
public void OnGet()
|
||||||
|
{
|
||||||
|
RetrieveAllRoutes();
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(from))
|
||||||
|
{
|
||||||
|
FilterRoutesByFrom();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(to))
|
||||||
|
{
|
||||||
|
FilterRoutesByTo();
|
||||||
|
}
|
||||||
|
|
||||||
|
FilterRoutesByDate();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnGetSortByNumber()
|
||||||
|
{
|
||||||
|
OnGet();
|
||||||
|
|
||||||
|
Route.Sort((x, y) => Math.Clamp(x.Number - y.Number, -1, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnGetSortByDeparture()
|
||||||
|
{
|
||||||
|
OnGet();
|
||||||
|
|
||||||
|
Route.Sort((x, y) => Math.Clamp((int)(x.Cities.First().DepartureTime - y.Cities.First().DepartureTime).Value.TotalMilliseconds, -1, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnGetSortByArrival()
|
||||||
|
{
|
||||||
|
OnGet();
|
||||||
|
|
||||||
|
Route.Sort((x, y) =>
|
||||||
|
{
|
||||||
|
TimeSpan? totalDuration = x.Cities.Last().ArrivalTime - y.Cities.Last().ArrivalTime;
|
||||||
|
|
||||||
|
return Math.Clamp((int)totalDuration.Value.TotalMilliseconds, -1, 1);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnGetSortByDuration()
|
||||||
|
{
|
||||||
|
OnGet();
|
||||||
|
|
||||||
|
Route.Sort((x, y) =>
|
||||||
|
{
|
||||||
|
TimeSpan? xDuration = x.Cities.Last().ArrivalTime - x.Cities.First().DepartureTime;
|
||||||
|
TimeSpan? yDuration = y.Cities.Last().ArrivalTime - y.Cities.First().DepartureTime;
|
||||||
|
TimeSpan? totalDuration = xDuration - yDuration;
|
||||||
|
|
||||||
|
return Math.Clamp((int)totalDuration.Value.TotalMilliseconds, -1, 1);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void RetrieveAllRoutes()
|
||||||
{
|
{
|
||||||
Route = _context.Route
|
Route = _context.Route
|
||||||
.Include(r => r.Cities)
|
.Include(r => r.Cities)
|
||||||
.Include(r => r.Tickets)
|
.Include(r => r.Tickets)
|
||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
if (!string.IsNullOrWhiteSpace(from) && !string.IsNullOrWhiteSpace(to))
|
|
||||||
{
|
|
||||||
FilterRoutes();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OnGetSortByDuration(bool isDescending)
|
private void FilterRoutesByFrom()
|
||||||
{
|
|
||||||
OnGet();
|
|
||||||
|
|
||||||
Route.Sort((x, y) => (isDescending ? -1 : 1) *
|
|
||||||
(x.Cities.Last().ArrivalTime - x.Cities.First().DepartureTime).Value
|
|
||||||
.CompareTo((y.Cities.Last().ArrivalTime - y.Cities.First().DepartureTime).Value) + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
private void FilterRoutes()
|
|
||||||
{
|
{
|
||||||
|
|
||||||
Route.ForEach(r => r.Cities = r.Cities
|
Route.ForEach(r => r.Cities = r.Cities
|
||||||
.SkipWhile(c => c.Name != from).Reverse()
|
.SkipWhile(c => c.Name != from)
|
||||||
.SkipWhile(c => c.Name != to).Reverse()
|
|
||||||
.ToList());
|
.ToList());
|
||||||
|
|
||||||
Route.RemoveAll(r =>
|
Route.RemoveAll(r => r.Cities.Count < 2);
|
||||||
r.Cities.Count < 2 || r.Cities.First().DepartureTime.Value.DayOfYear != date.DayOfYear);
|
}
|
||||||
|
|
||||||
|
private void FilterRoutesByTo()
|
||||||
|
{
|
||||||
|
|
||||||
|
Route.ForEach(r => r.Cities = r.Cities
|
||||||
|
.Reverse().SkipWhile(c => c.Name != to)
|
||||||
|
.Reverse().ToList());
|
||||||
|
|
||||||
|
Route.RemoveAll(r => r.Cities.Count < 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void FilterRoutesByDate()
|
||||||
|
{
|
||||||
|
Route.RemoveAll(r => r.Cities.First().DepartureTime.Value.DayOfYear != date.DayOfYear);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -20,6 +20,64 @@ body {
|
|||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.search-block {
|
||||||
|
background: #eaeef1;
|
||||||
|
box-shadow: 0 1px 2.4rem 0 #c3c9d0;
|
||||||
|
padding: 1.5rem 1.5rem 1.5rem 1.5rem;
|
||||||
|
margin: 1.5rem 0;
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.station, .date, div.search-btn {
|
||||||
|
display: inline-block;
|
||||||
|
margin: 0 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-weight: 500;
|
||||||
|
color: #777a7e;
|
||||||
|
margin-bottom: .3rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type=text], input[type=date] {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
color: #262626;
|
||||||
|
font-weight: 500;
|
||||||
|
line-height: 4.7rem;
|
||||||
|
width: 15rem;
|
||||||
|
height: 3rem;
|
||||||
|
box-sizing: border-box;
|
||||||
|
background: #dfe3e5;
|
||||||
|
border: 1px solid #b8bfc7;
|
||||||
|
box-shadow: 0 1px 0 0 #fff;
|
||||||
|
border-radius: .3rem;
|
||||||
|
padding: 0 1.1rem;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type=text]:focus, input[type=date]:focus {
|
||||||
|
outline: 0;
|
||||||
|
border-color: #68b2dd;
|
||||||
|
background-color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.search-btn {
|
||||||
|
color: #1d4965;
|
||||||
|
font-size: 1.6rem;
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 3rem;
|
||||||
|
padding: 0 1.5rem;
|
||||||
|
display: inline-block;
|
||||||
|
background: linear-gradient(0deg,#79b6db,#b3dbf2);
|
||||||
|
border: none;
|
||||||
|
border-radius: .3rem;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.search-btn:hover {
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
.route-list {
|
.route-list {
|
||||||
margin-top: 3rem;
|
margin-top: 3rem;
|
||||||
}
|
}
|
||||||
@ -45,8 +103,22 @@ th {
|
|||||||
line-height: 1.6rem;
|
line-height: 1.6rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.link-btn-sort {
|
||||||
|
color: #777a7e;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.link-btn-sort:hover {
|
||||||
|
color: #1d4965;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.link-btn-sort:visited, .link-btn-sort:active {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
tr {
|
tr {
|
||||||
line-height: 6rem;
|
line-height: 5rem;
|
||||||
background-color: white;
|
background-color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -66,6 +138,23 @@ td.num, td.capacity {
|
|||||||
line-height: 2rem;
|
line-height: 2rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.city-list-btn {
|
||||||
|
font-size: 0.8rem;
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 0.5rem;
|
||||||
|
color: #245c78;
|
||||||
|
text-decoration: none;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.city-list-btn:visited {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.city-list-btn:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
td.city {
|
td.city {
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
@ -75,28 +164,40 @@ td.city {
|
|||||||
td.time, td.duration {
|
td.time, td.duration {
|
||||||
font-size: 1.2rem;
|
font-size: 1.2rem;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
line-height: 1.8rem;
|
line-height: 1.4rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.route-date {
|
.route-date {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
border-bottom: 1px solid #c5c7cc;
|
border-bottom: 1px solid #c5c7cc;
|
||||||
font-size: 0.75rem;
|
font-size: 0.7rem;
|
||||||
|
font-weight: 700;
|
||||||
line-height: 1.25rem;
|
line-height: 1.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-block {
|
.link-btn-choose {
|
||||||
background: #eaeef1;
|
line-height: 2.5rem;
|
||||||
box-shadow: 0 1px 2.4rem 0 #c3c9d0;
|
padding: 0 1rem;
|
||||||
padding: 1.5rem 2.5rem 3.5rem 2.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;
|
||||||
|
}
|
||||||
|
|
||||||
|
.link-btn-choose:hover {
|
||||||
|
opacity: 0.8;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-error {
|
.search-error {
|
||||||
background: #f1f2f4;
|
background: #f1f2f4;
|
||||||
border: 1px solid #d7dce1;
|
border: 1px solid #d7dce1;
|
||||||
box-shadow: 0 0 4px 0 rgba(195,201,208,.5);
|
box-shadow: 0 0 4px 0 rgba(195,201,208,.5);
|
||||||
font-weight: 700;
|
font-weight: 500;
|
||||||
font-size: 1.5rem;
|
font-size: 1.5rem;
|
||||||
padding: 4rem 4rem;
|
padding: 4rem 4rem;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
Binary file not shown.
Loading…
Reference in New Issue
Block a user