Compare commits

...

80 Commits

Author SHA1 Message Date
c966ba26ab финалочка 2024-05-29 16:41:49 +04:00
dbf7368333 ворд и эксель 2024-05-28 21:36:41 +04:00
dbf37e6871 пдф не правильно создается, но на почту отправляется 2024-05-28 16:38:17 +04:00
698d467bf1 редактирование 2024-05-27 20:27:11 +04:00
5337232d0c привязка 2024-05-27 17:01:24 +04:00
c3d5dcbc78 изменила бд 2024-05-26 21:32:59 +04:00
8e9ecbf3b3 регистрация и вход 2024-05-25 16:31:54 +04:00
9afbae92a5 забыл до этого сделать коммит))) доделал веб 2024-05-01 22:04:27 +04:00
ddc21a6666 данные для отчетов 2024-05-01 14:16:12 +04:00
20049fb6b7 сделала веб 2024-04-30 19:50:49 +04:00
8321ae0bb4 исправила MainController 2024-04-30 00:33:22 +04:00
bb202799f6 исправление ошибок 2024-04-29 22:05:27 +04:00
cf0cfe3178 исправление ошибок 2024-04-29 21:13:20 +04:00
544a1868c3 Исправление ошибок 2024-04-29 21:09:12 +04:00
31287b8f1c сделала MainController 2024-04-29 20:26:27 +04:00
e6fe0eb446 Исправление Api 2024-04-29 19:36:47 +04:00
ad971e8dde Исправлены миграции 2024-04-29 19:27:00 +04:00
75d7be454c Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-29 19:25:42 +04:00
571ef2caca rest api 2024-04-29 19:25:13 +04:00
add645cde0 Правки 2024-04-29 19:25:10 +04:00
35ee169cda добавил миграции 2024-04-29 18:33:51 +04:00
10d5cc612a Исправил ошибку 2024-04-29 17:54:50 +04:00
1ed3ef3469 Исправления ошибок 2024-04-29 17:46:47 +04:00
7059604727 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-29 00:20:59 +04:00
f4b5935751 сделала Implements 2024-04-29 00:20:55 +04:00
35d81398ab + web 2024-04-29 00:19:47 +04:00
4e7c734a22 исправила модели в бд 2024-04-28 22:13:42 +04:00
6559110c71 исправление ошибок 2024-04-28 22:10:32 +04:00
2bd7121abb Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-28 21:29:15 +04:00
845aea110e тест 2024-04-28 21:28:39 +04:00
a93bdb7bb9 конфликт слияния 2024-04-28 21:27:33 +04:00
bd961535a2 сделала модели в бд 2024-04-28 21:14:42 +04:00
8b3f6b839e + DataBaseImplements 2024-04-28 21:13:12 +04:00
b1161502fa + Папочки 2024-04-28 19:12:06 +04:00
54d23f7166 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-28 19:07:11 +04:00
5fd73203cf + BusinessLogisc 2024-04-28 19:07:09 +04:00
03a7c2a36b изменила бизнес логику 2024-04-28 19:05:25 +04:00
057d088443 исправил ошибки 2024-04-28 17:54:17 +04:00
d935c309e1 сделала бизнес логику 2024-04-28 17:52:00 +04:00
4202c78a7f доработки 2024-04-28 16:57:59 +04:00
3998ed4bf8 + contracts 2024-04-28 16:00:22 +04:00
7a774b8c86 исправила контракты 2024-04-27 22:44:16 +04:00
40ba0f646b сделала contracts 2024-04-27 22:39:17 +04:00
2dc266ad03 + ссылка на контракты 2024-04-26 22:05:37 +04:00
dbe078ab91 + BusinessLogic 2024-04-26 22:04:15 +04:00
29d819d500 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-26 21:31:12 +04:00
e6aeacae24 добавила контракты 2024-04-26 21:30:35 +04:00
9a0d8d4e50 + BindingModels
+ Edit DataModels
2024-04-26 21:27:52 +04:00
50b737db72 создала папочки 2024-04-26 20:50:50 +04:00
b6f45d7039 test 2024-04-26 20:47:34 +04:00
4c5959d848 тест коммитов 2024-04-26 20:47:07 +04:00
8e75025863 создала контракты 2024-04-26 20:45:59 +04:00
070a07db24 убрала скобку 2024-04-26 20:23:06 +04:00
c5e513659e Исправления ошибок 2024-04-26 20:20:44 +04:00
704166efaa исправь 2024-04-26 20:06:37 +04:00
a6c5493980 Исправление ошибок 2024-04-26 20:02:35 +04:00
390071e31a sales ->sale 2024-04-26 20:00:44 +04:00
17ca45204f Мои сущности 2024-04-26 19:59:56 +04:00
27b3d0c77f Изменения типов данных 2024-04-26 19:58:18 +04:00
46b1357fa3 создала модели 2024-04-26 19:51:43 +04:00
ba75c2e410 Удалил надпись 2024-04-03 18:00:11 +04:00
dce2285614 Удалил класс 2024-04-03 17:59:42 +04:00
2b17a374c1 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-03 17:59:05 +04:00
d305341453 тест 2024-04-03 17:57:56 +04:00
8dc4e4f085 Тест 2024-04-03 17:57:51 +04:00
2be1d77659 добавила класс 2024-04-03 17:56:58 +04:00
c74ef32574 тестовый 4 2024-04-03 17:50:56 +04:00
17b4558b2e Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-03 17:50:07 +04:00
f50714c947 Тест 2024-04-03 17:50:03 +04:00
2247b5f0a2 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-03 17:49:02 +04:00
1432bbae72 тестовый 3 2024-04-03 17:48:47 +04:00
818f1628ac тестовый 3 2024-04-03 17:47:19 +04:00
198416dca8 Merge branch 'CarCenter' of https://git.is.ulstu.ru/spacyboy/CourseWork_CarCenter into CarCenter 2024-04-03 17:46:09 +04:00
02f764ee5d тесттт 2024-04-03 17:46:06 +04:00
290b14c45e тестовый 2 2024-04-03 17:44:43 +04:00
e2d1dd1699 Тест33 2024-04-03 17:44:19 +04:00
44afb9182e тестовый 2024-04-03 17:41:40 +04:00
803775d4b9 Тест2 2024-04-03 17:40:23 +04:00
223a52d4c9 проверочка 2024-04-03 17:39:06 +04:00
9051a05de2 Test1 2024-04-03 17:33:56 +04:00
322 changed files with 161011 additions and 0 deletions

67
CarCenter/CarCenter.sln Normal file
View File

