develop #5

Merged
luolan merged 26 commits from develop into master 2025-08-05 12:45:26 +08:00
48 changed files with 3541 additions and 235 deletions

13
.idea/.idea.AGSS/.idea/.gitignore generated vendored Normal file
View File

@ -0,0 +1,13 @@
# 默认忽略的文件
/shelf/
/workspace.xml
# Rider 忽略的文件
/projectSettingsUpdater.xml
/modules.xml
/.idea.AGSS.iml
/contentModel.xml
# 基于编辑器的 HTTP 客户端请求
/httpRequests/
# Datasource local storage ignored files
/dataSources/
/dataSources.local.xml

1
.idea/.idea.AGSS/.idea/.name generated Normal file
View File

@ -0,0 +1 @@
AGSS

View File

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="MaterialThemeProjectNewConfig">
<option name="metadata">
<MTProjectMetadataState>
<option name="migrated" value="true" />
<option name="pristineConfig" value="false" />
<option name="userId" value="7c74c904:197ef619e58:-7ffc" />
</MTProjectMetadataState>
</option>
</component>
</project>

6
.idea/.idea.AGSS/.idea/sqldialects.xml generated Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="SqlDialectMappings">
<file url="file://$PROJECT_DIR$/script.sql" dialect="PostgreSQL" />
</component>
</project>

View File

@ -1,6 +0,0 @@
@AGSS_HostAddress = http://localhost:5200
GET {{AGSS_HostAddress}}/weatherforecast/
Accept: application/json
###

View File

@ -6,16 +6,19 @@ using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.ComponentModel.DataAnnotations; using System.ComponentModel.DataAnnotations;
using System.Linq; using System.Linq;
using System.Security.Claims;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Authorization;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using AGSS.Utilities; using AGSS.Utilities;
using asg_form;
using Microsoft.AspNetCore.Authentication; using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Identity.UI.Services; using Microsoft.AspNetCore.Identity.UI.Services;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.RazorPages; using Microsoft.AspNetCore.Mvc.RazorPages;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
namespace AGSS.Areas.Identity.Pages.Account namespace AGSS.Areas.Identity.Pages.Account
{ {
@ -111,7 +114,7 @@ namespace AGSS.Areas.Identity.Pages.Account
ReturnUrl = returnUrl; ReturnUrl = returnUrl;
} }
public async Task<IActionResult> OnPostAsync(string returnUrl = null) public async Task<IActionResult> OnPostAsync([FromServices] IOptions<JWTOptions> jwtOptions,string returnUrl = null)
{ {
returnUrl ??= Url.Content("~/"); returnUrl ??= Url.Content("~/");
@ -125,12 +128,20 @@ namespace AGSS.Areas.Identity.Pages.Account
if (result.Succeeded) if (result.Succeeded)
{ {
_logger.LogInformation("User logged in."); _logger.LogInformation("User logged in.");
var user = await _userManager.FindByEmailAsync(Input.Email); var user = await _userManager.FindByEmailAsync(Input.Email);
var roles = await _userManager.GetRolesAsync(user);
var token = _jwt.GenerateJwtToken(user,roles);
var frontendCallback = $"{Request.Query["frontendCallback"]}?token={token}"; var claims = new List<Claim>();
claims.Add(new Claim(ClaimTypes.NameIdentifier, user.Id.ToString()));
claims.Add(new Claim(ClaimTypes.Name, user.UserName));
var roles = await _userManager.GetRolesAsync(user);
foreach (string role in roles)
{
claims.Add(new Claim(ClaimTypes.Role, role));
}
string jwtToken = _jwt.BuildToken(claims, jwtOptions.Value);
var frontendCallback = $"{Request.Query["frontendCallback"]}?token={jwtToken}";
return Redirect(frontendCallback); return Redirect(frontendCallback);
} }

View File

@ -12,6 +12,11 @@
<h2>创建新账户。</h2> <h2>创建新账户。</h2>
<hr /> <hr />
<div asp-validation-summary="ModelOnly" class="text-danger" role="alert"></div> <div asp-validation-summary="ModelOnly" class="text-danger" role="alert"></div>
<div class="form-floating mb-3">
<input asp-for="Input.UserName" class="form-control" autocomplete="username" aria-required="true" placeholder="luolan" />
<label asp-for="Input.UserName">用户名</label>
<span asp-validation-for="Input.UserName" class="text-danger"></span>
</div>
<div class="form-floating mb-3"> <div class="form-floating mb-3">
<input asp-for="Input.Email" class="form-control" autocomplete="username" aria-required="true" placeholder="name@example.com" /> <input asp-for="Input.Email" class="form-control" autocomplete="username" aria-required="true" placeholder="name@example.com" />
<label asp-for="Input.Email">电子邮件</label> <label asp-for="Input.Email">电子邮件</label>

View File

@ -6,6 +6,7 @@ using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.ComponentModel.DataAnnotations; using System.ComponentModel.DataAnnotations;
using System.Linq; using System.Linq;
using System.Security.Claims;
using System.Text; using System.Text;
using System.Text.Encodings.Web; using System.Text.Encodings.Web;
using System.Threading; using System.Threading;
@ -14,12 +15,14 @@ using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Authorization;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using AGSS.Utilities; using AGSS.Utilities;
using asg_form;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Identity.UI.Services; using Microsoft.AspNetCore.Identity.UI.Services;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.RazorPages; using Microsoft.AspNetCore.Mvc.RazorPages;
using Microsoft.AspNetCore.WebUtilities; using Microsoft.AspNetCore.WebUtilities;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
namespace AGSS.Areas.Identity.Pages.Account namespace AGSS.Areas.Identity.Pages.Account
{ {
@ -107,9 +110,12 @@ namespace AGSS.Areas.Identity.Pages.Account
[MaxLength(10)] [MaxLength(10)]
[Display(Name = "Confirm password")]
public string Sex { get; set; } public string Sex { get; set; }
[MaxLength(10)]
public string UserName { get; set; }
} }
@ -119,7 +125,7 @@ namespace AGSS.Areas.Identity.Pages.Account
ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList(); ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList();
} }
public async Task<IActionResult> OnPostAsync(string returnUrl = null) public async Task<IActionResult> OnPostAsync([FromServices] IOptions<JWTOptions> jwtOptions,string returnUrl = null)
{ {
returnUrl ??= Url.Content("~/"); returnUrl ??= Url.Content("~/");
ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList(); ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList();
@ -128,17 +134,24 @@ namespace AGSS.Areas.Identity.Pages.Account
var user = CreateUser(); var user = CreateUser();
user.Id = Guid.NewGuid().ToString(); user.Id = Guid.NewGuid().ToString();
user.Sex = Input.Sex; user.Sex = Input.Sex;
await _userStore.SetUserNameAsync(user, Input.Email, CancellationToken.None); await _userStore.SetUserNameAsync(user, Input.UserName, CancellationToken.None);
await _emailStore.SetEmailAsync(user, Input.Email, CancellationToken.None); await _emailStore.SetEmailAsync(user, Input.Email, CancellationToken.None);
var result = await _userManager.CreateAsync(user, Input.Password); var result = await _userManager.CreateAsync(user, Input.Password);
if (result.Succeeded) if (result.Succeeded)
{ {
_logger.LogInformation("User created a new account with password."); _logger.LogInformation("User created a new account with password.");
// var roles = await _userManager.GetRolesAsync(user);
var claims = new List<Claim>();
claims.Add(new Claim(ClaimTypes.NameIdentifier, user.Id.ToString()));
claims.Add(new Claim(ClaimTypes.Name, user.UserName));
var roles = await _userManager.GetRolesAsync(user); var roles = await _userManager.GetRolesAsync(user);
var token = _jwt.GenerateJwtToken(user,roles); foreach (string role in roles)
{
var frontendCallback = $"{Request.Query["frontendCallback"]}?token={token}"; claims.Add(new Claim(ClaimTypes.Role, role));
}
string jwtToken = _jwt.BuildToken(claims, jwtOptions.Value);
var frontendCallback = $"{Request.Query["frontendCallback"]}?token={jwtToken}";
return Redirect(frontendCallback); return Redirect(frontendCallback);
} }

View File

@ -1,9 +1,16 @@
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using AGSS.Models;
using AGSS.Models.DTOs; using AGSS.Models.DTOs;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using AGSS.Models.Template; using AGSS.Models.Template;
using AGSS.Services;
using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
namespace AGSS.Controllers.Admin; namespace AGSS.Controllers.Admin;
@ -12,9 +19,23 @@ namespace AGSS.Controllers.Admin;
/// 该控制器仅限具有"Admin"角色的用户访问。 /// 该控制器仅限具有"Admin"角色的用户访问。
/// </summary> /// </summary>
[Authorize(Roles = "Admin")] [Authorize(Roles = "Admin")]
[Route("api/v1/[controller]/[action]")] [Route("api/v1/[controller]")]
public class AdminRoleControllers:ControllerBase public class AdminRoleControllers:ControllerBase
{ {
/// <summary>
/// 用户服务实例,用于执行与用户相关的操作。
/// 该服务提供了一系列方法来处理用户的查询和更新等操作,
/// 包括但不限于获取用户详细信息、修改用户资料等功能。
/// </summary>
private readonly UserService _userService;
public AdminRoleControllers(UserService userService, RoleManager<RoleModel> roleManager, UserManager<UserModel> userManager)
{
_userService = userService;
_roleManager = roleManager;
_userManager = userManager;
}
/// <summary> /// <summary>
/// 角色管理器,用于处理角色相关的操作,如创建、查询等。 /// 角色管理器,用于处理角色相关的操作,如创建、查询等。
/// 此角色管理器实例主要用于与RoleModel类型的实体进行交互 /// 此角色管理器实例主要用于与RoleModel类型的实体进行交互
@ -32,30 +53,24 @@ public class AdminRoleControllers:ControllerBase
/// 管理员角色控制器,用于处理与角色相关的操作,如添加角色、分配角色给用户以及通过角色查询用户。 /// 管理员角色控制器,用于处理与角色相关的操作,如添加角色、分配角色给用户以及通过角色查询用户。
/// 该控制器下的所有方法都需要管理员权限才能访问。 /// 该控制器下的所有方法都需要管理员权限才能访问。
/// </summary> /// </summary>
public AdminRoleControllers(RoleManager<RoleModel> roleManager, UserManager<UserModel> userManager)
{
_roleManager = roleManager;
_userManager = userManager;
}
/// <summary> /// <summary>
/// 添加新角色 /// 添加新角色
/// </summary> /// </summary>
/// <param name="role">要添加的角色信息</param> /// <param name="role">要添加的角色信息</param>
/// <param name="rolename">角色名</param>
/// <param name="normalizedname">可以不填</param>
/// <param name="ChineseName">中文名</param>
/// <returns>返回操作结果,包含状态码、消息和数据</returns> /// <returns>返回操作结果,包含状态码、消息和数据</returns>
[HttpPost] [HttpPost("role")]
public async Task<IActionResult> AddRole([FromBody] RoleModel role) public async Task<IActionResult> AddRole(string rolename,string normalizedname,string ChineseName)
{ {
if (role == null || string.IsNullOrWhiteSpace(role.Name))
{
return Ok(new ReturnTemplate(400,"创建失败,请提供名字",""));
}
var result = await _roleManager.CreateAsync(role);
var result = await _roleManager.CreateAsync(new RoleModel(){Id = Guid.NewGuid().ToString(),Name = rolename,NormalizedName = normalizedname,ChineseName = ChineseName});
if (result.Succeeded) if (result.Succeeded)
{ {
return Ok(new ReturnTemplate(200,"创建成功",role)); return Ok(new ReturnTemplate(200,"创建成功",""));
} }
else else
@ -64,13 +79,38 @@ public class AdminRoleControllers:ControllerBase
} }
} }
[HttpDelete("role")]
public async Task<IActionResult> DelRole(string id)
{
var delrole = await _roleManager.FindByIdAsync(id);
var result = await _roleManager.DeleteAsync(delrole);
if (result.Succeeded)
{
return Ok(new ReturnTemplate(200,"成功了呢,贝贝",""));
}
else
{
return Ok(new ReturnTemplate(StatusCodes.Status500InternalServerError,"失败了呢,贝贝","Failed to delect role: " + string.Join(", ", result.Errors.Select(e => e.Description))));
}
}
/// <summary> /// <summary>
/// 为指定用户分配角色 /// 为指定用户分配角色
/// </summary> /// </summary>
/// <param name="userId">用户的唯一标识符</param> /// <param name="userId">用户的唯一标识符</param>
/// <param name="roleName">要分配的角色名称</param> /// <param name="roleName">要分配的角色名称</param>
/// <returns>返回一个包含操作结果的ReturnTemplate对象其中Code表示状态码Msg表示消息Data表示附加数据如果有的话</returns> /// <returns>返回一个包含操作结果的ReturnTemplate对象其中Code表示状态码Msg表示消息Data表示附加数据如果有的话</returns>
[HttpPost]
[HttpPost("role/endow")]
public async Task<IActionResult> EndowRole(string userId, string roleName) public async Task<IActionResult> EndowRole(string userId, string roleName)
{ {
var user = await _userManager.FindByIdAsync(userId); var user = await _userManager.FindByIdAsync(userId);
@ -88,7 +128,7 @@ public class AdminRoleControllers:ControllerBase
var result = await _userManager.AddToRoleAsync(user, role.Name); var result = await _userManager.AddToRoleAsync(user, role.Name);
if (result.Succeeded) if (result.Succeeded)
{ {
return Ok(new ReturnTemplate(200, "角色分配成功", user)); return Ok(new ReturnTemplate(200, "现在该用户已经被赋予这个角色了", user));
} }
else else
{ {
@ -102,18 +142,18 @@ public class AdminRoleControllers:ControllerBase
/// </summary> /// </summary>
/// <param name="userId">要删除的用户的唯一标识符。</param> /// <param name="userId">要删除的用户的唯一标识符。</param>
/// <returns>返回操作结果包含状态码、消息和数据。如果删除成功则返回200状态码如果用户ID为空或未找到指定用户则分别返回400或404状态码若删除过程中出现错误则返回500状态码并附带错误信息。</returns> /// <returns>返回操作结果包含状态码、消息和数据。如果删除成功则返回200状态码如果用户ID为空或未找到指定用户则分别返回400或404状态码若删除过程中出现错误则返回500状态码并附带错误信息。</returns>
[HttpPost] [HttpDelete("user")]
public async Task<IActionResult> DelUser(string userId) public async Task<IActionResult> DelUser(string userId)
{ {
if (string.IsNullOrWhiteSpace(userId)) if (string.IsNullOrWhiteSpace(userId))
{ {
return Ok(new ReturnTemplate(400, "你填写的用户ID是空的~", null)); return Ok(new ReturnTemplate(400, "你打算拿个空的Id来骗我吗", null));
} }
var user = await _userManager.FindByIdAsync(userId); var user = await _userManager.FindByIdAsync(userId);
if (user == null) if (user == null)
{ {
return Ok(new ReturnTemplate(404, "未找到指定用户哦·~", null)); return Ok(new ReturnTemplate(404, "你输了个假的用户吧......", null));
} }
// 删除用户 // 删除用户
@ -129,8 +169,8 @@ public class AdminRoleControllers:ControllerBase
} }
[HttpPost] [HttpPost("menu")]
public async Task<IActionResult> SetMenu([FromBody]MenuRequest request) public async Task<IActionResult> SetMenu([FromBody]UserMenuRequest request)
{ {
if (string.IsNullOrWhiteSpace(request.Id) || string.IsNullOrWhiteSpace(request.MenuName)) if (string.IsNullOrWhiteSpace(request.Id) || string.IsNullOrWhiteSpace(request.MenuName))
{ {
@ -150,13 +190,18 @@ public class AdminRoleControllers:ControllerBase
} }
else else
{ {
return StatusCode(500, new ReturnTemplate(500, "删除用户时发生错误", result.Errors)); return StatusCode(500, new ReturnTemplate(500, "删除用户时发生错误原因请看ErrorResult", result.Errors));
} }
} }
[HttpGet("role/all")]
public async Task<IActionResult> AllRole()
{
return Ok(new ReturnTemplate(200,"查询成功啦!", _roleManager.Roles.ToList()));
}
@ -165,21 +210,25 @@ public class AdminRoleControllers:ControllerBase
/// </summary> /// </summary>
/// <param name="request">包含角色名称、页码和每页大小的请求对象</param> /// <param name="request">包含角色名称、页码和每页大小的请求对象</param>
/// <returns>返回包含总用户数和当前页用户的响应对象</returns> /// <returns>返回包含总用户数和当前页用户的响应对象</returns>
[HttpPost] [HttpPost("user")]
public async Task<IActionResult> SearchUserFromRole([FromBody] SearchUserFromRoleRequest request) public async Task<IActionResult> SearchUserFromRole([FromBody] SearchUserFromRoleRequest request)
{ {
IList<UserProfile> usersInRole = null;
if (string.IsNullOrWhiteSpace(request.RoleName)) if (string.IsNullOrWhiteSpace(request.RoleName))
{ {
return Ok(new ReturnTemplate(400, "角色名称不能为空", null)); usersInRole = await _userService.GetUsersProfileByUserNameAsync(request.UserName);
} }
else
var role = await _roleManager.FindByNameAsync(request.RoleName);
if (role == null)
{ {
return Ok(new ReturnTemplate(400, "角色不存在", null)); var usersInRole1 = await _userService.GetUsersProfileInRoleAsync(request.RoleName);
usersInRole = usersInRole1.Where(a => a.UserName.Contains(request.UserName)).ToList();
} }
var usersInRole = await _userManager.GetUsersInRoleAsync(role.Name);
var totalUsers = usersInRole.Count; var totalUsers = usersInRole.Count;
var pagedUsers = usersInRole var pagedUsers = usersInRole
@ -201,6 +250,9 @@ public class AdminRoleControllers:ControllerBase
/// </summary> /// </summary>
public class SearchUserFromRoleRequest public class SearchUserFromRoleRequest
{ {
public string UserName { get; set; }
/// <summary> /// <summary>
/// 表示角色的名称。此属性用于指定或获取与用户管理相关的角色名称。 /// 表示角色的名称。此属性用于指定或获取与用户管理相关的角色名称。
/// 在进行角色分配、查询等操作时,需要提供正确的角色名称以确保操作的成功执行。 /// 在进行角色分配、查询等操作时,需要提供正确的角色名称以确保操作的成功执行。
@ -235,6 +287,6 @@ public class AdminRoleControllers:ControllerBase
/// 表示属于特定角色的用户列表。该属性用于存储和返回在给定角色下的所有用户。 /// 表示属于特定角色的用户列表。该属性用于存储和返回在给定角色下的所有用户。
/// </summary> /// </summary>
/// <remarks>此列表通常作为查询结果的一部分,例如通过角色名搜索用户时返回的数据。</remarks> /// <remarks>此列表通常作为查询结果的一部分,例如通过角色名搜索用户时返回的数据。</remarks>
public List<UserModel> Users { get; set; } public List<UserProfile> Users { get; set; }
} }
} }

View File

@ -0,0 +1,354 @@
using System.ComponentModel.DataAnnotations;
using AGSS.Models.Template;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
namespace AGSS.Controllers.Dict
{
[ApiController]
[Route("api/[controller]")]
// [Produces("application/json")]
public class DictController : ControllerBase
{
private readonly DictService _dictService;
private readonly ILogger<DictController> _logger;
public DictController(DictService dictService, ILogger<DictController> logger)
{
_dictService = dictService;
_logger = logger;
}
// 1. 获取父级列表
[HttpGet("parents")]
public async Task<ActionResult> GetParents([FromQuery] string? label)
{
try
{
var result = await _dictService.GetParentsAsync(label);
return Ok(new ReturnTemplate(200, "获取成功啦!", result));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(500, "失败,服务器类错误!", ex));
}
}
// 2. 创建父级项
// [Authorize(Roles = "Admin")]
/// <summary>
/// 创建一个新的父级项。
/// </summary>
/// <param name="request">包含创建父级所需信息的请求对象。</param>
/// <returns>返回一个ActionResult表示操作结果。成功时返回状态码200及新创建的父级项失败时返回错误信息和相应的状态码。</returns>
[HttpPost("parents")]
public async Task<ActionResult> CreateParent([FromBody] CreateParentRequest request)
{
var result = await _dictService.CreateParentAsync(request.Label, request.Value);
return Ok(new ReturnTemplate(200, "创建成功啦!", result));
}
// 请求模型
public class CreateParentRequest
{
[Required(ErrorMessage = "字典标签不能为空")]
[StringLength(100, ErrorMessage = "标签长度不能超过100字符")]
public string Label { get; set; } = null!;
/// <summary>
/// Represents the value of a dictionary item. This property is required and must not exceed 50 characters in length.
/// </summary>
/// <remarks>Ensure that the value provided is unique within its context to avoid conflicts.</remarks>
[Required(ErrorMessage = "字典值不能为空")]
[StringLength(50, ErrorMessage = "值长度不能超过50字符")]
public string Value { get; set; } = null!;
}
// 2.1 更新父级项
[HttpPut("parents/{uuid}")]
public async Task<ActionResult> UpdateParent(
[FromRoute] Guid uuid,
[FromBody] UpdateParentRequest request)
{
try
{
await _dictService.UpdateParentAsync(uuid, request.Label);
return Ok(new ReturnTemplate(200, "创建成功啦!", ""));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(200, "失败", ex));
}
}
/// <summary>
/// Represents the data contract for updating a parent dictionary item.
/// This class is used to pass updated details of a parent dictionary item through HTTP requests.
/// </summary>
public class UpdateParentRequest
{
/// <summary>
/// Represents the label of a dictionary item. This property is used to uniquely identify or name the item within its context.
/// It is required and must not be null, with a maximum length of 100 characters. The label plays a crucial role in both creating and updating dictionary items, serving as a key identifier for operations performed on the dictionary structure.
/// </summary>
[Required(ErrorMessage = "字典标签不能为空")]
[StringLength(100, ErrorMessage = "标签长度不能超过100字符")]
public string Label { get; set; } = null!;
}
// 3. 删除父级项
/// <summary>
/// Deletes a parent dictionary entry by its unique identifier.
/// </summary>
/// <param name="uuid">The unique identifier (UUID) of the parent dictionary entry to be deleted.</param>
/// <returns>An action result indicating the success or failure of the deletion operation. On success, returns a 200 status code with a confirmation message. In case of an exception, it returns a 200 status code with an error message and exception details.</returns>
[HttpDelete("parents/{uuid}")]
public async Task<ActionResult> DeleteParent([FromRoute] Guid uuid)
{
try
{
await _dictService.DeleteParentAsync(uuid);
return Ok(new ReturnTemplate(200, "删除成功啦!", ""));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(200, "失败", ex));
}
}
// 4. 创建子项
/// <summary>
/// 创建一个新的子项。
/// </summary>
/// <param name="request">包含创建子项所需信息的请求对象。</param>
/// <returns>如果创建成功,则返回一个包含成功消息和新创建子项数据的结果;如果失败,则返回一个错误消息。</returns>
[HttpPost("children")]
public async Task<ActionResult> CreateChild([FromBody] CreateChildRequest request)
{
try
{
var result = await _dictService.CreateChildAsync(
request.ParentId,
request.ParentValue,
request.Label,
request.Value,
request.LabelEn,
request.Remark,
request.Tag);
return Ok(new ReturnTemplate(200, "创建成功啦!",""));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(500, "我们遇到了一些问题,无法创建!", ex.ToString()));
}
}
/// <summary>
/// Represents the request model used for creating a child item in the dictionary.
/// This model contains the necessary properties to define a new child entry, including
/// its relationship to a parent, and additional metadata such as labels and values.
/// </summary>
public class CreateChildRequest
{
/// <summary>
/// Represents the unique identifier of the parent dictionary item to which a child dictionary item is associated.
/// This property is required and must be provided when creating or updating a child dictionary item.
/// </summary>
/// <remarks>
/// The value of this property should correspond to an existing dictionary item that is not a child itself (i.e., it does not have its own ParentId).
/// </remarks>
[Required(ErrorMessage = "父级ID不能为空")]
public Guid ParentId { get; set; }
[Required(ErrorMessage = "父级值不能为空")] public string ParentValue { get; set; } = null!;
/// <summary>
/// Represents the label of a dictionary item. This property is required and must not be empty, as it uniquely identifies the dictionary entry within its parent context. It is used in both creation and update operations for dictionary items.
/// </summary>
/// <remarks>
/// The label should be descriptive and meaningful to easily identify the purpose or content of the dictionary item. It plays a crucial role in the user interface and backend logic, serving as a key identifier for related operations.
/// </remarks>
[Required(ErrorMessage = "字典标签不能为空")]
public string Label { get; set; } = null!;
[Required(ErrorMessage = "字典值不能为空")] public string Value { get; set; } = null!;
/// <summary>
/// Represents the English label for a dictionary item. This property is optional and can be used to store an English translation or equivalent of the main label.
/// </summary>
/// <remarks>
/// This property is part of the request model for creating or updating a child dictionary item. It allows for the inclusion of an English label, enhancing multilingual support within the application.
/// </remarks>
public string? LabelEn { get; set; }
public string? Remark { get; set; }
/// <summary>
/// Represents an optional tag or identifier for a dictionary item. This property can be used to categorize or label the item in a way that is meaningful within the context of its usage, such as for filtering or searching purposes.
/// </summary>
public string? Tag { get; set; }
}
// 4.1 更新子项
/// <summary>
/// Updates the details of a child dictionary item.
/// </summary>
/// <param name="uuid">The unique identifier of the child dictionary item to update.</param>
/// <param name="request">The request object containing the new values for the child dictionary item.</param>
/// <returns>An action result indicating the success or failure of the operation, including a message and an optional data payload.</returns>
[HttpPut("children/{uuid}")]
public async Task<ActionResult> UpdateChild(
[FromRoute] Guid uuid,
[FromBody] UpdateChildRequest request)
{
try
{
await _dictService.UpdateChildAsync(
uuid,
request.Label,
request.Value,
request.LabelEn,
request.Remark,
request.Tag);
return Ok(new ReturnTemplate(200, "更新成功啦!", ""));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(200, "失败!", ""));
}
}
/// <summary>
/// Represents the request model used for updating a child item in the dictionary.
/// This class is utilized by the API endpoint responsible for modifying an existing child entry,
/// allowing changes to its label, value, and optional attributes such as English label, remarks, and tags.
/// </summary>
public class UpdateChildRequest
{
/// <summary>
/// Represents the label of a dictionary item. This property is required and must not be empty, as it uniquely identifies the dictionary item in a human-readable format.
/// </summary>
/// <remarks>
/// The Label is used to provide a meaningful name for the dictionary item that can be displayed in user interfaces or referenced in other parts of the application. It is crucial for the identification and management of dictionary items within the system.
/// </remarks>
[Required(ErrorMessage = "字典标签不能为空")]
public string Label { get; set; } = null!;
/// <summary>
/// Represents the value of a dictionary item. This property is required and must be provided when updating or creating a dictionary child item.
/// </summary>
/// <remarks>
/// The value is a unique identifier for the dictionary item within its parent context. It should be meaningful and reflect the nature of the item it represents.
/// </remarks>
[Required(ErrorMessage = "字典值不能为空")]
public string Value { get; set; } = null!;
/// <summary>
/// Represents the English label for a dictionary item. This property can be used to store an English translation or alternative name for the dictionary entry, facilitating internationalization and localization efforts.
/// </summary>
/// <remarks>
/// This is an optional field; if not provided, it defaults to null, indicating no specific English label is set for the dictionary item.
/// </remarks>
public string? LabelEn { get; set; }
/// <summary>
/// Gets or sets the remark for the dictionary item. This property is optional and can be used to store additional information or comments about the dictionary item.
/// </summary>
public string? Remark { get; set; }
/// <summary>
/// Represents an optional tag associated with a dictionary item. This property can be used to categorize or provide additional metadata about the dictionary item.
/// </summary>
/// <remarks>
/// The Tag is not required and can be null. It serves as a flexible way to add extra information that can be used for filtering, sorting, or any other purpose as needed by the application.
/// </remarks>
public string? Tag { get; set; }
}
// 5. 删除子项
/// <summary>
/// Deletes a child item from the dictionary based on the provided unique identifier.
/// </summary>
/// <param name="uuid">The unique identifier (UUID) of the child item to be deleted.</param>
/// <returns>An action result indicating the success or failure of the deletion operation, encapsulated in a ReturnTemplate object which includes a status code, message, and additional data if any.</returns>
[HttpDelete("children/{uuid}")]
public async Task<ActionResult> DeleteChild([FromRoute] Guid uuid)
{
try
{
await _dictService.DeleteChildAsync(uuid);
return Ok(new ReturnTemplate(200, "删除成功啦!", ""));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(200, "删除失败!", ""));
}
}
// 6. 获取子项
[HttpGet("children")]
public async Task<ActionResult> GetChildren(
[FromQuery, Required] string value,
[FromQuery] string? tag = null)
{
try
{
var result = await _dictService.GetChildrenAsync(value, tag);
return Ok(new ReturnTemplate(200, "查询成功啦!", result));
}
catch (Exception ex)
{
return Ok(new ReturnTemplate(200, "查询失败!", ""));
}
}
}
// 统一响应模型
/// <summary>
/// Represents a response object that is returned by API endpoints.
/// This class encapsulates the status code, message, and optionally data to provide a consistent structure for API responses.
/// </summary>
public class ApiResponse
{
/// <summary>
/// Represents the status code of the API response. This property is used to indicate the outcome of an operation, such as success, error, or a specific HTTP status.
/// </summary>
/// <remarks>
/// The value of this property should be set based on the result of the operation being performed. It can be used by the client to determine the next steps or to handle the response appropriately.
/// </remarks>
public int Code { get; set; }
/// <summary>
/// Represents a message or description associated with the API response. This property is used to provide additional information about the result of an operation, such as success messages, error details, or any other relevant information that needs to be communicated back to the client.
/// </summary>
public string Message { get; set; } = null!;
}
/// <summary>
/// Represents a response from the API, containing a status code and message.
/// This class is used to provide a uniform structure for all API responses,
/// ensuring that clients can consistently handle and interpret the results of their requests.
/// </summary>
public class ApiResponse<T> : ApiResponse
{
/// <summary>
/// Represents the data payload in the ApiResponse, containing the result of an API operation.
/// This property can hold any type of object that is being returned as part of the response from the server,
/// such as a list of items, a single item, or a custom object. The actual type of the data will be determined
/// by the generic type parameter T used with ApiResponse.
/// </summary>
/// <typeparam name="T">The type of the data being carried in the response.</typeparam>
public T? Data { get; set; }
}
}

