using JobsMedical.Web.Data; using JobsMedical.Web.Models; using JobsMedical.Web.Services; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc.RazorPages; using Microsoft.EntityFrameworkCore; namespace JobsMedical.Web.Pages.Jobs; public class IndexModel : PageModel { private readonly AppDbContext _db; public IndexModel(AppDbContext db) => _db = db; [BindProperty(SupportsGet = true)] public int? CityId { get; set; } [BindProperty(SupportsGet = true)] public int? DistrictId { get; set; } [BindProperty(SupportsGet = true)] public int? RoleId { get; set; } [BindProperty(SupportsGet = true)] public EmploymentType? EmploymentType { get; set; } [BindProperty(SupportsGet = true)] public Gender? GenderFilter { get; set; } [BindProperty(SupportsGet = true)] public double? Lat { get; set; } [BindProperty(SupportsGet = true)] public double? Lng { get; set; } // Pretty-URL segments (/استخدام/{roleSlug}/{citySlug?}); resolved to RoleId/CityId below. [BindProperty(SupportsGet = true)] public string? RoleSlug { get; set; } [BindProperty(SupportsGet = true)] public string? CitySlug { get; set; } public bool NearMeActive => Lat is not null && Lng is not null; public List Results { get; private set; } = new(); public List Cities { get; private set; } = new(); public List Districts { get; private set; } = new(); public List Roles { get; private set; } = new(); /// Dynamic page heading/H1 + title, set from the active role/city for SEO. public string PageHeading { get; private set; } = "موقعیت‌های استخدامی"; public async Task OnGetAsync() { Cities = await _db.Cities.Where(c => c.IsActive).OrderBy(c => c.Name).ToListAsync(); Roles = await _db.Roles.Where(r => r.IsActive).OrderBy(r => r.SortOrder).ToListAsync(); // Pretty-URL landing: resolve slugs → filters. A slug matching nothing is a 404 (don't // render a thin page under a junk URL). if (!string.IsNullOrWhiteSpace(RoleSlug)) { var role = Roles.FirstOrDefault(r => SeoSlug.Matches(r.Name, RoleSlug)); if (role is null) return NotFound(); RoleId = role.Id; } if (!string.IsNullOrWhiteSpace(CitySlug)) { var city = Cities.FirstOrDefault(c => SeoSlug.Matches(c.Name, CitySlug)); if (city is null) return NotFound(); CityId = city.Id; } Districts = await _db.Districts .Where(d => d.IsActive && (CityId == null || d.CityId == CityId)) .OrderBy(d => d.Name).ToListAsync(); var q = _db.JobOpenings .Include(j => j.Facility).ThenInclude(f => f.City) .Include(j => j.Facility).ThenInclude(f => f.District) .Include(j => j.Role) .Where(j => j.Status == ShiftStatus.Open && j.CreatedAt >= JobsMedical.Web.Services.Scraping.ListingPolicy.JobCutoffUtc); if (CityId is not null) q = q.Where(j => j.Facility.CityId == CityId); if (DistrictId is not null) q = q.Where(j => j.Facility.DistrictId == DistrictId); if (RoleId is not null) q = q.Where(j => j.RoleId == RoleId); if (EmploymentType is not null) q = q.Where(j => j.EmploymentType == EmploymentType); if (GenderFilter is Gender g && g != Gender.Any) q = q.Where(j => j.GenderRequirement == Gender.Any || j.GenderRequirement == g); var results = await q.ToListAsync(); if (NearMeActive) { foreach (var j in results) if (j.Facility.Lat is double flat && j.Facility.Lng is double flng) j.DistanceKm = Geo.DistanceKm(Lat!.Value, Lng!.Value, flat, flng); Results = results.OrderBy(j => j.DistanceKm ?? double.MaxValue) .ThenByDescending(j => j.CreatedAt).ToList(); } else { Results = results.OrderByDescending(j => j.CreatedAt).ToList(); } SetSeo(Roles.FirstOrDefault(r => r.Id == RoleId)?.Name, Cities.FirstOrDefault(c => c.Id == CityId)?.Name); return Page(); } /// Title/H1/meta from the active role+city so the page targets «استخدام [نقش] [شهر]». private void SetSeo(string? role, string? city) { PageHeading = role is not null && city is not null ? $"استخدام {role} در {city}" : role is not null ? $"استخدام {role}" : city is not null ? $"استخدام کادر درمان در {city}" : "موقعیت‌های استخدامی"; ViewData["Title"] = PageHeading; ViewData["Description"] = role is not null || city is not null ? $"جدیدترین آگهی‌های {PageHeading} در همکادر؛ مشاهده فرصت‌ها و تماس مستقیم با مراکز درمانی." : "موقعیت‌های استخدامی کادر درمان (پزشک، پرستار، ماما، تکنسین) در بیمارستان‌ها و کلینیک‌های تهران — همکادر."; } }