@ -0,0 +1,67 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.7.34031.279
MinimumVisualStudioVersion = 10.0.40219.1
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CarCenterView", "CarCenterView\CarCenterView.csproj", "{33FB84BE-4026-4B3E-92C5-CAE3BE4DFF25}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CarCenterDataModels", "CarCenterDataModels\CarCenterDataModels.csproj", "{F4F98E8B-8BF9-451B-9FC8-FE00D74001CD}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CarCenterContracts", "CarCenterContracts\CarCenterContracts.csproj", "{A1AAA9E0-F136-4327-B87E-602AC821E049}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CarCenterBusinessLogic", "CarCenterBusinessLogic\CarCenterBusinessLogic.csproj", "{3736E810-EE9C-4424-964B-FBAA50385E64}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "CarCenterDataBaseImplement", "CarCenterDataBaseImplement\CarCenterDataBaseImplement.csproj", "{BA715056-F26D-4828-B24D-C008733FB61A}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CarCenterManagerApp", "CarCenterManagerApp\CarCenterManagerApp.csproj", "{AC307AF0-A60A-49D9-ADB1-60918625510D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CarCenterRestApi", "CarCenterRestApi\CarCenterRestApi.csproj", "{216AF0B4-4EC3-44E9-A81C-98AFF268D70D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CarCenterAdministratorApp", "CarCenterAdministratorApp\CarCenterAdministratorApp.csproj", "{0A61B0F9-B0F6-448A-8831-EA7823575958}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{33FB84BE-4026-4B3E-92C5-CAE3BE4DFF25}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{33FB84BE-4026-4B3E-92C5-CAE3BE4DFF25}.Debug|Any CPU.Build.0 = Debug|Any CPU
{33FB84BE-4026-4B3E-92C5-CAE3BE4DFF25}.Release|Any CPU.ActiveCfg = Release|Any CPU
{33FB84BE-4026-4B3E-92C5-CAE3BE4DFF25}.Release|Any CPU.Build.0 = Release|Any CPU
{F4F98E8B-8BF9-451B-9FC8-FE00D74001CD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{F4F98E8B-8BF9-451B-9FC8-FE00D74001CD}.Debug|Any CPU.Build.0 = Debug|Any CPU
{F4F98E8B-8BF9-451B-9FC8-FE00D74001CD}.Release|Any CPU.ActiveCfg = Release|Any CPU
{F4F98E8B-8BF9-451B-9FC8-FE00D74001CD}.Release|Any CPU.Build.0 = Release|Any CPU
{A1AAA9E0-F136-4327-B87E-602AC821E049}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{A1AAA9E0-F136-4327-B87E-602AC821E049}.Debug|Any CPU.Build.0 = Debug|Any CPU
{A1AAA9E0-F136-4327-B87E-602AC821E049}.Release|Any CPU.ActiveCfg = Release|Any CPU
{A1AAA9E0-F136-4327-B87E-602AC821E049}.Release|Any CPU.Build.0 = Release|Any CPU
{3736E810-EE9C-4424-964B-FBAA50385E64}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{3736E810-EE9C-4424-964B-FBAA50385E64}.Debug|Any CPU.Build.0 = Debug|Any CPU
{3736E810-EE9C-4424-964B-FBAA50385E64}.Release|Any CPU.ActiveCfg = Release|Any CPU
{3736E810-EE9C-4424-964B-FBAA50385E64}.Release|Any CPU.Build.0 = Release|Any CPU
{BA715056-F26D-4828-B24D-C008733FB61A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{BA715056-F26D-4828-B24D-C008733FB61A}.Debug|Any CPU.Build.0 = Debug|Any CPU
{BA715056-F26D-4828-B24D-C008733FB61A}.Release|Any CPU.ActiveCfg = Release|Any CPU
{BA715056-F26D-4828-B24D-C008733FB61A}.Release|Any CPU.Build.0 = Release|Any CPU
{AC307AF0-A60A-49D9-ADB1-60918625510D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{AC307AF0-A60A-49D9-ADB1-60918625510D}.Debug|Any CPU.Build.0 = Debug|Any CPU
{AC307AF0-A60A-49D9-ADB1-60918625510D}.Release|Any CPU.ActiveCfg = Release|Any CPU
{AC307AF0-A60A-49D9-ADB1-60918625510D}.Release|Any CPU.Build.0 = Release|Any CPU
{216AF0B4-4EC3-44E9-A81C-98AFF268D70D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{216AF0B4-4EC3-44E9-A81C-98AFF268D70D}.Debug|Any CPU.Build.0 = Debug|Any CPU
{216AF0B4-4EC3-44E9-A81C-98AFF268D70D}.Release|Any CPU.ActiveCfg = Release|Any CPU
{216AF0B4-4EC3-44E9-A81C-98AFF268D70D}.Release|Any CPU.Build.0 = Release|Any CPU
{0A61B0F9-B0F6-448A-8831-EA7823575958}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{0A61B0F9-B0F6-448A-8831-EA7823575958}.Debug|Any CPU.Build.0 = Debug|Any CPU
{0A61B0F9-B0F6-448A-8831-EA7823575958}.Release|Any CPU.ActiveCfg = Release|Any CPU
{0A61B0F9-B0F6-448A-8831-EA7823575958}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {9F0C696A-1EB1-46EF-BCB2-1927DB215E31}
EndGlobalSection
EndGlobal

View File

@ -0,0 +1,50 @@
using CarCenterContracts.ViewModels;
using Newtonsoft.Json;
using System.Net.Http.Headers;
using System.Text;
namespace CarCenterAdministratorApp
{
public class APIClient
{
private static readonly HttpClient _administrator = new();
public static AdministratorViewModel? Administrator { get; set; } = null;
public static void Connect(IConfiguration configuration)
{
_administrator.BaseAddress = new Uri(configuration["IPAddress"]);
_administrator.DefaultRequestHeaders.Accept.Clear();
_administrator.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
}
public static T? GetRequest<T>(string requestUrl)
{
var response = _administrator.GetAsync(requestUrl);
var result = response.Result.Content.ReadAsStringAsync().Result;
if (response.Result.IsSuccessStatusCode)
{
return JsonConvert.DeserializeObject<T>(result);
}
else
{
throw new Exception(result);
}
}
public static void PostRequest<T>(string requestUrl, T model)
{
var json = JsonConvert.SerializeObject(model);
var data = new StringContent(json, Encoding.UTF8, "application/json");
var response = _administrator.PostAsync(requestUrl, data);
var result = response.Result.Content.ReadAsStringAsync().Result;
if (!response.Result.IsSuccessStatusCode)
{
throw new Exception(result);
}
}
}
}

View File

@ -0,0 +1,20 @@
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\CarCenterBusinessLogic\CarCenterBusinessLogic.csproj" />
<ProjectReference Include="..\CarCenterContracts\CarCenterContracts.csproj" />
<ProjectReference Include="..\CarCenterDataBaseImplement\CarCenterDataBaseImplement.csproj" />
<ProjectReference Include="..\CarCenterDataModels\CarCenterDataModels.csproj" />
</ItemGroup>
</Project>

View File

@ -0,0 +1,685 @@
using CarCenterAdministratorApp.Models;
using CarCenterBusinessLogic.BusinessLogics;
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.ViewModels;
using CarCenterDataBaseImplement;
using CarCenterDataBaseImplement.Models;
using CarCenterDataModels.Models;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.Rendering;
using Microsoft.IdentityModel.Tokens;
using System.Diagnostics;
using System.Globalization;
using System.Reflection;
namespace CarCenterAdministratorApp.Controllers
{
public class HomeController : Controller
{
private readonly ILogger<HomeController> _logger;
private readonly ICarLogic _car;
private readonly IEquipmentLogic _equipment;
private readonly IInspectionLogic _inspection;
public HomeController(ILogger<HomeController> logger, ICarLogic car, IEquipmentLogic equipment, IInspectionLogic inspection)
{
_logger = logger;
_car = car;
_equipment = equipment;
_inspection = inspection;
}
public IActionResult Index()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View();
}
[HttpGet]
public IActionResult Privacy()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View(APIClient.Administrator);
}
[HttpPost]
public void Privacy(string email, string password, string fio, string telephone)
{
if (APIClient.Administrator == null)
{
throw new Exception("Вы как сюда попали? Сюда вход только авторизованным");
}
if (string.IsNullOrEmpty(password) || string.IsNullOrEmpty(fio))
{
throw new Exception("Введите логин, пароль и ФИО");
}
APIClient.PostRequest("api/administrator/updatedata", new AdministratorBindingModel
{
Id = APIClient.Administrator.Id,
AdministratorFIO = fio,
AdministratorLogin = email,
AdministratorPassword = password,
AdministratorEmail = email,
AdministratorNumber = telephone
});
APIClient.Administrator.AdministratorFIO = fio;
APIClient.Administrator.AdministratorLogin = email;
APIClient.Administrator.AdministratorPassword = password;
APIClient.Administrator.AdministratorEmail = email;
APIClient.Administrator.AdministratorNumber = telephone;
Response.Redirect("Index");
}
[HttpGet]
public IActionResult Register()
{
return View();
}
[HttpPost]
public void Register(string email, string password, string fio, string telephone)
{
if (string.IsNullOrEmpty(password) || string.IsNullOrEmpty(fio))
{
throw new Exception("Введите логин, пароль и ФИО");
}
APIClient.PostRequest("api/administrator/register", new AdministratorBindingModel
{
AdministratorFIO = fio,
AdministratorNumber = telephone,
AdministratorLogin = email,
AdministratorEmail = email,
AdministratorPassword = password
});
Response.Redirect("Enter");
return;
}
[HttpGet]
public IActionResult Enter()
{
return View();
}
[HttpPost]
public void Enter(string login, string password)
{
if (string.IsNullOrEmpty(login) || string.IsNullOrEmpty(password))
{
throw new Exception("Введите логин и пароль");
}
APIClient.Administrator = APIClient.GetRequest<AdministratorViewModel>($"api/administrator/login?login={login}&password={password}");
if (APIClient.Administrator == null)
{
throw new Exception("Неверный логин/пароль");
}
Response.Redirect("Index");
}
public IActionResult CreateCar()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View();
}
[HttpPost]
public void CreateCar(string brandCar, string model)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(brandCar))
{
throw new Exception("Введите бренд машины");
}
if (string.IsNullOrEmpty(model))
{
throw new Exception("Введите модель машины");
}
APIClient.PostRequest("api/main/createcar", new CarBindingModel
{
BrandCar = brandCar,
Model = model,
AdministratorId = APIClient.Administrator.Id,
});
Response.Redirect("ListCars");
}
public IActionResult ListCars()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View(APIClient.GetRequest<List<CarViewModel>>($"api/main/getcarlist?administratorId={APIClient.Administrator.Id}"));
}
public IActionResult UpdateCar()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Cars = APIClient.GetRequest<List<CarViewModel>>($"api/main/getcarlist?administratorId={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void UpdateCar(int car, string brandCar, string model)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(brandCar))
{
throw new Exception("заполните все поля");
}
if (string.IsNullOrEmpty(model))
{
throw new Exception("заполните все поля");
}
APIClient.PostRequest("api/main/updatecar", new CarBindingModel
{
Id = car,
BrandCar = brandCar,
Model = model,
AdministratorId = APIClient.Administrator.Id,
});
Response.Redirect("ListCars");
}
public IActionResult DeleteCar()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Cars = APIClient.GetRequest<List<CarViewModel>>($"api/main/getcarlist?administratorId={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void DeleteCar(int car)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
APIClient.PostRequest("api/main/deletecar", new CarBindingModel
{
Id = car
});
Response.Redirect("ListCars");
}
public IActionResult ListEquipments()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View(APIClient.GetRequest<List<EquipmentViewModel>>($"api/main/getequipmentlist?administratorId={APIClient.Administrator.Id}"));
}
[HttpGet]
public IActionResult CreateEquipment()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
var list = _car.ReadList(new CarSearchModel { AdministratorId = APIClient.Administrator.Id });
var simpCar = list.Select(x => new { CarId = x.Id, BrandCar = x.BrandCar });
ViewBag.Cars = new MultiSelectList(simpCar, "CarId", "BrandCar");
return View();
}
[HttpPost]
public void CreateEquipment(string equipmentName, double equipmentPrice, int[] cars)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(equipmentName) || equipmentPrice <= 0 || cars.Length == 0)
{
throw new Exception("Введите название и стоимость");
}
Dictionary<int, ICarModel> equipmentCars = new Dictionary<int, ICarModel>();
foreach (int id in cars)
{
equipmentCars.Add(id, _car.ReadElement(new CarSearchModel { Id = id }));
}
APIClient.PostRequest("api/main/CreateEquipment", new EquipmentBindingModel
{
AdministratorId = APIClient.Administrator.Id,
EquipmentName = equipmentName,
EquipmentPrice = equipmentPrice,
DateCreateEquipment = DateTime.Now
});
for (int i = 0; i < cars.Length; i++)
{
APIClient.PostRequest("api/main/AddCarToEquipment", Tuple.Create(
new EquipmentSearchModel() { EquipmentName = equipmentName },
new CarViewModel() { Id = cars[i] }
));
}
Response.Redirect("ListEquipments");
}
public IActionResult UpdateEquipment()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Equipments = APIClient.GetRequest<List<EquipmentViewModel>>($"api/main/getequipmentlist?administratorId={APIClient.Administrator.Id}");
ViewBag.Cars = APIClient.GetRequest<List<CarViewModel>>($"api/main/getcarlist?administratorid={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void UpdateEquipment(int equipment, string equipmentName, double equipmentPrice, List<int> cars)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(equipmentName))
{
throw new Exception("Введите название");
}
if (string.IsNullOrEmpty(equipmentPrice.ToString()))
{
throw new Exception("Введите стоимость");
}
Dictionary<int, ICarModel> a = new Dictionary<int, ICarModel>();
foreach (int car in cars)
{
a.Add(car, new CarSearchModel { Id = car } as ICarModel);
}
APIClient.PostRequest("api/main/updateequipment", new EquipmentBindingModel
{
Id = equipment,
AdministratorId = APIClient.Administrator.Id,
EquipmentName = equipmentName,
EquipmentPrice = equipmentPrice,
EquipmentCars = a,
});
Response.Redirect("ListEquipments");
}
public IActionResult DeleteEquipment()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Equipments = APIClient.GetRequest<List<EquipmentViewModel>>($"api/main/getequipmentlist?administratorId={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void DeleteEquipment(int equipment)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
APIClient.PostRequest("api/main/deleteequipment", new EquipmentBindingModel
{
Id = equipment
});
Response.Redirect("ListEquipments");
}
public IActionResult ListInspections()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
return View(APIClient.GetRequest<List<InspectionViewModel>>($"api/main/getInspectionlist?administratorId={APIClient.Administrator.Id}"));
}
public IActionResult CreateInspection()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
var list = _car.ReadList(new CarSearchModel { AdministratorId = APIClient.Administrator.Id });
var simpCar = list.Select(x => new { CarId = x.Id, BrandCar = x.BrandCar });
ViewBag.Cars = new MultiSelectList(simpCar, "CarId", "BrandCar");
return View();
}
[HttpPost]
public void CreateInspection(string inspectionName, DateTime inspectionDate, int[] cars)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(inspectionName) || cars.Length == 0)
{
throw new Exception("Введите название и стоимость");
}
Dictionary<int, ICarModel> inspectionCars = new Dictionary<int, ICarModel>();
foreach (int id in cars)
{
inspectionCars.Add(id, _car.ReadElement(new CarSearchModel { Id = id }));
}
APIClient.PostRequest("api/main/Createinspection", new InspectionBindingModel
{
AdministratorId = APIClient.Administrator.Id,
InspectionName = inspectionName,
InspectionDate = inspectionDate
});
for (int i = 0; i < cars.Length; i++)
{
APIClient.PostRequest("api/main/AddCarToInspection", Tuple.Create(
new InspectionSearchModel() { InspectionName = inspectionName },
new CarViewModel() { Id = cars[i] }
));
}
Response.Redirect("ListInspections");
}
public IActionResult UpdateInspection()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Cars = APIClient.GetRequest<List<CarViewModel>>($"api/main/getcarlist?administratorid={APIClient.Administrator.Id}");
ViewBag.Inspections = APIClient.GetRequest<List<InspectionViewModel>>($"api/main/getinspectionlist?administratorId={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void UpdateInspection(int inspection, string inspectionName, DateTime inspectionDate, List<int> cars)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
if (string.IsNullOrEmpty(inspectionName))
{
throw new Exception("Введите название");
}
Dictionary<int, ICarModel> a = new Dictionary<int, ICarModel>();
foreach (int car in cars)
{
a.Add(car, new CarSearchModel { Id = car } as ICarModel);
}
APIClient.PostRequest("api/main/updateinspection", new InspectionBindingModel
{
Id = inspection,
AdministratorId = APIClient.Administrator.Id,
InspectionName = inspectionName,
InspectionDate = inspectionDate,
InspectionCars = a,
});
Response.Redirect("ListInspections");
}
public IActionResult DeleteInspection()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Inspections = APIClient.GetRequest<List<InspectionViewModel>>($"api/main/getinspectionlist?administratorId={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void DeleteInspection(int inspection)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
APIClient.PostRequest("api/main/deleteinspection", new InspectionBindingModel
{
Id = inspection
});
Response.Redirect("ListInspections");
}
public IActionResult AddEquipmentToPreSaleWork()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Equipments = APIClient.GetRequest<List<EquipmentViewModel>>($"api/main/getequipmentlist?administratorId={APIClient.Administrator.Id}");
ViewBag.PreSaleWorks = APIClient.GetRequest<List<PreSaleWorkViewModel>>($"api/main/getpreSaleWorks");
return View();
}
[HttpPost]
public void AddEquipmentToPreSaleWork(int preSaleWork, int equipment)
{
if (APIClient.Administrator == null)
{
throw new Exception("Необходима авторизация");
}
using var context = new CarCenterDataBase();
List<int> cars = context.EquipmentCars
.Where(rd => rd.EquipmentId == equipment)
.Select(rd => rd.CarId)
.ToList();
Dictionary<int, ICarModel> a = new Dictionary<int, ICarModel>();
foreach (int car in cars)
{
a.Add(car, new CarSearchModel { Id = car } as ICarModel);
}
var equipmentElem = APIClient.GetRequest<EquipmentViewModel>($"api/main/getequipmentbyid?EquipmentId={equipment}");
APIClient.PostRequest("api/main/updateequipment", new EquipmentBindingModel
{
Id = equipment,
PreSaleWorkId = preSaleWork,
EquipmentName = equipmentElem.EquipmentName,
EquipmentPrice = equipmentElem.EquipmentPrice,
AdministratorId = equipmentElem.AdministratorId,
EquipmentCars = a,
});
Response.Redirect("ListEquipments");
}
/*public IActionResult ListCarsToPdfFile()
{
return View();
}
public IActionResult ListCarEquipmentToFile()
{
return View();
}*/
[HttpGet]
public IActionResult PreSaleWorkCarReport()
{
if (APIClient.Administrator == null)
{
return Redirect("~/Home/Enter");
}
ViewBag.Cars = APIClient.GetRequest<List<CarViewModel>>($"api/main/GetCarList?Administratorid={APIClient.Administrator.Id}");
return View();
}
[HttpPost]
public void PreSaleWorkCarReport(List<int> cars, string type)
{
if (APIClient.Administrator == null)
{
throw new Exception("Вы как суда попали? Суда вход только авторизованным");
}
if (cars.Count <= 0)
{
throw new Exception("Количество должно быть больше 0");
}
if (string.IsNullOrEmpty(type))
{
throw new Exception("Неверный тип отчета");
}
if (type == "docx")
{
APIClient.PostRequest("api/main/createPreSaleWorklistwordfile", new ReportPreSaleWorkCarBindingModel
{
Cars = cars,
FileName = "C:\\Users\\kozlo\\Desktop\\reports\\wordfile.docx"
});
Response.Redirect("GetWordFile");
}
else
{
APIClient.PostRequest("api/main/createPreSaleWorklistexcelfile", new ReportPreSaleWorkCarBindingModel
{
Cars = cars,
FileName = "C:\\Users\\kozlo\\Desktop\\reports\\excelfile.xlsx"
});
Response.Redirect("GetExcelFile");
}
}
[HttpGet]
public IActionResult GetWordFile()
{
return new PhysicalFileResult("C:\\Users\\kozlo\\Desktop\\reports\\wordfile.docx", "application/vnd.openxmlformats-officedocument.wordprocessingml.document");
}
public IActionResult GetExcelFile()
{
return new PhysicalFileResult("C:\\Users\\kozlo\\Desktop\\reports\\excelfile.xlsx", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
}
[HttpGet]
public IActionResult Report()
{
ViewBag.Report = new List<ReportEquipmentsEmployeesBindingModel>();
return View();
}
[HttpGet]
public string GetCarsReport(DateTime dateFrom, DateTime dateTo)
{
if (APIClient.Administrator == null)
{
throw new Exception("Вы как суда попали? Суда вход только авторизованным");
}
List<ReportEquipmentsEmployeesViewModel> result;
try
{
string dateFromS = dateFrom.ToString("s", CultureInfo.InvariantCulture);
string dateToS = dateTo.ToString("s", CultureInfo.InvariantCulture);
result = APIClient.GetRequest<List<ReportEquipmentsEmployeesViewModel>>
($"api/main/GetEquipmentsEmployeesReport?datefrom={dateFromS}&dateto={dateToS}&administratorid={APIClient.Administrator.Id}")!;
}
catch (Exception ex)
{
_logger.LogError(ex, "Ошибка создания отчета");
throw;
}
string table = "";
table += "<h2 class=\"text-custom-color-1\">Предварительный отчет</h2>";
table += "<div class=\"table-responsive\">";
table += "<table class=\"table table-striped table-bordered table-hover\">";
table += "<thead class=\"table-dark\">";
table += "<tr>";
table += "<th scope=\"col\">Дата</th>";
table += "<th scope=\"col\">Название машины</th>";
table += "<th scope=\"col\">Комплектация</th>";
table += "<th scope=\"col\">Сотрудник</th>";
table += "</tr>";
table += "</thead>";
foreach (var car in result)
{
table += "<tbody>";
table += "<tr>";
table += $"<td></td>";
table += $"<td>{car.BrandCar}</td>";
table += $"<td></td>";
table += $"<td></td>";
table += "</tr>";
foreach (var equipment in car.Equipments)
{
table += "<tr>";
table += $"<td>{equipment.DateCreateEquipment}</td>";
table += $"<td></td>";
table += $"<td>{equipment.EquipmentName}</td>";
table += $"<td></td>";
table += "</tr>";
}
foreach (var employee in car.Employees)
{
table += "<tr>";
table += $"<td></td>";
table += $"<td></td>";
table += $"<td></td>";
table += $"<td>{employee.EmployeeFIO}</td>";
table += "</tr>";
}
table += "</tbody>";
}
table += "</table>";
table += "</div>";
return table;
}
[HttpPost]
public void Report(DateTime dateFrom, DateTime dateTo)
{
if (APIClient.Administrator == null)
{
throw new Exception("Вы как суда попали? Суда вход только авторизованным");
}
APIClient.PostRequest("api/main/sendEquipmentsEmployeesreporttoemail", new ReportEquipmentsEmployeesBindingModel
{
FileName = "C:\\Users\\kozlo\\Desktop\\reports\\pdffile.pdf",
AdministratorId = APIClient.Administrator.Id,
DateFrom = dateFrom,
DateTo = dateTo,
Email = APIClient.Administrator.AdministratorEmail
});
Response.Redirect("Report");
}
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
public IActionResult Error()
{
return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier });
}
}
}