View File

@ -0,0 +1,78 @@
using AGSS.Models.DTOs;
using AGSS.Models.Template;
using AGSS.Services;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
namespace AGSS.Controllers.Menu;
[ApiController]
[Route("api/[controller]")]
public class MenuController : ControllerBase
{
private readonly MenuService _menuService;
public MenuController(MenuService menuService)
{
_menuService = menuService;
}
/// <summary>
/// 新增父级菜单
/// </summary>
[HttpPost("createParent")]
[Authorize(Roles = "Admin")]
public async Task<ReturnTemplate> CreateParentMenu([FromBody] MenuInitialRequest request)
{
return await _menuService.CreateParentMenu(request);
}
/// <summary>
/// 编辑父级菜单
/// </summary>
[Authorize(Roles = "Admin")]
[HttpPut("updateParent")]
public async Task<ReturnTemplate> UpdateParentMenu([FromBody] MenuUpdateRequest request)
{
return await _menuService.UpdateParentMenu(request);
}
/// <summary>
/// 新增子级菜单
/// </summary>
[HttpPost("createChild")]
[Authorize(Roles = "Admin")]
public async Task<ReturnTemplate> CreateChildMenu([FromBody] MenuRequest request)
{
return await _menuService.CreateChildMenu(request);
}
/// <summary>
/// 编辑子级菜单
/// </summary>
[HttpPut("updateChild")]
[Authorize(Roles = "Admin")]
public async Task<ReturnTemplate> UpdateChildMenu([FromBody] MenuUpdateRequestSon request)
{
return await _menuService.UpdateChildMenu(request);
}
/// <summary>
/// 查询菜单全量返回
/// </summary>
[HttpGet("all")]
public async Task<ReturnTemplate> GetAllMenus()
{
return await _menuService.GetAllMenus();
}
/// <summary>
/// 删除菜单
/// </summary>
[HttpDelete("delete/{uuid}")]
[Authorize(Roles = "Admin")]
public async Task<ReturnTemplate> DeleteMenu(string uuid)
{
return await _menuService.DeleteMenu(uuid);
}
}

View File

@ -2,6 +2,7 @@ using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims; using System.Security.Claims;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using AGSS.Models.Template; using AGSS.Models.Template;
using AGSS.Services;
using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
@ -13,7 +14,7 @@ namespace AGSS.Controllers.User;
/// </summary> /// </summary>
/// <remarks>此控制器需要授权才能访问其方法。</remarks> /// <remarks>此控制器需要授权才能访问其方法。</remarks>
[Authorize] [Authorize]
[Route("api/v1/[controller]/[action]")] [Route("api/v1/[controller]")]
public class UserControllers:ControllerBase public class UserControllers:ControllerBase
{ {
/// <summary> /// <summary>
@ -29,31 +30,30 @@ public class UserControllers:ControllerBase
public UserControllers(UserService userService, UserManager<UserModel> userManager) public UserControllers(UserService userService, UserManager<UserModel> userManager)
{ {
_userService = userService; _userService = userService;
} }
/// <summary> /// <summary>
/// 获取当前登录用户的个人信息。 /// 获取当前登录用户的个人信息。
/// </summary> /// </summary>
/// <returns>返回一个包含状态码、消息和用户信息的ReturnTemplate对象。如果成功状态码为200如果失败状态码为500。</returns> /// <returns>返回一个包含状态码、消息和用户信息的ReturnTemplate对象。如果成功状态码为200如果失败状态码为500。</returns>
[HttpGet] [HttpGet("my")]
public async Task<IActionResult> My() public async Task<IActionResult> My()
{ {
string userId = this.User.FindFirst(JwtRegisteredClaimNames.Sub)!.Value;
if (string.IsNullOrEmpty(userId))
{
return Ok(new ReturnTemplate(500,"获取用户失败JWT解析错误",null));
} string userId = this.User.FindFirst(ClaimTypes.NameIdentifier)!.Value;
try try
{ {
var userProfile = await _userService.GetUserProfileAsync(userId); var userProfile = await _userService.GetUserProfileAsync(userId);
return Ok(new ReturnTemplate(200,"获取成功!",userProfile)); return Ok(new ReturnTemplate(200, "获取成功!", userProfile));
} }
catch (ArgumentException ex) catch (ArgumentException ex)
{ {
return NotFound(ex.Message); return NotFound(ex.Message);
} }
} }
} }

