| @@ -0,0 +1,7 @@ | |||
| <Project Sdk="Microsoft.NET.Sdk"> | |||
| <PropertyGroup> | |||
| <TargetFramework>netcoreapp3.1</TargetFramework> | |||
| </PropertyGroup> | |||
| </Project> | |||
| @@ -0,0 +1,7 @@ | |||
| <Project Sdk="Microsoft.NET.Sdk"> | |||
| <PropertyGroup> | |||
| <TargetFramework>netcoreapp3.1</TargetFramework> | |||
| </PropertyGroup> | |||
| </Project> | |||
| @@ -0,0 +1,7 @@ | |||
| <Project Sdk="Microsoft.NET.Sdk"> | |||
| <PropertyGroup> | |||
| <TargetFramework>netcoreapp3.1</TargetFramework> | |||
| </PropertyGroup> | |||
| </Project> | |||
| @@ -0,0 +1,55 @@ | |||
| | |||
| Microsoft Visual Studio Solution File, Format Version 12.00 | |||
| # Visual Studio Version 16 | |||
| VisualStudioVersion = 16.0.29911.84 | |||
| MinimumVisualStudioVersion = 10.0.40219.1 | |||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Diligent.Web", "Diligent.Web\Diligent.Web.csproj", "{4F0BF1BF-C621-49B2-B1C4-F50B19F05628}" | |||
| ProjectSection(ProjectDependencies) = postProject | |||
| {01314B0F-7116-4778-893C-9DB553F3B64E} = {01314B0F-7116-4778-893C-9DB553F3B64E} | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910} = {A770B88D-68E0-42D3-BC78-6AD630CED910} | |||
| {926F50AE-BB12-4458-9758-478DEFE01467} = {926F50AE-BB12-4458-9758-478DEFE01467} | |||
| EndProjectSection | |||
| EndProject | |||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Diligent.Services", "Diligent.Services\Diligent.Services.csproj", "{01314B0F-7116-4778-893C-9DB553F3B64E}" | |||
| ProjectSection(ProjectDependencies) = postProject | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910} = {A770B88D-68E0-42D3-BC78-6AD630CED910} | |||
| {926F50AE-BB12-4458-9758-478DEFE01467} = {926F50AE-BB12-4458-9758-478DEFE01467} | |||
| EndProjectSection | |||
| EndProject | |||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Diligent.Infrastructure", "Diligent.Infrastructure\Diligent.Infrastructure.csproj", "{A770B88D-68E0-42D3-BC78-6AD630CED910}" | |||
| ProjectSection(ProjectDependencies) = postProject | |||
| {926F50AE-BB12-4458-9758-478DEFE01467} = {926F50AE-BB12-4458-9758-478DEFE01467} | |||
| EndProjectSection | |||
| EndProject | |||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Diligent.Domain", "Diligent.Domain\Diligent.Domain.csproj", "{926F50AE-BB12-4458-9758-478DEFE01467}" | |||
| EndProject | |||
| Global | |||
| GlobalSection(SolutionConfigurationPlatforms) = preSolution | |||
| Debug|Any CPU = Debug|Any CPU | |||
| Release|Any CPU = Release|Any CPU | |||
| EndGlobalSection | |||
| GlobalSection(ProjectConfigurationPlatforms) = postSolution | |||
| {4F0BF1BF-C621-49B2-B1C4-F50B19F05628}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | |||
| {4F0BF1BF-C621-49B2-B1C4-F50B19F05628}.Debug|Any CPU.Build.0 = Debug|Any CPU | |||
| {4F0BF1BF-C621-49B2-B1C4-F50B19F05628}.Release|Any CPU.ActiveCfg = Release|Any CPU | |||
| {4F0BF1BF-C621-49B2-B1C4-F50B19F05628}.Release|Any CPU.Build.0 = Release|Any CPU | |||
| {01314B0F-7116-4778-893C-9DB553F3B64E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | |||
| {01314B0F-7116-4778-893C-9DB553F3B64E}.Debug|Any CPU.Build.0 = Debug|Any CPU | |||
| {01314B0F-7116-4778-893C-9DB553F3B64E}.Release|Any CPU.ActiveCfg = Release|Any CPU | |||
| {01314B0F-7116-4778-893C-9DB553F3B64E}.Release|Any CPU.Build.0 = Release|Any CPU | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910}.Debug|Any CPU.Build.0 = Debug|Any CPU | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910}.Release|Any CPU.ActiveCfg = Release|Any CPU | |||
| {A770B88D-68E0-42D3-BC78-6AD630CED910}.Release|Any CPU.Build.0 = Release|Any CPU | |||
| {926F50AE-BB12-4458-9758-478DEFE01467}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | |||
| {926F50AE-BB12-4458-9758-478DEFE01467}.Debug|Any CPU.Build.0 = Debug|Any CPU | |||
| {926F50AE-BB12-4458-9758-478DEFE01467}.Release|Any CPU.ActiveCfg = Release|Any CPU | |||
| {926F50AE-BB12-4458-9758-478DEFE01467}.Release|Any CPU.Build.0 = Release|Any CPU | |||
| EndGlobalSection | |||
| GlobalSection(SolutionProperties) = preSolution | |||
| HideSolutionNode = FALSE | |||
| EndGlobalSection | |||
| GlobalSection(ExtensibilityGlobals) = postSolution | |||
| SolutionGuid = {61A17CD4-6CC3-424E-85CA-BA3F66612D48} | |||
| EndGlobalSection | |||
| EndGlobal | |||
| @@ -0,0 +1,3 @@ | |||
| @{ | |||
| Layout = "/Views/Shared/_Layout.cshtml"; | |||
| } | |||
| @@ -0,0 +1,37 @@ | |||
| using System; | |||
| using System.Collections.Generic; | |||
| using System.Diagnostics; | |||
| using System.Linq; | |||
| using System.Threading.Tasks; | |||
| using Microsoft.AspNetCore.Mvc; | |||
| using Microsoft.Extensions.Logging; | |||
| using Diligent.Web.Models; | |||
| namespace Diligent.Web.Controllers | |||
| { | |||
| public class HomeController : Controller | |||
| { | |||
| private readonly ILogger<HomeController> _logger; | |||
| public HomeController(ILogger<HomeController> logger) | |||
| { | |||
| _logger = logger; | |||
| } | |||
| public IActionResult Index() | |||
| { | |||
| return View(); | |||
| } | |||
| public IActionResult Privacy() | |||
| { | |||
| return View(); | |||
| } | |||
| [ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)] | |||
| public IActionResult Error() | |||
| { | |||
| return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier }); | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,16 @@ | |||
| using System; | |||
| using System.Collections.Generic; | |||
| using System.Text; | |||
| using Microsoft.AspNetCore.Identity.EntityFrameworkCore; | |||
| using Microsoft.EntityFrameworkCore; | |||
| namespace Diligent.Web.Data | |||
| { | |||
| public class ApplicationDbContext : IdentityDbContext | |||
| { | |||
| public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) | |||
| : base(options) | |||
| { | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,277 @@ | |||
| // <auto-generated /> | |||
| using System; | |||
| using Diligent.Web.Data; | |||
| using Microsoft.EntityFrameworkCore; | |||
| using Microsoft.EntityFrameworkCore.Infrastructure; | |||
| using Microsoft.EntityFrameworkCore.Metadata; | |||
| using Microsoft.EntityFrameworkCore.Migrations; | |||
| using Microsoft.EntityFrameworkCore.Storage.ValueConversion; | |||
| namespace Diligent.Web.Data.Migrations | |||
| { | |||
| [DbContext(typeof(ApplicationDbContext))] | |||
| [Migration("00000000000000_CreateIdentitySchema")] | |||
| partial class CreateIdentitySchema | |||
| { | |||
| protected override void BuildTargetModel(ModelBuilder modelBuilder) | |||
| { | |||
| #pragma warning disable 612, 618 | |||
| modelBuilder | |||
| .HasAnnotation("ProductVersion", "3.0.0") | |||
| .HasAnnotation("Relational:MaxIdentifierLength", 128) | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRole", b => | |||
| { | |||
| b.Property<string>("Id") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("ConcurrencyStamp") | |||
| .IsConcurrencyToken() | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("Name") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("NormalizedName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("NormalizedName") | |||
| .IsUnique() | |||
| .HasName("RoleNameIndex") | |||
| .HasFilter("[NormalizedName] IS NOT NULL"); | |||
| b.ToTable("AspNetRoles"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => | |||
| { | |||
| b.Property<int>("Id") | |||
| .ValueGeneratedOnAdd() | |||
| .HasColumnType("int") | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| b.Property<string>("ClaimType") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("ClaimValue") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("RoleId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("RoleId"); | |||
| b.ToTable("AspNetRoleClaims"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUser", b => | |||
| { | |||
| b.Property<string>("Id") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<int>("AccessFailedCount") | |||
| .HasColumnType("int"); | |||
| b.Property<string>("ConcurrencyStamp") | |||
| .IsConcurrencyToken() | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("Email") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<bool>("EmailConfirmed") | |||
| .HasColumnType("bit"); | |||
| b.Property<bool>("LockoutEnabled") | |||
| .HasColumnType("bit"); | |||
| b.Property<DateTimeOffset?>("LockoutEnd") | |||
| .HasColumnType("datetimeoffset"); | |||
| b.Property<string>("NormalizedEmail") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("NormalizedUserName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("PasswordHash") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("PhoneNumber") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<bool>("PhoneNumberConfirmed") | |||
| .HasColumnType("bit"); | |||
| b.Property<string>("SecurityStamp") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<bool>("TwoFactorEnabled") | |||
| .HasColumnType("bit"); | |||
| b.Property<string>("UserName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("NormalizedEmail") | |||
| .HasName("EmailIndex"); | |||
| b.HasIndex("NormalizedUserName") | |||
| .IsUnique() | |||
| .HasName("UserNameIndex") | |||
| .HasFilter("[NormalizedUserName] IS NOT NULL"); | |||
| b.ToTable("AspNetUsers"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b => | |||
| { | |||
| b.Property<int>("Id") | |||
| .ValueGeneratedOnAdd() | |||
| .HasColumnType("int") | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| b.Property<string>("ClaimType") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("ClaimValue") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("UserId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("UserId"); | |||
| b.ToTable("AspNetUserClaims"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b => | |||
| { | |||
| b.Property<string>("LoginProvider") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("ProviderKey") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("ProviderDisplayName") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("UserId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("LoginProvider", "ProviderKey"); | |||
| b.HasIndex("UserId"); | |||
| b.ToTable("AspNetUserLogins"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b => | |||
| { | |||
| b.Property<string>("UserId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("RoleId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("UserId", "RoleId"); | |||
| b.HasIndex("RoleId"); | |||
| b.ToTable("AspNetUserRoles"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b => | |||
| { | |||
| b.Property<string>("UserId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("LoginProvider") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("Name") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("Value") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.HasKey("UserId", "LoginProvider", "Name"); | |||
| b.ToTable("AspNetUserTokens"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null) | |||
| .WithMany() | |||
| .HasForeignKey("RoleId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null) | |||
| .WithMany() | |||
| .HasForeignKey("RoleId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| #pragma warning restore 612, 618 | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,220 @@ | |||
| using System; | |||
| using Microsoft.EntityFrameworkCore.Metadata; | |||
| using Microsoft.EntityFrameworkCore.Migrations; | |||
| namespace Diligent.Web.Data.Migrations | |||
| { | |||
| public partial class CreateIdentitySchema : Migration | |||
| { | |||
| protected override void Up(MigrationBuilder migrationBuilder) | |||
| { | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetRoles", | |||
| columns: table => new | |||
| { | |||
| Id = table.Column<string>(nullable: false), | |||
| Name = table.Column<string>(maxLength: 256, nullable: true), | |||
| NormalizedName = table.Column<string>(maxLength: 256, nullable: true), | |||
| ConcurrencyStamp = table.Column<string>(nullable: true) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetRoles", x => x.Id); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetUsers", | |||
| columns: table => new | |||
| { | |||
| Id = table.Column<string>(nullable: false), | |||
| UserName = table.Column<string>(maxLength: 256, nullable: true), | |||
| NormalizedUserName = table.Column<string>(maxLength: 256, nullable: true), | |||
| Email = table.Column<string>(maxLength: 256, nullable: true), | |||
| NormalizedEmail = table.Column<string>(maxLength: 256, nullable: true), | |||
| EmailConfirmed = table.Column<bool>(nullable: false), | |||
| PasswordHash = table.Column<string>(nullable: true), | |||
| SecurityStamp = table.Column<string>(nullable: true), | |||
| ConcurrencyStamp = table.Column<string>(nullable: true), | |||
| PhoneNumber = table.Column<string>(nullable: true), | |||
| PhoneNumberConfirmed = table.Column<bool>(nullable: false), | |||
| TwoFactorEnabled = table.Column<bool>(nullable: false), | |||
| LockoutEnd = table.Column<DateTimeOffset>(nullable: true), | |||
| LockoutEnabled = table.Column<bool>(nullable: false), | |||
| AccessFailedCount = table.Column<int>(nullable: false) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetUsers", x => x.Id); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetRoleClaims", | |||
| columns: table => new | |||
| { | |||
| Id = table.Column<int>(nullable: false) | |||
| .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn), | |||
| RoleId = table.Column<string>(nullable: false), | |||
| ClaimType = table.Column<string>(nullable: true), | |||
| ClaimValue = table.Column<string>(nullable: true) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetRoleClaims", x => x.Id); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetRoleClaims_AspNetRoles_RoleId", | |||
| column: x => x.RoleId, | |||
| principalTable: "AspNetRoles", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetUserClaims", | |||
| columns: table => new | |||
| { | |||
| Id = table.Column<int>(nullable: false) | |||
| .Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn), | |||
| UserId = table.Column<string>(nullable: false), | |||
| ClaimType = table.Column<string>(nullable: true), | |||
| ClaimValue = table.Column<string>(nullable: true) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetUserClaims", x => x.Id); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetUserClaims_AspNetUsers_UserId", | |||
| column: x => x.UserId, | |||
| principalTable: "AspNetUsers", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetUserLogins", | |||
| columns: table => new | |||
| { | |||
| LoginProvider = table.Column<string>(maxLength: 128, nullable: false), | |||
| ProviderKey = table.Column<string>(maxLength: 128, nullable: false), | |||
| ProviderDisplayName = table.Column<string>(nullable: true), | |||
| UserId = table.Column<string>(nullable: false) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetUserLogins", x => new { x.LoginProvider, x.ProviderKey }); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetUserLogins_AspNetUsers_UserId", | |||
| column: x => x.UserId, | |||
| principalTable: "AspNetUsers", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetUserRoles", | |||
| columns: table => new | |||
| { | |||
| UserId = table.Column<string>(nullable: false), | |||
| RoleId = table.Column<string>(nullable: false) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetUserRoles", x => new { x.UserId, x.RoleId }); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetUserRoles_AspNetRoles_RoleId", | |||
| column: x => x.RoleId, | |||
| principalTable: "AspNetRoles", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetUserRoles_AspNetUsers_UserId", | |||
| column: x => x.UserId, | |||
| principalTable: "AspNetUsers", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| }); | |||
| migrationBuilder.CreateTable( | |||
| name: "AspNetUserTokens", | |||
| columns: table => new | |||
| { | |||
| UserId = table.Column<string>(nullable: false), | |||
| LoginProvider = table.Column<string>(maxLength: 128, nullable: false), | |||
| Name = table.Column<string>(maxLength: 128, nullable: false), | |||
| Value = table.Column<string>(nullable: true) | |||
| }, | |||
| constraints: table => | |||
| { | |||
| table.PrimaryKey("PK_AspNetUserTokens", x => new { x.UserId, x.LoginProvider, x.Name }); | |||
| table.ForeignKey( | |||
| name: "FK_AspNetUserTokens_AspNetUsers_UserId", | |||
| column: x => x.UserId, | |||
| principalTable: "AspNetUsers", | |||
| principalColumn: "Id", | |||
| onDelete: ReferentialAction.Cascade); | |||
| }); | |||
| migrationBuilder.CreateIndex( | |||
| name: "IX_AspNetRoleClaims_RoleId", | |||
| table: "AspNetRoleClaims", | |||
| column: "RoleId"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "RoleNameIndex", | |||
| table: "AspNetRoles", | |||
| column: "NormalizedName", | |||
| unique: true, | |||
| filter: "[NormalizedName] IS NOT NULL"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "IX_AspNetUserClaims_UserId", | |||
| table: "AspNetUserClaims", | |||
| column: "UserId"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "IX_AspNetUserLogins_UserId", | |||
| table: "AspNetUserLogins", | |||
| column: "UserId"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "IX_AspNetUserRoles_RoleId", | |||
| table: "AspNetUserRoles", | |||
| column: "RoleId"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "EmailIndex", | |||
| table: "AspNetUsers", | |||
| column: "NormalizedEmail"); | |||
| migrationBuilder.CreateIndex( | |||
| name: "UserNameIndex", | |||
| table: "AspNetUsers", | |||
| column: "NormalizedUserName", | |||
| unique: true, | |||
| filter: "[NormalizedUserName] IS NOT NULL"); | |||
| } | |||
| protected override void Down(MigrationBuilder migrationBuilder) | |||
| { | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetRoleClaims"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetUserClaims"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetUserLogins"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetUserRoles"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetUserTokens"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetRoles"); | |||
| migrationBuilder.DropTable( | |||
| name: "AspNetUsers"); | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,275 @@ | |||
| // <auto-generated /> | |||
| using System; | |||
| using Diligent.Web.Data; | |||
| using Microsoft.EntityFrameworkCore; | |||
| using Microsoft.EntityFrameworkCore.Infrastructure; | |||
| using Microsoft.EntityFrameworkCore.Metadata; | |||
| using Microsoft.EntityFrameworkCore.Storage.ValueConversion; | |||
| namespace Diligent.Web.Data.Migrations | |||
| { | |||
| [DbContext(typeof(ApplicationDbContext))] | |||
| partial class ApplicationDbContextModelSnapshot : ModelSnapshot | |||
| { | |||
| protected override void BuildModel(ModelBuilder modelBuilder) | |||
| { | |||
| #pragma warning disable 612, 618 | |||
| modelBuilder | |||
| .HasAnnotation("ProductVersion", "3.0.0") | |||
| .HasAnnotation("Relational:MaxIdentifierLength", 128) | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRole", b => | |||
| { | |||
| b.Property<string>("Id") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("ConcurrencyStamp") | |||
| .IsConcurrencyToken() | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("Name") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("NormalizedName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("NormalizedName") | |||
| .IsUnique() | |||
| .HasName("RoleNameIndex") | |||
| .HasFilter("[NormalizedName] IS NOT NULL"); | |||
| b.ToTable("AspNetRoles"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => | |||
| { | |||
| b.Property<int>("Id") | |||
| .ValueGeneratedOnAdd() | |||
| .HasColumnType("int") | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| b.Property<string>("ClaimType") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("ClaimValue") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("RoleId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("RoleId"); | |||
| b.ToTable("AspNetRoleClaims"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUser", b => | |||
| { | |||
| b.Property<string>("Id") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<int>("AccessFailedCount") | |||
| .HasColumnType("int"); | |||
| b.Property<string>("ConcurrencyStamp") | |||
| .IsConcurrencyToken() | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("Email") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<bool>("EmailConfirmed") | |||
| .HasColumnType("bit"); | |||
| b.Property<bool>("LockoutEnabled") | |||
| .HasColumnType("bit"); | |||
| b.Property<DateTimeOffset?>("LockoutEnd") | |||
| .HasColumnType("datetimeoffset"); | |||
| b.Property<string>("NormalizedEmail") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("NormalizedUserName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.Property<string>("PasswordHash") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("PhoneNumber") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<bool>("PhoneNumberConfirmed") | |||
| .HasColumnType("bit"); | |||
| b.Property<string>("SecurityStamp") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<bool>("TwoFactorEnabled") | |||
| .HasColumnType("bit"); | |||
| b.Property<string>("UserName") | |||
| .HasColumnType("nvarchar(256)") | |||
| .HasMaxLength(256); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("NormalizedEmail") | |||
| .HasName("EmailIndex"); | |||
| b.HasIndex("NormalizedUserName") | |||
| .IsUnique() | |||
| .HasName("UserNameIndex") | |||
| .HasFilter("[NormalizedUserName] IS NOT NULL"); | |||
| b.ToTable("AspNetUsers"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b => | |||
| { | |||
| b.Property<int>("Id") | |||
| .ValueGeneratedOnAdd() | |||
| .HasColumnType("int") | |||
| .HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn); | |||
| b.Property<string>("ClaimType") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("ClaimValue") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("UserId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("Id"); | |||
| b.HasIndex("UserId"); | |||
| b.ToTable("AspNetUserClaims"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b => | |||
| { | |||
| b.Property<string>("LoginProvider") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("ProviderKey") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("ProviderDisplayName") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.Property<string>("UserId") | |||
| .IsRequired() | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("LoginProvider", "ProviderKey"); | |||
| b.HasIndex("UserId"); | |||
| b.ToTable("AspNetUserLogins"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b => | |||
| { | |||
| b.Property<string>("UserId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("RoleId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.HasKey("UserId", "RoleId"); | |||
| b.HasIndex("RoleId"); | |||
| b.ToTable("AspNetUserRoles"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b => | |||
| { | |||
| b.Property<string>("UserId") | |||
| .HasColumnType("nvarchar(450)"); | |||
| b.Property<string>("LoginProvider") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("Name") | |||
| .HasColumnType("nvarchar(128)") | |||
| .HasMaxLength(128); | |||
| b.Property<string>("Value") | |||
| .HasColumnType("nvarchar(max)"); | |||
| b.HasKey("UserId", "LoginProvider", "Name"); | |||
| b.ToTable("AspNetUserTokens"); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null) | |||
| .WithMany() | |||
| .HasForeignKey("RoleId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null) | |||
| .WithMany() | |||
| .HasForeignKey("RoleId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b => | |||
| { | |||
| b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null) | |||
| .WithMany() | |||
| .HasForeignKey("UserId") | |||
| .OnDelete(DeleteBehavior.Cascade) | |||
| .IsRequired(); | |||
| }); | |||
| #pragma warning restore 612, 618 | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,17 @@ | |||
| <Project Sdk="Microsoft.NET.Sdk.Web"> | |||
| <PropertyGroup> | |||
| <TargetFramework>netcoreapp3.1</TargetFramework> | |||
| <UserSecretsId>aspnet-Diligent.Web-9D37C637-AA02-479D-AA93-8A1BFE4C460D</UserSecretsId> | |||
| </PropertyGroup> | |||
| <ItemGroup> | |||
| <PackageReference Include="Microsoft.AspNetCore.Diagnostics.EntityFrameworkCore" Version="3.1.2" /> | |||
| <PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="3.1.2" /> | |||
| <PackageReference Include="Microsoft.AspNetCore.Identity.UI" Version="3.1.2" /> | |||
| <PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="3.1.2" /> | |||
| <PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="3.1.2" /> | |||
| </ItemGroup> | |||
| </Project> | |||
| @@ -0,0 +1,11 @@ | |||
| using System; | |||
| namespace Diligent.Web.Models | |||
| { | |||
| public class ErrorViewModel | |||
| { | |||
| public string RequestId { get; set; } | |||
| public bool ShowRequestId => !string.IsNullOrEmpty(RequestId); | |||
| } | |||
| } | |||
| @@ -0,0 +1,26 @@ | |||
| using System; | |||
| using System.Collections.Generic; | |||
| using System.Linq; | |||
| using System.Threading.Tasks; | |||
| using Microsoft.AspNetCore.Hosting; | |||
| using Microsoft.Extensions.Configuration; | |||
| using Microsoft.Extensions.Hosting; | |||
| using Microsoft.Extensions.Logging; | |||
| namespace Diligent.Web | |||
| { | |||
| public class Program | |||
| { | |||
| public static void Main(string[] args) | |||
| { | |||
| CreateHostBuilder(args).Build().Run(); | |||
| } | |||
| public static IHostBuilder CreateHostBuilder(string[] args) => | |||
| Host.CreateDefaultBuilder(args) | |||
| .ConfigureWebHostDefaults(webBuilder => | |||
| { | |||
| webBuilder.UseStartup<Startup>(); | |||
| }); | |||
| } | |||
| } | |||
| @@ -0,0 +1,27 @@ | |||
| { | |||
| "iisSettings": { | |||
| "windowsAuthentication": false, | |||
| "anonymousAuthentication": true, | |||
| "iisExpress": { | |||
| "applicationUrl": "http://localhost:16144", | |||
| "sslPort": 44321 | |||
| } | |||
| }, | |||
| "profiles": { | |||
| "IIS Express": { | |||
| "commandName": "IISExpress", | |||
| "launchBrowser": true, | |||
| "environmentVariables": { | |||
| "ASPNETCORE_ENVIRONMENT": "Development" | |||
| } | |||
| }, | |||
| "Diligent.Web": { | |||
| "commandName": "Project", | |||
| "launchBrowser": true, | |||
| "applicationUrl": "https://localhost:5001;http://localhost:5000", | |||
| "environmentVariables": { | |||
| "ASPNETCORE_ENVIRONMENT": "Development" | |||
| } | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,70 @@ | |||
| using System; | |||
| using System.Collections.Generic; | |||
| using System.Linq; | |||
| using System.Threading.Tasks; | |||
| using Microsoft.AspNetCore.Builder; | |||
| using Microsoft.AspNetCore.Identity; | |||
| using Microsoft.AspNetCore.Identity.UI; | |||
| using Microsoft.AspNetCore.Hosting; | |||
| using Microsoft.AspNetCore.HttpsPolicy; | |||
| using Microsoft.EntityFrameworkCore; | |||
| using Diligent.Web.Data; | |||
| using Microsoft.Extensions.Configuration; | |||
| using Microsoft.Extensions.DependencyInjection; | |||
| using Microsoft.Extensions.Hosting; | |||
| namespace Diligent.Web | |||
| { | |||
| public class Startup | |||
| { | |||
| public Startup(IConfiguration configuration) | |||
| { | |||
| Configuration = configuration; | |||
| } | |||
| public IConfiguration Configuration { get; } | |||
| // This method gets called by the runtime. Use this method to add services to the container. | |||
| public void ConfigureServices(IServiceCollection services) | |||
| { | |||
| services.AddDbContext<ApplicationDbContext>(options => | |||
| options.UseSqlServer( | |||
| Configuration.GetConnectionString("DefaultConnection"))); | |||
| services.AddDefaultIdentity<IdentityUser>(options => options.SignIn.RequireConfirmedAccount = true) | |||
| .AddEntityFrameworkStores<ApplicationDbContext>(); | |||
| services.AddControllersWithViews(); | |||
| services.AddRazorPages(); | |||
| } | |||
| // This method gets called by the runtime. Use this method to configure the HTTP request pipeline. | |||
| public void Configure(IApplicationBuilder app, IWebHostEnvironment env) | |||
| { | |||
| if (env.IsDevelopment()) | |||
| { | |||
| app.UseDeveloperExceptionPage(); | |||
| app.UseDatabaseErrorPage(); | |||
| } | |||
| else | |||
| { | |||
| 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.UseAuthentication(); | |||
| app.UseAuthorization(); | |||
| app.UseEndpoints(endpoints => | |||
| { | |||
| endpoints.MapControllerRoute( | |||
| name: "default", | |||
| pattern: "{controller=Home}/{action=Index}/{id?}"); | |||
| endpoints.MapRazorPages(); | |||
| }); | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,8 @@ | |||
| @{ | |||
| ViewData["Title"] = "Home Page"; | |||
| } | |||
| <div class="text-center"> | |||
| <h1 class="display-4">Welcome</h1> | |||
| <p>Learn about <a href="https://docs.microsoft.com/aspnet/core">building Web apps with ASP.NET Core</a>.</p> | |||
| </div> | |||
| @@ -0,0 +1,6 @@ | |||
| @{ | |||
| ViewData["Title"] = "Privacy Policy"; | |||
| } | |||
| <h1>@ViewData["Title"]</h1> | |||
| <p>Use this page to detail your site's privacy policy.</p> | |||
| @@ -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> | |||
| @@ -0,0 +1,49 @@ | |||
| <!DOCTYPE html> | |||
| <html lang="en"> | |||
| <head> | |||
| <meta charset="utf-8" /> | |||
| <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | |||
| <title>@ViewData["Title"] - Diligent.Web</title> | |||
| <link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css" /> | |||
| <link rel="stylesheet" href="~/css/site.css" /> | |||
| </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"> | |||
| <a class="navbar-brand" asp-area="" asp-controller="Home" asp-action="Index">Diligent.Web</a> | |||
| <button class="navbar-toggler" type="button" data-toggle="collapse" data-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 d-sm-inline-flex flex-sm-row-reverse"> | |||
| <partial name="_LoginPartial" /> | |||
| <ul class="navbar-nav flex-grow-1"> | |||
| <li class="nav-item"> | |||
| <a class="nav-link text-dark" asp-area="" asp-controller="Home" asp-action="Index">Home</a> | |||
| </li> | |||
| <li class="nav-item"> | |||
| <a class="nav-link text-dark" asp-area="" asp-controller="Home" asp-action="Privacy">Privacy</a> | |||
| </li> | |||
| </ul> | |||
| </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"> | |||
| © 2020 - Diligent.Web - <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> | |||
| @RenderSection("Scripts", required: false) | |||
| </body> | |||
| </html> | |||
| @@ -0,0 +1,26 @@ | |||
| @using Microsoft.AspNetCore.Identity | |||
| @inject SignInManager<IdentityUser> SignInManager | |||
| @inject UserManager<IdentityUser> UserManager | |||
| <ul class="navbar-nav"> | |||
| @if (SignInManager.IsSignedIn(User)) | |||
| { | |||
| <li class="nav-item"> | |||
| <a class="nav-link text-dark" asp-area="Identity" asp-page="/Account/Manage/Index" title="Manage">Hello @User.Identity.Name!</a> | |||
| </li> | |||
| <li class="nav-item"> | |||
| <form class="form-inline" asp-area="Identity" asp-page="/Account/Logout" asp-route-returnUrl="@Url.Action("Index", "Home", new { area = "" })"> | |||
| <button type="submit" class="nav-link btn btn-link text-dark">Logout</button> | |||
| </form> | |||
| </li> | |||
| } | |||
| else | |||
| { | |||
| <li class="nav-item"> | |||
| <a class="nav-link text-dark" asp-area="Identity" asp-page="/Account/Register">Register</a> | |||
| </li> | |||
| <li class="nav-item"> | |||
| <a class="nav-link text-dark" asp-area="Identity" asp-page="/Account/Login">Login</a> | |||
| </li> | |||
| } | |||
| </ul> | |||
| @@ -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> | |||
| @@ -0,0 +1,3 @@ | |||
| @using Diligent.Web | |||
| @using Diligent.Web.Models | |||
| @addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers | |||
| @@ -0,0 +1,3 @@ | |||
| @{ | |||
| Layout = "_Layout"; | |||
| } | |||
| @@ -0,0 +1,9 @@ | |||
| { | |||
| "Logging": { | |||
| "LogLevel": { | |||
| "Default": "Information", | |||
| "Microsoft": "Warning", | |||
| "Microsoft.Hosting.Lifetime": "Information" | |||
| } | |||
| } | |||
| } | |||
| @@ -0,0 +1,13 @@ | |||
| { | |||
| "ConnectionStrings": { | |||
| "DefaultConnection": "Server=(localdb)\\mssqllocaldb;Database=aspnet-Diligent.Web-9D37C637-AA02-479D-AA93-8A1BFE4C460D;Trusted_Connection=True;MultipleActiveResultSets=true" | |||
| }, | |||
| "Logging": { | |||
| "LogLevel": { | |||
| "Default": "Information", | |||
| "Microsoft": "Warning", | |||
| "Microsoft.Hosting.Lifetime": "Information" | |||
| } | |||
| }, | |||
| "AllowedHosts": "*" | |||
| } | |||
| @@ -0,0 +1,71 @@ | |||
| /* 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; | |||
| } | |||
| /* Provide sufficient contrast against white background */ | |||
| a { | |||
| color: #0366d6; | |||
| } | |||
| .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; | |||
| } | |||
| /* Sticky footer styles | |||
| -------------------------------------------------- */ | |||
| html { | |||
| font-size: 14px; | |||
| } | |||
| @media (min-width: 768px) { | |||
| html { | |||
| font-size: 16px; | |||
| } | |||
| } | |||
| .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; | |||
| } | |||
| /* Sticky footer styles | |||
| -------------------------------------------------- */ | |||
| html { | |||
| position: relative; | |||
| min-height: 100%; | |||
| } | |||
| body { | |||
| /* Margin bottom by footer height */ | |||
| margin-bottom: 60px; | |||
| } | |||
| .footer { | |||
| position: absolute; | |||
| bottom: 0; | |||
| width: 100%; | |||
| white-space: nowrap; | |||
| line-height: 60px; /* Vertically center the text there */ | |||
| } | |||
| @@ -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. | |||
| @@ -0,0 +1,22 @@ | |||
| The MIT License (MIT) | |||
| Copyright (c) 2011-2018 Twitter, Inc. | |||
| Copyright (c) 2011-2018 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. | |||
| @@ -0,0 +1,331 @@ | |||
| /*! | |||
| * Bootstrap Reboot v4.3.1 (https://getbootstrap.com/) | |||
| * Copyright 2011-2019 The Bootstrap Authors | |||
| * Copyright 2011-2019 Twitter, Inc. | |||
| * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) | |||
| * Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md) | |||
| */ | |||
| *, | |||
| *::before, | |||
| *::after { | |||
| box-sizing: border-box; | |||
| } | |||
| html { | |||
| font-family: sans-serif; | |||
| line-height: 1.15; | |||
| -webkit-text-size-adjust: 100%; | |||
| -webkit-tap-highlight-color: rgba(0, 0, 0, 0); | |||
| } | |||
| article, aside, figcaption, figure, footer, header, hgroup, main, nav, section { | |||
| display: block; | |||
| } | |||
| body { | |||
| margin: 0; | |||
| font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji"; | |||
| font-size: 1rem; | |||
| font-weight: 400; | |||
| line-height: 1.5; | |||
| color: #212529; | |||
| text-align: left; | |||
| background-color: #fff; | |||
| } | |||
| [tabindex="-1"]:focus { | |||
| outline: 0 !important; | |||
| } | |||
| hr { | |||
| box-sizing: content-box; | |||
| height: 0; | |||
| overflow: visible; | |||
| } | |||
| h1, h2, h3, h4, h5, h6 { | |||
| margin-top: 0; | |||
| margin-bottom: 0.5rem; | |||
| } | |||
| p { | |||
| margin-top: 0; | |||
| margin-bottom: 1rem; | |||
| } | |||
| abbr[title], | |||
| abbr[data-original-title] { | |||
| text-decoration: underline; | |||
| -webkit-text-decoration: underline dotted; | |||
| text-decoration: underline dotted; | |||
| cursor: help; | |||
| border-bottom: 0; | |||
| -webkit-text-decoration-skip-ink: none; | |||
| text-decoration-skip-ink: none; | |||
| } | |||
| address { | |||
| margin-bottom: 1rem; | |||
| font-style: normal; | |||
| line-height: inherit; | |||
| } | |||
| 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: .5rem; | |||
| margin-left: 0; | |||
| } | |||
| blockquote { | |||
| margin: 0 0 1rem; | |||
| } | |||
| b, | |||
| strong { | |||
| font-weight: bolder; | |||
| } | |||
| small { | |||
| font-size: 80%; | |||
| } | |||
| sub, | |||
| sup { | |||
| position: relative; | |||
| font-size: 75%; | |||
| line-height: 0; | |||
| vertical-align: baseline; | |||
| } | |||
| sub { | |||
| bottom: -.25em; | |||
| } | |||
| sup { | |||
| top: -.5em; | |||
| } | |||
| a { | |||
| color: #007bff; | |||
| text-decoration: none; | |||
| background-color: transparent; | |||
| } | |||
| a:hover { | |||
| color: #0056b3; | |||
| text-decoration: underline; | |||
| } | |||
| a:not([href]):not([tabindex]) { | |||
| color: inherit; | |||
| text-decoration: none; | |||
| } | |||
| a:not([href]):not([tabindex]):hover, a:not([href]):not([tabindex]):focus { | |||
| color: inherit; | |||
| text-decoration: none; | |||
| } | |||
| a:not([href]):not([tabindex]):focus { | |||
| outline: 0; | |||
| } | |||
| pre, | |||
| code, | |||
| kbd, | |||
| samp { | |||
| font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace; | |||
| font-size: 1em; | |||
| } | |||
| pre { | |||
| margin-top: 0; | |||
| margin-bottom: 1rem; | |||
| overflow: auto; | |||
| } | |||
| figure { | |||
| margin: 0 0 1rem; | |||
| } | |||
| img { | |||
| vertical-align: middle; | |||
| border-style: none; | |||
| } | |||
| svg { | |||
| overflow: hidden; | |||
| vertical-align: middle; | |||
| } | |||
| table { | |||
| border-collapse: collapse; | |||
| } | |||
| caption { | |||
| padding-top: 0.75rem; | |||
| padding-bottom: 0.75rem; | |||
| color: #6c757d; | |||
| text-align: left; | |||
| caption-side: bottom; | |||
| } | |||
| th { | |||
| text-align: inherit; | |||
| } | |||
| label { | |||
| display: inline-block; | |||
| margin-bottom: 0.5rem; | |||
| } | |||
| button { | |||
| border-radius: 0; | |||
| } | |||
| button:focus { | |||
| outline: 1px dotted; | |||
| outline: 5px auto -webkit-focus-ring-color; | |||
| } | |||
| input, | |||
| button, | |||
| select, | |||
| optgroup, | |||
| textarea { | |||
| margin: 0; | |||
| font-family: inherit; | |||
| font-size: inherit; | |||
| line-height: inherit; | |||
| } | |||
| button, | |||
| input { | |||
| overflow: visible; | |||
| } | |||
| button, | |||
| select { | |||
| text-transform: none; | |||
| } | |||
| select { | |||
| word-wrap: normal; | |||
| } | |||
| 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; | |||
| } | |||
| button::-moz-focus-inner, | |||
| [type="button"]::-moz-focus-inner, | |||
| [type="reset"]::-moz-focus-inner, | |||
| [type="submit"]::-moz-focus-inner { | |||
| padding: 0; | |||
| border-style: none; | |||
| } | |||
| input[type="radio"], | |||
| input[type="checkbox"] { | |||
| box-sizing: border-box; | |||
| padding: 0; | |||
| } | |||
| input[type="date"], | |||
| input[type="time"], | |||
| input[type="datetime-local"], | |||
| input[type="month"] { | |||
| -webkit-appearance: listbox; | |||
| } | |||
| textarea { | |||
| overflow: auto; | |||
| resize: vertical; | |||
| } | |||
| fieldset { | |||
| min-width: 0; | |||
| padding: 0; | |||
| margin: 0; | |||
| border: 0; | |||
| } | |||
| legend { | |||
| display: block; | |||
| width: 100%; | |||
| max-width: 100%; | |||
| padding: 0; | |||
| margin-bottom: .5rem; | |||
| font-size: 1.5rem; | |||
| line-height: inherit; | |||
| color: inherit; | |||
| white-space: normal; | |||
| } | |||
| progress { | |||
| vertical-align: baseline; | |||
| } | |||
| [type="number"]::-webkit-inner-spin-button, | |||
| [type="number"]::-webkit-outer-spin-button { | |||
| height: auto; | |||
| } | |||
| [type="search"] { | |||
| outline-offset: -2px; | |||
| -webkit-appearance: none; | |||
| } | |||
| [type="search"]::-webkit-search-decoration { | |||
| -webkit-appearance: none; | |||
| } | |||
| ::-webkit-file-upload-button { | |||
| font: inherit; | |||
| -webkit-appearance: button; | |||
| } | |||
| output { | |||
| display: inline-block; | |||
| } | |||
| summary { | |||
| display: list-item; | |||
| cursor: pointer; | |||
| } | |||
| template { | |||
| display: none; | |||
| } | |||
| [hidden] { | |||
| display: none !important; | |||
| } | |||
| /*# sourceMappingURL=bootstrap-reboot.css.map */ | |||
| @@ -0,0 +1,8 @@ | |||
| /*! | |||
| * Bootstrap Reboot v4.3.1 (https://getbootstrap.com/) | |||
| * Copyright 2011-2019 The Bootstrap Authors | |||
| * Copyright 2011-2019 Twitter, Inc. | |||
| * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) | |||
| * Forked from Normalize.css, licensed MIT (https://github.com/necolas/normalize.css/blob/master/LICENSE.md) | |||
| */*,::after,::before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}article,aside,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:left;background-color:#fff}[tabindex="-1"]:focus{outline:0!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem}p{margin-top:0;margin-bottom:1rem}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}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:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#007bff;text-decoration:none;background-color:transparent}a:hover{color:#0056b3;text-decoration:underline}a:not([href]):not([tabindex]){color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus,a:not([href]):not([tabindex]):hover{color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus{outline:0}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg{overflow:hidden;vertical-align:middle}table{border-collapse:collapse}caption{padding-top:.75rem;padding-bottom:.75rem;color:#6c757d;text-align:left;caption-side:bottom}th{text-align:inherit}label{display:inline-block;margin-bottom:.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}select{word-wrap:normal}[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}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=date],input[type=datetime-local],input[type=month],input[type=time]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none!important} | |||
| /*# sourceMappingURL=bootstrap-reboot.min.css.map */ | |||
| @@ -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. | |||
| @@ -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; | |||
| })); | |||
| @@ -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. | |||
| @@ -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. | |||