View File

@ -0,0 +1,9 @@
namespace CarCenterAdministratorApp.Models
{
public class ErrorViewModel
{
public string? RequestId { get; set; }
public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
}
}

View File

@ -0,0 +1,53 @@
using CarCenterAdministratorApp;
using CarCenterBusinessLogic.BusinessLogics;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.StoragesContracts;
using CarCenterDataBaseImplement.Implements;
using CarCenterDataBaseImplement.Models;
var builder = WebApplication.CreateBuilder(args);
builder.Services.AddTransient<IAdministratorStorage, AdministratorStorage>();
builder.Services.AddTransient<IAdministratorLogic, AdministratorLogic>();
builder.Services.AddTransient<ICarStorage, CarStorage>();
builder.Services.AddTransient<ICarLogic, CarLogic>();
builder.Services.AddTransient<IEquipmentStorage, EquipmentStorage>();
builder.Services.AddTransient<IEquipmentLogic, EquipmentLogic>();
builder.Services.AddTransient<IInspectionStorage, InspectionStorage>();
builder.Services.AddTransient<IInspectionLogic, InspectionLogic>();
// Add services to the container.
builder.Services.AddControllersWithViews();
var app = builder.Build();
APIClient.Connect(builder.Configuration);
// Configure the HTTP request pipeline.
/*if (!app.Environment.IsDevelopment())
{*/
app.UseExceptionHandler("/Home/Error");
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
app.UseHsts();
/*}*/
app.UseHttpsRedirection();
app.UseStaticFiles();
app.UseRouting();
app.UseAuthorization();
app.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}");
app.Run();

View File

@ -0,0 +1,28 @@
{
"iisSettings": {
"windowsAuthentication": false,
"anonymousAuthentication": true,
"iisExpress": {
"applicationUrl": "http://localhost:11896",
"sslPort": 44358
}
},
"profiles": {
"CarCenterAdministratorApp": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"applicationUrl": "https://localhost:7044;http://localhost:5042",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
},
"IIS Express": {
"commandName": "IISExpress",
"launchBrowser": true,
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
}
}
}

View File

@ -0,0 +1,31 @@
@using CarCenterContracts.ViewModels;
@using CarCenterDataModels.Models;
@{
ViewData["Title"] = "AddEquipmentToPreSaleWork";
}
<form method="post">
<div class="u-form-group u-form-name u-label-top">
<br>
<br>
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Предпродажная работа: </label>
<div class="u-input u-input-rectangle">
<select id="preSaleWork" name="preSaleWork" class="form-control" asp-items="@(new SelectList(@ViewBag.PreSaleWorks, "Id", "PreSaleWorkType"))"></select>
</div>
</div>
<br>
<div class="u-form-group u-form-name u-label-top">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Комплектация: </label>
<br>
<div class="u-input u-input-rectangle">
<select id="equipment" name="equipment" class="form-control" asp-items="@(new SelectList(@ViewBag.Equipments, "Id", "EquipmentName"))"></select>
</div>
<br>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Привязать" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</form>

View File

@ -0,0 +1,25 @@
@{
ViewData["Title"] = "CreateCar";
}
<form method="post">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Бренд машины</label>
</div>
<input type="text"
placeholder="Введите бренд машины"
name="brandCar"
class="form-control" />
<br>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2">Модель</label>
</div>
<input type="text"
placeholder="Введите модель"
name="model"
class="form-control" />
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</form>

View File

@ -0,0 +1,34 @@
@{
ViewData["Title"] = "CreateEquipment";
}
<div class="text-center">
<h2 class="display-4">Создание комлектации</h2>
</div>
<form method="post">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Название комплектации</label>
</div>
<input type="text"
placeholder="Введите название комплектации"
name="equipmentName"
class="form-control" />
<br>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2">Стоимость комплектации</label>
</div>
<input type="number"
placeholder="Введите стоимость комплектации"
name="equipmentPrice"
class="form-control" />
<label class="u-label u-text-custom-color-1 u-label-1">Машины: </label>
<div>
@Html.ListBox("cars", (MultiSelectList)ViewBag.Cars, new { @class = "form-control", size = "10" })
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</form>

View File

@ -0,0 +1,34 @@
@{
ViewData["Title"] = "CreateInspection";
}
<div class="text-center">
<h2 class="display-4">Создание осмотра</h2>
</div>
<form method="post">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Название осмотра</label>
</div>
<input type="text"
placeholder="Введите название осмотра"
name="inspectionName"
class="form-control" />
<br>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2">Дата осмотра</label>
</div>
<input type="date"
placeholder="Выберите дату осмотра"
name="inspectionDate"
class="form-control" />
<label class="u-label u-text-custom-color-1 u-label-1">Машины: </label>
<div>
@Html.ListBox("cars", (MultiSelectList)ViewBag.Cars, new { @class = "form-control", size = "10" })
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</form>

View File

@ -0,0 +1,17 @@
@{
ViewData["Title"] = "DeleteCar";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center0">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Машины: </label>
<select id="car" name="car" class="form-control" asp-items="@(new SelectList(@ViewBag.Cars, "Id", "BrandCar"))"></select>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,18 @@
@{
ViewData["Title"] = "DeleteEquipment";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center0">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Комплектации: </label>
<select id="equipment" name="equipment" class="form-control" asp-items="@(new SelectList(@ViewBag.Equipments, "Id", "EquipmentName"))"></select>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Удалить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,18 @@
@{
ViewData["Title"] = "DeleteInspection";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center0">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1">Осмотр: </label>
<select id="inspection" name="inspection" class="form-control" asp-items="@(new SelectList(@ViewBag.Inspections, "Id", "InspectionName"))"></select>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,35 @@
@{
ViewData["Title"] = "Enter";
}
<div class="text-center">
<h2 class="u-text u-text-custom-color-1 u-text-default u-text-1">
Вход
</h2>
</div>
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 7px">Электронная почта:</label>
<input type="text"
placeholder="Введите свой логин"
name="login"
class="form-control" />
</div>
<br>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 7px">Пароль:</label>
<input type="password"
placeholder="Введите свой пароль"
name="password"
class="form-control" />
</div>
<br>
<div class="form-group text-center" style="padding-bottom: 120px;">
<button type="submit" class="btn btn-outline-dark text-center d-flex justify-content-md-center">Войти</button>
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,8 @@
@{
ViewData["Title"] = "Home Page";
}
<div class="text-center">
<h1 class="display-4">Добро пожаловать!</h1>
<h2 class="display-6">Автоцентр «Корыто»</h2>
</div>

View File

@ -0,0 +1,67 @@
@using CarCenterContracts.ViewModels
@model List<CarViewModel>
@{
ViewData["Title"] = "ListCars";
}
<section class="u-clearfix u-section-1" id="sec-e38b">
<div class="u-clearfix u-sheet u-sheet-1">
<div class="u-clearfix u-layout-wrap u-layout-wrap-1">
<div class="u-layout">
<div class="u-layout-row">
<div class="u-container-style u-layout-cell u-size-48 u-layout-cell-1">
<div class="u-container-layout u-container-layout-1">
<div class="u-table u-table-responsive u-table-1">
<table class="table">
<thead class="thead-dark">
<tr style="height: 31px">
<th class="u-border-1 u-border-grey-50 u-table-cell">
Номер
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Бренд машины
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Модель машины
</th>
</tr>
</thead>
<tbody class="u-table-body">
@foreach (var item in Model)
{
<tr style="height: 75px">
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.Id)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.BrandCar)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.Model)
</td>
</tr>
}
</tbody>
</table>
</div>
</div>
</div>
<div class="u-container-style u-layout-cell u-size-12 u-layout-cell-2">
<div class="u-container-layout u-container-layout-2">
<a asp-area="" asp-controller="Home" asp-action="CreateCar"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Добавить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="UpdateCar"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Изменить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="DeleteCar"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Удалить</a>
</div>
</div>
</div>
</div>
</div>
</div>
</section>

View File

@ -0,0 +1,68 @@
@using CarCenterContracts.ViewModels
@model List<EquipmentViewModel>
@{
ViewData["Title"] = "ListEquipments";
}
<section class="u-clearfix u-section-1" id="sec-e38b">
<div class="u-clearfix u-sheet u-sheet-1">
<div class="u-clearfix u-layout-wrap u-layout-wrap-1">
<div class="u-layout">
<div class="u-layout-row">
<div class="u-container-style u-layout-cell u-size-48 u-layout-cell-1">
<div class="u-container-layout u-container-layout-1">
<div class="u-table u-table-responsive u-table-1">
<table class="table">
<thead class="thead-dark">
<tr style="height: 31px">
<th class="u-border-1 u-border-grey-50 u-table-cell">
Номер
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Название
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Стоимость
</th>
</tr>
</thead>
<tbody class="u-table-body">
@foreach (var item in Model)
{
<tr style="height: 75px">
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.Id)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.EquipmentName)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.EquipmentPrice)
</td>
</tr>
}
</tbody>
</table>
</div>
</div>
</div>
<div class="u-container-style u-layout-cell u-size-12 u-layout-cell-2">
<div class="u-container-layout u-container-layout-2">
<a asp-area="" asp-controller="Home" asp-action="CreateEquipment"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Добавить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="UpdateEquipment"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Изменить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="DeleteEquipment"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Удалить</a>
&nbsp;
</div>
</div>
</div>
</div>
</div>
</div>
</section>

View File

@ -0,0 +1,68 @@
@using CarCenterContracts.ViewModels
@model List<InspectionViewModel>
@{
ViewData["Title"] = "ListInspections";
}
<section class="u-clearfix u-section-1" id="sec-e38b">
<div class="u-clearfix u-sheet u-sheet-1">
<div class="u-clearfix u-layout-wrap u-layout-wrap-1">
<div class="u-layout">
<div class="u-layout-row">
<div class="u-container-style u-layout-cell u-size-48 u-layout-cell-1">
<div class="u-container-layout u-container-layout-1">
<div class="u-table u-table-responsive u-table-1">
<table class="table">
<thead class="thead-dark">
<tr style="height: 31px">
<th class="u-border-1 u-border-grey-50 u-table-cell">
Номер
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Название
</th>
<th class="u-border-1 u-border-grey-50 u-table-cell">
Дата
</th>
</tr>
</thead>
<tbody class="u-table-body">
@foreach (var item in Model)
{
<tr style="height: 75px">
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.Id)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.InspectionName)
</td>
<td class="u-border-1 u-border-grey-40 u-border-no-left u-border-no-right u-table-cell">
@Html.DisplayFor(modelItem => item.InspectionDate)
</td>
</tr>
}
</tbody>
</table>
</div>
</div>
</div>
<div class="u-container-style u-layout-cell u-size-12 u-layout-cell-2">
<div class="u-container-layout u-container-layout-2">
<a asp-area="" asp-controller="Home" asp-action="CreateInspection"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Добавить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="UpdateInspection"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Изменить</a>
&nbsp;
<a asp-area="" asp-controller="Home" asp-action="DeleteInspection"
class="btn btn-outline-dark text-center d-flex justify-content-md-center">Удалить</a>
&nbsp;
</div>
</div>
</div>
</div>
</div>
</div>
</section>

View File