View File

@ -4,21 +4,67 @@ using Microsoft.EntityFrameworkCore;
namespace AGSS.DbSet namespace AGSS.DbSet
{ {
public class ApplicationDbContext : IdentityDbContext<UserModel,RoleModel,string> public class ApplicationDbContext : IdentityDbContext<UserModel, RoleModel, string>
{ {
public override DbSet<UserModel> Users { get; set; } public override DbSet<UserModel> Users { get; set; }
public override DbSet<RoleModel> Roles { get; set; } public override DbSet<RoleModel> Roles { get; set; }
public DbSet<DictItem> DictItems { get; set; }
public DbSet<MenuModel> Menus { get; set; }
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
: base(options) : base(options)
{ {
} }
protected override void OnModelCreating(ModelBuilder modelBuilder) protected override void OnModelCreating(ModelBuilder modelBuilder)
{ {
base.OnModelCreating(modelBuilder); base.OnModelCreating(modelBuilder);
modelBuilder.Entity<DictItem>(entity =>
{
entity.HasKey(e => e.Uuid);
entity.Property(e => e.ParentId).HasColumnName("parentid");
entity.Property(e => e.Tag).HasColumnName("tag");
entity.Property(e => e.CreateTime).HasColumnName("createtime");
entity.Property(e => e.ParentValue).HasColumnName("parentvalue");
entity.Property(e => e.Value).HasColumnName("value");
entity.Property(e => e.Label).HasColumnName("label");
entity.Property(e => e.Tag).HasColumnName("tag");
});
// 配置自引用关系
modelBuilder.Entity<DictItem>()
.HasOne(d => d.Parent)
.WithMany(d => d.Children)
.HasForeignKey(d => d.ParentId)
.OnDelete(DeleteBehavior.Cascade); // 级联删除
// 添加索引
modelBuilder.Entity<DictItem>()
.HasIndex(d => d.ParentId);
modelBuilder.Entity<DictItem>()
.HasIndex(d => d.ParentValue);
modelBuilder.Entity<DictItem>()
.HasIndex(d => d.Value);
modelBuilder.Entity<DictItem>()
.HasIndex(d => d.Tag);
// 确保顶级项的约束
modelBuilder.Entity<DictItem>()
.HasCheckConstraint("CK_TopLevelItems",
"ParentId IS NULL OR (Tag IS NOT NULL AND ParentValue IS NOT NULL)");
modelBuilder.Entity<MenuModel>()
.HasKey(m => m.Uuid);
modelBuilder.Entity<MenuModel>()
.HasOne<MenuModel>()
.WithMany()
.HasForeignKey(m => m.ParentId)
.OnDelete(DeleteBehavior.Restrict);
// 在这里添加额外的配置,如果需要的话 // 在这里添加额外的配置,如果需要的话
// 例如: // 例如:

View File

@ -0,0 +1,308 @@
// <auto-generated />
using System;
using AGSS.DbSet;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
#nullable disable
namespace AGSS.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
[Migration("20250709144855_Initial")]
partial class Initial
{
/// <inheritdoc />
protected override void BuildTargetModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "9.0.6")
.HasAnnotation("Relational:MaxIdentifierLength", 63);
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
modelBuilder.Entity("AGSS.Models.Entities.RoleModel", b =>
{
b.Property<string>("Id")
.HasColumnType("text");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("text");
b.Property<string>("Name")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("NormalizedName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.HasKey("Id");
b.HasIndex("NormalizedName")
.IsUnique()
.HasDatabaseName("RoleNameIndex");
b.ToTable("AspNetRoles", (string)null);
});
modelBuilder.Entity("AGSS.Models.Entities.UserModel", b =>
{
b.Property<string>("Id")
.HasColumnType("text");
b.Property<int>("AccessFailedCount")
.HasColumnType("integer");
b.Property<string>("Birthday")
.HasMaxLength(20)
.HasColumnType("character varying(20)");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("text");
b.Property<string>("Config")
.HasMaxLength(200)
.HasColumnType("character varying(200)");
b.Property<string>("Description")
.HasMaxLength(100)
.HasColumnType("character varying(100)");
b.Property<string>("Email")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<bool>("EmailConfirmed")
.HasColumnType("boolean");
b.Property<string>("JobCode")
.HasMaxLength(10)
.HasColumnType("character varying(10)");
b.Property<string>("JobName")
.HasMaxLength(10)
.HasColumnType("character varying(10)");
b.Property<bool>("LockoutEnabled")
.HasColumnType("boolean");
b.Property<DateTimeOffset?>("LockoutEnd")
.HasColumnType("timestamp with time zone");
b.Property<string>("MenuCode")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("MenuName")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("NormalizedEmail")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("NormalizedUserName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("PasswordHash")
.HasColumnType("text");
b.Property<string>("PhoneNumber")
.HasColumnType("text");
b.Property<bool>("PhoneNumberConfirmed")
.HasColumnType("boolean");
b.Property<string>("SecurityStamp")
.HasColumnType("text");
b.Property<string>("Sex")
.HasColumnType("text");
b.Property<bool>("TwoFactorEnabled")
.HasColumnType("boolean");
b.Property<string>("UserName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.HasKey("Id");
b.HasIndex("NormalizedEmail")
.HasDatabaseName("EmailIndex");
b.HasIndex("NormalizedUserName")
.IsUnique()
.HasDatabaseName("UserNameIndex");
b.ToTable("AspNetUsers", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("integer");
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
b.Property<string>("ClaimType")
.HasColumnType("text");
b.Property<string>("ClaimValue")
.HasColumnType("text");
b.Property<string>("RoleId")
.IsRequired()
.HasColumnType("text");
b.HasKey("Id");
b.HasIndex("RoleId");
b.ToTable("AspNetRoleClaims", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("integer");
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
b.Property<string>("ClaimType")
.HasColumnType("text");
b.Property<string>("ClaimValue")
.HasColumnType("text");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("text");
b.HasKey("Id");
b.HasIndex("UserId");
b.ToTable("AspNetUserClaims", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.Property<string>("LoginProvider")
.HasColumnType("text");
b.Property<string>("ProviderKey")
.HasColumnType("text");
b.Property<string>("ProviderDisplayName")
.HasColumnType("text");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("text");
b.HasKey("LoginProvider", "ProviderKey");
b.HasIndex("UserId");
b.ToTable("AspNetUserLogins", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("text");
b.Property<string>("RoleId")
.HasColumnType("text");
b.HasKey("UserId", "RoleId");
b.HasIndex("RoleId");
b.ToTable("AspNetUserRoles", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("text");
b.Property<string>("LoginProvider")
.HasColumnType("text");
b.Property<string>("Name")
.HasColumnType("text");
b.Property<string>("Value")
.HasColumnType("text");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("AGSS.Models.Entities.RoleModel", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("AGSS.Models.Entities.RoleModel", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
#pragma warning restore 612, 618
}
}
}

View File

@ -0,0 +1,231 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
#nullable disable
namespace AGSS.Migrations
{
/// <inheritdoc />
public partial class Initial : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "AspNetRoles",
columns: table => new
{
Id = table.Column<string>(type: "text", nullable: false),
Name = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
NormalizedName = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
ConcurrencyStamp = table.Column<string>(type: "text", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetRoles", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetUsers",
columns: table => new
{
Id = table.Column<string>(type: "text", nullable: false),
Sex = table.Column<string>(type: "text", nullable: true),
Description = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: true),
Config = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: true),
JobCode = table.Column<string>(type: "character varying(10)", maxLength: 10, nullable: true),
JobName = table.Column<string>(type: "character varying(10)", maxLength: 10, nullable: true),
Birthday = table.Column<string>(type: "character varying(20)", maxLength: 20, nullable: true),
MenuCode = table.Column<string>(type: "character varying(500)", maxLength: 500, nullable: true),
MenuName = table.Column<string>(type: "character varying(500)", maxLength: 500, nullable: true),
UserName = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
NormalizedUserName = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
Email = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
NormalizedEmail = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: true),
EmailConfirmed = table.Column<bool>(type: "boolean", nullable: false),
PasswordHash = table.Column<string>(type: "text", nullable: true),
SecurityStamp = table.Column<string>(type: "text", nullable: true),
ConcurrencyStamp = table.Column<string>(type: "text", nullable: true),
PhoneNumber = table.Column<string>(type: "text", nullable: true),
PhoneNumberConfirmed = table.Column<bool>(type: "boolean", nullable: false),
TwoFactorEnabled = table.Column<bool>(type: "boolean", nullable: false),
LockoutEnd = table.Column<DateTimeOffset>(type: "timestamp with time zone", nullable: true),
LockoutEnabled = table.Column<bool>(type: "boolean", nullable: false),
AccessFailedCount = table.Column<int>(type: "integer", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUsers", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetRoleClaims",
columns: table => new
{
Id = table.Column<int>(type: "integer", nullable: false)
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
RoleId = table.Column<string>(type: "text", nullable: false),
ClaimType = table.Column<string>(type: "text", nullable: true),
ClaimValue = table.Column<string>(type: "text", 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>(type: "integer", nullable: false)
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
UserId = table.Column<string>(type: "text", nullable: false),
ClaimType = table.Column<string>(type: "text", nullable: true),
ClaimValue = table.Column<string>(type: "text", 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>(type: "text", nullable: false),
ProviderKey = table.Column<string>(type: "text", nullable: false),
ProviderDisplayName = table.Column<string>(type: "text", nullable: true),
UserId = table.Column<string>(type: "text", 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>(type: "text", nullable: false),
RoleId = table.Column<string>(type: "text", 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>(type: "text", nullable: false),
LoginProvider = table.Column<string>(type: "text", nullable: false),
Name = table.Column<string>(type: "text", nullable: false),
Value = table.Column<string>(type: "text", 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);
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);
}
/// <inheritdoc />
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");
}
}
}

View File

@ -0,0 +1,425 @@
// <auto-generated />
using System;
using AGSS.DbSet;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
#nullable disable
namespace AGSS.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
[Migration("20250714031454_AddMenuTable")]
partial class AddMenuTable
{
/// <inheritdoc />
protected override void BuildTargetModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "9.0.6")
.HasAnnotation("Relational:MaxIdentifierLength", 63);
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
modelBuilder.Entity("AGSS.Models.Entities.DictionaryModel", b =>
{
b.Property<string>("Uuid")
.HasColumnType("text");
b.Property<DateTime>("CreateTime")
.HasColumnType("timestamp with time zone");
b.Property<string>("CreateUserId")
.IsRequired()
.HasColumnType("text");
b.Property<string>("Label")
.IsRequired()
.HasColumnType("text");
b.Property<string>("LabelEn")
.IsRequired()
.HasColumnType("text");
b.Property<string>("ParentId")
.IsRequired()
.HasColumnType("text");
b.Property<string>("ParentValue")
.IsRequired()
.HasColumnType("text");
b.Property<string>("Remark")
.IsRequired()
.HasColumnType("text");
b.Property<string>("Tag")
.IsRequired()
.HasColumnType("text");
b.Property<string>("Value")
.IsRequired()
.HasColumnType("text");
b.HasKey("Uuid");
b.ToTable("Dictionaries");
});
modelBuilder.Entity("AGSS.Models.Entities.MenuModel", b =>
{
b.Property<string>("Uuid")
.HasMaxLength(50)
.HasColumnType("character varying(50)");
b.Property<string>("Adaptability")
.IsRequired()
.HasMaxLength(20)
.HasColumnType("character varying(20)");
b.Property<string>("Component")
.IsRequired()
.HasMaxLength(200)
.HasColumnType("character varying(200)");
b.Property<DateTime>("CreateTime")
.HasColumnType("timestamp with time zone");
b.Property<string>("Icon")
.IsRequired()
.HasMaxLength(100)
.HasColumnType("character varying(100)");
b.Property<string>("Label")
.IsRequired()
.HasMaxLength(100)
.HasColumnType("character varying(100)");
b.Property<string>("MenuCode")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("ParentId")
.HasMaxLength(50)
.HasColumnType("character varying(50)");
b.Property<string>("Path")
.IsRequired()
.HasMaxLength(200)
.HasColumnType("character varying(200)");
b.Property<string>("Query")
.HasMaxLength(1000)
.HasColumnType("character varying(1000)");
b.Property<int>("Sort")
.HasColumnType("integer");
b.Property<string>("Status")
.IsRequired()
.HasMaxLength(1)
.HasColumnType("character varying(1)");
b.Property<DateTime?>("UpdateTime")
.HasColumnType("timestamp with time zone");
b.HasKey("Uuid");
b.HasIndex("ParentId");
b.ToTable("Menus");
});
modelBuilder.Entity("AGSS.Models.Entities.RoleModel", b =>
{
b.Property<string>("Id")
.HasColumnType("text");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("text");
b.Property<string>("Name")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("NormalizedName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.HasKey("Id");
b.HasIndex("NormalizedName")
.IsUnique()
.HasDatabaseName("RoleNameIndex");
b.ToTable("AspNetRoles", (string)null);
});
modelBuilder.Entity("AGSS.Models.Entities.UserModel", b =>
{
b.Property<string>("Id")
.HasColumnType("text");
b.Property<int>("AccessFailedCount")
.HasColumnType("integer");
b.Property<string>("Birthday")
.HasMaxLength(20)
.HasColumnType("character varying(20)");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("text");
b.Property<string>("Config")
.HasMaxLength(200)
.HasColumnType("character varying(200)");
b.Property<string>("Description")
.HasMaxLength(100)
.HasColumnType("character varying(100)");
b.Property<string>("Email")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<bool>("EmailConfirmed")
.HasColumnType("boolean");
b.Property<string>("JobCode")
.HasMaxLength(10)
.HasColumnType("character varying(10)");
b.Property<string>("JobName")
.HasMaxLength(10)
.HasColumnType("character varying(10)");
b.Property<bool>("LockoutEnabled")
.HasColumnType("boolean");
b.Property<DateTimeOffset?>("LockoutEnd")
.HasColumnType("timestamp with time zone");
b.Property<string>("MenuCode")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("MenuName")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("NormalizedEmail")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("NormalizedUserName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.Property<string>("PasswordHash")
.HasColumnType("text");
b.Property<string>("PhoneNumber")
.HasColumnType("text");
b.Property<bool>("PhoneNumberConfirmed")
.HasColumnType("boolean");
b.Property<string>("SecurityStamp")
.HasColumnType("text");
b.Property<string>("Sex")
.HasColumnType("text");
b.Property<bool>("TwoFactorEnabled")
.HasColumnType("boolean");
b.Property<string>("UserName")
.HasMaxLength(256)
.HasColumnType("character varying(256)");
b.HasKey("Id");
b.HasIndex("NormalizedEmail")
.HasDatabaseName("EmailIndex");
b.HasIndex("NormalizedUserName")
.IsUnique()
.HasDatabaseName("UserNameIndex");
b.ToTable("AspNetUsers", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("integer");
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
b.Property<string>("ClaimType")
.HasColumnType("text");
b.Property<string>("ClaimValue")
.HasColumnType("text");
b.Property<string>("RoleId")
.IsRequired()
.HasColumnType("text");
b.HasKey("Id");
b.HasIndex("RoleId");
b.ToTable("AspNetRoleClaims", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("integer");
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
b.Property<string>("ClaimType")
.HasColumnType("text");
b.Property<string>("ClaimValue")
.HasColumnType("text");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("text");
b.HasKey("Id");
b.HasIndex("UserId");
b.ToTable("AspNetUserClaims", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.Property<string>("LoginProvider")
.HasColumnType("text");
b.Property<string>("ProviderKey")
.HasColumnType("text");
b.Property<string>("ProviderDisplayName")
.HasColumnType("text");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("text");
b.HasKey("LoginProvider", "ProviderKey");
b.HasIndex("UserId");
b.ToTable("AspNetUserLogins", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("text");
b.Property<string>("RoleId")
.HasColumnType("text");
b.HasKey("UserId", "RoleId");
b.HasIndex("RoleId");
b.ToTable("AspNetUserRoles", (string)null);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("text");
b.Property<string>("LoginProvider")
.HasColumnType("text");
b.Property<string>("Name")
.HasColumnType("text");
b.Property<string>("Value")
.HasColumnType("text");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens", (string)null);
});
modelBuilder.Entity("AGSS.Models.Entities.MenuModel", b =>
{
b.HasOne("AGSS.Models.Entities.MenuModel", null)
.WithMany()
.HasForeignKey("ParentId")
.OnDelete(DeleteBehavior.Restrict);
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("AGSS.Models.Entities.RoleModel", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("AGSS.Models.Entities.RoleModel", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("AGSS.Models.Entities.UserModel", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
#pragma warning restore 612, 618
}
}
}

View File

@ -0,0 +1,56 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace AGSS.Migrations
{
/// <inheritdoc />
public partial class AddMenuTable : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "Menus",
columns: table => new
{
Uuid = table.Column<string>(type: "character varying(50)", maxLength: 50, nullable: false),
ParentId = table.Column<string>(type: "character varying(50)", maxLength: 50, nullable: true),
Path = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: false),
Label = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: false),
Icon = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: false),
MenuCode = table.Column<string>(type: "character varying(500)", maxLength: 500, nullable: true),
Adaptability = table.Column<string>(type: "character varying(20)", maxLength: 20, nullable: false),
Component = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: false),
Sort = table.Column<int>(type: "integer", nullable: false),
Status = table.Column<string>(type: "character varying(1)", maxLength: 1, nullable: false),
Query = table.Column<string>(type: "character varying(1000)", maxLength: 1000, nullable: true),
CreateTime = table.Column<DateTime>(type: "timestamp with time zone", nullable: false),
UpdateTime = table.Column<DateTime>(type: "timestamp with time zone", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_Menus", x => x.Uuid);
table.ForeignKey(
name: "FK_Menus_Menus_ParentId",
column: x => x.ParentId,
principalTable: "Menus",
principalColumn: "Uuid",
onDelete: ReferentialAction.Restrict);
});
migrationBuilder.CreateIndex(
name: "IX_Menus_ParentId",
table: "Menus",
column: "ParentId");
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "Menus");
}
}
}

View File

@ -22,11 +22,82 @@ namespace AGSS.Migrations
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
modelBuilder.Entity("AGSS.Models.Entities.MenuModel", b =>
{
b.Property<string>("Uuid")
.HasMaxLength(150)
.HasColumnType("character varying(150)");
b.Property<string>("Adaptability")
.IsRequired()
.HasMaxLength(120)
.HasColumnType("character varying(120)");
b.Property<string>("Component")
.IsRequired()
.HasMaxLength(1200)
.HasColumnType("character varying(1200)");
b.Property<DateTime>("CreateTime")
.HasColumnType("timestamp with time zone");
b.Property<string>("Icon")
.IsRequired()
.HasMaxLength(300)
.HasColumnType("character varying(300)");
b.Property<string>("Label")
.IsRequired()
.HasMaxLength(300)
.HasColumnType("character varying(300)");
b.Property<string>("MenuCode")
.HasMaxLength(1000)
.HasColumnType("character varying(1000)");
b.Property<string>("MenuName")
.HasMaxLength(1000)
.HasColumnType("character varying(1000)");
b.Property<string>("ParentId")
.HasMaxLength(150)
.HasColumnType("character varying(150)");
b.Property<string>("Path")
.IsRequired()
.HasMaxLength(300)
.HasColumnType("character varying(300)");
b.Property<string>("Query")
.HasMaxLength(10000)
.HasColumnType("character varying(10000)");
b.Property<int>("Sort")
.HasColumnType("integer");
b.Property<string>("Status")
.IsRequired()
.HasMaxLength(20)
.HasColumnType("character varying(20)");
b.Property<DateTime?>("UpdateTime")
.HasColumnType("timestamp with time zone");
b.HasKey("Uuid");
b.HasIndex("ParentId");
b.ToTable("Menus");
});
modelBuilder.Entity("AGSS.Models.Entities.RoleModel", b => modelBuilder.Entity("AGSS.Models.Entities.RoleModel", b =>
{ {
b.Property<string>("Id") b.Property<string>("Id")
.HasColumnType("text"); .HasColumnType("text");
b.Property<string>("ChineseName")
.HasColumnType("text");
b.Property<string>("ConcurrencyStamp") b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken() .IsConcurrencyToken()
.HasColumnType("text"); .HasColumnType("text");
@ -143,6 +214,68 @@ namespace AGSS.Migrations
b.ToTable("AspNetUsers", (string)null); b.ToTable("AspNetUsers", (string)null);
}); });
modelBuilder.Entity("DictItem", b =>
{
b.Property<Guid>("Uuid")
.ValueGeneratedOnAdd()
.HasColumnType("uuid");
b.Property<DateTime>("CreateTime")
.HasColumnType("timestamp with time zone")
.HasColumnName("createtime");
b.Property<Guid>("CreateUserId")
.HasColumnType("uuid");
b.Property<string>("Label")
.IsRequired()
.HasMaxLength(100)
.HasColumnType("character varying(100)")
.HasColumnName("label");
b.Property<string>("LabelEn")
.HasMaxLength(100)
.HasColumnType("character varying(100)");
b.Property<Guid?>("ParentId")
.HasColumnType("uuid")
.HasColumnName("parentid");
b.Property<string>("ParentValue")
.HasColumnType("text")
.HasColumnName("parentvalue");
b.Property<string>("Remark")
.HasMaxLength(500)
.HasColumnType("character varying(500)");
b.Property<string>("Tag")
.HasMaxLength(500)
.HasColumnType("character varying(500)")
.HasColumnName("tag");
b.Property<string>("Value")
.IsRequired()
.HasMaxLength(50)
.HasColumnType("character varying(50)")
.HasColumnName("value");
b.HasKey("Uuid");
b.HasIndex("ParentId");
b.HasIndex("ParentValue");
b.HasIndex("Tag");
b.HasIndex("Value");
b.ToTable("DictItems", t =>
{
t.HasCheckConstraint("CK_TopLevelItems", "ParentId IS NULL OR (Tag IS NOT NULL AND ParentValue IS NOT NULL)");
});
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{ {
b.Property<int>("Id") b.Property<int>("Id")
@ -249,6 +382,24 @@ namespace AGSS.Migrations
b.ToTable("AspNetUserTokens", (string)null); b.ToTable("AspNetUserTokens", (string)null);
}); });
modelBuilder.Entity("AGSS.Models.Entities.MenuModel", b =>
{
b.HasOne("AGSS.Models.Entities.MenuModel", null)
.WithMany()
.HasForeignKey("ParentId")
.OnDelete(DeleteBehavior.Restrict);
});
modelBuilder.Entity("DictItem", b =>
{
b.HasOne("DictItem", "Parent")
.WithMany("Children")
.HasForeignKey("ParentId")
.OnDelete(DeleteBehavior.Cascade);
b.Navigation("Parent");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b => modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{ {
b.HasOne("AGSS.Models.Entities.RoleModel", null) b.HasOne("AGSS.Models.Entities.RoleModel", null)
@ -299,6 +450,11 @@ namespace AGSS.Migrations
.OnDelete(DeleteBehavior.Cascade) .OnDelete(DeleteBehavior.Cascade)
.IsRequired(); .IsRequired();
}); });
modelBuilder.Entity("DictItem", b =>
{
b.Navigation("Children");
});
#pragma warning restore 612, 618 #pragma warning restore 612, 618
} }
} }

View File

@ -1,14 +0,0 @@
using AGSS.Models.Entities;
using Microsoft.EntityFrameworkCore;
namespace AGSS.Models;
public class DBContext : DbContext
{
public DBContext(DbContextOptions<DBContext> options)
: base(options)
{
}
}

View File

@ -0,0 +1,16 @@
using System;
using System.Collections.Generic;
namespace AGSS.Models.DTOs
{
public class DictionaryDto
{
public string Uuid { get; set; }
public string ParentId { get; set; }
public string ParentValue { get; set; }
public string Label { get; set; }
public string Remark { get; set; }
public string Value { get; set; }
public string Tag { get; set; }
}
}

View File

@ -0,0 +1,64 @@
namespace AGSS.Models.DTOs;
public class MenuInitialRequest
{
public string Path { get; set; } = string.Empty;
public string Label { get; set; } = string.Empty;
public string? MenuName { get; set; }
public string Icon { get; set; } = string.Empty;
public string? MenuCode { get; set; }
public string Adaptability { get; set; } = "pc";
public string Component { get; set; } = string.Empty;
public int Sort { get; set; }
public string Status { get; set; } = "1";
public string? Query { get; set; }
}
public class MenuRequest
{
public string? ParentId { get; set; }
public string Path { get; set; } = string.Empty;
public string Label { get; set; } = string.Empty;
public string? MenuName { get; set; }
public string Icon { get; set; } = string.Empty;
public string? MenuCode { get; set; }
public string Adaptability { get; set; } = "pc";
public string Component { get; set; } = string.Empty;
public int Sort { get; set; }
public string Status { get; set; } = "1";
public string? Query { get; set; }
}
public class MenuUpdateRequest : MenuInitialRequest
{
public string Uuid { get; set; } = string.Empty;
}
public class MenuUpdateRequestSon : MenuRequest
{
public string Uuid { get; set; } = string.Empty;
}
public class MenuResponse
{
public string Uuid { get; set; } = string.Empty;
public string? ParentId { get; set; }
public string Path { get; set; } = string.Empty;
public string Label { get; set; } = string.Empty;
public string? MenuName { get; set; }
public string Icon { get; set; } = string.Empty;
public string? MenuCode { get; set; }
public string Adaptability { get; set; } = string.Empty;
public string Component { get; set; } = string.Empty;
public int Sort { get; set; }
public string Status { get; set; } = string.Empty;
public string? Query { get; set; }
public DateTime CreateTime { get; set; }
public DateTime? UpdateTime { get; set; }
public List<MenuResponse> Children { get; set; } = new();
}
public class DeleteMenuRequest
{
public string Uuid { get; set; } = string.Empty;
}

View File

@ -1,6 +1,6 @@
namespace AGSS.Models.DTOs; namespace AGSS.Models.DTOs;
public struct MenuRequest public struct UserMenuRequest
{ {
public string Id { get; set; } public string Id { get; set; }
public string MenuName { get; set; } public string MenuName { get; set; }

View File

@ -11,4 +11,6 @@ public class UserProfile
public string? JobCode { get; set; } public string? JobCode { get; set; }
public string? JobName { get; set; } public string? JobName { get; set; }
public string? Birthday { get; set; } public string? Birthday { get; set; }
public string? MenuCode { get; set; }
public string? MenuName { get; set; }
} }

View File

@ -0,0 +1,37 @@
using System.ComponentModel.DataAnnotations;
public class DictItem
{
// 主键
public Guid Uuid { get; set; } = Guid.NewGuid();
// 层级关系
public Guid? ParentId { get; set; }
public string? ParentValue { get; set; }
// 基础信息
[Required]
[StringLength(100)]
public string Label { get; set; } = null!; // 中文标签
[StringLength(100)]
public string? LabelEn { get; set; } // 英文标签
[StringLength(500)]
public string? Remark { get; set; } // 备注
[Required]
[StringLength(50)]
public string Value { get; set; } = null!; // 字典值
[StringLength(500)]
public string? Tag { get; set; } // 逗号分隔的标签
// 审计字段
public DateTime CreateTime { get; set; } = DateTime.UtcNow;
public Guid CreateUserId { get; set; }
// 导航属性
public DictItem? Parent { get; set; }
public ICollection<DictItem> Children { get; set; } = new List<DictItem>();
}

View File

@ -0,0 +1,9 @@
namespace asg_form
{
public class JWTOptions
{
public string SigningKey { get; set; }
public int ExpireSeconds { get; set; }
}
}

View File

@ -0,0 +1,46 @@
using System.ComponentModel.DataAnnotations;
namespace AGSS.Models.Entities;
public class MenuModel
{
[Key]
[MaxLength(150)]
public string Uuid { get; set; } = string.Empty;
[MaxLength(150)]
public string? ParentId { get; set; }
[MaxLength(300)]
public string Path { get; set; } = string.Empty;
[MaxLength(300)]
public string Label { get; set; } = string.Empty;
[MaxLength(300)]
public string Icon { get; set; } = string.Empty;
[MaxLength(1000)]
public string? MenuName { get; set; }
[MaxLength(1000)]
public string? MenuCode { get; set; }
[MaxLength(120)]
public string Adaptability { get; set; } = "pc";
[MaxLength(1200)]
public string Component { get; set; } = string.Empty;
public int Sort { get; set; }
[MaxLength(20)]
public string Status { get; set; } = "1";
[MaxLength(10000)]
public string? Query { get; set; }
public DateTime CreateTime { get; set; } = DateTime.Now;
public DateTime? UpdateTime { get; set; }
}

View File

@ -5,8 +5,7 @@ namespace AGSS.Models.Entities;
public class UserModel:IdentityUser<string> public class UserModel:IdentityUser<string>
{ {
public string? Sex { get; set; }
public string? Sex { get; set; }
[MaxLength(100)] [MaxLength(100)]
public string? Description { get; set; } public string? Description { get; set; }
[MaxLength(200)] [MaxLength(200)]
@ -26,5 +25,6 @@ public class UserModel:IdentityUser<string>
public class RoleModel : IdentityRole<string> public class RoleModel : IdentityRole<string>
{ {
public string? ChineseName { get; set; }
} }

View File

@ -1,10 +1,13 @@
using System.Reflection; using System.Reflection;
using System.Security.Claims;
using System.Text; using System.Text;
using AGSS.DbSet; using AGSS.DbSet;
using AGSS.Models; using AGSS.Models;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using AGSS.Models.Template; using AGSS.Models.Template;
using AGSS.Services;
using AGSS.Utilities; using AGSS.Utilities;
using asg_form;
using Microsoft.AspNetCore.Authentication.JwtBearer; using Microsoft.AspNetCore.Authentication.JwtBearer;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
@ -33,49 +36,67 @@ builder.Services.AddDbContext<ApplicationDbContext>(opt =>
opt.UseNpgsql(builder.Configuration.GetConnectionString("DBContext"))); opt.UseNpgsql(builder.Configuration.GetConnectionString("DBContext")));
// Identity 配置 // Identity 配置
builder.Services.AddIdentity<UserModel, RoleModel>() builder.Services.AddIdentityCore<UserModel>(options =>
.AddEntityFrameworkStores<ApplicationDbContext>() {
.AddDefaultTokenProviders() options.Password.RequireDigit = false;
.AddDefaultUI(); options.Password.RequireLowercase = false;
options.Password.RequireNonAlphanumeric = false;
options.Password.RequireUppercase = false;
options.Password.RequiredLength = 6;
options.Tokens.PasswordResetTokenProvider = TokenOptions.DefaultEmailProvider;
options.Tokens.EmailConfirmationTokenProvider = TokenOptions.DefaultEmailProvider;
})
.AddRoles<RoleModel>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultUI()
;
// 注册 UserService // 注册 UserService
builder.Services.AddScoped<UserService>(); // builder.Services.AddScoped<UserService>();
builder.Services.AddScoped<Jwt>(); builder.Services.AddScoped<Jwt>();
builder.Services.AddScoped<UserService>();
builder.Services.AddScoped<MenuService>();
builder.Services.AddScoped<DictService>();
builder.Services.AddScoped<ICurrentUserService, CurrentUserService>();
builder.Services.AddAuthentication(options =>
builder.Services.Configure<JwtBearerOptions>(options =>
{
options.TokenValidationParameters.RoleClaimType = ClaimTypes.Role;
});
//builder.Services.AddHealthChecks().AddNpgSql(builder.Configuration.GetConnectionString("DBContext"));
//builder.Services.AddHealthChecksUI();
builder.Services.Configure<JWTOptions>(builder.Configuration.GetSection("JWT"));
builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
.AddJwtBearer(x =>
{ {
options.DefaultScheme = JwtBearerDefaults.AuthenticationScheme; var jwtOpt = builder.Configuration.GetSection("JWT").Get<JWTOptions>();
options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme; byte[] keyBytes = Encoding.UTF8.GetBytes(jwtOpt.SigningKey);
}) var secKey = new SymmetricSecurityKey(keyBytes);
.AddJwtBearer(options => x.TokenValidationParameters = new()
{
options.TokenValidationParameters = new TokenValidationParameters
{ {
ValidateIssuer = true, ValidateIssuer = false,
ValidateAudience = true, ValidateAudience = false,
ValidateLifetime = true, ValidateLifetime = true,
ValidateIssuerSigningKey = true, ValidateIssuerSigningKey = true,
ValidIssuer = builder.Configuration["Jwt:Issuer"], IssuerSigningKey = secKey
ValidAudience = builder.Configuration["Jwt:Audience"],
IssuerSigningKey = new SymmetricSecurityKey(
Encoding.UTF8.GetBytes(builder.Configuration["Jwt:Key"]))
}; };
options.Events = new JwtBearerEvents x.Events = new JwtBearerEvents
{ {
OnChallenge = context => OnChallenge = context =>
{ {
context.HandleResponse(); context.HandleResponse();
context.Response.StatusCode = 200; context.Response.StatusCode = 200;
context.Response.ContentType = "application/json"; context.Response.ContentType = "application/json";
return context.Response.WriteAsJsonAsync(new ReturnTemplate(401, "你提供了一个错误的Token所以我们无法验证你的身份唔......", null)); return context.Response.WriteAsJsonAsync(new ReturnTemplate(401,"你提供了一个错误的Token所以我们无法验证你的身份唔......",null));
} }
}; };
}).AddMicrosoftAccount(microsoftOptions => })
{ .AddCookie("Identity.External").AddCookie("Identity.Application");
microsoftOptions.ClientId = configuration["Authentication:Microsoft:ClientId"];
microsoftOptions.ClientSecret = configuration["Authentication:Microsoft:ClientSecret"];
});

View File

@ -0,0 +1,25 @@
// 用于获取当前用户信息
using System.Security.Claims;
public interface ICurrentUserService
{
Guid UserId { get; }
bool IsAdmin { get; }
}
public class CurrentUserService : ICurrentUserService
{
private readonly IHttpContextAccessor _httpContextAccessor;
public CurrentUserService(IHttpContextAccessor httpContextAccessor)
{
_httpContextAccessor = httpContextAccessor;
}
public Guid UserId =>
Guid.Parse(_httpContextAccessor.HttpContext?.User?.FindFirst(ClaimTypes.NameIdentifier)?.Value);
public bool IsAdmin =>
_httpContextAccessor.HttpContext?.User?.IsInRole("Admin") ?? false;
}

View File

@ -0,0 +1,175 @@
using AGSS.DbSet;
using Microsoft.EntityFrameworkCore;
public class DictService
{
private readonly ApplicationDbContext _context;
public DictService(ApplicationDbContext context, ICurrentUserService currentUserService)
{
_context = context;
_currentUserService = currentUserService;
}
private readonly ICurrentUserService _currentUserService;
// 1. 获取父级列表
public async Task<List<DictItem>> GetParentsAsync(string? label)
{
var query = _context.DictItems
.Where(d => d.ParentId == null);
if (!string.IsNullOrWhiteSpace(label))
{
query = query.Where(d => d.Label.Contains(label));
}
return await query.ToListAsync();
}
// 2. 创建父级项
public async Task<DictItem> CreateParentAsync(string label, string value)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var newItem = new DictItem
{
Label = label,
Value = value,
CreateUserId = _currentUserService.UserId
};
_context.DictItems.Add(newItem);
await _context.SaveChangesAsync();
return newItem;
}
// 2.1 更新父级项
public async Task UpdateParentAsync(Guid uuid, string label)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var item = await _context.DictItems
.FirstOrDefaultAsync(d => d.Uuid == uuid && d.ParentId == null);
if (item == null) throw new KeyNotFoundException("字典项不存在");
item.Label = label;
await _context.SaveChangesAsync();
}
// 3. 删除父级项(级联删除)
public async Task DeleteParentAsync(Guid uuid)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var item = await _context.DictItems
.Include(d => d.Children)
.FirstOrDefaultAsync(d => d.Uuid == uuid && d.ParentId == null);
if (item == null) throw new KeyNotFoundException("字典项不存在");
_context.DictItems.Remove(item);
await _context.SaveChangesAsync();
}
// 4. 创建子项
public async Task<DictItem> CreateChildAsync(
Guid parentId,
string parentValue,
string label,
string value,
string? labelEn = null,
string? remark = null,
string? tag = null)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var parent = await _context.DictItems
.FirstOrDefaultAsync(d => d.Uuid == parentId && d.ParentId == null);
if (parent == null) throw new KeyNotFoundException("父级字典项不存在");
var newItem = new DictItem
{
ParentId = parentId,
ParentValue = parentValue,
Label = label,
Value = value,
LabelEn = labelEn,
Remark = remark,
Tag = tag,
CreateUserId = _currentUserService.UserId
};
_context.DictItems.Add(newItem);
await _context.SaveChangesAsync();
return newItem;
}
// 4.1 更新子项
public async Task UpdateChildAsync(
Guid uuid,
string label,
string value,
string? labelEn = null,
string? remark = null,
string? tag = null)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var item = await _context.DictItems
.FirstOrDefaultAsync(d => d.Uuid == uuid && d.ParentId != null);
if (item == null) throw new KeyNotFoundException("字典项不存在");
item.Label = label;
item.Value = value;
item.LabelEn = labelEn;
item.Remark = remark;
item.Tag = tag;
await _context.SaveChangesAsync();
}
// 5. 删除子项
public async Task DeleteChildAsync(Guid uuid)
{
if (!_currentUserService.IsAdmin)
throw new UnauthorizedAccessException("您不是管理员,无权限");
var item = await _context.DictItems
.FirstOrDefaultAsync(d => d.Uuid == uuid && d.ParentId != null);
if (item == null) throw new KeyNotFoundException("字典项不存在");
_context.DictItems.Remove(item);
await _context.SaveChangesAsync();
}
// 6. 获取子项(带标签过滤)
public async Task<List<DictItem>> GetChildrenAsync(string value, string? tag = null)
{
var query = _context.DictItems
.Where(d => d.ParentValue == value);
if (!string.IsNullOrWhiteSpace(tag))
{
// 精确标签匹配逻辑
var tags = tag.Split(',', StringSplitOptions.RemoveEmptyEntries)
.Select(t => t.Trim())
.ToList();
query = query.Where(d => d.Tag != null &&
tags.All(t => d.Tag.Contains(t)));
}
return await query.ToListAsync();
}
}

View File

@ -0,0 +1,320 @@
using AGSS.DbSet;
using AGSS.Models.DTOs;
using AGSS.Models.Entities;
using AGSS.Models.Template;
using Microsoft.EntityFrameworkCore;
namespace AGSS.Services;
public class MenuService
{
private readonly ApplicationDbContext _context;
public MenuService(ApplicationDbContext context)
{
_context = context;
}
/// <summary>
/// 新增父级菜单
/// </summary>
public async Task<ReturnTemplate> CreateParentMenu(MenuInitialRequest request)
{
try
{
var menu = new MenuModel
{
Uuid = Guid.NewGuid().ToString(),
ParentId = null,
Path = request.Path,
Label = request.Label,
MenuName = request.MenuName,
Icon = request.Icon,
MenuCode = request.MenuCode,
Adaptability = request.Adaptability,
Component = request.Component,
Sort = request.Sort,
Status = request.Status,
Query = request.Query,
CreateTime = DateTime.UtcNow
};
_context.Menus.Add(menu);
await _context.SaveChangesAsync();
return new ReturnTemplate(200, "父级菜单创建成功", menu);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"创建父级菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 编辑父级菜单
/// </summary>
public async Task<ReturnTemplate> UpdateParentMenu(MenuUpdateRequest request)
{
try
{
if (string.IsNullOrEmpty(request.Uuid))
{
return new ReturnTemplate(400, "UUID不能为空", null);
}
var menu = await _context.Menus.FirstOrDefaultAsync(m => m.Uuid == request.Uuid);
if (menu == null)
{
return new ReturnTemplate(404, "菜单不存在", null);
}
// 确保是父级菜单
if (!string.IsNullOrEmpty(menu.ParentId))
{
return new ReturnTemplate(400, "只能编辑父级菜单", null);
}
menu.Path = request.Path;
menu.Label = request.Label;
menu.MenuName = request.MenuName;
menu.Icon = request.Icon;
menu.MenuCode = request.MenuCode;
menu.Adaptability = request.Adaptability;
menu.Component = request.Component;
menu.Sort = request.Sort;
menu.Status = request.Status;
menu.Query = request.Query;
menu.UpdateTime = DateTime.UtcNow;
await _context.SaveChangesAsync();
return new ReturnTemplate(200, "父级菜单更新成功", menu);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"更新父级菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 新增子级菜单
/// </summary>
public async Task<ReturnTemplate> CreateChildMenu(MenuRequest request)
{
try
{
if (string.IsNullOrEmpty(request.ParentId))
{
return new ReturnTemplate(400, "父级ID不能为空", null);
}
// 验证父级菜单是否存在
var parentMenu = await _context.Menus.FirstOrDefaultAsync(m => m.Uuid == request.ParentId);
if (parentMenu == null)
{
return new ReturnTemplate(404, "父级菜单不存在", null);
}
var menu = new MenuModel
{
Uuid = Guid.NewGuid().ToString(),
ParentId = request.ParentId,
Path = request.Path,
Label = request.Label,
MenuName = request.MenuName,
Icon = request.Icon,
MenuCode = request.MenuCode,
Adaptability = request.Adaptability,
Component = request.Component,
Sort = request.Sort,
Status = request.Status,
Query = request.Query,
CreateTime = DateTime.UtcNow
};
_context.Menus.Add(menu);
await _context.SaveChangesAsync();
return new ReturnTemplate(200, "子级菜单创建成功", menu);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"创建子级菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 编辑子级菜单
/// </summary>
public async Task<ReturnTemplate> UpdateChildMenu(MenuUpdateRequestSon request)
{
try
{
if (string.IsNullOrEmpty(request.Uuid))
{
return new ReturnTemplate(400, "UUID不能为空", null);
}
var menu = await _context.Menus.FirstOrDefaultAsync(m => m.Uuid == request.Uuid);
if (menu == null)
{
return new ReturnTemplate(404, "菜单不存在", null);
}
// 确保是子级菜单
if (string.IsNullOrEmpty(menu.ParentId))
{
return new ReturnTemplate(400, "只能编辑子级菜单", null);
}
menu.ParentId = request.ParentId;
menu.Path = request.Path;
menu.Label = request.Label;
menu.MenuName = request.MenuName;
menu.Icon = request.Icon;
menu.MenuCode = request.MenuCode;
menu.Adaptability = request.Adaptability;
menu.Component = request.Component;
menu.Sort = request.Sort;
menu.Status = request.Status;
menu.Query = request.Query;
menu.UpdateTime = DateTime.UtcNow;
await _context.SaveChangesAsync();
return new ReturnTemplate(200, "子级菜单更新成功", menu);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"更新子级菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 查询菜单全量返回(树形结构)
/// </summary>
public async Task<ReturnTemplate> GetAllMenus()
{
try
{
var allMenus = await _context.Menus
.OrderBy(m => m.Sort)
.ToListAsync();
var menuTree = BuildMenuTree(allMenus);
return new ReturnTemplate(200, "成功", menuTree);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"查询菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 删除菜单(递归删除)
/// </summary>
public async Task<ReturnTemplate> DeleteMenu(string uuid)
{
try
{
if (string.IsNullOrEmpty(uuid))
{
return new ReturnTemplate(400, "UUID不能为空", null);
}
var menuToDelete = await _context.Menus.FirstOrDefaultAsync(m => m.Uuid == uuid);
if (menuToDelete == null)
{
return new ReturnTemplate(404, "菜单不存在", null);
}
// 递归删除所有子菜单
await DeleteMenuRecursive(uuid);
await _context.SaveChangesAsync();
return new ReturnTemplate(200, "菜单删除成功", null);
}
catch (Exception ex)
{
return new ReturnTemplate(500, $"删除菜单失败: {ex.Message}", null);
}
}
/// <summary>
/// 递归删除菜单及其子菜单
/// </summary>
private async Task DeleteMenuRecursive(string parentUuid)
{
var children = await _context.Menus.Where(m => m.ParentId == parentUuid).ToListAsync();
foreach (var child in children)
{
await DeleteMenuRecursive(child.Uuid);
}
var menu = await _context.Menus.FirstOrDefaultAsync(m => m.Uuid == parentUuid);
if (menu != null)
{
_context.Menus.Remove(menu);
}
}
/// <summary>
/// 构建菜单树形结构
/// </summary>
private List<MenuResponse> BuildMenuTree(List<MenuModel> allMenus)
{
var menuResponseDict = new Dictionary<string, MenuResponse>();
var rootMenus = new List<MenuResponse>();
// 第一步:创建所有菜单的响应对象
foreach (var menu in allMenus)
{
var menuResponse = new MenuResponse
{
Uuid = menu.Uuid,
ParentId = menu.ParentId,
Path = menu.Path,
Label = menu.Label,
MenuName = menu.MenuName,
Icon = menu.Icon,
MenuCode = menu.MenuCode,
Adaptability = menu.Adaptability,
Component = menu.Component,
Sort = menu.Sort,
Status = menu.Status,
Query = menu.Query,
CreateTime = menu.CreateTime,
UpdateTime = menu.UpdateTime
};
menuResponseDict[menu.Uuid] = menuResponse;
}
// 第二步:构建树形结构
foreach (var menu in allMenus)
{
var menuResponse = menuResponseDict[menu.Uuid];
if (string.IsNullOrEmpty(menu.ParentId))
{
// 根菜单
rootMenus.Add(menuResponse);
}
else
{
// 子菜单
if (menuResponseDict.TryGetValue(menu.ParentId, out var parentResponse))
{
parentResponse.Children.Add(menuResponse);
}
}
}
return rootMenus;
}
}

View File

@ -1,6 +1,9 @@
using AGSS.Models; using AGSS.Models;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore;
namespace AGSS.Services;
public class UserService public class UserService
{ {
@ -29,7 +32,98 @@ public class UserService
Config = user.Config, Config = user.Config,
JobCode = user.JobCode, JobCode = user.JobCode,
JobName = user.JobName, JobName = user.JobName,
Birthday = user.Birthday Birthday = user.Birthday,
MenuCode = user.MenuCode,
MenuName = user.MenuName
}; };
} }
public async Task<List<UserProfile>> GetUsersProfileInRoleAsync(string roleName)
{
var usersInRole = await _userManager.GetUsersInRoleAsync(roleName);
if (usersInRole == null || !usersInRole.Any())
{
throw new ArgumentException("No users found in the specified role");
}
var userProfiles = new List<UserProfile>();
foreach (var user in usersInRole)
{
userProfiles.Add(new UserProfile
{
Id = user.Id,
UserName = user.UserName,
Email = user.Email,
Sex = user.Sex,
Description = user.Description,
Config = user.Config,
JobCode = user.JobCode,
JobName = user.JobName,
Birthday = user.Birthday,
MenuCode = user.MenuCode,
MenuName = user.MenuName
});
}
// Assuming you want to return a single UserProfile, you might need to adjust this logic
// For now, returning the first user's profile
return userProfiles;
}
public async Task<List<UserProfile>> GetUsersProfileByUserNameAsync(string userName)
{
var users = await _userManager.Users
.Where(u => u.UserName.Contains(userName))
.Select(u => new UserProfile
{
Id = u.Id,
UserName = u.UserName,
Email = u.Email,
Sex = u.Sex,
Description = u.Description,
Config = u.Config,
JobCode = u.JobCode,
JobName = u.JobName,
Birthday = u.Birthday,
MenuCode = u.MenuCode,
MenuName = u.MenuName
})
.ToListAsync();
if (users == null || !users.Any())
{
throw new ArgumentException("No users found with the specified username");
}
return users;
}
public async Task<List<UserProfile>> GetUsersProfileAllAsync()
{
var users = await _userManager.Users
.Select(u => new UserProfile
{
Id = u.Id,
UserName = u.UserName,
Email = u.Email,
Sex = u.Sex,
Description = u.Description,
Config = u.Config,
JobCode = u.JobCode,
JobName = u.JobName,
Birthday = u.Birthday,
MenuCode = u.MenuCode,
MenuName = u.MenuName
})
.ToListAsync();
if (users == null || !users.Any())
{
throw new ArgumentException("No users found");
}
return users;
}
} }

View File

@ -2,23 +2,19 @@ using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims; using System.Security.Claims;
using System.Text; using System.Text;
using AGSS.Models.Entities; using AGSS.Models.Entities;
using asg_form;
using Microsoft.IdentityModel.Tokens; using Microsoft.IdentityModel.Tokens;
namespace AGSS.Utilities; namespace AGSS.Utilities;
public class Jwt public class Jwt
{ {
private readonly IConfiguration _configuration;
public Jwt(IConfiguration configuration)
{
_configuration = configuration;
}
public string BuildToken(IEnumerable<Claim> claims) public string BuildToken(IEnumerable<Claim> claims, JWTOptions options)
{ {
DateTime expires = DateTime.Now.AddDays(int.Parse(_configuration["Jwt:ExpireMinutes"])); DateTime expires = DateTime.Now.AddSeconds(options.ExpireSeconds);
byte[] keyBytes = Encoding.UTF8.GetBytes(_configuration["Jwt:Key"]); byte[] keyBytes = Encoding.UTF8.GetBytes(options.SigningKey);
var secKey = new SymmetricSecurityKey(keyBytes); var secKey = new SymmetricSecurityKey(keyBytes);
var credentials = new SigningCredentials(secKey, var credentials = new SigningCredentials(secKey,
@ -27,18 +23,4 @@ public class Jwt
signingCredentials: credentials, claims: claims); signingCredentials: credentials, claims: claims);
return new JwtSecurityTokenHandler().WriteToken(tokenDescriptor); return new JwtSecurityTokenHandler().WriteToken(tokenDescriptor);
} }
public string GenerateJwtToken(UserModel user,IList<string> roles)
{
var claims = new List<Claim>();
claims.Add(new Claim(JwtRegisteredClaimNames.Sub, user.Id.ToString()));
claims.Add(new Claim(JwtRegisteredClaimNames.Jti, Guid.NewGuid().ToString()));
// var roles = await user.GetRolesAsync(user);
foreach (string role in roles)
{
claims.Add(new Claim(ClaimTypes.Role, role));
}
string jwtToken = BuildToken(claims);
return jwtToken;
}
} }

27
AGSS/Utilities/LINQ.cs Normal file
View File

@ -0,0 +1,27 @@
using System;
using System.Collections.Generic;
using System.Linq;
namespace AGSS.Utilities;
public static class LINQ
{
/// <summary>
/// 从给定的序列中分页获取元素。
/// </summary>
/// <typeparam name="T">序列中的元素类型。</typeparam>
/// <param name="source">要从中分页的源序列。</param>
/// <param name="pageIndex">请求的页码从0开始。</param>
/// <param name="pageSize">每页包含的元素数量。</param>
/// <returns>返回指定页码和页大小对应的子序列。</returns>
/// <exception cref="ArgumentNullException">如果source为null。</exception>
/// <exception cref="ArgumentOutOfRangeException">如果pageIndex是负数或者pageSize是非正数。</exception>
public static IEnumerable<T> Paginate<T>(this IEnumerable<T> source, int pageIndex, int pageSize)
{
if (source == null) throw new ArgumentNullException(nameof(source));
if (pageIndex < 0) throw new ArgumentOutOfRangeException(nameof(pageIndex), "Page index must be non-negative.");
if (pageSize <= 0) throw new ArgumentOutOfRangeException(nameof(pageSize), "Page size must be positive.");
return source.Skip(pageIndex * pageSize).Take(pageSize);
}
}

View File

@ -18,7 +18,7 @@
"ExpireMinutes": "4", "ExpireMinutes": "4",
"Issuer": "https://api.zeronode.cn/api", "Issuer": "https://api.zeronode.cn/api",
"Audience": "https://api.zeronode.cn/api", "Audience": "https://api.zeronode.cn/api",
"Key": "7wU9bdVfBsX3jITh0w4bgE6fkvLk8pIcZRSUw6r8HQUnXfslYxlx4c4E0ZAIw4Ak" "SigningKey": "7wU9bdVfBsX3jITh0w4bgE6fkvLk8pIcZRSUw6r8HQUnXfslYxlx4c4E0ZAIw4Ak"
}, },
"ConnectionStrings": { "ConnectionStrings": {
"DBContext": "Host=localhost;Port=5432;Database=postgres;Username=postgres;Password=luolan12323;" "DBContext": "Host=localhost;Port=5432;Database=postgres;Username=postgres;Password=luolan12323;"

View File

@ -14,11 +14,11 @@
}, },
"AllowedHosts": "*", "AllowedHosts": "*",
"Jwt": { "JWT": {
"ExpireMinutes": "4", "ExpireMinutes": "4",
"Issuer": "https://api.zeronode.cn/api", "Issuer": "https://api.zeronode.cn/api",
"Audience": "https://api.zeronode.cn/api", "Audience": "https://api.zeronode.cn/api",
"Key": "7wU9bdVfBsX3jITh0w4bgE6fkvLk8pIcZRSUw6r8HQUnXfslYxlx4c4E0ZAIw4Ak" "SigningKey": "7wU9bdVfBsX3jITh0w4bgE6fkvLk8pIcZRSUw6r8HQUnXfslYxlx4c4E0ZAIw4Ak"
}, },
"ConnectionStrings": { "ConnectionStrings": {
"DBContext": "Host=1Panel-postgresql-auKB;Port=5432;Database=zeronode;Username=zeronode;Password=luolan12323;" "DBContext": "Host=1Panel-postgresql-auKB;Port=5432;Database=zeronode;Username=zeronode;Password=luolan12323;"

9
AGSS/wwwroot/js/qr.js Normal file
View File

@ -0,0 +1,9 @@
window.addEventListener("load", () => {
const uri = document.getElementById("qrCodeData").getAttribute('data-url');
new QRCode(document.getElementById("qrCode"),
{
text: uri,
width: 150,
height: 150
});
});

4
AGSS/wwwroot/lib/qrcode/.gitignore vendored Normal file
View File

@ -0,0 +1,4 @@
.DS_Store
.idea
.project

View File

@ -0,0 +1,14 @@
The MIT License (MIT)
---------------------
Copyright (c) 2012 davidshimjs
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.

View File

@ -0,0 +1,46 @@
# QRCode.js
QRCode.js is javascript library for making QRCode. QRCode.js supports Cross-browser with HTML5 Canvas and table tag in DOM.
QRCode.js has no dependencies.
## Basic Usages
```
<div id="qrcode"></div>
<script type="text/javascript">
new QRCode(document.getElementById("qrcode"), "http://jindo.dev.naver.com/collie");
</script>
```
or with some options
```
<div id="qrcode"></div>
<script type="text/javascript">
var qrcode = new QRCode(document.getElementById("qrcode"), {
text: "http://jindo.dev.naver.com/collie",
width: 128,
height: 128,
colorDark : "#000000",
colorLight : "#ffffff",
correctLevel : QRCode.CorrectLevel.H
});
</script>
```
and you can use some methods
```
qrcode.clear(); // clear the code.
qrcode.makeCode("http://naver.com"); // make another code.
```
## Browser Compatibility
IE6~10, Chrome, Firefox, Safari, Opera, Mobile Safari, Android, Windows Mobile, ETC.
## License
MIT License
## Contact
twitter @davidshimjs
[![Bitdeli Badge](https://d2weczhvl823v0.cloudfront.net/davidshimjs/qrcodejs/trend.png)](https://bitdeli.com/free "Bitdeli Badge")

View File

@ -0,0 +1,18 @@
{
"name": "qrcode.js",
"version": "0.0.1",
"homepage": "https://github.com/davidshimjs/qrcodejs",
"authors": [
"Sangmin Shim", "Sangmin Shim <ssm0123@gmail.com> (http://jaguarjs.com)"
],
"description": "Cross-browser QRCode generator for javascript",
"main": "qrcode.js",
"ignore": [
"bower_components",
"node_modules",
"index.html",
"index.svg",
"jquery.min.js",
"qrcode.min.js"
]
}

View File

@ -0,0 +1,47 @@
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="ko" lang="ko">
<head>
<title>Cross-Browser QRCode generator for Javascript</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<meta name="viewport" content="width=device-width,initial-scale=1,user-scalable=no" />
<script type="text/javascript" src="jquery.min.js"></script>
<script type="text/javascript" src="qrcode.js"></script>
</head>
<body>
<input id="text" type="text" value="http://jindo.dev.naver.com/collie" style="width:80%" />
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<g id="qrcode"/>
</svg>
<script type="text/javascript">
var qrcode = new QRCode(document.getElementById("qrcode"), {
width : 100,
height : 100,
useSVG: true
});
function makeCode () {
var elText = document.getElementById("text");
if (!elText.value) {
alert("Input a text");
elText.focus();
return;
}
qrcode.makeCode(elText.value);
}
makeCode();
$("#text").
on("blur", function () {
makeCode();
}).
on("keydown", function (e) {
if (e.keyCode == 13) {
makeCode();
}
});
</script>
</body>
</html>

View File

@ -0,0 +1,44 @@
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="ko" lang="ko">
<head>
<title>Cross-Browser QRCode generator for Javascript</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<meta name="viewport" content="width=device-width,initial-scale=1,user-scalable=no" />
<script type="text/javascript" src="jquery.min.js"></script>
<script type="text/javascript" src="qrcode.js"></script>
</head>
<body>
<input id="text" type="text" value="http://jindo.dev.naver.com/collie" style="width:80%" /><br />
<div id="qrcode" style="width:100px; height:100px; margin-top:15px;"></div>
<script type="text/javascript">
var qrcode = new QRCode(document.getElementById("qrcode"), {
width : 100,
height : 100
});
function makeCode () {
var elText = document.getElementById("text");
if (!elText.value) {
alert("Input a text");
elText.focus();
return;
}
qrcode.makeCode(elText.value);
}
makeCode();
$("#text").
on("blur", function () {
makeCode();
}).
on("keydown", function (e) {
if (e.keyCode == 13) {
makeCode();
}
});
</script>
</body>

View File

@ -0,0 +1,37 @@
<?xml version="1.0" standalone="yes"?>
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="-50 0 200 100">
<g id="qrcode"/>
<foreignObject x="-50" y="0" width="100" height="100">
<body xmlns="http://www.w3.org/1999/xhtml" style="padding:0; margin:0">
<div style="padding:inherit; margin:inherit; height:100%">
<textarea id="text" style="height:100%; width:100%; position:absolute; margin:inherit; padding:inherit">james</textarea>
</div>
<script type="application/ecmascript" src="qrcode.js"></script>
<script type="application/ecmascript">
var elem = document.getElementById("qrcode");
var qrcode = new QRCode(elem, {
width : 100,
height : 100
});
function makeCode () {
var elText = document.getElementById("text");
if (elText.value === "") {
//alert("Input a text");
//elText.focus();
return;
}
qrcode.makeCode(elText.value);
}
makeCode();
document.getElementById("text").onkeyup = function (e) {
makeCode();
};
</script>
</body>
</foreignObject>
</svg>

After

Width:  |  Height:  |  Size: 1.1 KiB

2
AGSS/wwwroot/lib/qrcode/jquery.min.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,614 @@
/**
* @fileoverview
* - Using the 'QRCode for Javascript library'
* - Fixed dataset of 'QRCode for Javascript library' for support full-spec.
* - this library has no dependencies.
*
* @author davidshimjs
* @see <a href="http://www.d-project.com/" target="_blank">http://www.d-project.com/</a>
* @see <a href="http://jeromeetienne.github.com/jquery-qrcode/" target="_blank">http://jeromeetienne.github.com/jquery-qrcode/</a>
*/
var QRCode;
(function () {
//---------------------------------------------------------------------
// QRCode for JavaScript
//
// Copyright (c) 2009 Kazuhiko Arase
//
// URL: http://www.d-project.com/
//
// Licensed under the MIT license:
// http://www.opensource.org/licenses/mit-license.php
//
// The word "QR Code" is registered trademark of
// DENSO WAVE INCORPORATED
// http://www.denso-wave.com/qrcode/faqpatent-e.html
//
//---------------------------------------------------------------------
function QR8bitByte(data) {
this.mode = QRMode.MODE_8BIT_BYTE;
this.data = data;
this.parsedData = [];
// Added to support UTF-8 Characters
for (var i = 0, l = this.data.length; i < l; i++) {
var byteArray = [];
var code = this.data.charCodeAt(i);
if (code > 0x10000) {
byteArray[0] = 0xF0 | ((code & 0x1C0000) >>> 18);
byteArray[1] = 0x80 | ((code & 0x3F000) >>> 12);
byteArray[2] = 0x80 | ((code & 0xFC0) >>> 6);
byteArray[3] = 0x80 | (code & 0x3F);
} else if (code > 0x800) {
byteArray[0] = 0xE0 | ((code & 0xF000) >>> 12);
byteArray[1] = 0x80 | ((code & 0xFC0) >>> 6);
byteArray[2] = 0x80 | (code & 0x3F);
} else if (code > 0x80) {
byteArray[0] = 0xC0 | ((code & 0x7C0) >>> 6);
byteArray[1] = 0x80 | (code & 0x3F);
} else {
byteArray[0] = code;
}
this.parsedData.push(byteArray);
}
this.parsedData = Array.prototype.concat.apply([], this.parsedData);
if (this.parsedData.length != this.data.length) {
this.parsedData.unshift(191);
this.parsedData.unshift(187);
this.parsedData.unshift(239);
}
}
QR8bitByte.prototype = {
getLength: function (buffer) {
return this.parsedData.length;
},
write: function (buffer) {
for (var i = 0, l = this.parsedData.length; i < l; i++) {
buffer.put(this.parsedData[i], 8);
}
}
};
function QRCodeModel(typeNumber, errorCorrectLevel) {
this.typeNumber = typeNumber;
this.errorCorrectLevel = errorCorrectLevel;
this.modules = null;
this.moduleCount = 0;
this.dataCache = null;
this.dataList = [];
}
QRCodeModel.prototype={addData:function(data){var newData=new QR8bitByte(data);this.dataList.push(newData);this.dataCache=null;},isDark:function(row,col){if(row<0||this.moduleCount<=row||col<0||this.moduleCount<=col){throw new Error(row+","+col);}
return this.modules[row][col];},getModuleCount:function(){return this.moduleCount;},make:function(){this.makeImpl(false,this.getBestMaskPattern());},makeImpl:function(test,maskPattern){this.moduleCount=this.typeNumber*4+17;this.modules=new Array(this.moduleCount);for(var row=0;row<this.moduleCount;row++){this.modules[row]=new Array(this.moduleCount);for(var col=0;col<this.moduleCount;col++){this.modules[row][col]=null;}}
this.setupPositionProbePattern(0,0);this.setupPositionProbePattern(this.moduleCount-7,0);this.setupPositionProbePattern(0,this.moduleCount-7);this.setupPositionAdjustPattern();this.setupTimingPattern();this.setupTypeInfo(test,maskPattern);if(this.typeNumber>=7){this.setupTypeNumber(test);}
if(this.dataCache==null){this.dataCache=QRCodeModel.createData(this.typeNumber,this.errorCorrectLevel,this.dataList);}
this.mapData(this.dataCache,maskPattern);},setupPositionProbePattern:function(row,col){for(var r=-1;r<=7;r++){if(row+r<=-1||this.moduleCount<=row+r)continue;for(var c=-1;c<=7;c++){if(col+c<=-1||this.moduleCount<=col+c)continue;if((0<=r&&r<=6&&(c==0||c==6))||(0<=c&&c<=6&&(r==0||r==6))||(2<=r&&r<=4&&2<=c&&c<=4)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}},getBestMaskPattern:function(){var minLostPoint=0;var pattern=0;for(var i=0;i<8;i++){this.makeImpl(true,i);var lostPoint=QRUtil.getLostPoint(this);if(i==0||minLostPoint>lostPoint){minLostPoint=lostPoint;pattern=i;}}
return pattern;},createMovieClip:function(target_mc,instance_name,depth){var qr_mc=target_mc.createEmptyMovieClip(instance_name,depth);var cs=1;this.make();for(var row=0;row<this.modules.length;row++){var y=row*cs;for(var col=0;col<this.modules[row].length;col++){var x=col*cs;var dark=this.modules[row][col];if(dark){qr_mc.beginFill(0,100);qr_mc.moveTo(x,y);qr_mc.lineTo(x+cs,y);qr_mc.lineTo(x+cs,y+cs);qr_mc.lineTo(x,y+cs);qr_mc.endFill();}}}
return qr_mc;},setupTimingPattern:function(){for(var r=8;r<this.moduleCount-8;r++){if(this.modules[r][6]!=null){continue;}
this.modules[r][6]=(r%2==0);}
for(var c=8;c<this.moduleCount-8;c++){if(this.modules[6][c]!=null){continue;}
this.modules[6][c]=(c%2==0);}},setupPositionAdjustPattern:function(){var pos=QRUtil.getPatternPosition(this.typeNumber);for(var i=0;i<pos.length;i++){for(var j=0;j<pos.length;j++){var row=pos[i];var col=pos[j];if(this.modules[row][col]!=null){continue;}
for(var r=-2;r<=2;r++){for(var c=-2;c<=2;c++){if(r==-2||r==2||c==-2||c==2||(r==0&&c==0)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}}}},setupTypeNumber:function(test){var bits=QRUtil.getBCHTypeNumber(this.typeNumber);for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[Math.floor(i/3)][i%3+this.moduleCount-8-3]=mod;}
for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[i%3+this.moduleCount-8-3][Math.floor(i/3)]=mod;}},setupTypeInfo:function(test,maskPattern){var data=(this.errorCorrectLevel<<3)|maskPattern;var bits=QRUtil.getBCHTypeInfo(data);for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<6){this.modules[i][8]=mod;}else if(i<8){this.modules[i+1][8]=mod;}else{this.modules[this.moduleCount-15+i][8]=mod;}}
for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<8){this.modules[8][this.moduleCount-i-1]=mod;}else if(i<9){this.modules[8][15-i-1+1]=mod;}else{this.modules[8][15-i-1]=mod;}}
this.modules[this.moduleCount-8][8]=(!test);},mapData:function(data,maskPattern){var inc=-1;var row=this.moduleCount-1;var bitIndex=7;var byteIndex=0;for(var col=this.moduleCount-1;col>0;col-=2){if(col==6)col--;while(true){for(var c=0;c<2;c++){if(this.modules[row][col-c]==null){var dark=false;if(byteIndex<data.length){dark=(((data[byteIndex]>>>bitIndex)&1)==1);}
var mask=QRUtil.getMask(maskPattern,row,col-c);if(mask){dark=!dark;}
this.modules[row][col-c]=dark;bitIndex--;if(bitIndex==-1){byteIndex++;bitIndex=7;}}}
row+=inc;if(row<0||this.moduleCount<=row){row-=inc;inc=-inc;break;}}}}};QRCodeModel.PAD0=0xEC;QRCodeModel.PAD1=0x11;QRCodeModel.createData=function(typeNumber,errorCorrectLevel,dataList){var rsBlocks=QRRSBlock.getRSBlocks(typeNumber,errorCorrectLevel);var buffer=new QRBitBuffer();for(var i=0;i<dataList.length;i++){var data=dataList[i];buffer.put(data.mode,4);buffer.put(data.getLength(),QRUtil.getLengthInBits(data.mode,typeNumber));data.write(buffer);}
var totalDataCount=0;for(var i=0;i<rsBlocks.length;i++){totalDataCount+=rsBlocks[i].dataCount;}
if(buffer.getLengthInBits()>totalDataCount*8){throw new Error("code length overflow. ("
+buffer.getLengthInBits()
+">"
+totalDataCount*8
+")");}
if(buffer.getLengthInBits()+4<=totalDataCount*8){buffer.put(0,4);}
while(buffer.getLengthInBits()%8!=0){buffer.putBit(false);}
while(true){if(buffer.getLengthInBits()>=totalDataCount*8){break;}
buffer.put(QRCodeModel.PAD0,8);if(buffer.getLengthInBits()>=totalDataCount*8){break;}
buffer.put(QRCodeModel.PAD1,8);}
return QRCodeModel.createBytes(buffer,rsBlocks);};QRCodeModel.createBytes=function(buffer,rsBlocks){var offset=0;var maxDcCount=0;var maxEcCount=0;var dcdata=new Array(rsBlocks.length);var ecdata=new Array(rsBlocks.length);for(var r=0;r<rsBlocks.length;r++){var dcCount=rsBlocks[r].dataCount;var ecCount=rsBlocks[r].totalCount-dcCount;maxDcCount=Math.max(maxDcCount,dcCount);maxEcCount=Math.max(maxEcCount,ecCount);dcdata[r]=new Array(dcCount);for(var i=0;i<dcdata[r].length;i++){dcdata[r][i]=0xff&buffer.buffer[i+offset];}
offset+=dcCount;var rsPoly=QRUtil.getErrorCorrectPolynomial(ecCount);var rawPoly=new QRPolynomial(dcdata[r],rsPoly.getLength()-1);var modPoly=rawPoly.mod(rsPoly);ecdata[r]=new Array(rsPoly.getLength()-1);for(var i=0;i<ecdata[r].length;i++){var modIndex=i+modPoly.getLength()-ecdata[r].length;ecdata[r][i]=(modIndex>=0)?modPoly.get(modIndex):0;}}
var totalCodeCount=0;for(var i=0;i<rsBlocks.length;i++){totalCodeCount+=rsBlocks[i].totalCount;}
var data=new Array(totalCodeCount);var index=0;for(var i=0;i<maxDcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<dcdata[r].length){data[index++]=dcdata[r][i];}}}
for(var i=0;i<maxEcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<ecdata[r].length){data[index++]=ecdata[r][i];}}}
return data;};var QRMode={MODE_NUMBER:1<<0,MODE_ALPHA_NUM:1<<1,MODE_8BIT_BYTE:1<<2,MODE_KANJI:1<<3};var QRErrorCorrectLevel={L:1,M:0,Q:3,H:2};var QRMaskPattern={PATTERN000:0,PATTERN001:1,PATTERN010:2,PATTERN011:3,PATTERN100:4,PATTERN101:5,PATTERN110:6,PATTERN111:7};var QRUtil={PATTERN_POSITION_TABLE:[[],[6,18],[6,22],[6,26],[6,30],[6,34],[6,22,38],[6,24,42],[6,26,46],[6,28,50],[6,30,54],[6,32,58],[6,34,62],[6,26,46,66],[6,26,48,70],[6,26,50,74],[6,30,54,78],[6,30,56,82],[6,30,58,86],[6,34,62,90],[6,28,50,72,94],[6,26,50,74,98],[6,30,54,78,102],[6,28,54,80,106],[6,32,58,84,110],[6,30,58,86,114],[6,34,62,90,118],[6,26,50,74,98,122],[6,30,54,78,102,126],[6,26,52,78,104,130],[6,30,56,82,108,134],[6,34,60,86,112,138],[6,30,58,86,114,142],[6,34,62,90,118,146],[6,30,54,78,102,126,150],[6,24,50,76,102,128,154],[6,28,54,80,106,132,158],[6,32,58,84,110,136,162],[6,26,54,82,110,138,166],[6,30,58,86,114,142,170]],G15:(1<<10)|(1<<8)|(1<<5)|(1<<4)|(1<<2)|(1<<1)|(1<<0),G18:(1<<12)|(1<<11)|(1<<10)|(1<<9)|(1<<8)|(1<<5)|(1<<2)|(1<<0),G15_MASK:(1<<14)|(1<<12)|(1<<10)|(1<<4)|(1<<1),getBCHTypeInfo:function(data){var d=data<<10;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)>=0){d^=(QRUtil.G15<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)));}
return((data<<10)|d)^QRUtil.G15_MASK;},getBCHTypeNumber:function(data){var d=data<<12;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)>=0){d^=(QRUtil.G18<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)));}
return(data<<12)|d;},getBCHDigit:function(data){var digit=0;while(data!=0){digit++;data>>>=1;}
return digit;},getPatternPosition:function(typeNumber){return QRUtil.PATTERN_POSITION_TABLE[typeNumber-1];},getMask:function(maskPattern,i,j){switch(maskPattern){case QRMaskPattern.PATTERN000:return(i+j)%2==0;case QRMaskPattern.PATTERN001:return i%2==0;case QRMaskPattern.PATTERN010:return j%3==0;case QRMaskPattern.PATTERN011:return(i+j)%3==0;case QRMaskPattern.PATTERN100:return(Math.floor(i/2)+Math.floor(j/3))%2==0;case QRMaskPattern.PATTERN101:return(i*j)%2+(i*j)%3==0;case QRMaskPattern.PATTERN110:return((i*j)%2+(i*j)%3)%2==0;case QRMaskPattern.PATTERN111:return((i*j)%3+(i+j)%2)%2==0;default:throw new Error("bad maskPattern:"+maskPattern);}},getErrorCorrectPolynomial:function(errorCorrectLength){var a=new QRPolynomial([1],0);for(var i=0;i<errorCorrectLength;i++){a=a.multiply(new QRPolynomial([1,QRMath.gexp(i)],0));}
return a;},getLengthInBits:function(mode,type){if(1<=type&&type<10){switch(mode){case QRMode.MODE_NUMBER:return 10;case QRMode.MODE_ALPHA_NUM:return 9;case QRMode.MODE_8BIT_BYTE:return 8;case QRMode.MODE_KANJI:return 8;default:throw new Error("mode:"+mode);}}else if(type<27){switch(mode){case QRMode.MODE_NUMBER:return 12;case QRMode.MODE_ALPHA_NUM:return 11;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 10;default:throw new Error("mode:"+mode);}}else if(type<41){switch(mode){case QRMode.MODE_NUMBER:return 14;case QRMode.MODE_ALPHA_NUM:return 13;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 12;default:throw new Error("mode:"+mode);}}else{throw new Error("type:"+type);}},getLostPoint:function(qrCode){var moduleCount=qrCode.getModuleCount();var lostPoint=0;for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount;col++){var sameCount=0;var dark=qrCode.isDark(row,col);for(var r=-1;r<=1;r++){if(row+r<0||moduleCount<=row+r){continue;}
for(var c=-1;c<=1;c++){if(col+c<0||moduleCount<=col+c){continue;}
if(r==0&&c==0){continue;}
if(dark==qrCode.isDark(row+r,col+c)){sameCount++;}}}
if(sameCount>5){lostPoint+=(3+sameCount-5);}}}
for(var row=0;row<moduleCount-1;row++){for(var col=0;col<moduleCount-1;col++){var count=0;if(qrCode.isDark(row,col))count++;if(qrCode.isDark(row+1,col))count++;if(qrCode.isDark(row,col+1))count++;if(qrCode.isDark(row+1,col+1))count++;if(count==0||count==4){lostPoint+=3;}}}
for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount-6;col++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row,col+1)&&qrCode.isDark(row,col+2)&&qrCode.isDark(row,col+3)&&qrCode.isDark(row,col+4)&&!qrCode.isDark(row,col+5)&&qrCode.isDark(row,col+6)){lostPoint+=40;}}}
for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount-6;row++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row+1,col)&&qrCode.isDark(row+2,col)&&qrCode.isDark(row+3,col)&&qrCode.isDark(row+4,col)&&!qrCode.isDark(row+5,col)&&qrCode.isDark(row+6,col)){lostPoint+=40;}}}
var darkCount=0;for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount;row++){if(qrCode.isDark(row,col)){darkCount++;}}}
var ratio=Math.abs(100*darkCount/moduleCount/moduleCount-50)/5;lostPoint+=ratio*10;return lostPoint;}};var QRMath={glog:function(n){if(n<1){throw new Error("glog("+n+")");}
return QRMath.LOG_TABLE[n];},gexp:function(n){while(n<0){n+=255;}
while(n>=256){n-=255;}
return QRMath.EXP_TABLE[n];},EXP_TABLE:new Array(256),LOG_TABLE:new Array(256)};for(var i=0;i<8;i++){QRMath.EXP_TABLE[i]=1<<i;}
for(var i=8;i<256;i++){QRMath.EXP_TABLE[i]=QRMath.EXP_TABLE[i-4]^QRMath.EXP_TABLE[i-5]^QRMath.EXP_TABLE[i-6]^QRMath.EXP_TABLE[i-8];}
for(var i=0;i<255;i++){QRMath.LOG_TABLE[QRMath.EXP_TABLE[i]]=i;}
function QRPolynomial(num,shift){if(num.length==undefined){throw new Error(num.length+"/"+shift);}
var offset=0;while(offset<num.length&&num[offset]==0){offset++;}
this.num=new Array(num.length-offset+shift);for(var i=0;i<num.length-offset;i++){this.num[i]=num[i+offset];}}
QRPolynomial.prototype={get:function(index){return this.num[index];},getLength:function(){return this.num.length;},multiply:function(e){var num=new Array(this.getLength()+e.getLength()-1);for(var i=0;i<this.getLength();i++){for(var j=0;j<e.getLength();j++){num[i+j]^=QRMath.gexp(QRMath.glog(this.get(i))+QRMath.glog(e.get(j)));}}
return new QRPolynomial(num,0);},mod:function(e){if(this.getLength()-e.getLength()<0){return this;}
var ratio=QRMath.glog(this.get(0))-QRMath.glog(e.get(0));var num=new Array(this.getLength());for(var i=0;i<this.getLength();i++){num[i]=this.get(i);}
for(var i=0;i<e.getLength();i++){num[i]^=QRMath.gexp(QRMath.glog(e.get(i))+ratio);}
return new QRPolynomial(num,0).mod(e);}};function QRRSBlock(totalCount,dataCount){this.totalCount=totalCount;this.dataCount=dataCount;}
QRRSBlock.RS_BLOCK_TABLE=[[1,26,19],[1,26,16],[1,26,13],[1,26,9],[1,44,34],[1,44,28],[1,44,22],[1,44,16],[1,70,55],[1,70,44],[2,35,17],[2,35,13],[1,100,80],[2,50,32],[2,50,24],[4,25,9],[1,134,108],[2,67,43],[2,33,15,2,34,16],[2,33,11,2,34,12],[2,86,68],[4,43,27],[4,43,19],[4,43,15],[2,98,78],[4,49,31],[2,32,14,4,33,15],[4,39,13,1,40,14],[2,121,97],[2,60,38,2,61,39],[4,40,18,2,41,19],[4,40,14,2,41,15],[2,146,116],[3,58,36,2,59,37],[4,36,16,4,37,17],[4,36,12,4,37,13],[2,86,68,2,87,69],[4,69,43,1,70,44],[6,43,19,2,44,20],[6,43,15,2,44,16],[4,101,81],[1,80,50,4,81,51],[4,50,22,4,51,23],[3,36,12,8,37,13],[2,116,92,2,117,93],[6,58,36,2,59,37],[4,46,20,6,47,21],[7,42,14,4,43,15],[4,133,107],[8,59,37,1,60,38],[8,44,20,4,45,21],[12,33,11,4,34,12],[3,145,115,1,146,116],[4,64,40,5,65,41],[11,36,16,5,37,17],[11,36,12,5,37,13],[5,109,87,1,110,88],[5,65,41,5,66,42],[5,54,24,7,55,25],[11,36,12],[5,122,98,1,123,99],[7,73,45,3,74,46],[15,43,19,2,44,20],[3,45,15,13,46,16],[1,135,107,5,136,108],[10,74,46,1,75,47],[1,50,22,15,51,23],[2,42,14,17,43,15],[5,150,120,1,151,121],[9,69,43,4,70,44],[17,50,22,1,51,23],[2,42,14,19,43,15],[3,141,113,4,142,114],[3,70,44,11,71,45],[17,47,21,4,48,22],[9,39,13,16,40,14],[3,135,107,5,136,108],[3,67,41,13,68,42],[15,54,24,5,55,25],[15,43,15,10,44,16],[4,144,116,4,145,117],[17,68,42],[17,50,22,6,51,23],[19,46,16,6,47,17],[2,139,111,7,140,112],[17,74,46],[7,54,24,16,55,25],[34,37,13],[4,151,121,5,152,122],[4,75,47,14,76,48],[11,54,24,14,55,25],[16,45,15,14,46,16],[6,147,117,4,148,118],[6,73,45,14,74,46],[11,54,24,16,55,25],[30,46,16,2,47,17],[8,132,106,4,133,107],[8,75,47,13,76,48],[7,54,24,22,55,25],[22,45,15,13,46,16],[10,142,114,2,143,115],[19,74,46,4,75,47],[28,50,22,6,51,23],[33,46,16,4,47,17],[8,152,122,4,153,123],[22,73,45,3,74,46],[8,53,23,26,54,24],[12,45,15,28,46,16],[3,147,117,10,148,118],[3,73,45,23,74,46],[4,54,24,31,55,25],[11,45,15,31,46,16],[7,146,116,7,147,117],[21,73,45,7,74,46],[1,53,23,37,54,24],[19,45,15,26,46,16],[5,145,115,10,146,116],[19,75,47,10,76,48],[15,54,24,25,55,25],[23,45,15,25,46,16],[13,145,115,3,146,116],[2,74,46,29,75,47],[42,54,24,1,55,25],[23,45,15,28,46,16],[17,145,115],[10,74,46,23,75,47],[10,54,24,35,55,25],[19,45,15,35,46,16],[17,145,115,1,146,116],[14,74,46,21,75,47],[29,54,24,19,55,25],[11,45,15,46,46,16],[13,145,115,6,146,116],[14,74,46,23,75,47],[44,54,24,7,55,25],[59,46,16,1,47,17],[12,151,121,7,152,122],[12,75,47,26,76,48],[39,54,24,14,55,25],[22,45,15,41,46,16],[6,151,121,14,152,122],[6,75,47,34,76,48],[46,54,24,10,55,25],[2,45,15,64,46,16],[17,152,122,4,153,123],[29,74,46,14,75,47],[49,54,24,10,55,25],[24,45,15,46,46,16],[4,152,122,18,153,123],[13,74,46,32,75,47],[48,54,24,14,55,25],[42,45,15,32,46,16],[20,147,117,4,148,118],[40,75,47,7,76,48],[43,54,24,22,55,25],[10,45,15,67,46,16],[19,148,118,6,149,119],[18,75,47,31,76,48],[34,54,24,34,55,25],[20,45,15,61,46,16]];QRRSBlock.getRSBlocks=function(typeNumber,errorCorrectLevel){var rsBlock=QRRSBlock.getRsBlockTable(typeNumber,errorCorrectLevel);if(rsBlock==undefined){throw new Error("bad rs block @ typeNumber:"+typeNumber+"/errorCorrectLevel:"+errorCorrectLevel);}
var length=rsBlock.length/3;var list=[];for(var i=0;i<length;i++){var count=rsBlock[i*3+0];var totalCount=rsBlock[i*3+1];var dataCount=rsBlock[i*3+2];for(var j=0;j<count;j++){list.push(new QRRSBlock(totalCount,dataCount));}}
return list;};QRRSBlock.getRsBlockTable=function(typeNumber,errorCorrectLevel){switch(errorCorrectLevel){case QRErrorCorrectLevel.L:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+0];case QRErrorCorrectLevel.M:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+1];case QRErrorCorrectLevel.Q:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+2];case QRErrorCorrectLevel.H:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+3];default:return undefined;}};function QRBitBuffer(){this.buffer=[];this.length=0;}
QRBitBuffer.prototype={get:function(index){var bufIndex=Math.floor(index/8);return((this.buffer[bufIndex]>>>(7-index%8))&1)==1;},put:function(num,length){for(var i=0;i<length;i++){this.putBit(((num>>>(length-i-1))&1)==1);}},getLengthInBits:function(){return this.length;},putBit:function(bit){var bufIndex=Math.floor(this.length/8);if(this.buffer.length<=bufIndex){this.buffer.push(0);}
if(bit){this.buffer[bufIndex]|=(0x80>>>(this.length%8));}
this.length++;}};var QRCodeLimitLength=[[17,14,11,7],[32,26,20,14],[53,42,32,24],[78,62,46,34],[106,84,60,44],[134,106,74,58],[154,122,86,64],[192,152,108,84],[230,180,130,98],[271,213,151,119],[321,251,177,137],[367,287,203,155],[425,331,241,177],[458,362,258,194],[520,412,292,220],[586,450,322,250],[644,504,364,280],[718,560,394,310],[792,624,442,338],[858,666,482,382],[929,711,509,403],[1003,779,565,439],[1091,857,611,461],[1171,911,661,511],[1273,997,715,535],[1367,1059,751,593],[1465,1125,805,625],[1528,1190,868,658],[1628,1264,908,698],[1732,1370,982,742],[1840,1452,1030,790],[1952,1538,1112,842],[2068,1628,1168,898],[2188,1722,1228,958],[2303,1809,1283,983],[2431,1911,1351,1051],[2563,1989,1423,1093],[2699,2099,1499,1139],[2809,2213,1579,1219],[2953,2331,1663,1273]];
function _isSupportCanvas() {
return typeof CanvasRenderingContext2D != "undefined";
}
// android 2.x doesn't support Data-URI spec
function _getAndroid() {
var android = false;
var sAgent = navigator.userAgent;
if (/android/i.test(sAgent)) { // android
android = true;
var aMat = sAgent.toString().match(/android ([0-9]\.[0-9])/i);
if (aMat && aMat[1]) {
android = parseFloat(aMat[1]);
}
}
return android;
}
var svgDrawer = (function() {
var Drawing = function (el, htOption) {
this._el = el;
this._htOption = htOption;
};
Drawing.prototype.draw = function (oQRCode) {
var _htOption = this._htOption;
var _el = this._el;
var nCount = oQRCode.getModuleCount();
var nWidth = Math.floor(_htOption.width / nCount);
var nHeight = Math.floor(_htOption.height / nCount);
this.clear();
function makeSVG(tag, attrs) {
var el = document.createElementNS('http://www.w3.org/2000/svg', tag);
for (var k in attrs)
if (attrs.hasOwnProperty(k)) el.setAttribute(k, attrs[k]);
return el;
}
var svg = makeSVG("svg" , {'viewBox': '0 0 ' + String(nCount) + " " + String(nCount), 'width': '100%', 'height': '100%', 'fill': _htOption.colorLight});
svg.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:xlink", "http://www.w3.org/1999/xlink");
_el.appendChild(svg);
svg.appendChild(makeSVG("rect", {"fill": _htOption.colorLight, "width": "100%", "height": "100%"}));
svg.appendChild(makeSVG("rect", {"fill": _htOption.colorDark, "width": "1", "height": "1", "id": "template"}));
for (var row = 0; row < nCount; row++) {
for (var col = 0; col < nCount; col++) {
if (oQRCode.isDark(row, col)) {
var child = makeSVG("use", {"x": String(col), "y": String(row)});
child.setAttributeNS("http://www.w3.org/1999/xlink", "href", "#template")
svg.appendChild(child);
}
}
}
};
Drawing.prototype.clear = function () {
while (this._el.hasChildNodes())
this._el.removeChild(this._el.lastChild);
};
return Drawing;
})();
var useSVG = document.documentElement.tagName.toLowerCase() === "svg";
// Drawing in DOM by using Table tag
var Drawing = useSVG ? svgDrawer : !_isSupportCanvas() ? (function () {
var Drawing = function (el, htOption) {
this._el = el;
this._htOption = htOption;
};
/**
* Draw the QRCode
*
* @param {QRCode} oQRCode
*/
Drawing.prototype.draw = function (oQRCode) {
var _htOption = this._htOption;
var _el = this._el;
var nCount = oQRCode.getModuleCount();
var nWidth = Math.floor(_htOption.width / nCount);
var nHeight = Math.floor(_htOption.height / nCount);
var aHTML = ['<table style="border:0;border-collapse:collapse;">'];
for (var row = 0; row < nCount; row++) {
aHTML.push('<tr>');
for (var col = 0; col < nCount; col++) {
aHTML.push('<td style="border:0;border-collapse:collapse;padding:0;margin:0;width:' + nWidth + 'px;height:' + nHeight + 'px;background-color:' + (oQRCode.isDark(row, col) ? _htOption.colorDark : _htOption.colorLight) + ';"></td>');
}
aHTML.push('</tr>');
}
aHTML.push('</table>');
_el.innerHTML = aHTML.join('');
// Fix the margin values as real size.
var elTable = _el.childNodes[0];
var nLeftMarginTable = (_htOption.width - elTable.offsetWidth) / 2;
var nTopMarginTable = (_htOption.height - elTable.offsetHeight) / 2;
if (nLeftMarginTable > 0 && nTopMarginTable > 0) {
elTable.style.margin = nTopMarginTable + "px " + nLeftMarginTable + "px";
}
};
/**
* Clear the QRCode
*/
Drawing.prototype.clear = function () {
this._el.innerHTML = '';
};
return Drawing;
})() : (function () { // Drawing in Canvas
function _onMakeImage() {
this._elImage.src = this._elCanvas.toDataURL("image/png");
this._elImage.style.display = "block";
this._elCanvas.style.display = "none";
}
// Android 2.1 bug workaround
// http://code.google.com/p/android/issues/detail?id=5141
if (this._android && this._android <= 2.1) {
var factor = 1 / window.devicePixelRatio;
var drawImage = CanvasRenderingContext2D.prototype.drawImage;
CanvasRenderingContext2D.prototype.drawImage = function (image, sx, sy, sw, sh, dx, dy, dw, dh) {
if (("nodeName" in image) && /img/i.test(image.nodeName)) {
for (var i = arguments.length - 1; i >= 1; i--) {
arguments[i] = arguments[i] * factor;
}
} else if (typeof dw == "undefined") {
arguments[1] *= factor;
arguments[2] *= factor;
arguments[3] *= factor;
arguments[4] *= factor;
}
drawImage.apply(this, arguments);
};
}
/**
* Check whether the user's browser supports Data URI or not
*
* @private
* @param {Function} fSuccess Occurs if it supports Data URI
* @param {Function} fFail Occurs if it doesn't support Data URI
*/
function _safeSetDataURI(fSuccess, fFail) {
var self = this;
self._fFail = fFail;
self._fSuccess = fSuccess;
// Check it just once
if (self._bSupportDataURI === null) {
var el = document.createElement("img");
var fOnError = function() {
self._bSupportDataURI = false;
if (self._fFail) {
self._fFail.call(self);
}
};
var fOnSuccess = function() {
self._bSupportDataURI = true;
if (self._fSuccess) {
self._fSuccess.call(self);
}
};
el.onabort = fOnError;
el.onerror = fOnError;
el.onload = fOnSuccess;
el.src = "data:image/gif;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg=="; // the Image contains 1px data.
return;
} else if (self._bSupportDataURI === true && self._fSuccess) {
self._fSuccess.call(self);
} else if (self._bSupportDataURI === false && self._fFail) {
self._fFail.call(self);
}
};
/**
* Drawing QRCode by using canvas
*
* @constructor
* @param {HTMLElement} el
* @param {Object} htOption QRCode Options
*/
var Drawing = function (el, htOption) {
this._bIsPainted = false;
this._android = _getAndroid();
this._htOption = htOption;
this._elCanvas = document.createElement("canvas");
this._elCanvas.width = htOption.width;
this._elCanvas.height = htOption.height;
el.appendChild(this._elCanvas);
this._el = el;
this._oContext = this._elCanvas.getContext("2d");
this._bIsPainted = false;
this._elImage = document.createElement("img");
this._elImage.alt = "Scan me!";
this._elImage.style.display = "none";
this._el.appendChild(this._elImage);
this._bSupportDataURI = null;
};
/**
* Draw the QRCode
*
* @param {QRCode} oQRCode
*/
Drawing.prototype.draw = function (oQRCode) {
var _elImage = this._elImage;
var _oContext = this._oContext;
var _htOption = this._htOption;
var nCount = oQRCode.getModuleCount();
var nWidth = _htOption.width / nCount;
var nHeight = _htOption.height / nCount;
var nRoundedWidth = Math.round(nWidth);
var nRoundedHeight = Math.round(nHeight);
_elImage.style.display = "none";
this.clear();
for (var row = 0; row < nCount; row++) {
for (var col = 0; col < nCount; col++) {
var bIsDark = oQRCode.isDark(row, col);
var nLeft = col * nWidth;
var nTop = row * nHeight;
_oContext.strokeStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
_oContext.lineWidth = 1;
_oContext.fillStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
_oContext.fillRect(nLeft, nTop, nWidth, nHeight);
// 안티 앨리어싱 방지 처리
_oContext.strokeRect(
Math.floor(nLeft) + 0.5,
Math.floor(nTop) + 0.5,
nRoundedWidth,
nRoundedHeight
);
_oContext.strokeRect(
Math.ceil(nLeft) - 0.5,
Math.ceil(nTop) - 0.5,
nRoundedWidth,
nRoundedHeight
);
}
}
this._bIsPainted = true;
};
/**
* Make the image from Canvas if the browser supports Data URI.
*/
Drawing.prototype.makeImage = function () {
if (this._bIsPainted) {
_safeSetDataURI.call(this, _onMakeImage);
}
};
/**
* Return whether the QRCode is painted or not
*
* @return {Boolean}
*/
Drawing.prototype.isPainted = function () {
return this._bIsPainted;
};
/**
* Clear the QRCode
*/
Drawing.prototype.clear = function () {
this._oContext.clearRect(0, 0, this._elCanvas.width, this._elCanvas.height);
this._bIsPainted = false;
};
/**
* @private
* @param {Number} nNumber
*/
Drawing.prototype.round = function (nNumber) {
if (!nNumber) {
return nNumber;
}
return Math.floor(nNumber * 1000) / 1000;
};
return Drawing;
})();
/**
* Get the type by string length
*
* @private
* @param {String} sText
* @param {Number} nCorrectLevel
* @return {Number} type
*/
function _getTypeNumber(sText, nCorrectLevel) {
var nType = 1;
var length = _getUTF8Length(sText);
for (var i = 0, len = QRCodeLimitLength.length; i <= len; i++) {
var nLimit = 0;
switch (nCorrectLevel) {
case QRErrorCorrectLevel.L :
nLimit = QRCodeLimitLength[i][0];
break;
case QRErrorCorrectLevel.M :
nLimit = QRCodeLimitLength[i][1];
break;
case QRErrorCorrectLevel.Q :
nLimit = QRCodeLimitLength[i][2];
break;
case QRErrorCorrectLevel.H :
nLimit = QRCodeLimitLength[i][3];
break;
}
if (length <= nLimit) {
break;
} else {
nType++;
}
}
if (nType > QRCodeLimitLength.length) {
throw new Error("Too long data");
}
return nType;
}
function _getUTF8Length(sText) {
var replacedText = encodeURI(sText).toString().replace(/\%[0-9a-fA-F]{2}/g, 'a');
return replacedText.length + (replacedText.length != sText ? 3 : 0);
}
/**
* @class QRCode
* @constructor
* @example
* new QRCode(document.getElementById("test"), "http://jindo.dev.naver.com/collie");
*
* @example
* var oQRCode = new QRCode("test", {
* text : "http://naver.com",
* width : 128,
* height : 128
* });
*
* oQRCode.clear(); // Clear the QRCode.
* oQRCode.makeCode("http://map.naver.com"); // Re-create the QRCode.
*
* @param {HTMLElement|String} el target element or 'id' attribute of element.
* @param {Object|String} vOption
* @param {String} vOption.text QRCode link data
* @param {Number} [vOption.width=256]
* @param {Number} [vOption.height=256]
* @param {String} [vOption.colorDark="#000000"]
* @param {String} [vOption.colorLight="#ffffff"]
* @param {QRCode.CorrectLevel} [vOption.correctLevel=QRCode.CorrectLevel.H] [L|M|Q|H]
*/
QRCode = function (el, vOption) {
this._htOption = {
width : 256,
height : 256,
typeNumber : 4,
colorDark : "#000000",
colorLight : "#ffffff",
correctLevel : QRErrorCorrectLevel.H
};
if (typeof vOption === 'string') {
vOption = {
text : vOption
};
}
// Overwrites options
if (vOption) {
for (var i in vOption) {
this._htOption[i] = vOption[i];
}
}
if (typeof el == "string") {
el = document.getElementById(el);
}
if (this._htOption.useSVG) {
Drawing = svgDrawer;
}
this._android = _getAndroid();
this._el = el;
this._oQRCode = null;
this._oDrawing = new Drawing(this._el, this._htOption);
if (this._htOption.text) {
this.makeCode(this._htOption.text);
}
};
/**
* Make the QRCode
*
* @param {String} sText link data
*/
QRCode.prototype.makeCode = function (sText) {
this._oQRCode = new QRCodeModel(_getTypeNumber(sText, this._htOption.correctLevel), this._htOption.correctLevel);
this._oQRCode.addData(sText);
this._oQRCode.make();
this._el.title = sText;
this._oDrawing.draw(this._oQRCode);
this.makeImage();
};
/**
* Make the Image from Canvas element
* - It occurs automatically
* - Android below 3 doesn't support Data-URI spec.
*
* @private
*/
QRCode.prototype.makeImage = function () {
if (typeof this._oDrawing.makeImage == "function" && (!this._android || this._android >= 3)) {
this._oDrawing.makeImage();
}
};
/**
* Clear the QRCode
*/
QRCode.prototype.clear = function () {
this._oDrawing.clear();
};
/**
* @name QRCode.CorrectLevel
*/
QRCode.CorrectLevel = QRErrorCorrectLevel;
})();

