using System.Security.Claims; using JobsMedical.Web.Data; using JobsMedical.Web.Models; using Microsoft.AspNetCore.Authentication; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc.RazorPages; using Microsoft.EntityFrameworkCore; namespace JobsMedical.Web.Pages.Me; [Authorize] public class ProfileModel : PageModel { private readonly AppDbContext _db; public ProfileModel(AppDbContext db) => _db = db; public List Roles { get; private set; } = new(); public List Cities { get; private set; } = new(); public bool HasAvatar { get; private set; } public string? ResumeName { get; private set; } public string Phone { get; private set; } = ""; [TempData] public string? Msg { get; set; } [BindProperty] public string? FullName { get; set; } [BindProperty] public int? RoleId { get; set; } [BindProperty] public int? CityId { get; set; } [BindProperty] public string? Specialty { get; set; } [BindProperty] public string? LicenseNo { get; set; } [BindProperty] public int YearsExperience { get; set; } [BindProperty] public string? Bio { get; set; } private int Uid => int.Parse(User.FindFirstValue(ClaimTypes.NameIdentifier)!); private static readonly string[] ImgTypes = { "image/jpeg", "image/png", "image/webp" }; private static readonly string[] DocTypes = { "image/jpeg", "image/png", "image/webp", "application/pdf" }; private const long MaxImg = 2 * 1024 * 1024; // 2 MB private const long MaxDoc = 5 * 1024 * 1024; // 5 MB public async Task OnGetAsync() { await LoadListsAsync(); var u = await _db.Users.Include(x => x.DoctorProfile).FirstAsync(x => x.Id == Uid); Phone = u.Phone; FullName = u.FullName; HasAvatar = u.Avatar != null; ResumeName = u.ResumeFileName; var p = u.DoctorProfile; RoleId = p?.RoleId; CityId = p?.CityId; Specialty = p?.Specialty; LicenseNo = p?.LicenseNo; YearsExperience = p?.YearsExperience ?? 0; Bio = p?.Bio; } public async Task OnPostAsync(IFormFile? avatar, IFormFile? resume) { var u = await _db.Users.Include(x => x.DoctorProfile).FirstAsync(x => x.Id == Uid); u.FullName = string.IsNullOrWhiteSpace(FullName) ? null : FullName.Trim(); var p = u.DoctorProfile ??= new DoctorProfile { UserId = Uid }; p.RoleId = RoleId; p.CityId = CityId; p.Specialty = string.IsNullOrWhiteSpace(Specialty) ? "پزشک عمومی" : Specialty.Trim(); p.LicenseNo = LicenseNo?.Trim(); p.YearsExperience = Math.Clamp(YearsExperience, 0, 70); p.Bio = Bio?.Trim(); string? warn = null; if (avatar is { Length: > 0 }) { if (avatar.Length > MaxImg || !ImgTypes.Contains((avatar.ContentType ?? "").ToLowerInvariant())) warn = "تصویر باید JPG/PNG/WebP و کمتر از ۲ مگابایت باشد."; else { using var ms = new MemoryStream(); await avatar.CopyToAsync(ms); u.Avatar = ms.ToArray(); u.AvatarContentType = avatar.ContentType!.ToLowerInvariant(); } } if (resume is { Length: > 0 }) { if (resume.Length > MaxDoc || !DocTypes.Contains((resume.ContentType ?? "").ToLowerInvariant())) warn = (warn is null ? "" : warn + " ") + "رزومه باید PDF یا تصویر و کمتر از ۵ مگابایت باشد."; else { using var ms = new MemoryStream(); await resume.CopyToAsync(ms); u.Resume = ms.ToArray(); u.ResumeContentType = resume.ContentType!.ToLowerInvariant(); u.ResumeFileName = Path.GetFileName(resume.FileName); } } await _db.SaveChangesAsync(); Msg = warn ?? "پروفایل ذخیره شد."; return RedirectToPage(); } public async Task OnPostDeleteResumeAsync() { var u = await _db.Users.FirstAsync(x => x.Id == Uid); u.Resume = null; u.ResumeFileName = null; u.ResumeContentType = null; await _db.SaveChangesAsync(); Msg = "رزومه حذف شد."; return RedirectToPage(); } public async Task OnPostDeleteAvatarAsync() { var u = await _db.Users.FirstAsync(x => x.Id == Uid); u.Avatar = null; u.AvatarContentType = null; await _db.SaveChangesAsync(); Msg = "تصویر حذف شد."; return RedirectToPage(); } /// Permanently delete the account + its data (per the privacy policy). public async Task OnPostDeleteAccountAsync() { var uid = Uid; // Detach anonymous browsing history (keep events, drop the user link), then remove the user. await _db.Visitors.Where(v => v.UserId == uid) .ExecuteUpdateAsync(s => s.SetProperty(v => v.UserId, (int?)null)); await _db.Users.Where(u => u.Id == uid).ExecuteDeleteAsync(); // cascades profile/alerts/reviews/applications await HttpContext.SignOutAsync(Microsoft.AspNetCore.Authentication.Cookies.CookieAuthenticationDefaults.AuthenticationScheme); return RedirectToPage("/Index"); } private async Task LoadListsAsync() { Roles = await _db.Roles.Where(r => r.IsActive).OrderBy(r => r.SortOrder).ToListAsync(); Cities = await _db.Cities.Where(c => c.IsActive).OrderBy(c => c.Name).ToListAsync(); } }