@ -0,0 +1,81 @@
@using CarCenterContracts.ViewModels
@model List<CarViewModel>
@{
ViewData["Title"] = "PreSaleWorkCarReport";
}
<div class="text-center">
<h2 class="display-4">Создать список предпродажных работ по машинам</h2>
</div>
<form method="post">
<div class="row">
<div class="col-4">Машина:</div>
<div class="col-8">
<select name="cars" class="form-control" multiple size="5" id="cars">
@foreach (var service in ViewBag.Cars)
{
<option value="@service.Id">@service.BrandCar</option>
}
</select>
</div>
</div>
<div class="file-format">
<label class="form-label">Выберите формат файла:</label>
<div class="form-check">
<input class="form-check-input" type="radio" name="type" value="docx" id="docx">
<label class="form-check-label" for="docx">Word-файл</label>
</div>
<div class="form-check">
<input class="form-check-input" type="radio" name="type" value="xlsx" id="xlsx" checked>
<label class="form-check-label" for="xlsx">Excel-файл</label>
</div>
</div>
<div class="d-flex justify-content-center">
<button type="submit" class="btn btn-block btn-outline-dark w-100">Создать</button>
</div>
</form>
@* <div class="text-center">
<div class="title">
<h2>Создание отчета по машинам</h2>
</div>
</div>
<div class="text-center">
<form method="post">
<div class="file-format">
<label class="form-label">Выберите формат файла:</label>
<div class="form-check">
<input class="form-check-input" type="radio" name="type" value="docx" id="docx">
<label class="form-check-label" for="docx">Word-файл</label>
</div>
<div class="form-check">
<input class="form-check-input" type="radio" name="type" value="xlsx" id="xlsx" checked>
<label class="form-check-label" for="xlsx">Excel-файл</label>
</div>
</div>
<div class="table">
<table class="table table-hover">
<thead class="thead-light">
<tr>
<th scope="col"></th>
<th scope="col">бренд машины</th>
<th scope="col">модель</th>
</tr>
</thead>
<tbody>
</tbody>
</table>
</div>
<br>
<div class="d-flex justify-content-center">
<button type="submit" class="btn btn-block btn-outline-dark w-100">Создать</button>
</div>
</form>
</div> *@

View File

@ -0,0 +1,61 @@
@using CarCenterContracts.ViewModels
@model AdministratorViewModel
@{
ViewData["Title"] = "Privacy";
}
<div class="text-center">
<h2 class="u-text u-text-custom-color-1 u-text-default u-text-1"> Мои данные </h2>
</div>
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<div class="u-form-email u-form-group u-label-top">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 8px">Электронная почта</label>
<input type="email"
placeholder="Введите электронную почту"
name="email"
value="@Model.AdministratorEmail"
class="form-control" />
</div>
</div>
<div class="form-group">
<div class="u-form-group u-label-top u-form-group-3">
<label class="u-label u-text-custom-color-1 u-label-3" style="margin-bottom: 8px">ФИО</label>
<input type="text"
placeholder="Введите ФИО"
name="fio"
value="@Model.AdministratorFIO"
class="form-control" />
</div>
</div>
<div class="form-group">
<div class="u-form-group u-label-top u-form-group-4">
<label class="u-label u-text-custom-color-1 u-label-4" style="margin-bottom: 8px">Номер телефона</label>
<input type="text"
name="telephone"
class="form-control"
value="@Model.AdministratorNumber"
placeholder="Введите номер телефона" />
</div>
</div>
<div class="form-group">
<div class="u-form-group u-label-top u-form-group-5">
<label class="u-label u-text-custom-color-1 u-label-5" style="margin-bottom: 8px">Пароль</label>
<input type="text"
placeholder="Введите пароль"
name="password"
value="@Model.AdministratorPassword"
class="form-control" />
</div>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Изменить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,48 @@
@{
ViewData["Title"] = "Register";
}
<div class="text-center">
<h2 class="u-text u-text-custom-color-1 u-text-default u-text-1"> Регистрация </h2>
</div>
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 6px">Электронная почта:</label>
<input type="email"
placeholder="Введите электронную почту"
name="email"
class="form-control"
style="margin-bottom: 10px" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-3" style="margin-bottom: 7px">ФИО:</label>
<input type="text"
placeholder="Введите ФИО"
name="fio"
class="form-control"
style="margin-bottom: 10px" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-4" style="margin-bottom: 7px">Номер телефона:</label>
<input type="text"
name="telephone"
class="form-control"
placeholder="Введите номер телефона"
style="margin-bottom: 10px" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-5" style="margin-bottom: 7px">Пароль:</label>
<input type="password"
placeholder="Введите пароль"
name="password"
class="form-control"
style="margin-bottom: 10px" />
</div>
<br>
<div class="form-group text-center" style="padding-bottom: 120px;">
<button type="submit" class="btn btn-outline-dark text-center d-flex justify-content-md-center">Зарегистрироваться</button>
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,86 @@
@using CarCenterContracts.ViewModels
@{
ViewData["Title"] = "Report";
}
<div class="container">
<div class="text-center mb-4">
<h2 class="text-custom-color-1">Отчет по машинам за период</h2>
</div>
<form method="post">
<div class="row">
<div class="col-md-6">
<div class="form-group">
<label for="dateFrom" class="form-label text-custom-color-1">Начало периода:</label>
<input type="date" id="dateFrom" name="dateFrom" class="form-control" placeholder="Выберите дату начала периода">
</div>
</div>
<div class="col-md-6">
<div class="form-group">
<label for="dateTo" class="form-label text-custom-color-1">Окончание периода:</label>
<input type="date" id="dateTo" name="dateTo" class="form-control" placeholder="Выберите дату окончания периода">
</div>
</div>
</div>
@* <div class="form-group mb-4">
<label for="headwaiterEmail" class="form-label text-custom-color-1">Введите почту:</label>
<input type="email" id="headwaiterEmail" name="headwaiterEmail" class="form-control" placeholder="Введите вашу почту">
</div>
<div class="row mb-4">
<div class="col-md-8"></div>
<div class="col-md-4">
<button type="submit" class="btn btn-outline-dark w-100 text-center d-flex justify-content-md-center">Отправить на почту</button>
</div>
</div> *@
<div class="row mb-4">
<div class="col-md-8"></div>
<div class="col-md-4">
<button type="submit" class="btn btn-outline-dark w-100 text-center d-flex justify-content-md-center">Отправить на почту</button>
</div>
</div>
<div class="row mb-4">
<div class="col-md-8"></div>
<div class="col-md-4">
<button type="button" id="demonstrate" class="btn btn-outline-dark w-100 text-center d-flex justify-content-md-center">Продемонстрировать</button>
</div>
</div>
@*
<div class="row mb-4">
<div class="col-md-8"></div>
<div class="col-md-4">
<button type="button" id="demonstrate" class="btn btn-outline-dark w-100 text-center d-flex justify-content-md-center">Продемонстрировать</button>
</div>
</div> *@
<div id="report"></div>
</form>
</div>
@section Scripts {
<script>
function check() {
var dateFrom = $('#dateFrom').val();
var dateTo = $('#dateTo').val();
if (dateFrom && dateTo) {
$.ajax({
method: "GET",
url: "/Home/GetCarsReport",
data: { dateFrom: dateFrom, dateTo: dateTo },
success: function (result) {
if (result != null) {
$('#report').html(result);
}
}
});
};
}
check();
$('#demonstrate').on('click', (e) => check());
</script>
}

View File

@ -0,0 +1,60 @@
@using CarCenterContracts.ViewModels;
@using CarCenterDataModels.Models;
@{
ViewData["Title"] = "UpdateCar";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Машины: </label>
<select id="car" name="car" class="form-control" asp-items="@(new SelectList(@ViewBag.Cars, "Id", "BrandCar"))"></select>
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Бренд машины</label>
<input type="text"
id="brandCar"
placeholder="Введите название машины"
name="brandCar"
class="form-control" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 8px">Модель машины</label>
<input type="text"
id="model"
placeholder="Введите модель"
name="model"
class="form-control" />
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>
@section Scripts
{
<script>
function check() {
var car = $('#car').val();
if (car) {
$.ajax({
method: "GET",
url: "/Home/GetCar",
data: { carId: car },
success: function (result) {
$('#brandCar').val(result.BrandCar);
$('#model').val(result.Model);
}
});
};
}
check();
$('#car').on('change', function () {
check();
});
</script>
}

View File

@ -0,0 +1,78 @@
@using CarCenterContracts.ViewModels;
@using CarCenterDataModels.Models;
@{
ViewData["Title"] = "UpdateEquipment";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Комплектации: </label>
<select id="equipment" name="equipment" class="form-control" asp-items="@(new SelectList(@ViewBag.Equipments, "Id", "EquipmentName"))"></select>
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Название комплектации</label>
<input type="text"
id="equipmentName"
placeholder="Введите название конференции"
name="equipmentName"
class="form-control" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 8px">Стоимость комплектации</label>
<input type="number"
id="equipmentPrice"
placeholder="Введите стоимость комплектации"
name="equipmentPrice"
class="form-control" />
</div>
<div class="row">
<div class="col-4">Машины:</div>
<div class="col-8">
<select name="cars" class="form-control" multiple size="5" id="cars">
@foreach (var car in ViewBag.Cars)
{
<option value="@car.Id" data-name="@car.Id">@car.BrandCar</option>
}
</select>
</div>
</div>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>
@section Scripts
{
<script>
function check() {
var equipment = $('#equipment').val();
$("#cars option:selected").removeAttr("selected");
if (equipment) {
$.ajax({
method: "GET",
url: "/Home/GetEquipment",
data: { equipmentId: equipment },
success: function (result) {
$('#equipmentName').val(result.item1.equipmentName);
$('#equipmentPrice').val(result.item1.equipmentPrice);
$('#table-elements').html(result.item1.item2);
$.map(result.item2, function (n) {
$(`option[data-name=${n.item2}]`).attr("selected", "selected")
});
}
});
};
}
check();
$('#equipment').on('change', function () {
check();
});
</script>
}

View File

@ -0,0 +1,80 @@
@using CarCenterContracts.ViewModels;
@using CarCenterDataModels.Models;
@{
ViewData["Title"] = "UpdateInspection";
}
<form method="post">
<div class="container d-flex justify-content-center align-items-center">
<div class="card-body">
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Осмотры: </label>
<select id="inspection" name="inspection" class="form-control" asp-items="@(new SelectList(@ViewBag.Inspections, "Id", "InspectionName"))"></select>
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-1" style="margin-bottom: 8px">Название Осмотра</label>
<input type="text"
id="inspectionName"
placeholder="Введите название осмотра"
name="inspectionName"
class="form-control" />
</div>
<div class="form-group">
<label class="u-label u-text-custom-color-1 u-label-2" style="margin-bottom: 8px">Дата осмотра</label>
<input type="date"
id="inspectionDate"
placeholder="Выберите дату осмотра"
name="inspectionDate"
class="form-control" />
</div>
<div class="row">
<div class="col-4">Машины:</div>
<div class="col-8">
<select name="cars" class="form-control" multiple size="5" id="cars">
@foreach (var car in ViewBag.Cars)
{
<option value="@car.Id" data-name="@car.Id">@car.BrandCar</option>
}
</select>
</div>
</div>
</div>
<br>
<div class="u-container-layout u-container-layout-2">
<input type="submit" value="Сохранить" class="btn btn-outline-dark text-center d-flex justify-content-md-center" />
</div>
</div>
</div>
</form>
@section Scripts
{
<script>
function check() {
var inspection = $('#inspection').val();
$("#cars option:selected").removeAttr("selected");
if (inspection) {
$.ajax({
method: "GET",
url: "/Home/GetInspection",
data: { inspectionId: inspection },
success: function (result) {
$('#inspectionName').val(result.item1.inspectionName);
$('#inspectionDate').val(result.item1.inspectionDate);
$('#table-elements').html(result.item1.item2);
$.map(result.item2, function (n) {
$(`option[data-name=${n.item2}]`).attr("selected", "selected")
});
}
});
};
}
check();
$('#inspection').on('change', function () {
check();
});
</script>
}

View File

@ -0,0 +1,25 @@
@model ErrorViewModel
@{
ViewData["Title"] = "Error";
}
<h1 class="text-danger">Error.</h1>
<h2 class="text-danger">An error occurred while processing your request.</h2>
@if (Model.ShowRequestId)
{
<p>
<strong>Request ID:</strong> <code>@Model.RequestId</code>
</p>
}
<h3>Development Mode</h3>
<p>
Swapping to <strong>Development</strong> environment will display more detailed information about the error that occurred.
</p>
<p>
<strong>The Development environment shouldn't be enabled for deployed applications.</strong>
It can result in displaying sensitive information from exceptions to end users.
For local debugging, enable the <strong>Development</strong> environment by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment variable to <strong>Development</strong>
and restarting the app.
</p>

View File

@ -0,0 +1,79 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>@ViewData["Title"] - CarCenterAdministratorApp</title>
<link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css" />
<link rel="stylesheet" href="~/css/site.css" asp-append-version="true" />
<link rel="stylesheet" href="~/CarCenterAdministratorApp.styles.css" asp-append-version="true" />
</head>
<body>
<header>
<nav class="navbar navbar-expand-sm navbar-toggleable-sm navbar-light bg-white border-bottom box-shadow mb-3">
<div class="container-fluid">
<a class="navbar-brand" asp-area="" asp-controller="Home" asp-action="Index">CarCenter</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target=".navbar-collapse" aria-controls="navbarSupportedContent"
aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="navbar-collapse collapse justify-content-end" id="navbarNav">
<div class="navbar-nav">
<a class="nav-link " asp-area="" asp-controller="Home" asp-action="Index">Главное меню</a>
<li class="nav-item dropdown">
<a class="nav-link dropdown-toggle" href="#" id="navbarDropdownMenuLink" role="button" data-bs-toggle="dropdown" aria-expanded="false">
Справочники
</a>
<ul class="dropdown-menu" aria-labelledby="navbarDropdownMenuLink">
<li><a class="nav-link " asp-area="" asp-controller="Home" asp-action="ListCars">Машины</a></li>
<li><a class="nav-link " asp-area="" asp-controller="Home" asp-action="ListEquipments">Комплектации</a></li>
<li><a class="nav-link " asp-area="" asp-controller="Home" asp-action="ListInspections">Осмотры</a></li>
</ul>
</li>
<li class="nav-item dropdown">
<a class="nav-link dropdown-toggle" href="#" id="navbarDropdownMenuLink" role="button" data-bs-toggle="dropdown" aria-expanded="false">
Отчеты
</a>
<ul class="dropdown-menu" aria-labelledby="navbarDropdownMenuLink">
<li><a class="nav-link " asp-area="" asp-controller="Home" asp-action="PreSaleWorkCarReport">Отчет (word/excel) </a></li>
<li><a class="nav-link " asp-area="" asp-controller="Home" asp-action="Report">Отчет (pdf) </a></li>
</ul>
</li>
<a class="nav-link " asp-area="" asp-controller="Home" asp-action="AddEquipmentToPreSaleWork">Привязка комплектаций</a>
<a class="nav-link " asp-area="" asp-controller="Home" asp-action="Privacy">Администратор</a>
<a class="nav-link " asp-area="" asp-controller="Home" asp-action="Register">Регистрация</a>
<a class="nav-link " asp-area="" asp-controller="Home" asp-action="Enter">Вход</a>
</div>
</div>
</div>
</nav>
</header>
<div class="container">
<main role="main" class="pb-3">
@RenderBody()
</main>
</div>
<footer class="border-top footer text-muted">
<div class="container">
&copy; 2024 - CarCenter - <a asp-area="" asp-controller="Home" asp-action="Privacy">Privacy</a>
</div>
</footer>
<script src="~/lib/jquery/dist/jquery.min.js"></script>
<script src="~/lib/bootstrap/dist/js/bootstrap.bundle.min.js"></script>
<script src="~/js/site.js" asp-append-version="true"></script>
@await RenderSectionAsync("Scripts", required: false)
</body>
</html>

View File

@ -0,0 +1,48 @@
/* Please see documentation at https://docs.microsoft.com/aspnet/core/client-side/bundling-and-minification
for details on configuring this project to bundle and minify static web assets. */
a.navbar-brand {
white-space: normal;
text-align: center;
word-break: break-all;
}
a {
color: #0077cc;
}
.btn-primary {
color: #fff;
background-color: #1b6ec2;
border-color: #1861ac;
}
.nav-pills .nav-link.active, .nav-pills .show > .nav-link {
color: #fff;
background-color: #1b6ec2;
border-color: #1861ac;
}
.border-top {
border-top: 1px solid #e5e5e5;
}
.border-bottom {
border-bottom: 1px solid #e5e5e5;
}
.box-shadow {
box-shadow: 0 .25rem .75rem rgba(0, 0, 0, .05);
}
button.accept-policy {
font-size: 1rem;
line-height: inherit;
}
.footer {
position: absolute;
bottom: 0;
width: 100%;
white-space: nowrap;
line-height: 60px;
}

View File

@ -0,0 +1,2 @@
<script src="~/lib/jquery-validation/dist/jquery.validate.min.js"></script>
<script src="~/lib/jquery-validation-unobtrusive/jquery.validate.unobtrusive.min.js"></script>

View File

@ -0,0 +1,3 @@
@using CarCenterAdministratorApp
@using CarCenterAdministratorApp.Models
@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers

View File

@ -0,0 +1,3 @@
@{
Layout = "_Layout";
}

View File

@ -0,0 +1,8 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
}
}