1
AGSS/wwwroot/lib/qrcode/qrcode.min.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -1,105 +1,10 @@
CREATE TABLE IF NOT EXISTS "__EFMigrationsHistory" ( START TRANSACTION;
"MigrationId" character varying(150) NOT NULL, ALTER TABLE "Menus" ADD "MenuName" character varying(1000);
"ProductVersion" character varying(32) NOT NULL,
CONSTRAINT "PK___EFMigrationsHistory" PRIMARY KEY ("MigrationId")
);
START TRANSACTION; ALTER TABLE "AspNetRoles" ADD "ChineseName" text;
CREATE TABLE "AspNetRoles" (
"Id" text NOT NULL,
"Name" character varying(256),
"NormalizedName" character varying(256),
"ConcurrencyStamp" text,
CONSTRAINT "PK_AspNetRoles" PRIMARY KEY ("Id")
);
CREATE TABLE "AspNetUsers" (
"Id" text NOT NULL,
"Sex" text,
"Description" character varying(100),
"Config" character varying(200),
"JobCode" character varying(10),
"JobName" character varying(10),
"Birthday" character varying(20),
"MenuCode" character varying(500),
"MenuName" character varying(500),
"UserName" character varying(256),
"NormalizedUserName" character varying(256),
"Email" character varying(256),
"NormalizedEmail" character varying(256),
"EmailConfirmed" boolean NOT NULL,
"PasswordHash" text,
"SecurityStamp" text,
"ConcurrencyStamp" text,
"PhoneNumber" text,
"PhoneNumberConfirmed" boolean NOT NULL,
"TwoFactorEnabled" boolean NOT NULL,
"LockoutEnd" timestamp with time zone,
"LockoutEnabled" boolean NOT NULL,
"AccessFailedCount" integer NOT NULL,
CONSTRAINT "PK_AspNetUsers" PRIMARY KEY ("Id")
);
CREATE TABLE "AspNetRoleClaims" (
"Id" integer GENERATED BY DEFAULT AS IDENTITY,
"RoleId" text NOT NULL,
"ClaimType" text,
"ClaimValue" text,
CONSTRAINT "PK_AspNetRoleClaims" PRIMARY KEY ("Id"),
CONSTRAINT "FK_AspNetRoleClaims_AspNetRoles_RoleId" FOREIGN KEY ("RoleId") REFERENCES "AspNetRoles" ("Id") ON DELETE CASCADE
);
CREATE TABLE "AspNetUserClaims" (
"Id" integer GENERATED BY DEFAULT AS IDENTITY,
"UserId" text NOT NULL,
"ClaimType" text,
"ClaimValue" text,
CONSTRAINT "PK_AspNetUserClaims" PRIMARY KEY ("Id"),
CONSTRAINT "FK_AspNetUserClaims_AspNetUsers_UserId" FOREIGN KEY ("UserId") REFERENCES "AspNetUsers" ("Id") ON DELETE CASCADE
);
CREATE TABLE "AspNetUserLogins" (
"LoginProvider" text NOT NULL,
"ProviderKey" text NOT NULL,
"ProviderDisplayName" text,
"UserId" text NOT NULL,
CONSTRAINT "PK_AspNetUserLogins" PRIMARY KEY ("LoginProvider", "ProviderKey"),
CONSTRAINT "FK_AspNetUserLogins_AspNetUsers_UserId" FOREIGN KEY ("UserId") REFERENCES "AspNetUsers" ("Id") ON DELETE CASCADE
);
CREATE TABLE "AspNetUserRoles" (
"UserId" text NOT NULL,
"RoleId" text NOT NULL,
CONSTRAINT "PK_AspNetUserRoles" PRIMARY KEY ("UserId", "RoleId"),
CONSTRAINT "FK_AspNetUserRoles_AspNetRoles_RoleId" FOREIGN KEY ("RoleId") REFERENCES "AspNetRoles" ("Id") ON DELETE CASCADE,
CONSTRAINT "FK_AspNetUserRoles_AspNetUsers_UserId" FOREIGN KEY ("UserId") REFERENCES "AspNetUsers" ("Id") ON DELETE CASCADE
);
CREATE TABLE "AspNetUserTokens" (
"UserId" text NOT NULL,
"LoginProvider" text NOT NULL,
"Name" text NOT NULL,
"Value" text,
CONSTRAINT "PK_AspNetUserTokens" PRIMARY KEY ("UserId", "LoginProvider", "Name"),
CONSTRAINT "FK_AspNetUserTokens_AspNetUsers_UserId" FOREIGN KEY ("UserId") REFERENCES "AspNetUsers" ("Id") ON DELETE CASCADE
);
CREATE INDEX "IX_AspNetRoleClaims_RoleId" ON "AspNetRoleClaims" ("RoleId");
CREATE UNIQUE INDEX "RoleNameIndex" ON "AspNetRoles" ("NormalizedName");
CREATE INDEX "IX_AspNetUserClaims_UserId" ON "AspNetUserClaims" ("UserId");
CREATE INDEX "IX_AspNetUserLogins_UserId" ON "AspNetUserLogins" ("UserId");
CREATE INDEX "IX_AspNetUserRoles_RoleId" ON "AspNetUserRoles" ("RoleId");
CREATE INDEX "EmailIndex" ON "AspNetUsers" ("NormalizedEmail");
CREATE UNIQUE INDEX "UserNameIndex" ON "AspNetUsers" ("NormalizedUserName");
INSERT INTO "__EFMigrationsHistory" ("MigrationId", "ProductVersion") INSERT INTO "__EFMigrationsHistory" ("MigrationId", "ProductVersion")
VALUES ('20250709144855_Initial', '9.0.6'); VALUES ('20250719094443_roleaddchinses', '9.0.6');
COMMIT; COMMIT;