View File

@ -0,0 +1,10 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
},
"AllowedHosts": "*",
"IPAddress": "http://localhost:5148/"
}

View File

@ -0,0 +1,18 @@
html {
font-size: 14px;
}
@media (min-width: 768px) {
html {
font-size: 16px;
}
}
html {
position: relative;
min-height: 100%;
}
body {
margin-bottom: 60px;
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.3 KiB

View File

@ -0,0 +1,4 @@
// Please see documentation at https://docs.microsoft.com/aspnet/core/client-side/bundling-and-minification
// for details on configuring this project to bundle and minify static web assets.
// Write your JavaScript code.

View File

@ -0,0 +1,22 @@
The MIT License (MIT)
Copyright (c) 2011-2021 Twitter, Inc.
Copyright (c) 2011-2021 The Bootstrap Authors
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,427 @@
/*!
* Bootstrap Reboot v5.1.0 (https://getbootstrap.com/)
* Copyright 2011-2021 The Bootstrap Authors
* Copyright 2011-2021 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
* Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md)
*/
*,
*::before,
*::after {
box-sizing: border-box;
}
@media (prefers-reduced-motion: no-preference) {
:root {
scroll-behavior: smooth;
}
}
body {
margin: 0;
font-family: var(--bs-body-font-family);
font-size: var(--bs-body-font-size);
font-weight: var(--bs-body-font-weight);
line-height: var(--bs-body-line-height);
color: var(--bs-body-color);
text-align: var(--bs-body-text-align);
background-color: var(--bs-body-bg);
-webkit-text-size-adjust: 100%;
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
hr {
margin: 1rem 0;
color: inherit;
background-color: currentColor;
border: 0;
opacity: 0.25;
}
hr:not([size]) {
height: 1px;
}
h6, h5, h4, h3, h2, h1 {
margin-top: 0;
margin-bottom: 0.5rem;
font-weight: 500;
line-height: 1.2;
}
h1 {
font-size: calc(1.375rem + 1.5vw);
}
@media (min-width: 1200px) {
h1 {
font-size: 2.5rem;
}
}
h2 {
font-size: calc(1.325rem + 0.9vw);
}
@media (min-width: 1200px) {
h2 {
font-size: 2rem;
}
}
h3 {
font-size: calc(1.3rem + 0.6vw);
}
@media (min-width: 1200px) {
h3 {
font-size: 1.75rem;
}
}
h4 {
font-size: calc(1.275rem + 0.3vw);
}
@media (min-width: 1200px) {
h4 {
font-size: 1.5rem;
}
}
h5 {
font-size: 1.25rem;
}
h6 {
font-size: 1rem;
}
p {
margin-top: 0;
margin-bottom: 1rem;
}
abbr[title],
abbr[data-bs-original-title] {
-webkit-text-decoration: underline dotted;
text-decoration: underline dotted;
cursor: help;
-webkit-text-decoration-skip-ink: none;
text-decoration-skip-ink: none;
}
address {
margin-bottom: 1rem;
font-style: normal;
line-height: inherit;
}
ol,
ul {
padding-left: 2rem;
}
ol,
ul,
dl {
margin-top: 0;
margin-bottom: 1rem;
}
ol ol,
ul ul,
ol ul,
ul ol {
margin-bottom: 0;
}
dt {
font-weight: 700;
}
dd {
margin-bottom: 0.5rem;
margin-left: 0;
}
blockquote {
margin: 0 0 1rem;
}
b,
strong {
font-weight: bolder;
}
small {
font-size: 0.875em;
}
mark {
padding: 0.2em;
background-color: #fcf8e3;
}
sub,
sup {
position: relative;
font-size: 0.75em;
line-height: 0;
vertical-align: baseline;
}
sub {
bottom: -0.25em;
}
sup {
top: -0.5em;
}
a {
color: #0d6efd;
text-decoration: underline;
}
a:hover {
color: #0a58ca;
}
a:not([href]):not([class]), a:not([href]):not([class]):hover {
color: inherit;
text-decoration: none;
}
pre,
code,
kbd,
samp {
font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
font-size: 1em;
direction: ltr /* rtl:ignore */;
unicode-bidi: bidi-override;
}
pre {
display: block;
margin-top: 0;
margin-bottom: 1rem;
overflow: auto;
font-size: 0.875em;
}
pre code {
font-size: inherit;
color: inherit;
word-break: normal;
}
code {
font-size: 0.875em;
color: #d63384;
word-wrap: break-word;
}
a > code {
color: inherit;
}
kbd {
padding: 0.2rem 0.4rem;
font-size: 0.875em;
color: #fff;
background-color: #212529;
border-radius: 0.2rem;
}
kbd kbd {
padding: 0;
font-size: 1em;
font-weight: 700;
}
figure {
margin: 0 0 1rem;
}
img,
svg {
vertical-align: middle;
}
table {
caption-side: bottom;
border-collapse: collapse;
}
caption {
padding-top: 0.5rem;
padding-bottom: 0.5rem;
color: #6c757d;
text-align: left;
}
th {
text-align: inherit;
text-align: -webkit-match-parent;
}
thead,
tbody,
tfoot,
tr,
td,
th {
border-color: inherit;
border-style: solid;
border-width: 0;
}
label {
display: inline-block;
}
button {
border-radius: 0;
}
button:focus:not(:focus-visible) {
outline: 0;
}
input,
button,
select,
optgroup,
textarea {
margin: 0;
font-family: inherit;
font-size: inherit;
line-height: inherit;
}
button,
select {
text-transform: none;
}
[role=button] {
cursor: pointer;
}
select {
word-wrap: normal;
}
select:disabled {
opacity: 1;
}
[list]::-webkit-calendar-picker-indicator {
display: none;
}
button,
[type=button],
[type=reset],
[type=submit] {
-webkit-appearance: button;
}
button:not(:disabled),
[type=button]:not(:disabled),
[type=reset]:not(:disabled),
[type=submit]:not(:disabled) {
cursor: pointer;
}
::-moz-focus-inner {
padding: 0;
border-style: none;
}
textarea {
resize: vertical;
}
fieldset {
min-width: 0;
padding: 0;
margin: 0;
border: 0;
}
legend {
float: left;
width: 100%;
padding: 0;
margin-bottom: 0.5rem;
font-size: calc(1.275rem + 0.3vw);
line-height: inherit;
}
@media (min-width: 1200px) {
legend {
font-size: 1.5rem;
}
}
legend + * {
clear: left;
}
::-webkit-datetime-edit-fields-wrapper,
::-webkit-datetime-edit-text,
::-webkit-datetime-edit-minute,
::-webkit-datetime-edit-hour-field,
::-webkit-datetime-edit-day-field,
::-webkit-datetime-edit-month-field,
::-webkit-datetime-edit-year-field {
padding: 0;
}
::-webkit-inner-spin-button {
height: auto;
}
[type=search] {
outline-offset: -2px;
-webkit-appearance: textfield;
}
/* rtl:raw:
[type="tel"],
[type="url"],
[type="email"],
[type="number"] {
direction: ltr;
}
*/
::-webkit-search-decoration {
-webkit-appearance: none;
}
::-webkit-color-swatch-wrapper {
padding: 0;
}
::file-selector-button {
font: inherit;
}
::-webkit-file-upload-button {
font: inherit;
-webkit-appearance: button;
}
output {
display: inline-block;
}
iframe {
border: 0;
}
summary {
display: list-item;
cursor: pointer;
}
progress {
vertical-align: baseline;
}
[hidden] {
display: none !important;
}
/*# sourceMappingURL=bootstrap-reboot.css.map */

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,8 @@
/*!
* Bootstrap Reboot v5.1.0 (https://getbootstrap.com/)
* Copyright 2011-2021 The Bootstrap Authors
* Copyright 2011-2021 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
* Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md)
*/*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}hr{margin:1rem 0;color:inherit;background-color:currentColor;border:0;opacity:.25}hr:not([size]){height:1px}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2}h1{font-size:calc(1.375rem + 1.5vw)}@media (min-width:1200px){h1{font-size:2.5rem}}h2{font-size:calc(1.325rem + .9vw)}@media (min-width:1200px){h2{font-size:2rem}}h3{font-size:calc(1.3rem + .6vw)}@media (min-width:1200px){h3{font-size:1.75rem}}h4{font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){h4{font-size:1.5rem}}h5{font-size:1.25rem}h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[data-bs-original-title],abbr[title]{-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-left:2rem}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:.875em}mark{padding:.2em;background-color:#fcf8e3}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#0d6efd;text-decoration:underline}a:hover{color:#0a58ca}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em;direction:ltr;unicode-bidi:bidi-override}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:.875em}pre code{font-size:inherit;color:inherit;word-break:normal}code{font-size:.875em;color:#d63384;word-wrap:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:.875em;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:1em;font-weight:700}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.5rem;padding-bottom:.5rem;color:#6c757d;text-align:left}th{text-align:inherit;text-align:-webkit-match-parent}tbody,td,tfoot,th,thead,tr{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]::-webkit-calendar-picker-indicator{display:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:left;width:100%;padding:0;margin-bottom:.5rem;font-size:calc(1.275rem + .3vw);line-height:inherit}@media (min-width:1200px){legend{font-size:1.5rem}}legend+*{clear:left}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-text,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:textfield}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::file-selector-button{font:inherit}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none!important}
/*# sourceMappingURL=bootstrap-reboot.min.css.map */

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,424 @@
/*!
* Bootstrap Reboot v5.1.0 (https://getbootstrap.com/)
* Copyright 2011-2021 The Bootstrap Authors
* Copyright 2011-2021 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
* Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md)
*/
*,
*::before,
*::after {
box-sizing: border-box;
}
@media (prefers-reduced-motion: no-preference) {
:root {
scroll-behavior: smooth;
}
}
body {
margin: 0;
font-family: var(--bs-body-font-family);
font-size: var(--bs-body-font-size);
font-weight: var(--bs-body-font-weight);
line-height: var(--bs-body-line-height);
color: var(--bs-body-color);
text-align: var(--bs-body-text-align);
background-color: var(--bs-body-bg);
-webkit-text-size-adjust: 100%;
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
hr {
margin: 1rem 0;
color: inherit;
background-color: currentColor;
border: 0;
opacity: 0.25;
}
hr:not([size]) {
height: 1px;
}
h6, h5, h4, h3, h2, h1 {
margin-top: 0;
margin-bottom: 0.5rem;
font-weight: 500;
line-height: 1.2;
}
h1 {
font-size: calc(1.375rem + 1.5vw);
}
@media (min-width: 1200px) {
h1 {
font-size: 2.5rem;
}
}
h2 {
font-size: calc(1.325rem + 0.9vw);
}
@media (min-width: 1200px) {
h2 {
font-size: 2rem;
}
}
h3 {
font-size: calc(1.3rem + 0.6vw);
}
@media (min-width: 1200px) {
h3 {
font-size: 1.75rem;
}
}
h4 {
font-size: calc(1.275rem + 0.3vw);
}
@media (min-width: 1200px) {
h4 {
font-size: 1.5rem;
}
}
h5 {
font-size: 1.25rem;
}
h6 {
font-size: 1rem;
}
p {
margin-top: 0;
margin-bottom: 1rem;
}
abbr[title],
abbr[data-bs-original-title] {
-webkit-text-decoration: underline dotted;
text-decoration: underline dotted;
cursor: help;
-webkit-text-decoration-skip-ink: none;
text-decoration-skip-ink: none;
}
address {
margin-bottom: 1rem;
font-style: normal;
line-height: inherit;
}
ol,
ul {
padding-right: 2rem;
}
ol,
ul,
dl {
margin-top: 0;
margin-bottom: 1rem;
}
ol ol,
ul ul,
ol ul,
ul ol {
margin-bottom: 0;
}
dt {
font-weight: 700;
}
dd {
margin-bottom: 0.5rem;
margin-right: 0;
}
blockquote {
margin: 0 0 1rem;
}
b,
strong {
font-weight: bolder;
}
small {
font-size: 0.875em;
}
mark {
padding: 0.2em;
background-color: #fcf8e3;
}
sub,
sup {
position: relative;
font-size: 0.75em;
line-height: 0;
vertical-align: baseline;
}
sub {
bottom: -0.25em;
}
sup {
top: -0.5em;
}
a {
color: #0d6efd;
text-decoration: underline;
}
a:hover {
color: #0a58ca;
}
a:not([href]):not([class]), a:not([href]):not([class]):hover {
color: inherit;
text-decoration: none;
}
pre,
code,
kbd,
samp {
font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
font-size: 1em;
direction: ltr ;
unicode-bidi: bidi-override;
}
pre {
display: block;
margin-top: 0;
margin-bottom: 1rem;
overflow: auto;
font-size: 0.875em;
}
pre code {
font-size: inherit;
color: inherit;
word-break: normal;
}
code {
font-size: 0.875em;
color: #d63384;
word-wrap: break-word;
}
a > code {
color: inherit;
}
kbd {
padding: 0.2rem 0.4rem;
font-size: 0.875em;
color: #fff;
background-color: #212529;
border-radius: 0.2rem;
}
kbd kbd {
padding: 0;
font-size: 1em;
font-weight: 700;
}
figure {
margin: 0 0 1rem;
}
img,
svg {
vertical-align: middle;
}
table {
caption-side: bottom;
border-collapse: collapse;
}
caption {
padding-top: 0.5rem;
padding-bottom: 0.5rem;
color: #6c757d;
text-align: right;
}
th {
text-align: inherit;
text-align: -webkit-match-parent;
}
thead,
tbody,
tfoot,
tr,
td,
th {
border-color: inherit;
border-style: solid;
border-width: 0;
}
label {
display: inline-block;
}
button {
border-radius: 0;
}
button:focus:not(:focus-visible) {
outline: 0;
}
input,
button,
select,
optgroup,
textarea {
margin: 0;
font-family: inherit;
font-size: inherit;
line-height: inherit;
}
button,
select {
text-transform: none;
}
[role=button] {
cursor: pointer;
}
select {
word-wrap: normal;
}
select:disabled {
opacity: 1;
}
[list]::-webkit-calendar-picker-indicator {
display: none;
}
button,
[type=button],
[type=reset],
[type=submit] {
-webkit-appearance: button;
}
button:not(:disabled),
[type=button]:not(:disabled),
[type=reset]:not(:disabled),
[type=submit]:not(:disabled) {
cursor: pointer;
}
::-moz-focus-inner {
padding: 0;
border-style: none;
}
textarea {
resize: vertical;
}
fieldset {
min-width: 0;
padding: 0;
margin: 0;
border: 0;
}
legend {
float: right;
width: 100%;
padding: 0;
margin-bottom: 0.5rem;
font-size: calc(1.275rem + 0.3vw);
line-height: inherit;
}
@media (min-width: 1200px) {
legend {
font-size: 1.5rem;
}
}
legend + * {
clear: right;
}
::-webkit-datetime-edit-fields-wrapper,
::-webkit-datetime-edit-text,
::-webkit-datetime-edit-minute,
::-webkit-datetime-edit-hour-field,
::-webkit-datetime-edit-day-field,
::-webkit-datetime-edit-month-field,
::-webkit-datetime-edit-year-field {
padding: 0;
}
::-webkit-inner-spin-button {
height: auto;
}
[type=search] {
outline-offset: -2px;
-webkit-appearance: textfield;
}
[type="tel"],
[type="url"],
[type="email"],
[type="number"] {
direction: ltr;
}
::-webkit-search-decoration {
-webkit-appearance: none;
}
::-webkit-color-swatch-wrapper {
padding: 0;
}
::file-selector-button {
font: inherit;
}
::-webkit-file-upload-button {
font: inherit;
-webkit-appearance: button;
}
output {
display: inline-block;
}
iframe {
border: 0;
}
summary {
display: list-item;
cursor: pointer;
}
progress {
vertical-align: baseline;
}
[hidden] {
display: none !important;
}
/*# sourceMappingURL=bootstrap-reboot.rtl.css.map */

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,8 @@
/*!
* Bootstrap Reboot v5.1.0 (https://getbootstrap.com/)
* Copyright 2011-2021 The Bootstrap Authors
* Copyright 2011-2021 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
* Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md)
*/*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}hr{margin:1rem 0;color:inherit;background-color:currentColor;border:0;opacity:.25}hr:not([size]){height:1px}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2}h1{font-size:calc(1.375rem + 1.5vw)}@media (min-width:1200px){h1{font-size:2.5rem}}h2{font-size:calc(1.325rem + .9vw)}@media (min-width:1200px){h2{font-size:2rem}}h3{font-size:calc(1.3rem + .6vw)}@media (min-width:1200px){h3{font-size:1.75rem}}h4{font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){h4{font-size:1.5rem}}h5{font-size:1.25rem}h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[data-bs-original-title],abbr[title]{-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-right:2rem}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-right:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:.875em}mark{padding:.2em;background-color:#fcf8e3}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#0d6efd;text-decoration:underline}a:hover{color:#0a58ca}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em;direction:ltr;unicode-bidi:bidi-override}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:.875em}pre code{font-size:inherit;color:inherit;word-break:normal}code{font-size:.875em;color:#d63384;word-wrap:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:.875em;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:1em;font-weight:700}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.5rem;padding-bottom:.5rem;color:#6c757d;text-align:right}th{text-align:inherit;text-align:-webkit-match-parent}tbody,td,tfoot,th,thead,tr{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]::-webkit-calendar-picker-indicator{display:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:right;width:100%;padding:0;margin-bottom:.5rem;font-size:calc(1.275rem + .3vw);line-height:inherit}@media (min-width:1200px){legend{font-size:1.5rem}}legend+*{clear:right}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-text,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:textfield}[type=email],[type=number],[type=tel],[type=url]{direction:ltr}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::file-selector-button{font:inherit}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none!important}
/*# sourceMappingURL=bootstrap-reboot.rtl.min.css.map */

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,12 @@
Copyright (c) .NET Foundation. All rights reserved.
Licensed under the Apache License, Version 2.0 (the "License"); you may not use
these files except in compliance with the License. You may obtain a copy of the
License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed
under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
CONDITIONS OF ANY KIND, either express or implied. See the License for the
specific language governing permissions and limitations under the License.

View File

@ -0,0 +1,432 @@
// Unobtrusive validation support library for jQuery and jQuery Validate
// Copyright (c) .NET Foundation. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
// @version v3.2.11
/*jslint white: true, browser: true, onevar: true, undef: true, nomen: true, eqeqeq: true, plusplus: true, bitwise: true, regexp: true, newcap: true, immed: true, strict: false */
/*global document: false, jQuery: false */
(function (factory) {
if (typeof define === 'function' && define.amd) {
// AMD. Register as an anonymous module.
define("jquery.validate.unobtrusive", ['jquery-validation'], factory);
} else if (typeof module === 'object' && module.exports) {
// CommonJS-like environments that support module.exports
module.exports = factory(require('jquery-validation'));
} else {
// Browser global
jQuery.validator.unobtrusive = factory(jQuery);
}
}(function ($) {
var $jQval = $.validator,
adapters,
data_validation = "unobtrusiveValidation";
function setValidationValues(options, ruleName, value) {
options.rules[ruleName] = value;
if (options.message) {
options.messages[ruleName] = options.message;
}
}
function splitAndTrim(value) {
return value.replace(/^\s+|\s+$/g, "").split(/\s*,\s*/g);
}
function escapeAttributeValue(value) {
// As mentioned on http://api.jquery.com/category/selectors/
return value.replace(/([!"#$%&'()*+,./:;<=>?@\[\\\]^`{|}~])/g, "\\$1");
}
function getModelPrefix(fieldName) {
return fieldName.substr(0, fieldName.lastIndexOf(".") + 1);
}
function appendModelPrefix(value, prefix) {
if (value.indexOf("*.") === 0) {
value = value.replace("*.", prefix);
}
return value;
}
function onError(error, inputElement) { // 'this' is the form element
var container = $(this).find("[data-valmsg-for='" + escapeAttributeValue(inputElement[0].name) + "']"),
replaceAttrValue = container.attr("data-valmsg-replace"),
replace = replaceAttrValue ? $.parseJSON(replaceAttrValue) !== false : null;
container.removeClass("field-validation-valid").addClass("field-validation-error");
error.data("unobtrusiveContainer", container);
if (replace) {
container.empty();
error.removeClass("input-validation-error").appendTo(container);
}
else {
error.hide();
}
}
function onErrors(event, validator) { // 'this' is the form element
var container = $(this).find("[data-valmsg-summary=true]"),
list = container.find("ul");
if (list && list.length && validator.errorList.length) {
list.empty();
container.addClass("validation-summary-errors").removeClass("validation-summary-valid");
$.each(validator.errorList, function () {
$("<li />").html(this.message).appendTo(list);
});
}
}
function onSuccess(error) { // 'this' is the form element
var container = error.data("unobtrusiveContainer");
if (container) {
var replaceAttrValue = container.attr("data-valmsg-replace"),
replace = replaceAttrValue ? $.parseJSON(replaceAttrValue) : null;
container.addClass("field-validation-valid").removeClass("field-validation-error");
error.removeData("unobtrusiveContainer");
if (replace) {
container.empty();
}
}
}
function onReset(event) { // 'this' is the form element
var $form = $(this),
key = '__jquery_unobtrusive_validation_form_reset';
if ($form.data(key)) {
return;
}
// Set a flag that indicates we're currently resetting the form.
$form.data(key, true);
try {
$form.data("validator").resetForm();
} finally {
$form.removeData(key);
}
$form.find(".validation-summary-errors")
.addClass("validation-summary-valid")
.removeClass("validation-summary-errors");
$form.find(".field-validation-error")
.addClass("field-validation-valid")
.removeClass("field-validation-error")
.removeData("unobtrusiveContainer")
.find(">*") // If we were using valmsg-replace, get the underlying error
.removeData("unobtrusiveContainer");
}
function validationInfo(form) {
var $form = $(form),
result = $form.data(data_validation),
onResetProxy = $.proxy(onReset, form),
defaultOptions = $jQval.unobtrusive.options || {},
execInContext = function (name, args) {
var func = defaultOptions[name];
func && $.isFunction(func) && func.apply(form, args);
};
if (!result) {
result = {
options: { // options structure passed to jQuery Validate's validate() method
errorClass: defaultOptions.errorClass || "input-validation-error",
errorElement: defaultOptions.errorElement || "span",
errorPlacement: function () {
onError.apply(form, arguments);
execInContext("errorPlacement", arguments);
},
invalidHandler: function () {
onErrors.apply(form, arguments);
execInContext("invalidHandler", arguments);
},
messages: {},
rules: {},
success: function () {
onSuccess.apply(form, arguments);
execInContext("success", arguments);
}
},
attachValidation: function () {
$form
.off("reset." + data_validation, onResetProxy)
.on("reset." + data_validation, onResetProxy)
.validate(this.options);
},
validate: function () { // a validation function that is called by unobtrusive Ajax
$form.validate();
return $form.valid();
}
};
$form.data(data_validation, result);
}
return result;
}
$jQval.unobtrusive = {
adapters: [],
parseElement: function (element, skipAttach) {
/// <summary>
/// Parses a single HTML element for unobtrusive validation attributes.
/// </summary>
/// <param name="element" domElement="true">The HTML element to be parsed.</param>
/// <param name="skipAttach" type="Boolean">[Optional] true to skip attaching the
/// validation to the form. If parsing just this single element, you should specify true.
/// If parsing several elements, you should specify false, and manually attach the validation
/// to the form when you are finished. The default is false.</param>
var $element = $(element),
form = $element.parents("form")[0],
valInfo, rules, messages;
if (!form) { // Cannot do client-side validation without a form
return;
}
valInfo = validationInfo(form);
valInfo.options.rules[element.name] = rules = {};
valInfo.options.messages[element.name] = messages = {};
$.each(this.adapters, function () {
var prefix = "data-val-" + this.name,
message = $element.attr(prefix),
paramValues = {};
if (message !== undefined) { // Compare against undefined, because an empty message is legal (and falsy)
prefix += "-";
$.each(this.params, function () {
paramValues[this] = $element.attr(prefix + this);
});
this.adapt({
element: element,
form: form,
message: message,
params: paramValues,
rules: rules,
messages: messages
});
}
});
$.extend(rules, { "__dummy__": true });
if (!skipAttach) {
valInfo.attachValidation();
}
},
parse: function (selector) {
/// <summary>
/// Parses all the HTML elements in the specified selector. It looks for input elements decorated
/// with the [data-val=true] attribute value and enables validation according to the data-val-*
/// attribute values.
/// </summary>
/// <param name="selector" type="String">Any valid jQuery selector.</param>
// $forms includes all forms in selector's DOM hierarchy (parent, children and self) that have at least one
// element with data-val=true
var $selector = $(selector),
$forms = $selector.parents()
.addBack()
.filter("form")
.add($selector.find("form"))
.has("[data-val=true]");
$selector.find("[data-val=true]").each(function () {
$jQval.unobtrusive.parseElement(this, true);
});
$forms.each(function () {
var info = validationInfo(this);
if (info) {
info.attachValidation();
}
});
}
};
adapters = $jQval.unobtrusive.adapters;
adapters.add = function (adapterName, params, fn) {
/// <summary>Adds a new adapter to convert unobtrusive HTML into a jQuery Validate validation.</summary>
/// <param name="adapterName" type="String">The name of the adapter to be added. This matches the name used
/// in the data-val-nnnn HTML attribute (where nnnn is the adapter name).</param>
/// <param name="params" type="Array" optional="true">[Optional] An array of parameter names (strings) that will
/// be extracted from the data-val-nnnn-mmmm HTML attributes (where nnnn is the adapter name, and
/// mmmm is the parameter name).</param>
/// <param name="fn" type="Function">The function to call, which adapts the values from the HTML
/// attributes into jQuery Validate rules and/or messages.</param>
/// <returns type="jQuery.validator.unobtrusive.adapters" />
if (!fn) { // Called with no params, just a function
fn = params;
params = [];
}
this.push({ name: adapterName, params: params, adapt: fn });
return this;
};
adapters.addBool = function (adapterName, ruleName) {
/// <summary>Adds a new adapter to convert unobtrusive HTML into a jQuery Validate validation, where
/// the jQuery Validate validation rule has no parameter values.</summary>
/// <param name="adapterName" type="String">The name of the adapter to be added. This matches the name used
/// in the data-val-nnnn HTML attribute (where nnnn is the adapter name).</param>
/// <param name="ruleName" type="String" optional="true">[Optional] The name of the jQuery Validate rule. If not provided, the value
/// of adapterName will be used instead.</param>
/// <returns type="jQuery.validator.unobtrusive.adapters" />
return this.add(adapterName, function (options) {
setValidationValues(options, ruleName || adapterName, true);
});
};
adapters.addMinMax = function (adapterName, minRuleName, maxRuleName, minMaxRuleName, minAttribute, maxAttribute) {
/// <summary>Adds a new adapter to convert unobtrusive HTML into a jQuery Validate validation, where
/// the jQuery Validate validation has three potential rules (one for min-only, one for max-only, and
/// one for min-and-max). The HTML parameters are expected to be named -min and -max.</summary>
/// <param name="adapterName" type="String">The name of the adapter to be added. This matches the name used
/// in the data-val-nnnn HTML attribute (where nnnn is the adapter name).</param>
/// <param name="minRuleName" type="String">The name of the jQuery Validate rule to be used when you only
/// have a minimum value.</param>
/// <param name="maxRuleName" type="String">The name of the jQuery Validate rule to be used when you only
/// have a maximum value.</param>
/// <param name="minMaxRuleName" type="String">The name of the jQuery Validate rule to be used when you
/// have both a minimum and maximum value.</param>
/// <param name="minAttribute" type="String" optional="true">[Optional] The name of the HTML attribute that
/// contains the minimum value. The default is "min".</param>
/// <param name="maxAttribute" type="String" optional="true">[Optional] The name of the HTML attribute that
/// contains the maximum value. The default is "max".</param>
/// <returns type="jQuery.validator.unobtrusive.adapters" />
return this.add(adapterName, [minAttribute || "min", maxAttribute || "max"], function (options) {
var min = options.params.min,
max = options.params.max;
if (min && max) {
setValidationValues(options, minMaxRuleName, [min, max]);
}
else if (min) {
setValidationValues(options, minRuleName, min);
}
else if (max) {
setValidationValues(options, maxRuleName, max);
}
});
};
adapters.addSingleVal = function (adapterName, attribute, ruleName) {
/// <summary>Adds a new adapter to convert unobtrusive HTML into a jQuery Validate validation, where
/// the jQuery Validate validation rule has a single value.</summary>
/// <param name="adapterName" type="String">The name of the adapter to be added. This matches the name used
/// in the data-val-nnnn HTML attribute(where nnnn is the adapter name).</param>
/// <param name="attribute" type="String">[Optional] The name of the HTML attribute that contains the value.
/// The default is "val".</param>
/// <param name="ruleName" type="String" optional="true">[Optional] The name of the jQuery Validate rule. If not provided, the value
/// of adapterName will be used instead.</param>
/// <returns type="jQuery.validator.unobtrusive.adapters" />
return this.add(adapterName, [attribute || "val"], function (options) {
setValidationValues(options, ruleName || adapterName, options.params[attribute]);
});
};
$jQval.addMethod("__dummy__", function (value, element, params) {
return true;
});
$jQval.addMethod("regex", function (value, element, params) {
var match;
if (this.optional(element)) {
return true;
}
match = new RegExp(params).exec(value);
return (match && (match.index === 0) && (match[0].length === value.length));
});
$jQval.addMethod("nonalphamin", function (value, element, nonalphamin) {
var match;
if (nonalphamin) {
match = value.match(/\W/g);
match = match && match.length >= nonalphamin;
}
return match;
});
if ($jQval.methods.extension) {
adapters.addSingleVal("accept", "mimtype");
adapters.addSingleVal("extension", "extension");
} else {
// for backward compatibility, when the 'extension' validation method does not exist, such as with versions
// of JQuery Validation plugin prior to 1.10, we should use the 'accept' method for
// validating the extension, and ignore mime-type validations as they are not supported.
adapters.addSingleVal("extension", "extension", "accept");
}
adapters.addSingleVal("regex", "pattern");
adapters.addBool("creditcard").addBool("date").addBool("digits").addBool("email").addBool("number").addBool("url");
adapters.addMinMax("length", "minlength", "maxlength", "rangelength").addMinMax("range", "min", "max", "range");
adapters.addMinMax("minlength", "minlength").addMinMax("maxlength", "minlength", "maxlength");
adapters.add("equalto", ["other"], function (options) {
var prefix = getModelPrefix(options.element.name),
other = options.params.other,
fullOtherName = appendModelPrefix(other, prefix),
element = $(options.form).find(":input").filter("[name='" + escapeAttributeValue(fullOtherName) + "']")[0];
setValidationValues(options, "equalTo", element);
});
adapters.add("required", function (options) {
// jQuery Validate equates "required" with "mandatory" for checkbox elements
if (options.element.tagName.toUpperCase() !== "INPUT" || options.element.type.toUpperCase() !== "CHECKBOX") {
setValidationValues(options, "required", true);
}
});
adapters.add("remote", ["url", "type", "additionalfields"], function (options) {
var value = {
url: options.params.url,
type: options.params.type || "GET",
data: {}
},
prefix = getModelPrefix(options.element.name);
$.each(splitAndTrim(options.params.additionalfields || options.element.name), function (i, fieldName) {
var paramName = appendModelPrefix(fieldName, prefix);
value.data[paramName] = function () {
var field = $(options.form).find(":input").filter("[name='" + escapeAttributeValue(paramName) + "']");
// For checkboxes and radio buttons, only pick up values from checked fields.
if (field.is(":checkbox")) {
return field.filter(":checked").val() || field.filter(":hidden").val() || '';
}
else if (field.is(":radio")) {
return field.filter(":checked").val() || '';
}
return field.val();
};
});
setValidationValues(options, "remote", value);
});
adapters.add("password", ["min", "nonalphamin", "regex"], function (options) {
if (options.params.min) {
setValidationValues(options, "minlength", options.params.min);
}
if (options.params.nonalphamin) {
setValidationValues(options, "nonalphamin", options.params.nonalphamin);
}
if (options.params.regex) {
setValidationValues(options, "regex", options.params.regex);
}
});
adapters.add("fileextensions", ["extensions"], function (options) {
setValidationValues(options, "extension", options.params.extensions);
});
$(function () {
$jQval.unobtrusive.parse(document);
});
return $jQval.unobtrusive;
}));

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,22 @@
The MIT License (MIT)
=====================
Copyright Jörn Zaefferer
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,36 @@
Copyright JS Foundation and other contributors, https://js.foundation/
This software consists of voluntary contributions made by many
individuals. For exact contribution history, see the revision history
available at https://github.com/jquery/jquery
The following license applies to all parts of this software except as
documented below:
====
Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:
The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
====
All files located in the node_modules and external directories are
externally maintained libraries used by this software which have their
own licenses; we recommend you read them, as their terms may differ from
the terms above.

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,178 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using CarCenterContracts.BusinessLogicsContracts;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class AdministratorLogic : IAdministratorLogic
{
private readonly int _loginMaxLength = 50;
private readonly int _passwordMaxLength = 50;
private readonly int _passwordMinLength = 5;
private readonly ILogger _logger;
private readonly IAdministratorStorage _administratorStorage;
public AdministratorLogic(ILogger<AdministratorLogic> logger, IAdministratorStorage administratorStorage)
{
_logger = logger;
_administratorStorage = administratorStorage;
}
public bool Create(AdministratorBindingModel model)
{
CheckModel(model);
if (_administratorStorage.Insert(model) == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(AdministratorBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id: {Id}", model.Id);
if (_administratorStorage.Delete(model) == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public AdministratorViewModel? ReadElement(AdministratorSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. AdministratorFIO: {AdministratorFIO}. AdministratorLogin: {AdministratorLogin}. Id: {Id}.", model.AdministratorFIO, model.AdministratorLogin, model.Id);
var element = _administratorStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id: {Id}", element.Id);
return element;
}
public List<AdministratorViewModel>? ReadList(AdministratorSearchModel? model)
{
_logger.LogInformation("ReadList. AdministratorFIO: {AdministratorFIO}. AdministratorLogin: {AdministratorLogin}. Id: {Id}.", model?.AdministratorFIO, model?.AdministratorLogin, model?.Id);
var list = model == null ? _administratorStorage.GetFullList() : _administratorStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count: {Count}", list.Count);
return list;
}
public bool Update(AdministratorBindingModel model)
{
CheckModel(model);
if (_administratorStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
private void CheckModel(AdministratorBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.AdministratorFIO))
{
throw new ArgumentNullException("Нет ФИО администратора", nameof(model.AdministratorFIO));
}
if (string.IsNullOrEmpty(model.AdministratorLogin))
{
throw new ArgumentNullException("Нет логина администратора", nameof(model.AdministratorLogin));
}
if (model.AdministratorLogin.Length > _loginMaxLength)
{
throw new ArgumentNullException("Логин слишком длинный", nameof(model.AdministratorLogin));
}
if (string.IsNullOrEmpty(model.AdministratorNumber))
{
throw new ArgumentNullException("Нет номера телефона администратора", nameof(model.AdministratorNumber));
}
if (model.AdministratorEmail.Length > _loginMaxLength || !Regex.IsMatch(model.AdministratorEmail, @"([a-zA-Z0-9]+@[a-zA-Z0-9]+\.[a-zA-Z0-9]+)"))
{
throw new Exception($"В качестве логина должна быть указана почта и иметь длинну не более {_loginMaxLength} символов");
}
if (string.IsNullOrEmpty(model.AdministratorEmail))
{
throw new ArgumentNullException("Нет почты администратора", nameof(model.AdministratorEmail));
}
if (string.IsNullOrEmpty(model.AdministratorPassword))
{
throw new ArgumentNullException("Нет пароля администратора", nameof(model.AdministratorPassword));
}
if (model.AdministratorPassword.Length > _passwordMaxLength || model.AdministratorPassword.Length < _passwordMinLength
|| !Regex.IsMatch(model.AdministratorPassword, @"^((\w+\d+\W+)|(\w+\W+\d+)|(\d+\w+\W+)|(\d+\W+\w+)|(\W+\w+\d+)|(\W+\d+\w+))[\w\d\W]*$"))
{
throw new Exception($"Пароль длиной от {_passwordMinLength} до {_passwordMaxLength} должен состоять из цифр, букв и небуквенных символов");
}
_logger.LogInformation("Administrator. AdministratorFIO: {AdministratorFIO}. AdministratorLogin: {AdministratorLogin}. Id: {Id}", model.AdministratorFIO, model.AdministratorLogin, model.Id);
var element = _administratorStorage.GetElement(new AdministratorSearchModel
{
AdministratorEmail = model.AdministratorEmail
});
if (element != null && element.Id != model.Id)
{
throw new InvalidOperationException("администратор с таким логином уже есть");
}
}
}
}

View File

@ -0,0 +1,136 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class CarLogic : ICarLogic
{
private readonly ILogger _logger;
private readonly ICarStorage _carStorage;
private readonly IAdministratorLogic _administratorLogic;
public CarLogic(ILogger<CarLogic> logger, ICarStorage carStorage, IAdministratorLogic administratorLogic)
{
_logger = logger;
_carStorage = carStorage;
_administratorLogic = administratorLogic;
}
public bool Create(CarBindingModel model)
{
CheckModel(model);
var result = _carStorage.Insert(model);
if (result == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(CarBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id:{Id}", model.Id);
var result = _carStorage.Delete(model);
if (result == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public CarViewModel? ReadElement(CarSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. BrandCar:{BrandCar}.Id:{Id}", model.BrandCar, model.Id);
var element = _carStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id:{Id}", element.Id);
return element;
}
public List<CarViewModel>? ReadList(CarSearchModel? model)
{
_logger.LogInformation("ReadList. BrandCar:{BrandCar}.Id:{ Id}", model?.BrandCar, model?.Id);
var list = model == null ? _carStorage.GetFullList() : _carStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count:{Count}", list.Count);
return list;
}
public bool Update(CarBindingModel model)
{
CheckModel(model);
if (_carStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
private void CheckModel(CarBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.BrandCar))
{
throw new ArgumentNullException("Нет названия бренда", nameof(model.BrandCar));
}
if (string.IsNullOrEmpty(model.Model))
{
throw new ArgumentNullException("Нет названия модели", nameof(model.Model));
}
_logger.LogInformation("Car. BrandCar:{BrandCar}.Model:{ Model}. Id: { Id}", model.BrandCar, model.BrandCar, model.Id);
}
}
}

View File

@ -0,0 +1,162 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using CarCenterDataModels.Models;
using Microsoft.Extensions.Logging;
using System.IO.Packaging;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class EmployeeLogic : IEmployeeLogic
{
private readonly ILogger _logger;
private readonly IEmployeeStorage _employeeStorage;
private readonly IManagerLogic _managerLogic;
public EmployeeLogic(ILogger<EmployeeLogic> logger, IEmployeeStorage EmployeeStorage, IManagerLogic ManagerLogic)
{
_logger = logger;
_employeeStorage = EmployeeStorage;
_managerLogic = ManagerLogic;
}
public bool Create(EmployeeBindingModel model)
{
CheckModel(model);
model.EmployeeSales = new();
var result = _employeeStorage.Insert(model);
if (result == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(EmployeeBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id:{Id}", model.Id);
var result = _employeeStorage.Delete(model);
if (result == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public EmployeeViewModel? ReadElement(EmployeeSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. EmployeeFIO:{EmployeeFIO}.Id:{Id}", model.EmployeeFIO, model.Id);
var element = _employeeStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id:{Id}", element.Id);
return element;
}
public List<EmployeeViewModel>? ReadList(EmployeeSearchModel? model)
{
_logger.LogInformation("ReadList. EmployeeFIO:{EmployeeFIO}.Id:{ Id}", model?.EmployeeFIO, model?.Id);
var list = model == null ? _employeeStorage.GetFullList() : _employeeStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count:{Count}", list.Count);
return list;
}
public bool AddSaleToEmployee(EmployeeSearchModel model, ISaleModel Sale)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("AddSaleToEmployee. EmployeeFIO:{EmployeeFIO}.Id:{ Id}", model.EmployeeFIO, model.Id);
var element = _employeeStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("AddSaleToEmployee element not found");
return false;
}
_logger.LogInformation("AddSaleToEmployee find. Id:{Id}", element.Id);
element.EmployeeSales[Sale.Id] = Sale;
_employeeStorage.Update(new()
{
Id = element.Id,
EmployeeFIO = element.EmployeeFIO,
Specialization = element.Specialization,
ManagerId = element.ManagerId,
EmployeeSales = element.EmployeeSales,
});
return true;
}
public bool Update(EmployeeBindingModel model)
{
CheckModel(model);
if (_employeeStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
private void CheckModel(EmployeeBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.EmployeeFIO))
{
throw new ArgumentNullException("Нет ФИО сотрудника", nameof(model.EmployeeFIO));
}
_logger.LogInformation("Employee. EmployeeFIO:{EmployeeFIO}.StartDate:{ StartDate}. Id: { Id}", model.EmployeeFIO, model.Specialization, model.Id);
}
}
}

View File

@ -0,0 +1,169 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using CarCenterDataModels.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class EquipmentLogic : IEquipmentLogic
{
private readonly ILogger _logger;
private readonly IEquipmentStorage _equipmentStorage;
public EquipmentLogic(ILogger<EquipmentLogic> logger, IEquipmentStorage equipmentStorage)
{
_logger = logger;
_equipmentStorage = equipmentStorage;
}
public bool Create(EquipmentBindingModel model)
{
CheckModel(model);
model.EquipmentCars = new();
var result = _equipmentStorage.Insert(model);
if (result == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(EquipmentBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id:{Id}", model.Id);
var result = _equipmentStorage.Delete(model);
if (result == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public EquipmentViewModel? ReadElement(EquipmentSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. EquipmentName:{EquipmentName}.Id:{Id}", model.EquipmentName, model.Id);
var element = _equipmentStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id:{Id}", element.Id);
return element;
}
public List<EquipmentViewModel>? ReadList(EquipmentSearchModel? model)
{
_logger.LogInformation("ReadList. EquipmentName:{EquipmentName}.Id:{ Id}", model?.EquipmentName, model?.Id);
var list = model == null ? _equipmentStorage.GetFullList() : _equipmentStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count:{Count}", list.Count);
return list;
}
public bool Update(EquipmentBindingModel model)
{
CheckModel(model);
if (_equipmentStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
public bool AddCarToEquipment(EquipmentSearchModel model, ICarModel car)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("AddCarToEquipment. EquipmentName:{EquipmentName}.Id:{ Id}", model.EquipmentName, model.Id);
var element = _equipmentStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("AddCarToEquipment element not found");
return false;
}
_logger.LogInformation("AddCarToEquipment find. Id:{Id}", element.Id);
element.EquipmentCars[car.Id] = car;
_equipmentStorage.Update(new()
{
Id = element.Id,
EquipmentName = element.EquipmentName,
EquipmentPrice = element.EquipmentPrice,
AdministratorId = element.AdministratorId,
PreSaleWorkId = element.PreSaleWorkId,
EquipmentCars = element.EquipmentCars,
});
return true;
}
private void CheckModel(EquipmentBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.EquipmentName))
{
throw new ArgumentNullException("Нет названия комплектации", nameof(model.EquipmentName));
}
if (model.EquipmentPrice < 0)
{
throw new ArgumentNullException("Стоимость не может быть меньше 0", nameof(model.EquipmentPrice));
}
_logger.LogInformation("Equipment. EquipmentName:{EquipmentName}.EquipmentPrice:{ EquipmentPrice}. Id: { Id}", model.EquipmentName, model.EquipmentPrice, model.Id);
}
}
}

View File

@ -0,0 +1,166 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using CarCenterDataModels.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Microsoft.Extensions.Logging;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class InspectionLogic : IInspectionLogic
{
private readonly ILogger _logger;
private readonly IInspectionStorage _inspectionStorage;
public InspectionLogic(ILogger<InspectionLogic> logger, IInspectionStorage inspectionStorage)
{
_logger = logger;
_inspectionStorage = inspectionStorage;
}
public bool Create(InspectionBindingModel model)
{
CheckModel(model);
model.InspectionCars = new();
var result = _inspectionStorage.Insert(model);
if (result == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(InspectionBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id:{Id}", model.Id);
var result = _inspectionStorage.Delete(model);
if (result == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public InspectionViewModel? ReadElement(InspectionSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. Id:{Id}", model.Id);
var element = _inspectionStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id:{Id}", element.Id);
return element;
}
public List<InspectionViewModel>? ReadList(InspectionSearchModel? model)
{
_logger.LogInformation("ReadList. Id:{ Id}", model?.Id);
var list = model == null ? _inspectionStorage.GetFullList() : _inspectionStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count:{Count}", list.Count);
return list;
}
public bool Update(InspectionBindingModel model)
{
CheckModel(model);
if (_inspectionStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
public bool AddCarToInspection(InspectionSearchModel model, ICarModel car)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("AddCarToInspection. InspectionName:{InspectionName}.Id:{ Id}", model.InspectionName, model.Id);
var element = _inspectionStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("AddCarToInspection element not found");
return false;
}
_logger.LogInformation("AddCarToInspection find. Id:{Id}", element.Id);
element.InspectionCars[car.Id] = car;
_inspectionStorage.Update(new()
{
Id = element.Id,
InspectionName = element.InspectionName,
InspectionDate = element.InspectionDate,
AdministratorId = element.AdministratorId,
EmployeeId = element.EmployeeId,
InspectionCars = element.InspectionCars
});
return true;
}
private void CheckModel(InspectionBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.InspectionName))
{
throw new ArgumentNullException("Нет названия осмотра", nameof(model.InspectionName));
}
_logger.LogInformation("Inspection. Id: { Id}", model.Id);
}
}
}

View File

@ -0,0 +1,173 @@
using CarCenterContracts.BindingModels;
using CarCenterContracts.BusinessLogicsContracts;
using CarCenterContracts.SearchModels;
using CarCenterContracts.StoragesContracts;
using CarCenterContracts.ViewModels;
using Microsoft.Extensions.Logging;
using System.Text.RegularExpressions;
namespace CarCenterBusinessLogic.BusinessLogics
{
public class ManagerLogic : IManagerLogic
{
private readonly int _loginMaxLength = 50;
private readonly int _passwordMaxLength = 50;
private readonly int _passwordMinLength = 10;
private readonly ILogger _logger;
private readonly IManagerStorage _managerStorage;
public ManagerLogic(ILogger<ManagerLogic> logger, IManagerStorage ManagerStorage)
{
_logger = logger;
_managerStorage = ManagerStorage;
}
public bool Create(ManagerBindingModel model)
{
CheckModel(model);
if (_managerStorage.Insert(model) == null)
{
_logger.LogWarning("Insert operation failed");
return false;
}
return true;
}
public bool Delete(ManagerBindingModel model)
{
CheckModel(model, false);
_logger.LogInformation("Delete. Id: {Id}", model.Id);
if (_managerStorage.Delete(model) == null)
{
_logger.LogWarning("Delete operation failed");
return false;
}
return true;
}
public ManagerViewModel? ReadElement(ManagerSearchModel model)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
_logger.LogInformation("ReadElement. ManagerFIO: {ManagerFIO}. ManagerLogin: {ManagerLogin}. Id: {Id}.", model.ManagerFIO, model.ManagerLogin, model.Id);
var element = _managerStorage.GetElement(model);
if (element == null)
{
_logger.LogWarning("ReadElement element not found");
return null;
}
_logger.LogInformation("ReadElement find. Id: {Id}", element.Id);
return element;
}
public List<ManagerViewModel>? ReadList(ManagerSearchModel? model)
{
_logger.LogInformation("ReadList. ManagerFIO: {ManagerFIO}. ManagerLogin: {ManagerLogin}. Id: {Id}.", model?.ManagerFIO, model?.ManagerLogin, model?.Id);
var list = model == null ? _managerStorage.GetFullList() : _managerStorage.GetFilteredList(model);
if (list == null)
{
_logger.LogWarning("ReadList return null list");
return null;
}
_logger.LogInformation("ReadList. Count: {Count}", list.Count);
return list;
}
public bool Update(ManagerBindingModel model)
{
CheckModel(model);
if (_managerStorage.Update(model) == null)
{
_logger.LogWarning("Update operation failed");
return false;
}
return true;
}
private void CheckModel(ManagerBindingModel model, bool withParams = true)
{
if (model == null)
{
throw new ArgumentNullException(nameof(model));
}
if (!withParams)
{
return;
}
if (string.IsNullOrEmpty(model.ManagerFIO))
{
throw new ArgumentNullException("Нет ФИО менеджера", nameof(model.ManagerFIO));
}
if (string.IsNullOrEmpty(model.ManagerLogin))
{
throw new ArgumentNullException("Нет логина менеджера", nameof(model.ManagerLogin));
}
if (model.ManagerLogin.Length>_loginMaxLength)
{
throw new ArgumentNullException("Логин слишком длинный", nameof(model.ManagerLogin));
}
if (string.IsNullOrEmpty(model.ManagerNumber))
{
throw new ArgumentNullException("Нет номера телефона менеджера", nameof(model.ManagerNumber));
}
if (model.ManagerEmail.Length > _loginMaxLength || !Regex.IsMatch(model.ManagerEmail, @"([a-zA-Z0-9]+@[a-zA-Z0-9]+\.[a-zA-Z0-9]+)"))
{
throw new Exception($"В качестве логина должна быть указана почта и иметь длинну не более {_loginMaxLength} символов");
}
if (string.IsNullOrEmpty(model.ManagerEmail))
{
throw new ArgumentNullException("Нет почты менеджера", nameof(model.ManagerEmail));
}
if (string.IsNullOrEmpty(model.ManagerPassword))
{
throw new ArgumentNullException("Нет пароля менеджера", nameof(model.ManagerPassword));
}
if (model.ManagerPassword.Length > _passwordMaxLength || model.ManagerPassword.Length < _passwordMinLength
|| !Regex.IsMatch(model.ManagerPassword, @"^((\w+\d+\W+)|(\w+\W+\d+)|(\d+\w+\W+)|(\d+\W+\w+)|(\W+\w+\d+)|(\W+\d+\w+))[\w\d\W]*$"))
{
throw new Exception($"Пароль длиной от {_passwordMinLength} до {_passwordMaxLength} должен состоять из цифр, букв и небуквенных символов");
}
_logger.LogInformation("Manager. ManagerFIO: {ManagerFIO}. ManagerLogin: {ManagerLogin}. Id: {Id}", model.ManagerFIO, model.ManagerLogin, model.Id);
var element = _managerStorage.GetElement(new ManagerSearchModel
{
ManagerEmail = model.ManagerEmail
});
if (element != null && element.Id != model.Id)
{
throw new InvalidOperationException("менеджер с таким логином уже есть");
}
}
}
}

Some files were not shown because too many files have changed in this diff Show More