From 6b070a375f80854902f53179a53ee5e1d3f1ee8a Mon Sep 17 00:00:00 2001 From: Vachan <40485260+vd1992@users.noreply.github.com> Date: Thu, 12 Dec 2024 14:38:46 -0700 Subject: [PATCH 01/31] null case added (#12283) --- api/app/Traits/Generator/GeneratesUserDoc.php | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/api/app/Traits/Generator/GeneratesUserDoc.php b/api/app/Traits/Generator/GeneratesUserDoc.php index 27ea407a8c8..1c28ab22622 100644 --- a/api/app/Traits/Generator/GeneratesUserDoc.php +++ b/api/app/Traits/Generator/GeneratesUserDoc.php @@ -337,8 +337,7 @@ public function experience(Section $section, AwardExperience|CommunityExperience $this->localize('experiences.seniority_role'), $this->localizeEnum($experience->ext_role_seniority, ExternalRoleSeniority::class) ); - } - if ($experience->employment_category === EmploymentCategory::CANADIAN_ARMED_FORCES->name) { + } elseif ($experience->employment_category === EmploymentCategory::CANADIAN_ARMED_FORCES->name) { $section->addTitle( sprintf( '%s %s %s', @@ -361,8 +360,7 @@ public function experience(Section $section, AwardExperience|CommunityExperience $this->localize('experiences.rank_category'), $this->localizeEnum($experience->caf_rank, CafRank::class) ); - } - if ($experience->employment_category === EmploymentCategory::GOVERNMENT_OF_CANADA->name) { + } elseif ($experience->employment_category === EmploymentCategory::GOVERNMENT_OF_CANADA->name) { /** @var Department | null $department */ $department = Department::find($experience->department_id); $section->addTitle( @@ -424,6 +422,11 @@ public function experience(Section $section, AwardExperience|CommunityExperience $classification ? $classification->group.'-'.$classification->level : Lang::get('common.not_found', [], $this->lang), ); } + } else { + // null case, so experiences prior to adding employment_category + $section->addTitle($experience->getTitle($this->lang), $headingRank); + $section->addText($experience->getDateRange($this->lang)); + $this->addLabelText($section, $this->localize('experiences.team_group_division'), $experience->division); } } From cff0e28e17d143fdb18aff40ae6a5c68c8d24ae1 Mon Sep 17 00:00:00 2001 From: Eric Sizer Date: Fri, 13 Dec 2024 10:59:45 -0500 Subject: [PATCH 02/31] [Feat] Return locale aware string from `LocalizedString` (#12291) * add custom cast for localized string * update graphql type for localized string * update model casts for localized strings * improve localized string getter * fix tests * fix test name case * fallback to en then null --- api/app/Casts/LocalizedString.php | 40 +++++++++++++++++++++ api/app/Models/AssessmentStep.php | 3 +- api/app/Models/Classification.php | 3 +- api/app/Models/Community.php | 7 ++-- api/app/Models/Department.php | 3 +- api/app/Models/GeneralQuestion.php | 3 +- api/app/Models/GenericJobTitle.php | 3 +- api/app/Models/JobPosterTemplate.php | 17 ++++----- api/app/Models/Pool.php | 17 ++++----- api/app/Models/Role.php | 5 +-- api/app/Models/ScreeningQuestion.php | 3 +- api/app/Models/Skill.php | 5 +-- api/app/Models/SkillFamily.php | 5 +-- api/app/Models/Team.php | 5 +-- api/app/Models/TrainingOpportunity.php | 7 ++-- api/app/Models/WorkStream.php | 5 +-- api/graphql/schema.graphql | 1 + api/storage/app/lighthouse-schema.graphql | 1 + api/tests/Feature/ApplicantFilterTest.php | 3 ++ api/tests/Feature/CommunityTest.php | 5 ++- api/tests/Feature/JobPosterTemplateTest.php | 17 ++++----- api/tests/Feature/PoolCandidateTest.php | 1 + api/tests/Feature/PoolTest.php | 2 +- api/tests/Feature/SnapshotTest.php | 1 + api/tests/Unit/PoolCandidatePolicyTest.php | 4 +-- api/tests/Unit/PoolPolicyTest.php | 4 +-- 26 files changed, 116 insertions(+), 54 deletions(-) create mode 100644 api/app/Casts/LocalizedString.php diff --git a/api/app/Casts/LocalizedString.php b/api/app/Casts/LocalizedString.php new file mode 100644 index 00000000000..a4452a70ff5 --- /dev/null +++ b/api/app/Casts/LocalizedString.php @@ -0,0 +1,40 @@ + $attributes + */ + public function get(Model $model, string $key, mixed $value, array $attributes): mixed + { + if (is_null($value) || $value === '' || $value === 'null') { + return null; + } + + $decodedValue = is_array($value) ? $value : json_decode($value, true); + $locale = App::getLocale() ?? 'en'; + + return [ + ...$decodedValue, + 'localized' => $decodedValue[$locale] ?? null, + ]; + } + + /** + * Prepare the given value for storage. + * + * @param array $attributes + */ + public function set(Model $model, string $key, mixed $value, array $attributes): mixed + { + return json_encode($value); + } +} diff --git a/api/app/Models/AssessmentStep.php b/api/app/Models/AssessmentStep.php index 799e6f3182a..8117cf845f5 100644 --- a/api/app/Models/AssessmentStep.php +++ b/api/app/Models/AssessmentStep.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use App\Enums\AssessmentStepType; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; @@ -33,7 +34,7 @@ class AssessmentStep extends Model * The attributes that should be cast. */ protected $casts = [ - 'title' => 'array', + 'title' => LocalizedString::class, ]; /** diff --git a/api/app/Models/Classification.php b/api/app/Models/Classification.php index 9c93456ec4d..81067b806ef 100644 --- a/api/app/Models/Classification.php +++ b/api/app/Models/Classification.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Casts\Attribute; use Illuminate\Database\Eloquent\Factories\HasFactory; @@ -35,7 +36,7 @@ class Classification extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', + 'name' => LocalizedString::class, ]; /** @return HasMany */ diff --git a/api/app/Models/Community.php b/api/app/Models/Community.php index 45cc0220080..f5221d9afaf 100644 --- a/api/app/Models/Community.php +++ b/api/app/Models/Community.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\HasMany; @@ -26,9 +27,9 @@ class Community extends Model protected $keyType = 'string'; protected $casts = [ - 'name' => 'array', - 'description' => 'array', - 'mandate_authority' => 'array', + 'name' => LocalizedString::class, + 'description' => LocalizedString::class, + 'mandate_authority' => LocalizedString::class, ]; protected $fillable = [ diff --git a/api/app/Models/Department.php b/api/app/Models/Department.php index 809c49e1c6d..9c7a17dc721 100644 --- a/api/app/Models/Department.php +++ b/api/app/Models/Department.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; @@ -32,7 +33,7 @@ class Department extends Model * The attributes that should be case. */ protected $casts = [ - 'name' => 'array', + 'name' => LocalizedString::class, ]; /** @return HasMany */ diff --git a/api/app/Models/GeneralQuestion.php b/api/app/Models/GeneralQuestion.php index b09d79eb24e..4aa24d6b139 100644 --- a/api/app/Models/GeneralQuestion.php +++ b/api/app/Models/GeneralQuestion.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsTo; @@ -30,7 +31,7 @@ class GeneralQuestion extends Model * The attributes that should be cast. */ protected $casts = [ - 'question' => 'array', + 'question' => LocalizedString::class, ]; /** diff --git a/api/app/Models/GenericJobTitle.php b/api/app/Models/GenericJobTitle.php index 2c91bdea3ed..b633a706269 100644 --- a/api/app/Models/GenericJobTitle.php +++ b/api/app/Models/GenericJobTitle.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsTo; @@ -28,7 +29,7 @@ class GenericJobTitle extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', + 'name' => LocalizedString::class, ]; diff --git a/api/app/Models/JobPosterTemplate.php b/api/app/Models/JobPosterTemplate.php index ef459eea91e..e5357629bc4 100644 --- a/api/app/Models/JobPosterTemplate.php +++ b/api/app/Models/JobPosterTemplate.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsTo; @@ -36,14 +37,14 @@ class JobPosterTemplate extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', - 'description' => 'array', - 'work_description' => 'array', - 'tasks' => 'array', - 'keywords' => 'array', - 'essential_technical_skills_notes' => 'array', - 'essential_behavioural_skills_notes' => 'array', - 'nonessential_technical_skills_notes' => 'array', + 'name' => LocalizedString::class, + 'description' => LocalizedString::class, + 'work_description' => LocalizedString::class, + 'tasks' => LocalizedString::class, + 'keywords' => LocalizedString::class, + 'essential_technical_skills_notes' => LocalizedString::class, + 'essential_behavioural_skills_notes' => LocalizedString::class, + 'nonessential_technical_skills_notes' => LocalizedString::class, ]; /** diff --git a/api/app/Models/Pool.php b/api/app/Models/Pool.php index e5a7540a4d2..02dcea8609d 100644 --- a/api/app/Models/Pool.php +++ b/api/app/Models/Pool.php @@ -3,6 +3,7 @@ namespace App\Models; use App\Builders\PoolBuilder; +use App\Casts\LocalizedString; use App\Enums\AssessmentStepType; use App\Enums\PoolSkillType; use App\Enums\PoolStatus; @@ -70,15 +71,15 @@ class Pool extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', + 'name' => LocalizedString::class, 'operational_requirements' => 'array', - 'key_tasks' => 'array', - 'advertisement_location' => 'array', - 'your_impact' => 'array', - 'what_to_expect' => 'array', - 'special_note' => 'array', - 'what_to_expect_admission' => 'array', - 'about_us' => 'array', + 'key_tasks' => LocalizedString::class, + 'advertisement_location' => LocalizedString::class, + 'your_impact' => LocalizedString::class, + 'what_to_expect' => LocalizedString::class, + 'special_note' => LocalizedString::class, + 'what_to_expect_admission' => LocalizedString::class, + 'about_us' => LocalizedString::class, 'closing_date' => 'datetime', 'published_at' => 'datetime', 'is_remote' => 'boolean', diff --git a/api/app/Models/Role.php b/api/app/Models/Role.php index 8b310f1cb85..b1c44accfba 100644 --- a/api/app/Models/Role.php +++ b/api/app/Models/Role.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Relations\HasMany; use Laratrust\Models\Role as LaratrustRole; @@ -24,8 +25,8 @@ class Role extends LaratrustRole protected $keyType = 'string'; protected $casts = [ - 'display_name' => 'array', - 'description' => 'array', + 'display_name' => LocalizedString::class, + 'description' => LocalizedString::class, ]; protected $fillable = [ diff --git a/api/app/Models/ScreeningQuestion.php b/api/app/Models/ScreeningQuestion.php index 2f2b77b3829..b877e9e2d57 100644 --- a/api/app/Models/ScreeningQuestion.php +++ b/api/app/Models/ScreeningQuestion.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsTo; @@ -31,7 +32,7 @@ class ScreeningQuestion extends Model * The attributes that should be cast. */ protected $casts = [ - 'question' => 'array', + 'question' => LocalizedString::class, ]; /** diff --git a/api/app/Models/Skill.php b/api/app/Models/Skill.php index 34a79d6a5ce..46ea9a9f818 100644 --- a/api/app/Models/Skill.php +++ b/api/app/Models/Skill.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use App\Enums\PoolSkillType; use App\Enums\SkillCategory; use Illuminate\Database\Eloquent\Builder; @@ -36,8 +37,8 @@ class Skill extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', - 'description' => 'array', + 'name' => LocalizedString::class, + 'description' => LocalizedString::class, 'keywords' => 'array', ]; diff --git a/api/app/Models/SkillFamily.php b/api/app/Models/SkillFamily.php index 81adac8a8ea..99b8bc8d9ec 100644 --- a/api/app/Models/SkillFamily.php +++ b/api/app/Models/SkillFamily.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsToMany; @@ -28,8 +29,8 @@ class SkillFamily extends Model * The attributes that should be cast. */ protected $casts = [ - 'name' => 'array', - 'description' => 'array', + 'name' => LocalizedString::class, + 'description' => LocalizedString::class, ]; /** @return BelongsToMany */ diff --git a/api/app/Models/Team.php b/api/app/Models/Team.php index b0831fbdee8..dc9a740bd66 100644 --- a/api/app/Models/Team.php +++ b/api/app/Models/Team.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Relations\BelongsToMany; @@ -27,8 +28,8 @@ class Team extends LaratrustTeam protected $keyType = 'string'; protected $casts = [ - 'display_name' => 'array', - 'description' => 'array', + 'display_name' => LocalizedString::class, + 'description' => LocalizedString::class, ]; protected $fillable = [ diff --git a/api/app/Models/TrainingOpportunity.php b/api/app/Models/TrainingOpportunity.php index f3c5a7904b2..1f4b3dbff3a 100644 --- a/api/app/Models/TrainingOpportunity.php +++ b/api/app/Models/TrainingOpportunity.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use App\Enums\CourseLanguage; use App\Enums\DeadlineStatus; use Illuminate\Database\Eloquent\Builder; @@ -34,12 +35,12 @@ class TrainingOpportunity extends Model * The attributes that should be cast. */ protected $casts = [ - 'title' => 'array', + 'title' => LocalizedString::class, 'registration_deadline' => 'date', 'training_start' => 'date', 'training_end' => 'date', - 'description' => 'array', - 'application_url' => 'array', + 'description' => LocalizedString::class, + 'application_url' => LocalizedString::class, ]; /** diff --git a/api/app/Models/WorkStream.php b/api/app/Models/WorkStream.php index 093a2b01c78..66608ee4e40 100644 --- a/api/app/Models/WorkStream.php +++ b/api/app/Models/WorkStream.php @@ -2,6 +2,7 @@ namespace App\Models; +use App\Casts\LocalizedString; use Illuminate\Database\Eloquent\Factories\HasFactory; use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Relations\BelongsTo; @@ -24,8 +25,8 @@ class WorkStream extends Model protected $keyType = 'string'; protected $casts = [ - 'name' => 'array', - 'plain_language_name' => 'array', + 'name' => LocalizedString::class, + 'plain_language_name' => LocalizedString::class, ]; protected $fillable = [ diff --git a/api/graphql/schema.graphql b/api/graphql/schema.graphql index 2084f9de921..e0ffb458613 100644 --- a/api/graphql/schema.graphql +++ b/api/graphql/schema.graphql @@ -23,6 +23,7 @@ scalar UUID @scalar(class: "UUID") type LocalizedString { en: String fr: String + localized: String } type LocalizedEnumString { diff --git a/api/storage/app/lighthouse-schema.graphql b/api/storage/app/lighthouse-schema.graphql index 351b417b9ac..c537c8f302d 100755 --- a/api/storage/app/lighthouse-schema.graphql +++ b/api/storage/app/lighthouse-schema.graphql @@ -540,6 +540,7 @@ scalar UUID type LocalizedString { en: String fr: String + localized: String } type LocalizedEnumString { diff --git a/api/tests/Feature/ApplicantFilterTest.php b/api/tests/Feature/ApplicantFilterTest.php index 4a6d8663900..e83989dab30 100644 --- a/api/tests/Feature/ApplicantFilterTest.php +++ b/api/tests/Feature/ApplicantFilterTest.php @@ -273,6 +273,7 @@ public function testQueryRelationships() name { en fr + localized } } pools { @@ -280,6 +281,7 @@ public function testQueryRelationships() name { en fr + localized } } qualifiedStreams { value } @@ -288,6 +290,7 @@ public function testQueryRelationships() name { en fr + localized } } community { diff --git a/api/tests/Feature/CommunityTest.php b/api/tests/Feature/CommunityTest.php index bd892884644..ea828be181f 100644 --- a/api/tests/Feature/CommunityTest.php +++ b/api/tests/Feature/CommunityTest.php @@ -47,11 +47,10 @@ protected function setUp(): void // Create communities. $this->toBeDeletedUUID = $this->faker->UUID(); - $this->community1 = Community::factory()->create(['name' => 'community1']); - $this->community2 = Community::factory()->create(['name' => 'community2']); + $this->community1 = Community::factory()->create(); + $this->community2 = Community::factory()->create(); $this->community3 = Community::factory()->create([ 'id' => $this->toBeDeletedUUID, // need specific ID for delete community testing. - 'name' => 'community3', ]); // Create users. diff --git a/api/tests/Feature/JobPosterTemplateTest.php b/api/tests/Feature/JobPosterTemplateTest.php index b41223dfb3a..b41a1f80ada 100644 --- a/api/tests/Feature/JobPosterTemplateTest.php +++ b/api/tests/Feature/JobPosterTemplateTest.php @@ -13,6 +13,7 @@ use Database\Seeders\SkillSeeder; use Database\Seeders\WorkStreamSeeder; use Illuminate\Foundation\Testing\RefreshDatabase; +use Illuminate\Support\Arr; use Nuwave\Lighthouse\Testing\MakesGraphQLRequests; use Nuwave\Lighthouse\Testing\RefreshesSchemaCache; use Tests\TestCase; @@ -301,16 +302,16 @@ private function getCreateInput(): array return [ 'referenceId' => $template->reference_id, - 'name' => $template->name, - 'description' => $template->description, + 'name' => Arr::only($template->name, ['en', 'fr']), + 'description' => Arr::only($template->description, ['en', 'fr']), 'supervisoryStatus' => $template->supervisory_status, 'stream' => $template->stream, - 'tasks' => $template->tasks, - 'workDescription' => $template->work_description, - 'keywords' => $template->keywords, - 'essentialBehaviouralSkillsNotes' => $template->essential_behavioural_skills_notes, - 'essentialTechnicalSkillsNotes' => $template->essential_technical_skills_notes, - 'nonessentialTechnicalSkillsNotes' => $template->nonessential_technical_skills_notes, + 'tasks' => Arr::only($template->tasks, ['en', 'fr']), + 'workDescription' => Arr::only($template->work_description, ['en', 'fr']), + 'keywords' => Arr::only($template->keywords, ['en', 'fr']), + 'essentialBehaviouralSkillsNotes' => Arr::only($template->essential_behavioural_skills_notes, ['en', 'fr']), + 'essentialTechnicalSkillsNotes' => Arr::only($template->essential_technical_skills_notes, ['en', 'fr']), + 'nonessentialTechnicalSkillsNotes' => Arr::only($template->nonessential_technical_skills_notes, ['en', 'fr']), 'classification' => [ 'connect' => $template->classification->id, ], diff --git a/api/tests/Feature/PoolCandidateTest.php b/api/tests/Feature/PoolCandidateTest.php index 3428eb22f83..2451366fe7f 100644 --- a/api/tests/Feature/PoolCandidateTest.php +++ b/api/tests/Feature/PoolCandidateTest.php @@ -639,6 +639,7 @@ public function testOrderByPoolName(): void name { en fr + localized } } } diff --git a/api/tests/Feature/PoolTest.php b/api/tests/Feature/PoolTest.php index 1fcf719ac4e..f0cbc9292fc 100644 --- a/api/tests/Feature/PoolTest.php +++ b/api/tests/Feature/PoolTest.php @@ -1099,7 +1099,7 @@ public function testPoolNameScope(): void poolsPaginated(where: $where) { data { id - name { en fr } + name { en fr localized } } } } diff --git a/api/tests/Feature/SnapshotTest.php b/api/tests/Feature/SnapshotTest.php index 83e10b6d485..9e9b6f89075 100644 --- a/api/tests/Feature/SnapshotTest.php +++ b/api/tests/Feature/SnapshotTest.php @@ -88,6 +88,7 @@ public function testCreateSnapshot() )->json('data.poolCandidate.profileSnapshot'); $decodedActual = json_decode($actualSnapshot, true); + unset($decodedActual['pool']['department']['name']['localized']); // Add version number $expectedSnapshot['version'] = ProfileSnapshot::$VERSION; diff --git a/api/tests/Unit/PoolCandidatePolicyTest.php b/api/tests/Unit/PoolCandidatePolicyTest.php index 76f94f6bb35..a8068a1be4d 100644 --- a/api/tests/Unit/PoolCandidatePolicyTest.php +++ b/api/tests/Unit/PoolCandidatePolicyTest.php @@ -73,8 +73,8 @@ protected function setUp(): void ]); $this->team = Team::factory()->create(['name' => 'test-team']); - $this->community = Community::factory()->create(['name' => 'test-team']); - $this->otherCommunity = Community::factory()->create(['name' => 'suspicious-team']); + $this->community = Community::factory()->create(['name' => ['en' => 'test-team EN', 'fr' => 'test-team FR']]); + $this->otherCommunity = Community::factory()->create(['name' => ['en' => 'suspicious-team EN', 'fr' => 'suspicious-team FR']]); $this->poolOperatorUser = User::factory() ->asPoolOperator($this->team->name) diff --git a/api/tests/Unit/PoolPolicyTest.php b/api/tests/Unit/PoolPolicyTest.php index 6f0bca92972..688cfd58aa3 100644 --- a/api/tests/Unit/PoolPolicyTest.php +++ b/api/tests/Unit/PoolPolicyTest.php @@ -67,8 +67,8 @@ protected function setUp(): void ]); $this->team = Team::factory()->create(['name' => 'test-team']); - $this->community = Community::factory()->create(['name' => 'test-team']); - $this->otherCommunity = Community::factory()->create(['name' => 'suspicious-team']); + $this->community = Community::factory()->create(); + $this->otherCommunity = Community::factory()->create(); $this->poolOperatorUser = User::factory() ->asApplicant() From 5041102fce22f588b346806185457a627a77fda1 Mon Sep 17 00:00:00 2001 From: Yoni K Date: Fri, 13 Dec 2024 11:02:42 -0500 Subject: [PATCH 03/31] [Feature] Update work experience UI (#12208) * Add new fields to work experience form * Update work content on experience card * Add sentence to help text in additional details section * Add new fields to queries * Split up work content by category * Fix layout on work fields * Fix classification level * Add translations * Fix resetting fields on initial load when editting * Fix lint error * Add new tests for work experiences * Remove redundant fallback message * Refactor gov fields * Add decorative prop to separators * Remove else statement for style * Comment out edit test code * Remove division from caf form * Move term check in gov content * Improve useWatch types * Refactor end date logic * Remove redundant notEmpty helper * Fix tests * Make employment category required * Fix duplicate translations * Add substative not_in rule to validation * Add php unit test * Update classification and department in schema and validator * Update work fields * Add descriptions to employment categories * Update PHP test * Update experience name on link career timeline page * Only validate employyment category if present * Fix lint error * Add more translations * Add e2e test to edit work experience * Remove duplicate translation * Remove export on ExperienceFormProps --------- Co-authored-by: Yoni K --- .../CreateUpdateWorkExperienceValidator.php | 44 +- api/graphql/schema.graphql | 8 +- api/storage/app/lighthouse-schema.graphql | 4 +- api/tests/Feature/WorkExperienceTest.php | 31 +- apps/playwright/fixtures/ExperiencePage.ts | 429 +++++++++++++++- .../tests/applicant/experience.spec.ts | 123 ++++- .../ExperienceCard/ExperienceCard.tsx | 53 +- .../components/ExperienceCard/WorkContent.tsx | 36 +- .../ExperienceCard/WorkContent/CafContent.tsx | 40 ++ .../WorkContent/ExternalContent.tsx | 52 ++ .../ExperienceCard/WorkContent/GovContent.tsx | 191 ++++++++ .../AdditionalDetails.tsx | 4 +- .../ExperienceDetails.tsx | 2 +- .../ExperienceFormFields/WorkFields.tsx | 102 ---- .../WorkFields/CafFields.tsx | 153 ++++++ .../WorkFields/ExternalFields.tsx | 159 ++++++ .../WorkFields/GovFields.tsx | 456 ++++++++++++++++++ .../WorkFields/WorkFields.tsx | 194 ++++++++ .../PoolCandidatesTable/SkillMatchDialog.tsx | 84 ++++ apps/web/src/lang/fr.json | 96 +++- apps/web/src/lang/whitelist.yml | 1 + apps/web/src/messages/experienceMessages.ts | 10 + .../LinkCareerTimeline.tsx | 11 +- apps/web/src/pages/Applications/fragment.ts | 168 +++++++ .../CareerTimelineAndRecruitment.tsx | 84 ++++ .../ExperienceForm.test.tsx | 393 --------------- .../ExperienceFormPage/ExperienceFormPage.tsx | 86 +++- .../src/pages/Skills/UpdateUserSkillPage.tsx | 168 +++++++ .../AdminUserProfilePage.tsx | 84 ++++ .../UserInformationPage.tsx | 84 ++++ apps/web/src/types/experience.ts | 66 ++- apps/web/src/utils/experienceUtils.tsx | 219 ++++++++- 32 files changed, 3043 insertions(+), 592 deletions(-) create mode 100644 apps/web/src/components/ExperienceCard/WorkContent/CafContent.tsx create mode 100644 apps/web/src/components/ExperienceCard/WorkContent/ExternalContent.tsx create mode 100644 apps/web/src/components/ExperienceCard/WorkContent/GovContent.tsx delete mode 100644 apps/web/src/components/ExperienceFormFields/WorkFields.tsx create mode 100644 apps/web/src/components/ExperienceFormFields/WorkFields/CafFields.tsx create mode 100644 apps/web/src/components/ExperienceFormFields/WorkFields/ExternalFields.tsx create mode 100644 apps/web/src/components/ExperienceFormFields/WorkFields/GovFields.tsx create mode 100644 apps/web/src/components/ExperienceFormFields/WorkFields/WorkFields.tsx delete mode 100644 apps/web/src/pages/Profile/ExperienceFormPage/ExperienceForm.test.tsx diff --git a/api/app/GraphQL/Validators/CreateUpdateWorkExperienceValidator.php b/api/app/GraphQL/Validators/CreateUpdateWorkExperienceValidator.php index a4fd70c1fcd..6f81e3c4eb5 100644 --- a/api/app/GraphQL/Validators/CreateUpdateWorkExperienceValidator.php +++ b/api/app/GraphQL/Validators/CreateUpdateWorkExperienceValidator.php @@ -4,6 +4,7 @@ use App\Enums\EmploymentCategory; use App\Enums\GovContractorType; +use App\Enums\GovPositionType; use App\Enums\WorkExperienceGovEmployeeType; use Illuminate\Validation\Rule; use Nuwave\Lighthouse\Validation\Validator; @@ -18,9 +19,10 @@ final class CreateUpdateWorkExperienceValidator extends Validator public function rules(): array { return [ - // 'workExperience.employmentCategory' => [ - // 'required', - // ], + 'workExperience.employmentCategory' => [ + 'sometimes', + 'required', + ], 'workExperience.extSizeOfOrganization' => [ Rule::requiredIf( ( @@ -60,7 +62,8 @@ public function rules(): array 'workExperience.govPositionType' => [ Rule::requiredIf( ( - $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::INDETERMINATE->name + $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::INDETERMINATE->name || + $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::TERM->name ) ), Rule::prohibitedIf( @@ -68,6 +71,7 @@ public function rules(): array $this->arg('workExperience.employmentCategory') !== EmploymentCategory::GOVERNMENT_OF_CANADA->name ) ), + $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::TERM->name ? Rule::notIn(GovPositionType::SUBSTANTIVE->name) : null, ], 'workExperience.govContractorRoleSeniority' => [ Rule::requiredIf( @@ -141,8 +145,10 @@ public function rules(): array ) ), ], - 'workExperience.classification' => [ + 'workExperience.classificationId' => [ Rule::requiredIf( + $this->arg('workExperience.employmentCategory') === EmploymentCategory::GOVERNMENT_OF_CANADA->name + && ( $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::CASUAL->name ) || @@ -153,31 +159,15 @@ public function rules(): array $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::INDETERMINATE->name ) ), - Rule::prohibitedIf( - ( - $this->arg('workExperience.employmentCategory') !== EmploymentCategory::GOVERNMENT_OF_CANADA->name - ) - ), - Rule::exists('classifications', 'id'), + // This does not work without proper connect/disconnect in the schema + // Rule::exists('classifications', 'id'), ], - 'workExperience.department' => [ + 'workExperience.departmentId' => [ Rule::requiredIf( - ( - $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::CASUAL->name - ) || - ( - $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::TERM->name - ) || - ( - $this->arg('workExperience.govEmploymentType') === WorkExperienceGovEmployeeType::INDETERMINATE->name - ) - ), - Rule::prohibitedIf( - ( - $this->arg('workExperience.employmentCategory') !== EmploymentCategory::GOVERNMENT_OF_CANADA->name - ) + $this->arg('workExperience.employmentCategory') === EmploymentCategory::GOVERNMENT_OF_CANADA->name ), - Rule::exists('departments', 'id'), + // This does not work without proper connect/disconnect in the schema + // Rule::exists('departments', 'id'), ], ]; } diff --git a/api/graphql/schema.graphql b/api/graphql/schema.graphql index e0ffb458613..1d63db08d32 100644 --- a/api/graphql/schema.graphql +++ b/api/graphql/schema.graphql @@ -1695,12 +1695,8 @@ input WorkExperienceInput { @rename(attribute: "contractor_firm_agency_name") cafForce: CafForce @rename(attribute: "caf_force") cafRank: CafRank @rename(attribute: "caf_rank") - classification: ClassificationBelongsTo - @pluck(key: "connect") - @rename(attribute: "classification_id") - department: DepartmentBelongsTo - @pluck(key: "connect") - @rename(attribute: "department_id") + classificationId: String @rename(attribute: "classification_id") + departmentId: String @rename(attribute: "department_id") } input PersonalExperienceInput { title: String diff --git a/api/storage/app/lighthouse-schema.graphql b/api/storage/app/lighthouse-schema.graphql index c537c8f302d..97b3f979f44 100755 --- a/api/storage/app/lighthouse-schema.graphql +++ b/api/storage/app/lighthouse-schema.graphql @@ -1736,8 +1736,8 @@ input WorkExperienceInput { contractorFirmAgencyName: String cafForce: CafForce cafRank: CafRank - classification: ClassificationBelongsTo - department: DepartmentBelongsTo + classificationId: String + departmentId: String } input PersonalExperienceInput { diff --git a/api/tests/Feature/WorkExperienceTest.php b/api/tests/Feature/WorkExperienceTest.php index 8a4edc64cf9..cf463bfee8c 100644 --- a/api/tests/Feature/WorkExperienceTest.php +++ b/api/tests/Feature/WorkExperienceTest.php @@ -6,6 +6,7 @@ use App\Enums\EmploymentCategory; use App\Enums\ExternalRoleSeniority; use App\Enums\ExternalSizeOfOrganization; +use App\Enums\GovPositionType; use App\Enums\WorkExperienceGovEmployeeType; use App\Models\Classification; use App\Models\Department; @@ -146,6 +147,31 @@ public function testCreatingExperienceFailsValidatingRequired(): void )->assertGraphQLValidationError('workExperience.extRoleSeniority', 'The work experience.ext role seniority field is required.'); } + // test that validation rejects creating experiences with substantive position type when gov employee type is term + public function testCreatingExperienceFailsValidatingGovPositionType(): void + { + $this->actingAs($this->admin, 'api')->graphQL( + /** @lang GraphQL */ + ' + mutation createWorkExperience($userId: ID!, $workExperience: WorkExperienceInput!) { + createWorkExperience(userId: $userId, workExperience: $workExperience) { + user { + id + } + } + } + ', + [ + 'userId' => $this->admin->id, + 'workExperience' => [ + 'employmentCategory' => EmploymentCategory::GOVERNMENT_OF_CANADA->name, + 'govEmploymentType' => WorkExperienceGovEmployeeType::TERM->name, + 'govPositionType' => GovPositionType::SUBSTANTIVE->name, + ], + ] + )->assertGraphQLValidationError('workExperience.govPositionType', 'The selected work experience.gov position type is invalid.'); + } + // test that a created work experience of government type queries without issue public function testQueryingCreatedExperienceGovernment(): void { @@ -169,8 +195,9 @@ public function testQueryingCreatedExperienceGovernment(): void 'workExperience' => [ 'employmentCategory' => EmploymentCategory::GOVERNMENT_OF_CANADA->name, 'govEmploymentType' => WorkExperienceGovEmployeeType::TERM->name, - 'classification' => ['connect' => $classification->id], - 'department' => ['connect' => $department->id], + 'govPositionType' => GovPositionType::ACTING->name, + 'classificationId' => $classification->id, + 'departmentId' => $department->id, ], ] )->assertJsonFragment( diff --git a/apps/playwright/fixtures/ExperiencePage.ts b/apps/playwright/fixtures/ExperiencePage.ts index 235022e06e6..0fe6bdf3d31 100644 --- a/apps/playwright/fixtures/ExperiencePage.ts +++ b/apps/playwright/fixtures/ExperiencePage.ts @@ -1,4 +1,4 @@ -import { Locator, type Page } from "@playwright/test"; +import { Locator, type Page, expect } from "@playwright/test"; import { InputMaybe, @@ -41,7 +41,12 @@ class ExperiencePage extends AppPage { await this.waitForGraphqlResponse("ExperienceFormData"); } - async addWorkExperience(input: WorkExperienceInput) { + async edit(id: string) { + await this.page.goto(`/en/applicant/career-timeline/${id}/edit`); + await this.waitForGraphqlResponse("ExperienceFormData"); + } + + async addExternalWorkExperience(input: WorkExperienceInput) { await this.create(); await this.typeLocator.selectOption("work"); @@ -49,16 +54,148 @@ class ExperiencePage extends AppPage { .getByRole("textbox", { name: /my role/i }) .fill(input.role ?? "test role"); + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /This was a role with an external organization/i, + }) + .click(); + await this.page .getByRole("textbox", { name: /organization/i }) - .fill(input?.organization ?? "test org"); + .fill(input.organization ?? "test org"); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + await this.page + .getByRole("group", { name: /size of the organization/i }) + .getByRole("radio", { + name: /1-35 employees/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /seniority of the role/i }) + .getByRole("radio", { + name: /intern or co-op/i, + }) + .click(); - if (input.division) { + await this.fillDate(input.startDate); + + if (!input.endDate) { await this.page - .getByRole("textbox", { name: /team, group, or division/i }) - .fill(input.division); + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + } else { + await this.fillDate(input.endDate, true); } + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("CreateWorkExperience"); + } + + async addGovStudentWorkExperience(input: WorkExperienceInput) { + await this.create(); + await this.typeLocator.selectOption("work"); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the government of canada/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /department/i }) + .selectOption({ label: "Treasury Board Secretariat" }); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /student/i, + }) + .click(); + + await this.fillDate(input.startDate); + + await this.page + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + + // Ensure label changes to "Expected end date" when currently active in the role is selected + await expect( + this.page.getByRole("group", { name: /end date/i }), + ).toContainText("Expected end date"); + + await this.fillDate(input.endDate, true); + + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("CreateWorkExperience"); + } + + async addGovCasualWorkExperience(input: WorkExperienceInput) { + await this.create(); + await this.typeLocator.selectOption("work"); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the government of canada/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /department/i }) + .selectOption({ label: "Treasury Board Secretariat" }); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /casual/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /group/i }) + .selectOption({ label: "IT" }); + await this.page + .getByRole("combobox", { name: /level/i }) + .selectOption({ label: "1" }); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /casual/i, + }) + .click(); + await this.fillDate(input.startDate); if (!input.endDate) { @@ -77,6 +214,284 @@ class ExperiencePage extends AppPage { await this.waitForGraphqlResponse("CreateWorkExperience"); } + async addGovTermOrIndeterminateWorkExperience(input: WorkExperienceInput) { + await this.create(); + await this.typeLocator.selectOption("work"); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the government of canada/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /department/i }) + .selectOption({ label: "Treasury Board Secretariat" }); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + // Set the employment type to "Term" + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /^term$/i, + }) + .click(); + + // Ensure "Substantive" option is removed from position type group + // when employment type is "Term" + await expect( + this.page.getByRole("group", { name: /position type/i }), + ).not.toContainText("Substantive"); + + await this.page + .getByRole("group", { name: /position type/i }) + .getByRole("radio", { + name: /acting/i, + }) + .click(); + + // Change the employment type to "Indeterminate" + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /^indeterminate$/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /position type/i }) + .getByRole("radio", { + name: /substantive/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /group/i }) + .selectOption({ label: "IT" }); + await this.page + .getByRole("combobox", { name: /level/i }) + .selectOption({ label: "1" }); + + await this.fillDate(input.startDate); + + if (!input.endDate) { + await this.page + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + } else { + await this.fillDate(input.endDate, true); + } + + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("CreateWorkExperience"); + } + + async addGovContractorWorkExperience(input: WorkExperienceInput) { + await this.create(); + await this.typeLocator.selectOption("work"); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the government of canada/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /department/i }) + .selectOption({ label: "Treasury Board Secretariat" }); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + // Set the employment type to "Term" + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /contractor/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /seniority of the role/i }) + .getByRole("radio", { + name: /intern or co-op/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /contractor type/i }) + .getByRole("radio", { + name: /self-employed/i, + }) + .click(); + + // Ensure contracting firm or agency text input isn't rendered + await expect( + this.page.getByRole("textbox", { name: /contracting firm or agency/i }), + ).toBeHidden(); + + await this.page + .getByRole("group", { name: /contractor type/i }) + .getByRole("radio", { + name: /firm or agency/i, + }) + .click(); + + await this.page + .getByRole("textbox", { name: /contracting firm or agency/i }) + .fill("test contracting firm"); + + await this.fillDate(input.startDate); + + if (!input.endDate) { + await this.page + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + } else { + await this.fillDate(input.endDate, true); + } + + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("CreateWorkExperience"); + } + + async addCafWorkExperience(input: WorkExperienceInput) { + await this.create(); + await this.typeLocator.selectOption("work"); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the canadian armed forces/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /regular force/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /military force/i }) + .getByRole("radio", { + name: /canadian army/i, + }) + .click(); + + await this.page + .getByRole("group", { name: /rank category/i }) + .getByRole("radio", { + name: /general or flag officer/i, + }) + .click(); + + await this.fillDate(input.startDate); + + if (!input.endDate) { + await this.page + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + } else { + await this.fillDate(input.endDate, true); + } + + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("CreateWorkExperience"); + } + + async editWorkExperience(id: string, input: WorkExperienceInput) { + await this.edit(id); + + await this.page + .getByRole("textbox", { name: /my role/i }) + .fill(input.role ?? "edit test role"); + + await this.page + .getByRole("group", { name: /employment category/i }) + .getByRole("radio", { + name: /this was a role with the government of canada/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /department/i }) + .selectOption({ label: "Treasury Board Secretariat" }); + + await this.page + .getByRole("textbox", { name: /team, group, or division/i }) + .fill(input.division ?? "test team"); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /casual/i, + }) + .click(); + + await this.page + .getByRole("combobox", { name: /group/i }) + .selectOption({ label: "IT" }); + await this.page + .getByRole("combobox", { name: /level/i }) + .selectOption({ label: "1" }); + + await this.page + .getByRole("group", { name: /employment type/i }) + .getByRole("radio", { + name: /casual/i, + }) + .click(); + + await this.fillDate(input.startDate); + + if (!input.endDate) { + await this.page + .getByRole("checkbox", { name: /i am currently active in this role/i }) + .click(); + } else { + await this.fillDate(input.endDate, true); + } + + await this.page + .getByRole("textbox", { name: /additional details/i }) + .fill(input.details ?? "test details"); + + await this.save(); + await this.waitForGraphqlResponse("UpdateWorkExperience"); + } + async addPersonalExperience(input: PersonalExperienceInput) { await this.create(); await this.typeLocator.selectOption("personal"); @@ -223,7 +638,7 @@ class ExperiencePage extends AppPage { await this.waitForGraphqlResponse("CreateEducationExperience"); } - async linkSkilltoExperience(input: { + async linkSkillToExperience(input: { experienceType: string; skill: string; }) { diff --git a/apps/playwright/tests/applicant/experience.spec.ts b/apps/playwright/tests/applicant/experience.spec.ts index 7972623de1f..32dfe3d7009 100644 --- a/apps/playwright/tests/applicant/experience.spec.ts +++ b/apps/playwright/tests/applicant/experience.spec.ts @@ -1,16 +1,20 @@ +import { WorkExperience } from "@gc-digital-talent/graphql"; + import { test, expect } from "~/fixtures"; import ExperiencePage from "~/fixtures/ExperiencePage"; import { loginBySub } from "~/utils/auth"; +import graphql from "~/utils/graphql"; +import { me } from "~/utils/user"; test.describe("Experiences", () => { const uniqueTestId = Date.now().valueOf(); - test("Can create work experience", async ({ appPage }) => { - const role = `Test add work experience (${uniqueTestId})`; + test("Can create external work experience", async ({ appPage }) => { + const role = `Test add external work experience (${uniqueTestId})`; const experiencePage = new ExperiencePage(appPage.page); await loginBySub(experiencePage.page, "applicant@test.com"); - await experiencePage.addWorkExperience({ + await experiencePage.addExternalWorkExperience({ role, startDate: "2001-01", }); @@ -20,6 +24,117 @@ test.describe("Experiences", () => { ); }); + test("Can create goc student work experience", async ({ appPage }) => { + const role = `Test add goc student work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addGovStudentWorkExperience({ + role, + startDate: "2001-01", + endDate: "2200-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + }); + + test("Can create goc casual work experience", async ({ appPage }) => { + const role = `Test add goc casual work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addGovCasualWorkExperience({ + role, + startDate: "2001-01", + endDate: "2023-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + }); + + test("Can create goc term or indeterminate work experience", async ({ + appPage, + }) => { + const role = `Test add goc term or indeterminate work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addGovTermOrIndeterminateWorkExperience({ + role, + startDate: "2001-01", + endDate: "2023-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + }); + + test("Can create goc contractor work experience", async ({ appPage }) => { + const role = `Test add goc contractor work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addGovContractorWorkExperience({ + role, + startDate: "2001-01", + endDate: "2023-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + }); + + test("Can create caf work experience", async ({ appPage }) => { + const role = `Test add caf work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addCafWorkExperience({ + role, + startDate: "2001-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + }); + + test("Can edit work experience", async ({ appPage }) => { + const role = `Test edit work experience (${uniqueTestId})`; + const experiencePage = new ExperiencePage(appPage.page); + await loginBySub(experiencePage.page, "applicant@test.com"); + + await experiencePage.addCafWorkExperience({ + role, + startDate: "2001-01", + }); + + await expect(experiencePage.page.getByRole("alert")).toContainText( + /successfully added experience/i, + ); + + await experiencePage.goToIndex(); + + const applicantCtx = await graphql.newContext("applicant@test.com"); + const applicant = await me(applicantCtx, {}); + + const workExperience = applicant.experiences?.find( + (ex: WorkExperience) => ex?.role === role, + ); + + await experiencePage.editWorkExperience(`${workExperience?.id}`, { + role, + startDate: "2001-01", + endDate: "2200-01", + }); + }); + test("Can create personal experience", async ({ appPage }) => { const title = `Test add personal experience (${uniqueTestId})`; const experiencePage = new ExperiencePage(appPage.page); @@ -84,7 +199,7 @@ test.describe("Experiences", () => { const skill = "Courage"; - await experiencePage.linkSkilltoExperience({ + await experiencePage.linkSkillToExperience({ experienceType: "work", skill: skill, }); diff --git a/apps/web/src/components/ExperienceCard/ExperienceCard.tsx b/apps/web/src/components/ExperienceCard/ExperienceCard.tsx index febc5102170..bd9d1c6cd9b 100644 --- a/apps/web/src/components/ExperienceCard/ExperienceCard.tsx +++ b/apps/web/src/components/ExperienceCard/ExperienceCard.tsx @@ -15,7 +15,11 @@ import { useControllableState, } from "@gc-digital-talent/ui"; import { commonMessages, getLocalizedName } from "@gc-digital-talent/i18n"; -import { Skill } from "@gc-digital-talent/graphql"; +import { + EmploymentCategory, + Skill, + WorkExperienceGovEmployeeType, +} from "@gc-digital-talent/graphql"; import { AnyExperience } from "~/types/experience"; import { @@ -165,6 +169,53 @@ const ExperienceCard = ({ data-h2-color="base(black.light)" > {typeMessage} + {isWorkExperience(experience) && + experience.employmentCategory?.value === + EmploymentCategory.GovernmentOfCanada && ( + <> + + + {intl.formatMessage({ + defaultMessage: "Government of Canada", + id: "OKqOVT", + description: + "Label for goc employment category on work experience card metadata", + })} + + + )} + {isWorkExperience(experience) && + experience.employmentCategory?.value === + EmploymentCategory.GovernmentOfCanada && + experience.govEmploymentType?.value === + WorkExperienceGovEmployeeType.Contractor && ( + <> + + + {intl.formatMessage({ + defaultMessage: "Contractor", + id: "dpZ2B9", + description: + "Label for contractor employment type on work experience card metadata", + })} + + + )} + {isWorkExperience(experience) && + experience.employmentCategory?.value === + EmploymentCategory.CanadianArmedForces && ( + <> + + + {intl.formatMessage({ + defaultMessage: "Canadian Armed Forces", + id: "dBpcNA", + description: + "Label for caf employment category on work experience card metadata", + })} + + + )} {date && ( <> diff --git a/apps/web/src/components/ExperienceCard/WorkContent.tsx b/apps/web/src/components/ExperienceCard/WorkContent.tsx index 078a7fa08ae..6ab7b9d4751 100644 --- a/apps/web/src/components/ExperienceCard/WorkContent.tsx +++ b/apps/web/src/components/ExperienceCard/WorkContent.tsx @@ -1,28 +1,44 @@ import { useIntl } from "react-intl"; import { commonMessages } from "@gc-digital-talent/i18n"; -import { WorkExperience } from "@gc-digital-talent/graphql"; +import { EmploymentCategory, WorkExperience } from "@gc-digital-talent/graphql"; import { getExperienceFormLabels } from "~/utils/experienceUtils"; import ContentSection from "./ContentSection"; import { ContentProps } from "./types"; +import ExternalContent from "./WorkContent/ExternalContent"; +import CafContent from "./WorkContent/CafContent"; +import GovContent from "./WorkContent/GovContent"; const WorkContent = ({ - experience: { division }, + experience, headingLevel, }: ContentProps>) => { const intl = useIntl(); const experienceFormLabels = getExperienceFormLabels(intl); + const { division, employmentCategory } = experience; - return ( - - {division ?? intl.formatMessage(commonMessages.notAvailable)} - - ); + switch (employmentCategory?.value) { + case EmploymentCategory.ExternalOrganization: + return ( + + ); + case EmploymentCategory.GovernmentOfCanada: + return ; + case EmploymentCategory.CanadianArmedForces: + return ; + default: + return ( + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + ); + } }; export default WorkContent; diff --git a/apps/web/src/components/ExperienceCard/WorkContent/CafContent.tsx b/apps/web/src/components/ExperienceCard/WorkContent/CafContent.tsx new file mode 100644 index 00000000000..a3966432305 --- /dev/null +++ b/apps/web/src/components/ExperienceCard/WorkContent/CafContent.tsx @@ -0,0 +1,40 @@ +import { useIntl } from "react-intl"; + +import { getLocalizedName } from "@gc-digital-talent/i18n"; +import { WorkExperience } from "@gc-digital-talent/graphql"; +import { Separator } from "@gc-digital-talent/ui"; + +import { getExperienceFormLabels } from "~/utils/experienceUtils"; + +import ContentSection from "../ContentSection"; +import { ContentProps } from "../types"; + +const CafContent = ({ + experience: { cafEmploymentType, cafRank }, + headingLevel, +}: ContentProps>) => { + const intl = useIntl(); + const experienceFormLabels = getExperienceFormLabels(intl); + + return ( + <> + + {getLocalizedName(cafEmploymentType?.label, intl)} + + + + {getLocalizedName(cafRank?.label, intl)} + + + ); +}; + +export default CafContent; diff --git a/apps/web/src/components/ExperienceCard/WorkContent/ExternalContent.tsx b/apps/web/src/components/ExperienceCard/WorkContent/ExternalContent.tsx new file mode 100644 index 00000000000..dca72b5b7d2 --- /dev/null +++ b/apps/web/src/components/ExperienceCard/WorkContent/ExternalContent.tsx @@ -0,0 +1,52 @@ +import { useIntl } from "react-intl"; + +import { commonMessages, getLocalizedName } from "@gc-digital-talent/i18n"; +import { WorkExperience } from "@gc-digital-talent/graphql"; +import { Separator } from "@gc-digital-talent/ui"; + +import { getExperienceFormLabels } from "~/utils/experienceUtils"; + +import ContentSection from "../ContentSection"; +import { ContentProps } from "../types"; + +const ExternalContent = ({ + experience: { division, extSizeOfOrganization, extRoleSeniority }, + headingLevel, +}: ContentProps>) => { + const intl = useIntl(); + const experienceFormLabels = getExperienceFormLabels(intl); + + return ( + <> + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + +
+ + {getLocalizedName(extSizeOfOrganization?.label, intl)} + + + {getLocalizedName(extRoleSeniority?.label, intl)} + +
+ + ); +}; + +export default ExternalContent; diff --git a/apps/web/src/components/ExperienceCard/WorkContent/GovContent.tsx b/apps/web/src/components/ExperienceCard/WorkContent/GovContent.tsx new file mode 100644 index 00000000000..45514c8ffe1 --- /dev/null +++ b/apps/web/src/components/ExperienceCard/WorkContent/GovContent.tsx @@ -0,0 +1,191 @@ +import { useIntl } from "react-intl"; + +import { commonMessages, getLocalizedName } from "@gc-digital-talent/i18n"; +import { + GovContractorType, + WorkExperience, + WorkExperienceGovEmployeeType, +} from "@gc-digital-talent/graphql"; +import { Separator } from "@gc-digital-talent/ui"; + +import { getExperienceFormLabels } from "~/utils/experienceUtils"; + +import ContentSection from "../ContentSection"; +import { ContentProps } from "../types"; + +const GovContent = ({ + experience: { + division, + govEmploymentType, + classification, + govPositionType, + govContractorRoleSeniority, + govContractorType, + contractorFirmAgencyName, + }, + headingLevel, +}: ContentProps>) => { + const intl = useIntl(); + const experienceFormLabels = getExperienceFormLabels(intl); + + if (govEmploymentType?.value === WorkExperienceGovEmployeeType.Student) { + return ( + <> + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + + + {getLocalizedName(govEmploymentType.label, intl)} + + + ); + } else if ( + govEmploymentType?.value === WorkExperienceGovEmployeeType.Casual + ) { + return ( + <> + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + +
+ + {getLocalizedName(govEmploymentType.label, intl)} + + + {classification + ? `${classification.group}-0${classification.level}` + : intl.formatMessage(commonMessages.notAvailable)} + +
+ + ); + } else if ( + govEmploymentType?.value === WorkExperienceGovEmployeeType.Indeterminate || + govEmploymentType?.value === WorkExperienceGovEmployeeType.Term + ) { + return ( + <> + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + +
+ + {getLocalizedName(govEmploymentType.label, intl)} + + + {getLocalizedName(govPositionType?.label, intl)} + + + {classification + ? `${classification.group}-0${classification.level}` + : intl.formatMessage(commonMessages.notAvailable)} + +
+ + ); + } else if ( + govEmploymentType?.value === WorkExperienceGovEmployeeType.Contractor + ) { + return ( + <> + + {division ?? intl.formatMessage(commonMessages.notAvailable)} + + +
+ + {getLocalizedName(govEmploymentType.label, intl)} + + + {getLocalizedName(govContractorRoleSeniority?.label, intl)} + + + {getLocalizedName(govContractorType?.label, intl)} + +
+ {govContractorType?.value === GovContractorType.FirmOrAgency && ( + <> + + + {contractorFirmAgencyName ?? + intl.formatMessage(commonMessages.notAvailable)} + + + )} + + ); + } + + return null; +}; + +export default GovContent; diff --git a/apps/web/src/components/ExperienceFormFields/AdditionalDetails.tsx b/apps/web/src/components/ExperienceFormFields/AdditionalDetails.tsx index d2835ad2443..523b55c3caa 100644 --- a/apps/web/src/components/ExperienceFormFields/AdditionalDetails.tsx +++ b/apps/web/src/components/ExperienceFormFields/AdditionalDetails.tsx @@ -51,8 +51,8 @@ const AdditionalDetails = ({ experienceType }: AdditionalDetailsProps) => {

{intl.formatMessage({ defaultMessage: - "Describe key tasks, responsibilities, or other information you feel were crucial in making this experience important.", - id: "KKLx+Z", + "Describe key tasks, responsibilities, or other information you feel were crucial in making this experience important. Try to keep this field concise as you'll be able to provide more detailed information when linking skills to this experience.", + id: "yZ0kfQ", description: "Help text for the experience additional details field", })} diff --git a/apps/web/src/components/ExperienceFormFields/ExperienceDetails.tsx b/apps/web/src/components/ExperienceFormFields/ExperienceDetails.tsx index 4ddc0dd3361..ca7d48bcb08 100644 --- a/apps/web/src/components/ExperienceFormFields/ExperienceDetails.tsx +++ b/apps/web/src/components/ExperienceFormFields/ExperienceDetails.tsx @@ -10,7 +10,7 @@ import AwardFields from "./AwardFields"; import CommunityFields from "./CommunityFields"; import EducationFields from "./EducationFields"; import PersonalFields from "./PersonalFields"; -import WorkFields from "./WorkFields"; +import WorkFields from "./WorkFields/WorkFields"; import NullExperienceType from "./NullExperienceType"; interface ExperienceDetailsProps { diff --git a/apps/web/src/components/ExperienceFormFields/WorkFields.tsx b/apps/web/src/components/ExperienceFormFields/WorkFields.tsx deleted file mode 100644 index 40a39d1e29d..00000000000 --- a/apps/web/src/components/ExperienceFormFields/WorkFields.tsx +++ /dev/null @@ -1,102 +0,0 @@ -import { useIntl } from "react-intl"; -import { useWatch } from "react-hook-form"; - -import { - Checkbox, - DATE_SEGMENT, - DateInput, - Input, -} from "@gc-digital-talent/forms"; -import { errorMessages } from "@gc-digital-talent/i18n"; -import { strToFormDate } from "@gc-digital-talent/date-helpers"; - -import { SubExperienceFormProps } from "~/types/experience"; - -const WorkFields = ({ labels }: SubExperienceFormProps) => { - const intl = useIntl(); - const todayDate = new Date(); - // to toggle whether End date is required, the state of the Current role checkbox must be monitored and have to adjust the form accordingly - const isCurrent = useWatch<{ currentRole: string }>({ name: "currentRole" }); - // ensuring end date isn't before the start date, using this as a minimum value - const startDate = useWatch<{ startDate: string }>({ name: "startDate" }); - - return ( -

-
-
- -
-
- -
-
- -
-
- -
-
- -
-
- {/* conditionally render the end-date based off the state attached to the checkbox input */} - {!isCurrent && ( - - )} -
-
-
- ); -}; - -export default WorkFields; diff --git a/apps/web/src/components/ExperienceFormFields/WorkFields/CafFields.tsx b/apps/web/src/components/ExperienceFormFields/WorkFields/CafFields.tsx new file mode 100644 index 00000000000..c568db9e153 --- /dev/null +++ b/apps/web/src/components/ExperienceFormFields/WorkFields/CafFields.tsx @@ -0,0 +1,153 @@ +import { useIntl } from "react-intl"; +import { useWatch } from "react-hook-form"; +import { useQuery } from "urql"; + +import { + Checkbox, + DATE_SEGMENT, + DateInput, + localizedEnumToOptions, + RadioGroup, +} from "@gc-digital-talent/forms"; +import { errorMessages } from "@gc-digital-talent/i18n"; +import { strToFormDate } from "@gc-digital-talent/date-helpers"; +import { CafFieldsOptionsQuery, graphql } from "@gc-digital-talent/graphql"; +import { Loading } from "@gc-digital-talent/ui"; + +import { SubExperienceFormProps, WorkFormValues } from "~/types/experience"; + +const CafFieldsOptions_Query = graphql(/* GraphQL */ ` + query CafFieldsOptions { + cafEmploymentTypes: localizedEnumStrings(enumName: "CafEmploymentType") { + value + label { + en + fr + } + } + cafForces: localizedEnumStrings(enumName: "CafForce") { + value + label { + en + fr + } + } + cafRanks: localizedEnumStrings(enumName: "CafRank") { + value + label { + en + fr + } + } + } +`); + +const CafFields = ({ labels }: SubExperienceFormProps) => { + const intl = useIntl(); + const [{ data, fetching }] = useQuery({ + query: CafFieldsOptions_Query, + }); + + const todayDate = new Date(); + // to toggle whether End date is required, the state of the Current role checkbox must be monitored and have to adjust the form accordingly + const watchCurrentRole = useWatch({ name: "currentRole" }); + // ensuring end date isn't before the start date, using this as a minimum value + const watchStartDate = useWatch({ name: "startDate" }); + return ( + <> + {fetching ? ( +
+ +
+ ) : ( + <> +
+ +
+
+ +
+
+ +
+
+ +
+
+
+ +
+
+
+ {/* conditionally render the end-date based off the state attached to the checkbox input */} + {!watchCurrentRole && ( + + )} +
+ + )} + + ); +}; + +export default CafFields; diff --git a/apps/web/src/components/ExperienceFormFields/WorkFields/ExternalFields.tsx b/apps/web/src/components/ExperienceFormFields/WorkFields/ExternalFields.tsx new file mode 100644 index 00000000000..9f601a80d30 --- /dev/null +++ b/apps/web/src/components/ExperienceFormFields/WorkFields/ExternalFields.tsx @@ -0,0 +1,159 @@ +import { useIntl } from "react-intl"; +import { useWatch } from "react-hook-form"; +import { useQuery } from "urql"; + +import { + Checkbox, + DATE_SEGMENT, + DateInput, + Input, + localizedEnumToOptions, + RadioGroup, +} from "@gc-digital-talent/forms"; +import { errorMessages } from "@gc-digital-talent/i18n"; +import { strToFormDate } from "@gc-digital-talent/date-helpers"; +import { + ExternalWorkFieldOptionsQuery, + graphql, +} from "@gc-digital-talent/graphql"; +import { Loading } from "@gc-digital-talent/ui"; + +import { SubExperienceFormProps, WorkFormValues } from "~/types/experience"; + +const ExternalWorkFieldOptions_Query = graphql(/* GraphQL */ ` + query ExternalWorkFieldOptions { + extSizeOfOrganizations: localizedEnumStrings( + enumName: "ExternalSizeOfOrganization" + ) { + value + label { + en + fr + } + } + extRoleSeniorities: localizedEnumStrings( + enumName: "ExternalRoleSeniority" + ) { + value + label { + en + fr + } + } + } +`); + +const ExternalFields = ({ labels }: SubExperienceFormProps) => { + const intl = useIntl(); + const [{ data, fetching }] = useQuery({ + query: ExternalWorkFieldOptions_Query, + }); + + const todayDate = new Date(); + // to toggle whether End date is required, the state of the Current role checkbox must be monitored and have to adjust the form accordingly + const watchCurrentRole = useWatch({ name: "currentRole" }); + // ensuring end date isn't before the start date, using this as a minimum value + const watchStartDate = useWatch({ name: "startDate" }); + return ( + <> + {fetching ? ( +
+ +
+ ) : ( + <> +
+ +
+
+ +
+
+ +
+
+ +
+
+ +
+
+
+ +
+
+
+ {/* conditionally render the end-date based off the state attached to the checkbox input */} + {!watchCurrentRole && ( + + )} +
+ + )} + + ); +}; + +export default ExternalFields; diff --git a/apps/web/src/components/ExperienceFormFields/WorkFields/GovFields.tsx b/apps/web/src/components/ExperienceFormFields/WorkFields/GovFields.tsx new file mode 100644 index 00000000000..9a5fc905da4 --- /dev/null +++ b/apps/web/src/components/ExperienceFormFields/WorkFields/GovFields.tsx @@ -0,0 +1,456 @@ +import { useIntl } from "react-intl"; +import { useFormContext, useWatch } from "react-hook-form"; +import { useQuery } from "urql"; +import uniqBy from "lodash/uniqBy"; +import { useEffect } from "react"; + +import { + Checkbox, + DATE_SEGMENT, + DateInput, + Input, + localizedEnumToOptions, + RadioGroup, + Select, +} from "@gc-digital-talent/forms"; +import { + commonMessages, + errorMessages, + getLocalizedName, + uiMessages, +} from "@gc-digital-talent/i18n"; +import { strToFormDate } from "@gc-digital-talent/date-helpers"; +import { + GovContractorType, + GovFieldOptionsQuery, + GovPositionType, + graphql, + WorkExperienceGovEmployeeType, +} from "@gc-digital-talent/graphql"; +import { Loading } from "@gc-digital-talent/ui"; +import { unpackMaybes } from "@gc-digital-talent/helpers"; + +import { SubExperienceFormProps, WorkFormValues } from "~/types/experience"; +import { splitAndJoin } from "~/utils/nameUtils"; + +const GovFieldOptions_Query = graphql(/* GraphQL */ ` + query GovFieldOptions { + departments { + id + name { + en + fr + } + } + classifications { + id + name { + en + fr + } + group + level + } + govEmploymentTypes: localizedEnumStrings( + enumName: "WorkExperienceGovEmployeeType" + ) { + value + label { + en + fr + } + } + govPositionTypes: localizedEnumStrings(enumName: "GovPositionType") { + value + label { + en + fr + } + } + govContractorRoleSeniorities: localizedEnumStrings( + enumName: "GovContractorRoleSeniority" + ) { + value + label { + en + fr + } + } + govContractorTypes: localizedEnumStrings(enumName: "GovContractorType") { + value + label { + en + fr + } + } + } +`); + +const GovFields = ({ labels }: SubExperienceFormProps) => { + const intl = useIntl(); + const [{ data, fetching }] = useQuery({ + query: GovFieldOptions_Query, + }); + const { resetField } = useFormContext(); + + const todayDate = new Date(); + const watchStartDate = useWatch({ name: "startDate" }); + const watchCurrentRole = useWatch({ + name: "currentRole", + }); + const watchGroupSelection = useWatch({ + name: "classificationGroup", + }); + const watchGovEmploymentType = useWatch({ + name: "govEmploymentType", + }); + const watchGovPositionType = useWatch({ + name: "govPositionType", + }); + const watchGovContractorType = useWatch({ + name: "govContractorType", + }); + + // If the government employee type is "Term", + // then remove the "Substantive" option from the govPositionTypes + const allPositionTypes = localizedEnumToOptions(data?.govPositionTypes, intl); + const conditionalPositionTypes = + watchGovEmploymentType === WorkExperienceGovEmployeeType.Term + ? allPositionTypes.filter( + (positionType) => + positionType.value !== String(GovPositionType.Substantive), + ) + : allPositionTypes; + + const departmentOptions = unpackMaybes(data?.departments).map( + ({ id, name }) => ({ + value: id, + label: getLocalizedName(name, intl), + }), + ); + + // Consolidate James's classification-> group and level form logic + const classifications = unpackMaybes(data?.classifications); + + const classGroupsWithDupes: { + label: string; + ariaLabel: string; + }[] = classifications.map((classification) => { + return { + label: + classification.group || + intl.formatMessage({ + defaultMessage: "Error: classification group not found.", + id: "YA/7nb", + description: "Error message if classification group is not defined.", + }), + ariaLabel: `${getLocalizedName(classification.name, intl)} ${splitAndJoin( + classification.group, + )}`, + }; + }); + const noDupes = uniqBy(classGroupsWithDupes, "label"); + const groupOptions = noDupes.map(({ label, ariaLabel }) => { + return { + value: label, + label, + ariaLabel, + }; + }); + + // generate classification levels from the selected group + const levelOptions = classifications + .filter((x) => x.group === watchGroupSelection) + .map((iterator) => { + return { + value: iterator.id.toString(), // change the value to id for the query + label: iterator.level.toString(), + }; + }); + + /** + * Reset classification level when group changes + * because level options change + */ + useEffect(() => { + resetField("classificationLevel", { + keepDirty: false, + }); + }, [resetField, watchGroupSelection]); + + const isIndeterminate = + watchGovEmploymentType === WorkExperienceGovEmployeeType.Indeterminate; + const indeterminateActing = + isIndeterminate && watchGovPositionType === GovPositionType.Acting; + const indeterminateAssignment = + isIndeterminate && watchGovPositionType === GovPositionType.Assignment; + const indeterminateSecondment = + isIndeterminate && watchGovPositionType === GovPositionType.Secondment; + + const expectedEndDate = + watchCurrentRole && + (watchGovEmploymentType === WorkExperienceGovEmployeeType.Student || + watchGovEmploymentType === WorkExperienceGovEmployeeType.Casual || + watchGovEmploymentType === WorkExperienceGovEmployeeType.Term || + indeterminateActing || + indeterminateAssignment || + indeterminateSecondment); + + /** + * Reset classification group and level + * when the employment type changes to casual or contract + */ + useEffect(() => { + const resetDirtyField = (name: keyof WorkFormValues) => { + resetField(name, { + keepDirty: false, + }); + }; + + if ( + watchGovEmploymentType === WorkExperienceGovEmployeeType.Student || + watchGovEmploymentType === WorkExperienceGovEmployeeType.Contractor + ) { + resetDirtyField("classificationGroup"); + resetDirtyField("classificationLevel"); + } + + if (watchGovEmploymentType) { + resetDirtyField("govPositionType"); + } + + if (watchGovEmploymentType !== WorkExperienceGovEmployeeType.Contractor) { + resetDirtyField("govContractorRoleSeniority"); + resetDirtyField("govContractorType"); + } + + if ( + watchGovContractorType === GovContractorType.SelfEmployed || + watchGovEmploymentType !== WorkExperienceGovEmployeeType.Contractor + ) { + resetField("contractorFirmAgencyName", { + keepDirty: false, + defaultValue: undefined, + }); + } + }, [resetField, watchGovEmploymentType, watchGovContractorType]); + + return ( + <> + {fetching ? ( +
+ +
+ ) : ( + <> +
+ +
+
+ +
+ {(watchGovEmploymentType === + WorkExperienceGovEmployeeType.Indeterminate || + watchGovEmploymentType === WorkExperienceGovEmployeeType.Term) && ( +
+ +
+ )} + {watchGovEmploymentType === + WorkExperienceGovEmployeeType.Contractor && ( + <> +
+ +
+
+ +
+ + )} + {watchGovContractorType === GovContractorType.FirmOrAgency && ( +
+ +
+ )} + {(watchGovEmploymentType === WorkExperienceGovEmployeeType.Casual || + watchGovEmploymentType === + WorkExperienceGovEmployeeType.Indeterminate || + watchGovEmploymentType === WorkExperienceGovEmployeeType.Term) && ( + <> +
+ +
+ + )} +
+ +
+
+
+ +
+
+
+ {expectedEndDate ? ( + + ) : ( + <> + {!watchCurrentRole && ( + + )} + + )} +
+ + )} + + ); +}; + +export default GovFields; diff --git a/apps/web/src/components/ExperienceFormFields/WorkFields/WorkFields.tsx b/apps/web/src/components/ExperienceFormFields/WorkFields/WorkFields.tsx new file mode 100644 index 00000000000..af0083262f6 --- /dev/null +++ b/apps/web/src/components/ExperienceFormFields/WorkFields/WorkFields.tsx @@ -0,0 +1,194 @@ +import { useIntl, defineMessage, MessageDescriptor } from "react-intl"; +import { useQuery } from "urql"; +import { useFormContext, useWatch } from "react-hook-form"; +import { useEffect } from "react"; + +import { + FieldLabels, + Input, + Radio, + RadioGroup, +} from "@gc-digital-talent/forms"; +import { errorMessages, getLocalizedName } from "@gc-digital-talent/i18n"; +import { Loading } from "@gc-digital-talent/ui"; +import { + EmploymentCategory, + graphql, + WorkFieldOptionsQuery, +} from "@gc-digital-talent/graphql"; +import { unpackMaybes } from "@gc-digital-talent/helpers"; + +import { SubExperienceFormProps, WorkFormValues } from "~/types/experience"; + +import CafFields from "./CafFields"; +import ExternalFields from "./ExternalFields"; +import GovFields from "./GovFields"; + +const WorkFieldOptions_Query = graphql(/* GraphQL */ ` + query WorkFieldOptions { + employmentCategoryTypes: localizedEnumStrings( + enumName: "EmploymentCategory" + ) { + value + label { + en + fr + } + } + } +`); + +const EmploymentCategoryFields = ({ + employmentCategory, + labels, +}: { + employmentCategory: EmploymentCategory; + labels: FieldLabels; +}) => { + switch (employmentCategory) { + case EmploymentCategory.CanadianArmedForces: + return ; + case EmploymentCategory.ExternalOrganization: + return ; + case EmploymentCategory.GovernmentOfCanada: + return ; + default: + return null; + } +}; + +const employmentCategoryDescriptions: Record< + EmploymentCategory, + MessageDescriptor +> = { + EXTERNAL_ORGANIZATION: defineMessage({ + defaultMessage: + "Select this option if the employment had no affiliation with the Government of Canada.", + id: "0MakGC", + description: + "Description for the external employment category option in work experience", + }), + GOVERNMENT_OF_CANADA: defineMessage({ + defaultMessage: + "Select this option if the employment was with a Government of Canada department, agency, crown corporation, or if you were a contractor working with one of these organizations.", + id: "nmx1ym", + description: + "Description for the goc employment category option in work experience", + }), + CANADIAN_ARMED_FORCES: defineMessage({ + defaultMessage: + "Select this option if the employment was with Canadian Army, the Royal Canadian Air Force, or the Royal Canadian Navy, either as regular force, or reserve force.", + id: "uZuEHk", + description: + "Description for the caf employment category option in work experience", + }), +}; + +const WorkFields = ({ labels }: SubExperienceFormProps) => { + const intl = useIntl(); + const [{ data, fetching }] = useQuery({ + query: WorkFieldOptions_Query, + }); + + const { resetField, formState } = useFormContext(); + + const watchEmploymentCategory = useWatch<{ + employmentCategory: EmploymentCategory; + }>({ + name: "employmentCategory", + }); + + const employmentCategories: Radio[] = unpackMaybes( + data?.employmentCategoryTypes, + ).map(({ value, label }) => { + const contentBelow = + employmentCategoryDescriptions[value as EmploymentCategory]; + return { + label: getLocalizedName(label, intl), + value, + contentBelow: intl.formatMessage(contentBelow), + }; + }); + + /** + * Reset all fields when employmentCategory field is changed + */ + useEffect(() => { + const resetDirtyField = (name: keyof WorkFormValues) => { + resetField(name, { keepDirty: false, defaultValue: null }); + }; + + if (formState.dirtyFields.employmentCategory) { + resetDirtyField("team"); // both external and goc + + // external fields + resetDirtyField("organization"); + resetDirtyField("extSizeOfOrganization"); + resetDirtyField("extRoleSeniority"); + + // goc fields + resetDirtyField("department"); + resetDirtyField("govEmploymentType"); + resetDirtyField("govPositionType"); + resetDirtyField("govContractorRoleSeniority"); + resetDirtyField("govContractorType"); + resetDirtyField("contractorFirmAgencyName"); + resetDirtyField("classificationGroup"); + resetDirtyField("classificationLevel"); + + // caf fields + resetDirtyField("cafEmploymentType"); + resetDirtyField("cafForce"); + resetDirtyField("cafRank"); + + // all categories + resetDirtyField("startDate"); + resetDirtyField("currentRole"); + resetDirtyField("endDate"); + } + }, [formState.dirtyFields, watchEmploymentCategory, resetField]); + + return ( +
+ {fetching ? ( + + ) : ( +
+
+
+ +
+
+ +
+ +
+
+ )} +
+ ); +}; + +export default WorkFields; diff --git a/apps/web/src/components/PoolCandidatesTable/SkillMatchDialog.tsx b/apps/web/src/components/PoolCandidatesTable/SkillMatchDialog.tsx index 77b4e982cd9..2edbe42145a 100644 --- a/apps/web/src/components/PoolCandidatesTable/SkillMatchDialog.tsx +++ b/apps/web/src/components/PoolCandidatesTable/SkillMatchDialog.tsx @@ -100,6 +100,90 @@ const SkillMatchDialog_Query = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } } diff --git a/apps/web/src/lang/fr.json b/apps/web/src/lang/fr.json index bb404b123a2..5d7f4acce3e 100644 --- a/apps/web/src/lang/fr.json +++ b/apps/web/src/lang/fr.json @@ -443,6 +443,10 @@ "defaultMessage": "Deux plumes attachées l’une à l’autre.", "description": "Indigenous Apprenticeship feathers image text alternative" }, + "0Dp1N4": { + "defaultMessage": "Type de poste", + "description": "Label for the position type radio group" + }, "0E9hiS": { "defaultMessage": "Groupes de compétences", "description": "Title for skill families" @@ -479,6 +483,10 @@ "defaultMessage": "De la même manière que vous avez sélectionné des éléments de votre parcours professionnel pour confirmer les exigences en matière d’expérience et d’études, nous vous demanderons de décrire une ou plusieurs expériences de votre parcours professionnel au cours desquelles vous avez activement utilisé la compétence requise.", "description": "Application step for skill requirements, introduction, description, paragraph two" }, + "0MakGC": { + "defaultMessage": "Sélectionnez cette option si l'emploi n'avait aucun lien avec le gouvernement du Canada.", + "description": "Description for the external employment category option in work experience" + }, "0NbdGD": { "defaultMessage": "Erreur : Échec de la suppression de la compétence", "description": "Message displayed to user after skill fails to be deleted." @@ -567,6 +575,10 @@ "defaultMessage": "Compétences techniques", "description": "Title for the technical skill rank card" }, + "0qwyH4": { + "defaultMessage": "Date de fin prévue", + "description": "Label displayed on an Experience form for expected end date input" + }, "0w59dG": { "defaultMessage": "Votre parcours professionnel", "description": "Heading for career timeline section of the application review page." @@ -699,6 +711,10 @@ "defaultMessage": "Signature", "description": "Title for the signature snapshot section" }, + "1b+6V1": { + "defaultMessage": "{role} à {group}", + "description": "Role with group" + }, "1bWLa3": { "defaultMessage": "Compétences que le candidat aimerait améliorer", "description": "Heading for a users skills they would like to improve" @@ -743,6 +759,10 @@ "defaultMessage": "Groupes de compétences (anglais)", "description": "Title for skill families in English" }, + "1syFdp": { + "defaultMessage": "{role} à {group}", + "description": "Role with group, HTML" + }, "1xI8uo": { "defaultMessage": "Utilisez le bouton « Ajoutez un rôle dans l’équipe » pour commencer.", "description": "Instructions for adding a role to a user." @@ -812,7 +832,7 @@ "description": "Subject for email to gain hiring experience" }, "2Oubfe": { - "defaultMessage": "Type d’emploi", + "defaultMessage": "Type d'emploi", "description": "Label for applicant's employment type" }, "2QDT5C": { @@ -971,6 +991,10 @@ "defaultMessage": "Changer votre disponibilité", "description": "Button text to open form to change availability in a generic recruitment" }, + "34NvoS": { + "defaultMessage": "L'ancienneté du poste", + "description": "Label for the seniority of the role radio group" + }, "37mBAU": { "defaultMessage": "Faites croître votre carrière et trouvez des talents pour votre équipe.", "description": "Subtitle for the manager homepage" @@ -1295,6 +1319,10 @@ "defaultMessage": "Les renseignements que vous fournissez peuvent également être utilisés à des fins statistiques et de recherche, et ils peuvent être communiqués à laDirection des enquêtes de la Commission de la fonction publique au besoin.", "description": "Paragraph for privacy policy page" }, + "4fV+wX": { + "defaultMessage": "Catégorie de grade", + "description": "Label for the rank category radio group" + }, "4ii2WZ": { "defaultMessage": "Révisez votre candidature et envoyez-la!", "description": "Subtitle for the application review page." @@ -2615,6 +2643,10 @@ "defaultMessage": "Demande de recherche {searchRequestId} introuvable.", "description": "Message displayed for search request not found on single search request page." }, + "BdpXAF": { + "defaultMessage": "Catégorie d'emploi", + "description": "Label for the employment category radio group" + }, "BdsZwe": { "defaultMessage": "Candidature soumise", "description": "Label for showing the submitted date of an application." @@ -3735,6 +3767,10 @@ "defaultMessage": "1 million de dollars à 2,5 millions de dollars", "description": "Contract value range between one-million and two-point-five-million" }, + "HP5PEg": { + "defaultMessage": "Taille de l'organisation", + "description": "Label for the size of the organization radio group" + }, "HRmtdK": { "defaultMessage": "Bien que les Procédures obligatoires soient également efficaces, il est possible d’en améliorer la clarté, de réduire la charge de travail des ministères en matière de rapports et de mieux les aligner sur les nouvelles procédures obligatoires pour les propriétaires fonctionnels lors de l’approvisionnement en services professionnels menées par le Bureau du contrôleur général du Canada (BCG).", "description": "third paragraph describing the 2024 changes to the directive on digital talent" @@ -4279,10 +4315,6 @@ "defaultMessage": "État du processus", "description": "Title for card for actions related to changing the status of a process" }, - "KKLx+Z": { - "defaultMessage": "Tâches, responsabilités ou autre information clés qui, selon vous, sont essentielles à l’importance de cette expérience.", - "description": "Help text for the experience additional details field" - }, "KKXJFE": { "defaultMessage": "Fiabilité", "description": "Reliability screening level" @@ -4747,6 +4779,10 @@ "defaultMessage": "Modifier votre parcours professionnel", "description": "Edit link text for career timeline section of the application review page." }, + "Mea0Vt": { + "defaultMessage": "Entreprise ou organisme", + "description": "Label for the contracting firm or agency text field" + }, "MejjjQ": { "defaultMessage": "Voici un tableau des équipes, de même que de leurs détails pertinents. Vous pouvez également créer une équipe ou en modifier une qui existe déjà.", "description": "Descriptive text about the list of teams in the admin portal." @@ -5063,6 +5099,10 @@ "defaultMessage": "Titre du poste", "description": "Label for an opportunity's job title." }, + "OKqOVT": { + "defaultMessage": "Gouvernement du Canada", + "description": "Label for goc employment category on work experience card metadata" + }, "OM75j3": { "defaultMessage": "La démonstration des compétences permet à un candidat de fournir une série conserve de listes qui mettent en évidence leurs forces, lacunes particulières, de même que leurs occasions de croissances. Ces listes peuvent vous donner un aperçu de l’ensemble plus vaste de compétences d’un candidat, et préciser où ils pourraient être intéressés à apprendre de nouvelles compétences. ", "description": "Lead in text for a users skill showcase for admins." @@ -7042,6 +7082,10 @@ "defaultMessage": "Communication générale", "description": "Label for preferred language in profile details box." }, + "Y7Qop6": { + "defaultMessage": "Niveau", + "description": "Label displayed on Work Experience form for classification level input" + }, "Y96JXz": { "defaultMessage": "La compétence que vous avez choisie sera aussi ajoutée à votre présentation de compétences, si elle ne s’y trouve pas déjà.", "description": "Subtitle for the find a skill dialog within an experience" @@ -7206,6 +7250,10 @@ "defaultMessage": "Télécopieur : 613-996-9661", "description": "Fax contact info" }, + "Ym2fFN": { + "defaultMessage": "Type d'entrepreneur ou d'entrepreneuse", + "description": "Label for the role seniority radio group" + }, "YmWKlv": { "defaultMessage": "Classification", "description": "Label for a process' classification" @@ -8002,6 +8050,10 @@ "defaultMessage": "4. Hourra! Vous avez terminé de configurer votre compte CléGC et serez ramené à la plateforme Talents numériques du GC.", "description": "Text for fourth sign up -> mfa step." }, + "d1FYv4": { + "defaultMessage": "Classification", + "description": "Label displayed on Work Experience card for classification" + }, "d3HIMV": { "defaultMessage": "Renseignez-vous sur CléGC et trouvez des liens vers des renseignements sur les comptes.", "description": "Introductory text displayed in login and authentication accordion." @@ -8034,6 +8086,10 @@ "defaultMessage": "l’optimisation des résultats de la recherche en matière de l'expérience utilisateur", "description": "List item four, things designers consider for accessibility" }, + "dBpcNA": { + "defaultMessage": "Forces armées canadiennes", + "description": "Label for caf employment category on work experience card metadata" + }, "dD3S0i": { "defaultMessage": "Les employés auront-ils accès à la formation et au perfectionnement pour les ensembles de compétences requis dans le contrat?", "description": "Label for _employees have access to knowledge_ fieldset in the _digital services contracting questionnaire_" @@ -8174,6 +8230,10 @@ "defaultMessage": "Statut du placement", "description": "Label for the job placement status field" }, + "dpZ2B9": { + "defaultMessage": "Entrepreneur ou entrepreneuse", + "description": "Label for contractor employment type on work experience card metadata" + }, "drDPf3": { "defaultMessage": "Parcourez les offres dans le domaine des TI pour la communauté autochtone", "description": "Title for Indigenous community job opportunities on Browse IT jobs page" @@ -9630,6 +9690,10 @@ "defaultMessage": "Vous pouvez utiliser le bouton « Ajouter une compétence » pour mettre une compétence en vedette ici.", "description": "Secondary message to user when no skills have been attached to experience." }, + "kUqaoo": { + "defaultMessage": "Groupe", + "description": "Label displayed on Work Experience form for classification group input" + }, "kVavip": { "defaultMessage": "Ministère de placement", "description": "Title displayed on the Pool Candidates table placed department column" @@ -9646,6 +9710,10 @@ "defaultMessage": "Lieu de travail et terminologie", "description": "Heading for the pool work location dialog" }, + "kdXBAS": { + "defaultMessage": "Force militaire", + "description": "Label for the military force radio group" + }, "kf5Td+": { "defaultMessage": "Assurez-vous de sauvegarder ou de fermer tout formulaire ouvert avant de continuer.", "description": "Message displayed to users when they attempt to quit the profile form with unsaved changes" @@ -10278,6 +10346,10 @@ "defaultMessage": "Se déconnecter", "description": "Message displayed to users to sign out of the application" }, + "nmx1ym": { + "defaultMessage": " Sélectionnez cette option si l'emploi était dans un ministère ou un organisme fédéral ou une société d'État, ou si vous étiez un entrepreneur ou une entrepreneuse travaillant pour l'une de ces organisations.", + "description": "Description for the goc employment category option in work experience" + }, "nn9B4R": { "defaultMessage": "Postulez dès aujourd’hui pour entamer votre parcours de carrière en TI.", "description": "Homepage subtitle for IT Apprenticeship Program for Indigenous Peoples" @@ -11482,6 +11554,14 @@ "defaultMessage": "Exigences essentielles", "description": "Sub title for the pool core requirements" }, + "uZuEHk": { + "defaultMessage": "Sélectionnez cette option si l'emploi était dans l'Armée canadienne, dans l'Aviation royale canadienne ou dans la Marine royale canadienne, dans la Force régulière ou dans la Force de réserve.", + "description": "Description for the caf employment category option in work experience" + }, + "uaEMMO": { + "defaultMessage": "Type d'emploi", + "description": "Label for the employment type radio group" + }, "uiuMqi": { "defaultMessage": "Utilisez le bouton « Ajouter une nouvelle compétence » pour commencer.", "description": "Null message description for essential skills table." @@ -12035,7 +12115,7 @@ "description": "Message for total estimated candidates box next to search form." }, "xzSXz9": { - "defaultMessage": "Type d’emploi", + "defaultMessage": "Type d'emploi", "description": "Employment type label" }, "y+Pq1f": { @@ -12162,6 +12242,10 @@ "defaultMessage": "Commencez à remplir la section sur vos renseignements en tant que fonctionnaire ", "description": "Call to action to begin editing government information" }, + "yZ0kfQ": { + "defaultMessage": "Tâches, responsabilités ou autre information clés qui, selon vous, sont essentielles à l’importance de cette expérience. La concision est de mise ici : vous aurez plus d'espace pour fournir des détails lorsque vous établirez un lien entre des compétences et cette expérience.", + "description": "Help text for the experience additional details field" + }, "yZMQ6j": { "defaultMessage": "Nous voulons en apprendre davantage sur vous et sur votre intérêt ou votre passion dans le domaine de la TI!", "description": "How it works, step 2 content sentence 1" diff --git a/apps/web/src/lang/whitelist.yml b/apps/web/src/lang/whitelist.yml index 9d7157ae2d4..02ee8758242 100644 --- a/apps/web/src/lang/whitelist.yml +++ b/apps/web/src/lang/whitelist.yml @@ -29,3 +29,4 @@ - IhPS6D # Important - CdJQ7z # Administration - yWen8A # Format - The format of the training opportunity +- d1FYv4 # Classification diff --git a/apps/web/src/messages/experienceMessages.ts b/apps/web/src/messages/experienceMessages.ts index 5a24e982058..469c16e4ebd 100644 --- a/apps/web/src/messages/experienceMessages.ts +++ b/apps/web/src/messages/experienceMessages.ts @@ -76,6 +76,16 @@ const messages = defineMessages({ id: "lhV7kM", description: "Role at organization, HTML", }, + workWith: { + defaultMessage: "{role} with {group}", + id: "1b+6V1", + description: "Role with group", + }, + workWithHtml: { + defaultMessage: "{role} with {group}", + id: "1syFdp", + description: "Role with group, HTML", + }, collapseDetails: { defaultMessage: "Collapse all experience details", id: "pp+b1H", diff --git a/apps/web/src/pages/Applications/ApplicationEducationPage/LinkCareerTimeline.tsx b/apps/web/src/pages/Applications/ApplicationEducationPage/LinkCareerTimeline.tsx index 8fa30681775..e50ea205e57 100644 --- a/apps/web/src/pages/Applications/ApplicationEducationPage/LinkCareerTimeline.tsx +++ b/apps/web/src/pages/Applications/ApplicationEducationPage/LinkCareerTimeline.tsx @@ -13,6 +13,7 @@ import { } from "@gc-digital-talent/graphql"; import { + getExperienceName, isAwardExperience, isCommunityExperience, isEducationExperience, @@ -255,15 +256,7 @@ const LinkCareerTimeline = ({ if (isWorkExperience(experience)) { const workExperience = { value: experience.id, - label: - intl.formatMessage( - { - defaultMessage: "{role} at {organization}", - id: "wTAdQe", - description: "Role at organization", - }, - { role: experience.role, organization: experience.organization }, - ) || "", + label: getExperienceName(experience, intl), }; return { diff --git a/apps/web/src/pages/Applications/fragment.ts b/apps/web/src/pages/Applications/fragment.ts index 94a289ed9e9..0800cce6d63 100644 --- a/apps/web/src/pages/Applications/fragment.ts +++ b/apps/web/src/pages/Applications/fragment.ts @@ -249,6 +249,90 @@ const Application_PoolCandidateFragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } } @@ -408,6 +492,90 @@ const Application_PoolCandidateFragment = graphql(/* GraphQL */ ` startDate endDate details + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } submittedSteps diff --git a/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx b/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx index 95c8db4cd0a..9495986d869 100644 --- a/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx +++ b/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx @@ -120,6 +120,90 @@ export const CareerTimelineExperience_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } `); diff --git a/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceForm.test.tsx b/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceForm.test.tsx deleted file mode 100644 index 2267978a61a..00000000000 --- a/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceForm.test.tsx +++ /dev/null @@ -1,393 +0,0 @@ -/** - * @jest-environment jsdom - */ -import "@testing-library/jest-dom"; -import { Provider as GraphqlProvider } from "urql"; -import { act, screen, waitFor } from "@testing-library/react"; -import { userEvent } from "@testing-library/user-event"; -import { never, fromValue } from "wonka"; - -import { fakeSkills } from "@gc-digital-talent/fake-data"; -import { - axeTest, - renderWithProviders, - updateDate, -} from "@gc-digital-talent/jest-helpers"; -import { - AwardedScope, - AwardedTo, - makeFragmentData, -} from "@gc-digital-talent/graphql"; - -import type { ExperienceType } from "~/types/experience"; - -import { - ExperienceForm, - ExperienceFormProps, - ExperienceFormSkill_Fragment, -} from "./ExperienceFormPage"; - -const mockUserId = "user-id"; -const mockSkills = fakeSkills(50); -const mockClient = { - executeMutation: jest.fn(() => never), - executeQuery: jest.fn(() => - fromValue({ - data: { - awardedTo: [{ value: AwardedTo.Me, label: { en: "Me", fr: "Me" } }], - awardedScopes: [ - { value: AwardedScope.Local, label: { en: "Local", fr: "Local" } }, - ], - }, - }), - ), -}; - -const skillFragments = mockSkills.map((skill) => - makeFragmentData(skill, ExperienceFormSkill_Fragment), -); - -const renderExperienceForm = (props: ExperienceFormProps) => - renderWithProviders( - - - , - ); - -describe("ExperienceForm", () => { - const user = userEvent.setup(); - - it("award type should have no accessibility errors", async () => { - const { container } = renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - }); - await axeTest(container); - }); - - it("community type should have no accessibility errors", async () => { - const { container } = renderExperienceForm({ - userId: mockUserId, - experienceType: "community", - skillsQuery: skillFragments, - }); - await axeTest(container); - }); - - it("education type should have no accessibility errors", async () => { - const { container } = renderExperienceForm({ - userId: mockUserId, - experienceType: "education", - skillsQuery: skillFragments, - }); - await axeTest(container); - }); - - it("personal type should have no accessibility errors", async () => { - const { container } = renderExperienceForm({ - userId: mockUserId, - experienceType: "personal", - skillsQuery: skillFragments, - }); - await axeTest(container); - }); - - it("work type should have no accessibility errors", async () => { - const { container } = renderExperienceForm({ - userId: mockUserId, - experienceType: "work", - skillsQuery: skillFragments, - }); - await axeTest(container); - }); - - it("should render award fields", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("textbox", { name: /award title/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("combobox", { name: /awarded to/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /issuing/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("combobox", { name: /award scope/i }), - ).toBeInTheDocument(); - - // Note: Date inputs have no role by default - expect( - screen.getByRole("group", { name: /date awarded/i }), - ).toBeInTheDocument(); - }); - - it("should render community fields", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "community", - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("textbox", { name: /my role/i }), - ).toBeInTheDocument(); - - expect(screen.getByRole("textbox", { name: /group/i })).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /project/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /start date/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /end date/i }), - ).toBeInTheDocument(); - }); - - it("should render education fields", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "education", - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("combobox", { name: /type of education/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /area of study/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /institution/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("combobox", { name: /status/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /thesis title/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /start date/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /end date/i }), - ).toBeInTheDocument(); - }); - - it("should render personal fields", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "personal", - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("textbox", { name: /short title/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /experience description/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /disclaimer/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /current experience/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /start date/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /end date/i }), - ).toBeInTheDocument(); - }); - - it("should render work fields", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "work", - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("textbox", { name: /my role/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("textbox", { name: /organization/i }), - ).toBeInTheDocument(); - - expect(screen.getByRole("textbox", { name: /team/i })).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /current role/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /start date/i }), - ).toBeInTheDocument(); - - expect( - screen.getByRole("group", { name: /end date/i }), - ).toBeInTheDocument(); - }); - - it("should render additional details", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "work", // Type of form shouldn't matter here - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("heading", { name: /highlight additional details/i }), - ).toBeInTheDocument(); - }); - - it("should render link featured skills", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "work", // Type of form shouldn't matter here - skillsQuery: skillFragments, - }); - - expect( - screen.getByRole("heading", { name: /link featured skills/i }), - ).toBeInTheDocument(); - }); - - it("should not submit award with empty fields", async () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - }); - - expect(await screen.findByText(/save and return/i)).toBeInTheDocument(); - - await user.click(screen.getByRole("button", { name: /save and return/i })); - - expect(mockClient.executeMutation).not.toHaveBeenCalled(); - }); - - it("should submit with required fields", async () => { - const experienceType: ExperienceType = "award"; - renderExperienceForm({ - userId: mockUserId, - experienceType, - skillsQuery: skillFragments, - }); - - expect(await screen.findByText(/save and return/i)).toBeInTheDocument(); - - const awardTitle = screen.getByRole("textbox", { name: /award title/i }); - await user.type(awardTitle, "AwardTitle"); - expect(awardTitle).toHaveValue("AwardTitle"); - - const awardedTo = screen.getByRole("combobox", { name: /awarded to/i }); - await user.selectOptions(awardedTo, "ME"); - expect(awardedTo).toHaveValue("ME"); - - const organization = screen.getByRole("textbox", { - name: /issuing organization/i, - }); - await user.clear(organization); - expect(organization).toHaveValue(""); - await user.type(organization, "Org"); - expect(organization).toHaveValue("Org"); - - const dateAwarded = screen.getByRole("group", { name: /date awarded/i }); - await updateDate(dateAwarded, { - year: "1111", - month: "11", - }); - expect(screen.getByRole("spinbutton", { name: /year/i })).toHaveValue(1111); - expect(screen.getByRole("combobox", { name: /month/i })).toHaveValue("11"); - - const scope = screen.getByRole("combobox", { name: /award scope/i }); - await user.selectOptions(scope, "LOCAL"); - expect(scope).toHaveValue("LOCAL"); - - const details = screen.getByRole("textbox", { - name: /additional details/i, - }); - await user.clear(details); - await user.type(details, "details"); - expect(details).toHaveValue("details"); - - await user.click(screen.getByRole("button", { name: /save and return/i })); - - await waitFor(() => expect(screen.queryAllByRole("alert")).toHaveLength(0)); - }); - - it("should add skill", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - }); - - act(() => { - screen - .getAllByRole("button", { - name: /add a skill/i, - })[0] - .click(); - }); - }); - - it("delete should not render when edit is false", () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - edit: false, - }); - expect(screen.queryByText("Delete this experience")).toBeFalsy(); - }); - - it("delete should render when edit is true and be called properly", async () => { - renderExperienceForm({ - userId: mockUserId, - experienceType: "award", - skillsQuery: skillFragments, - edit: true, - }); - // get and open Dialog Component - const deleteButton = screen.getByRole("button", { - name: /delete this experience/i, - }); - expect(deleteButton).toBeTruthy(); - - await user.click(deleteButton); - // get and click on Delete in Dialog - const deleteSubmit = screen.getByRole("button", { name: /delete/i }); - expect(deleteSubmit).toBeTruthy(); - await user.click(deleteSubmit); - - expect(mockClient.executeMutation).toHaveBeenCalled(); - }); -}); diff --git a/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceFormPage.tsx b/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceFormPage.tsx index 7e97e52f730..9b283ed1de8 100644 --- a/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceFormPage.tsx +++ b/apps/web/src/pages/Profile/ExperienceFormPage/ExperienceFormPage.tsx @@ -205,11 +205,95 @@ const ExperienceFormExperience_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } `); -export interface ExperienceFormProps { +interface ExperienceFormProps { edit?: boolean; experienceQuery?: FragmentType; experienceId?: string; diff --git a/apps/web/src/pages/Skills/UpdateUserSkillPage.tsx b/apps/web/src/pages/Skills/UpdateUserSkillPage.tsx index 8cf6554ba01..ed1badd079f 100644 --- a/apps/web/src/pages/Skills/UpdateUserSkillPage.tsx +++ b/apps/web/src/pages/Skills/UpdateUserSkillPage.tsx @@ -186,6 +186,90 @@ export const UpdateUserSkillExperience_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } `); @@ -278,6 +362,90 @@ export const UpdateUserSkill_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } skills { id diff --git a/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx b/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx index 6aecfc4ed7f..eac9c4563eb 100644 --- a/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx +++ b/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx @@ -300,6 +300,90 @@ const AdminUserProfileUser_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } topTechnicalSkillsRanking { diff --git a/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx b/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx index da319858236..05f0ff27e89 100644 --- a/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx @@ -284,6 +284,90 @@ export const UserInfo_Fragment = graphql(/* GraphQL */ ` division startDate endDate + employmentCategory { + value + label { + en + fr + } + } + extSizeOfOrganization { + value + label { + en + fr + } + } + extRoleSeniority { + value + label { + en + fr + } + } + govEmploymentType { + value + label { + en + fr + } + } + govPositionType { + value + label { + en + fr + } + } + govContractorRoleSeniority { + value + label { + en + fr + } + } + govContractorType { + value + label { + en + fr + } + } + contractorFirmAgencyName + cafEmploymentType { + value + label { + en + fr + } + } + cafForce { + value + label { + en + fr + } + } + cafRank { + value + label { + en + fr + } + } + classification { + id + group + level + } + department { + id + departmentNumber + name { + en + fr + } + } } } topTechnicalSkillsRanking { diff --git a/apps/web/src/types/experience.ts b/apps/web/src/types/experience.ts index 3b86329e61a..c0f5eca498d 100644 --- a/apps/web/src/types/experience.ts +++ b/apps/web/src/types/experience.ts @@ -25,6 +25,16 @@ import { Scalars, WorkExperienceInput, WorkExperience, + EmploymentCategory, + ExternalSizeOfOrganization, + ExternalRoleSeniority, + GovEmployeeType, + CafForce, + CafRank, + CafEmploymentType, + GovPositionType, + GovContractorRoleSeniority, + GovContractorType, } from "@gc-digital-talent/graphql"; export type ExperienceType = @@ -86,10 +96,25 @@ export type PersonalFormValues = FormValueDateRange & { currentRole: boolean; }; -type WorkFormValues = FormValueDateRange & { - role: string; - organization: string; - team?: string; +export type WorkFormValues = FormValueDateRange & { + role: string | null; + organization: string | null; + team?: string | null; + employmentCategory?: EmploymentCategory | null; + extSizeOfOrganization?: ExternalSizeOfOrganization | null; + extRoleSeniority?: ExternalRoleSeniority | null; + department?: string | null; + classificationGroup: string | null; + classificationLevel: string | null; + govEmploymentType?: GovEmployeeType | null; + govPositionType?: GovPositionType | null; + govContractorRoleSeniority?: GovContractorRoleSeniority | null; + govContractorType?: GovContractorType | null; + contractorFirmAgencyName?: string | null; + cafEmploymentType?: CafEmploymentType | null; + cafForce?: CafForce | null; + cafRank?: CafRank | null; + currentRole?: boolean | null; }; export type AllExperienceFormValues = AwardFormValues & @@ -124,19 +149,32 @@ export interface ExperienceDetailsSubmissionData { awardedScope?: AwardedScope; description?: string; details?: string; - division?: string; + division?: string | null; currentRole?: boolean; endDate?: Scalars["Date"]["input"] | null; institution?: string; issuedBy?: string; organization?: string; project?: string; - role?: string; + role?: string | null; startDate?: Scalars["Date"]["input"]; status?: EducationStatus; thesisTitle?: string; - title?: string; + title?: string | null; type?: EducationType; + employmentCategory?: EmploymentCategory | null; + extSizeOfOrganization?: ExternalSizeOfOrganization | null; + extRoleSeniority?: ExternalRoleSeniority | null; + departmentId?: string | null; + classificationId?: string | null; + govEmploymentType?: GovEmployeeType | null; + govPositionType?: GovPositionType | null; + govContractorRoleSeniority?: GovContractorRoleSeniority | null; + govContractorType?: GovContractorType | null; + contractorFirmAgencyName?: string | null; + cafEmploymentType?: CafEmploymentType | null; + cafForce?: CafForce | null; + cafRank?: CafRank | null; skills?: { sync?: | ({ id: string; details: Maybe | undefined } | undefined)[] @@ -201,5 +239,19 @@ export interface ExperienceDetailsDefaultValues { thesisTitle?: string; title?: string; educationType?: EducationType; + employmentCategory?: EmploymentCategory; + extSizeOfOrganization?: ExternalSizeOfOrganization; + extRoleSeniority?: ExternalRoleSeniority; + department?: string; + classificationGroup?: string; + classificationLevel?: string; + govEmploymentType?: GovEmployeeType; + govPositionType?: GovPositionType; + govContractorRoleSeniority?: GovContractorRoleSeniority; + govContractorType?: GovContractorType; + contractorFirmAgencyName?: string; + cafEmploymentType?: CafEmploymentType; + cafForce?: CafForce; + cafRank?: CafRank; skills?: FormSkills; } diff --git a/apps/web/src/utils/experienceUtils.tsx b/apps/web/src/utils/experienceUtils.tsx index 784295ced9c..59a9cdfd29b 100644 --- a/apps/web/src/utils/experienceUtils.tsx +++ b/apps/web/src/utils/experienceUtils.tsx @@ -7,17 +7,21 @@ import UserGroupIcon from "@heroicons/react/20/solid/UserGroupIcon"; import InformationCircleIcon from "@heroicons/react/24/solid/InformationCircleIcon"; import { ReactNode } from "react"; -import { commonMessages } from "@gc-digital-talent/i18n"; +import { commonMessages, getLocalizedName } from "@gc-digital-talent/i18n"; import { IconType } from "@gc-digital-talent/ui"; import { AwardExperience, CommunityExperience, EducationExperience, + EmploymentCategory, + GovPositionType, Maybe, PersonalExperience, Skill, WorkExperience, + WorkExperienceGovEmployeeType, } from "@gc-digital-talent/graphql"; +import { strToFormDate } from "@gc-digital-talent/date-helpers"; import { AllExperienceFormValues, @@ -160,6 +164,12 @@ export const getExperienceFormLabels = ( id: "cD3QKi", description: "Label displayed on an Experience form for end date input", }), + expectedEndDate: intl.formatMessage({ + defaultMessage: "Expected end date", + id: "0qwyH4", + description: + "Label displayed on an Experience form for expected end date input", + }), dateRange: intl.formatMessage({ defaultMessage: "Start/end date", id: "PVzyQl", @@ -222,6 +232,68 @@ export const getExperienceFormLabels = ( description: "Label displayed on experience form/card for how a skill was applied section", }), + classificationGroup: intl.formatMessage({ + defaultMessage: "Group", + id: "kUqaoo", + description: + "Label displayed on Work Experience form for classification group input", + }), + classificationLevel: intl.formatMessage({ + defaultMessage: "Level", + id: "Y7Qop6", + description: + "Label displayed on Work Experience form for classification level input", + }), + extSizeOfOrganization: intl.formatMessage({ + defaultMessage: "Size of the organization", + id: "HP5PEg", + description: "Label for the size of the organization radio group", + }), + extRoleSeniority: intl.formatMessage({ + defaultMessage: "Seniority of the role", + id: "34NvoS", + description: "Label for the seniority of the role radio group", + }), + govEmploymentType: intl.formatMessage({ + defaultMessage: "Employment type", + id: "uaEMMO", + description: "Label for the employment type radio group", + }), + classification: intl.formatMessage({ + defaultMessage: "Classification", + id: "d1FYv4", + description: "Label displayed on Work Experience card for classification", + }), + positionType: intl.formatMessage({ + defaultMessage: "Position type", + id: "0Dp1N4", + description: "Label for the position type radio group", + }), + govContractorRoleSeniority: intl.formatMessage({ + defaultMessage: "Seniority of the role", + id: "34NvoS", + description: "Label for the seniority of the role radio group", + }), + govContractorType: intl.formatMessage({ + defaultMessage: "Contractor type", + id: "Ym2fFN", + description: "Label for the role seniority radio group", + }), + contractorFirmAgencyName: intl.formatMessage({ + defaultMessage: "Contracting firm or agency", + id: "Mea0Vt", + description: "Label for the contracting firm or agency text field", + }), + cafEmploymentType: intl.formatMessage({ + defaultMessage: "Employment type", + id: "uaEMMO", + description: "Label for the employment type radio group", + }), + cafRank: intl.formatMessage({ + defaultMessage: "Rank category", + id: "4fV+wX", + description: "Label for the rank category radio group", + }), }; }; @@ -260,6 +332,19 @@ export const formValuesToSubmitData = ( experienceTitle, experienceDescription, currentRole, + employmentCategory, + extSizeOfOrganization, + extRoleSeniority, + department: departmentId, + govEmploymentType, + govPositionType, + govContractorRoleSeniority, + govContractorType, + contractorFirmAgencyName, + classificationLevel: classificationId, + cafEmploymentType, + cafForce, + cafRank, } = data; const newEndDate = !currentRole && endDate ? endDate : null; @@ -299,7 +384,20 @@ export const formValuesToSubmitData = ( organization, division: team, startDate, - endDate: newEndDate, + endDate: endDate, + employmentCategory, + extSizeOfOrganization, + extRoleSeniority, + departmentId: departmentId ?? null, + govEmploymentType, + govPositionType, + govContractorRoleSeniority, + govContractorType, + contractorFirmAgencyName, + classificationId: classificationId ?? null, + cafEmploymentType, + cafForce, + cafRank, }, }; @@ -521,14 +619,49 @@ const getPersonalExperienceDefaultValues = ( const getWorkExperienceDefaultValues = ( experience: Omit, ) => { - const { role, organization, division, startDate, endDate } = experience; + const { + role, + organization, + division, + startDate, + endDate, + employmentCategory, + extSizeOfOrganization, + extRoleSeniority, + department, + classification, + govEmploymentType, + govPositionType, + govContractorRoleSeniority, + govContractorType, + contractorFirmAgencyName, + cafEmploymentType, + cafForce, + cafRank, + } = experience; return { role, organization, team: division, startDate, - currentRole: endDate === null, + currentRole: endDate + ? endDate >= strToFormDate(new Date().toISOString()) // today's date + : true, endDate, + employmentCategory: employmentCategory?.value, + extSizeOfOrganization: extSizeOfOrganization?.value, + extRoleSeniority: extRoleSeniority?.value, + department: department?.id, + classificationGroup: classification?.group, + classificationLevel: classification?.id, + govEmploymentType: govEmploymentType?.value, + govPositionType: govPositionType?.value, + govContractorRoleSeniority: govContractorRoleSeniority?.value, + govContractorType: govContractorType?.value, + contractorFirmAgencyName, + cafEmploymentType: cafEmploymentType?.value, + cafForce: cafForce?.value, + cafRank: cafRank?.value, }; }; @@ -620,14 +753,42 @@ export const getExperienceName = ( } if (isWorkExperience(experience)) { - const { role, organization } = experience; - return intl.formatMessage( - html ? experienceMessages.workAtHtml : experienceMessages.workAt, - { - role, - organization, - }, - ); + const { role, organization, employmentCategory, department, cafForce } = + experience; + switch (employmentCategory?.value) { + case EmploymentCategory.ExternalOrganization: + return intl.formatMessage( + html ? experienceMessages.workWithHtml : experienceMessages.workWith, + { + role, + group: organization, + }, + ); + case EmploymentCategory.GovernmentOfCanada: + return intl.formatMessage( + html ? experienceMessages.workWithHtml : experienceMessages.workWith, + { + role, + group: getLocalizedName(department?.name, intl), + }, + ); + case EmploymentCategory.CanadianArmedForces: + return intl.formatMessage( + html ? experienceMessages.workWithHtml : experienceMessages.workWith, + { + role, + group: getLocalizedName(cafForce?.label, intl), + }, + ); + default: + return intl.formatMessage( + html ? experienceMessages.workAtHtml : experienceMessages.workAt, + { + role, + organization, + }, + ); + } } // We should never get here but just in case we do, return no provided @@ -657,6 +818,40 @@ export const getExperienceDate = ( } const { startDate, endDate } = experience; + + if (isWorkExperience(experience)) { + const isIndeterminate = + experience.govEmploymentType?.value === + WorkExperienceGovEmployeeType.Indeterminate; + const indeterminateActing = + isIndeterminate && + experience.govPositionType?.value === GovPositionType.Acting; + const indeterminateAssignment = + isIndeterminate && + experience.govPositionType?.value === GovPositionType.Assignment; + const indeterminateSecondment = + isIndeterminate && + experience.govPositionType?.value === GovPositionType.Secondment; + + const todayDate = strToFormDate(new Date().toISOString()); + const expectedEndDate = + endDate && + endDate >= todayDate && + (experience.govEmploymentType?.value === + WorkExperienceGovEmployeeType.Student || + experience.govEmploymentType?.value === + WorkExperienceGovEmployeeType.Casual || + experience.govEmploymentType?.value === + WorkExperienceGovEmployeeType.Term || + indeterminateActing || + indeterminateAssignment || + indeterminateSecondment); + + return expectedEndDate + ? `${getDateRange({ startDate, endDate, intl })} (${getExperienceFormLabels(intl, "work").expectedEndDate})` + : getDateRange({ startDate, endDate, intl }); + } + return getDateRange({ startDate, endDate, intl }); }; From f57a0b81634c32f58ebe83c207ee5dd71e722e42 Mon Sep 17 00:00:00 2001 From: Peter Giles <8978655+petertgiles@users.noreply.github.com> Date: Fri, 13 Dec 2024 12:21:42 -0500 Subject: [PATCH 04/31] [Feature] Utility function to detect the VPN (#12226) * update error exchange detection * util function * Update infrastructure/conf/nginx-conf-local/default Co-authored-by: tristan-orourke * reword firewall simulator --------- Co-authored-by: tristan-orourke --- infrastructure/conf/nginx-conf-local/default | 8 +++++++ .../src/exchanges/specialErrorExchange.ts | 9 ++++++-- packages/client/src/index.tsx | 3 ++- .../src/utils/canAccessProtectedEndpoint.ts | 21 +++++++++++++++++++ 4 files changed, 38 insertions(+), 3 deletions(-) create mode 100644 packages/client/src/utils/canAccessProtectedEndpoint.ts diff --git a/infrastructure/conf/nginx-conf-local/default b/infrastructure/conf/nginx-conf-local/default index fc1e18aa382..aa4c1ff81cb 100644 --- a/infrastructure/conf/nginx-conf-local/default +++ b/infrastructure/conf/nginx-conf-local/default @@ -139,6 +139,13 @@ server { fastcgi_index index.php; } + # Simulate being off the VPN and the firewall redirecting to the restricted page. + # Don't forget to also comment out the location rule for /admin/graphql + # location ~* /admin(/|$) { + # absolute_redirect on; + # return 301 http://localhost:8000/restricted.html; + # } + # api location = /graphql { access_log off; @@ -153,6 +160,7 @@ server { include fastcgi_params; fastcgi_param SCRIPT_FILENAME $document_root/api/public/index.php; } + # graphql online web interface location = /graphiql { fastcgi_pass 127.0.0.1:9000; diff --git a/packages/client/src/exchanges/specialErrorExchange.ts b/packages/client/src/exchanges/specialErrorExchange.ts index dc833a13600..e260ecfcf82 100644 --- a/packages/client/src/exchanges/specialErrorExchange.ts +++ b/packages/client/src/exchanges/specialErrorExchange.ts @@ -18,13 +18,18 @@ const specialErrorExchange = ({ intl }: { intl: IntlShape }) => { tap((result) => { const err = result.error; if (err) { - const res = err.response as Response; + const errRes = err.response as Response; + // I think this is old error condition from when the firewall responded directly when blocking if ( - res?.status === 403 && + errRes?.status === 403 && err?.networkError?.message?.includes("Request Rejected") ) { toast.error(intl.formatMessage(errorMessages.requestRejected)); } + // This is the new error condition from when the firewall responds a redirect to restricted.html + if (errRes?.url?.endsWith("/restricted.html")) { + toast.error(intl.formatMessage(errorMessages.requestRejected)); + } } }), ); diff --git a/packages/client/src/index.tsx b/packages/client/src/index.tsx index 200432e2e7d..a481d6250c9 100644 --- a/packages/client/src/index.tsx +++ b/packages/client/src/index.tsx @@ -1,5 +1,6 @@ import ClientProvider from "./components/ClientProvider/ClientProvider"; import { isUuidError } from "./utils/errors"; +import canAccessProtectedEndpoint from "./utils/canAccessProtectedEndpoint"; export default ClientProvider; -export { isUuidError }; +export { isUuidError, canAccessProtectedEndpoint }; diff --git a/packages/client/src/utils/canAccessProtectedEndpoint.ts b/packages/client/src/utils/canAccessProtectedEndpoint.ts new file mode 100644 index 00000000000..0be281666f4 --- /dev/null +++ b/packages/client/src/utils/canAccessProtectedEndpoint.ts @@ -0,0 +1,21 @@ +import { apiHost, protectedUrl } from "../constants"; + +async function canAccessProtectedEndpoint(): Promise { + const response = await fetch(`${apiHost}${protectedUrl}`, { + method: "POST", + body: `{ "query": "{ __typename }" }`, + headers: { + "Content-Type": "application/json", + }, + }); + + // The firewall will redirect to a restricted page if the user is not allowed to access the endpoint + if (response.redirected && response.url.endsWith("/restricted.html")) { + return false; + } + + // otherwise check if we got an OK + return response.ok; +} + +export default canAccessProtectedEndpoint; From b5b4094754ec66c96ae881a8eb14b810bed40949 Mon Sep 17 00:00:00 2001 From: Vachan <40485260+vd1992@users.noreply.github.com> Date: Fri, 13 Dec 2024 11:53:20 -0700 Subject: [PATCH 05/31] [Feature] Stub out applicant dashboard (#12242) * create stub * commented out router addition * adjust placeholder * eslint ignore --- apps/web/src/components/Router.tsx | 7 ++ .../ApplicantDashboardPage.tsx | 77 +++++++++++++++++++ 2 files changed, 84 insertions(+) create mode 100644 apps/web/src/pages/ApplicantDashboardPage/ApplicantDashboardPage.tsx diff --git a/apps/web/src/components/Router.tsx b/apps/web/src/components/Router.tsx index 7136dabd25b..f8154d338c7 100644 --- a/apps/web/src/components/Router.tsx +++ b/apps/web/src/components/Router.tsx @@ -210,6 +210,13 @@ const createRoute = (locale: Locales) => "../pages/ProfileAndApplicationsPage/ProfileAndApplicationsPage" ), }, + // { + // path: "dashboard-test", + // lazy: () => + // import( + // "../pages/ApplicantDashboardPage/ApplicantDashboardPage" + // ), + // }, { path: "settings", lazy: () => diff --git a/apps/web/src/pages/ApplicantDashboardPage/ApplicantDashboardPage.tsx b/apps/web/src/pages/ApplicantDashboardPage/ApplicantDashboardPage.tsx new file mode 100644 index 00000000000..778e1ac46c0 --- /dev/null +++ b/apps/web/src/pages/ApplicantDashboardPage/ApplicantDashboardPage.tsx @@ -0,0 +1,77 @@ +/* eslint-disable import/no-unused-modules */ +import { useIntl } from "react-intl"; +import { useQuery } from "urql"; + +import { Pending } from "@gc-digital-talent/ui"; +import { ROLE_NAME } from "@gc-digital-talent/auth"; +import { User, graphql } from "@gc-digital-talent/graphql"; +import { commonMessages } from "@gc-digital-talent/i18n"; + +import SEO from "~/components/SEO/SEO"; +import { getFullNameHtml } from "~/utils/nameUtils"; +import RequireAuth from "~/components/RequireAuth/RequireAuth"; +import Hero from "~/components/Hero"; + +export interface DashboardPageProps { + currentUser?: User | null; +} + +export const DashboardPage = ({ currentUser }: DashboardPageProps) => { + const intl = useIntl(); + + return ( + <> + + + + ); +}; + +const ApplicantDashboard_Query = graphql(/* GraphQL */ ` + query ApplicantDashboard_Query { + me { + id + firstName + lastName + } + } +`); + +export const ApplicantDashboardPageApi = () => { + const [{ data, fetching, error }] = useQuery({ + query: ApplicantDashboard_Query, + }); + + return ( + + + + ); +}; + +export const Component = () => ( + + + +); + +Component.displayName = "ApplicantDashboardPage"; From 28794052ab9347b78af68a0646fccaea2378d06c Mon Sep 17 00:00:00 2001 From: Eric Sizer Date: Fri, 13 Dec 2024 14:28:32 -0500 Subject: [PATCH 06/31] [Debt] Replace pool stream enum with work stream model (#12222) * add data migration script * update pool grapqhl type to use work stream * update pool publish validation * add missing work stream id column selection * adjust pool candidates test * update playwright tests * remove unused query field * fix pool utils test * run prettier * remove unecessary null safe selector * fix pool filter input * update applicant filters with work stream * fix search requests * fix some tests * fix relationships in factories * update filter input type * fix PHPUnit tests * update frontend with new input * update additional tests * fix lint errors * fix e2e test * document migration command * add work stream filter comment back in * update input fields for consistency * fix scopes, tests for new inputs * remove new references to PoolStream * add error logging in command for missing streams * fix pool table sort * fix count query, improve test * fix test names * fix search form story * check for count on applicant filter migration * fix search request table filter * remove unused function --- api/app/Builders/PoolBuilder.php | 27 ++++- .../Console/Commands/MigratePoolStream.php | 111 ++++++++++++++++++ .../Queries/CountPoolCandidatesByPool.php | 4 +- .../Validators/PoolIsCompleteValidator.php | 2 +- api/app/Models/Pool.php | 2 + api/app/Models/PoolCandidate.php | 2 +- api/app/Models/PoolCandidateSearchRequest.php | 11 +- api/app/Models/User.php | 2 +- .../factories/ApplicantFilterFactory.php | 25 +++- api/database/factories/PoolFactory.php | 8 +- api/graphql/schema.graphql | 21 ++-- api/storage/app/lighthouse-schema.graphql | 20 ++-- api/tests/Feature/ApplicantFilterTest.php | 24 ++-- api/tests/Feature/ApplicantTest.php | 22 ++-- .../Feature/CountPoolCandidatesByPoolTest.php | 18 ++- ...oolCandidateSearchRequestPaginatedTest.php | 27 +++-- api/tests/Feature/PoolTest.php | 36 +++--- .../tests/admin/pool-candidate.spec.ts | 5 +- .../playwright/tests/search-workflows.spec.ts | 11 +- apps/playwright/utils/pools.ts | 12 +- apps/playwright/utils/workStreams.ts | 29 +++++ .../ApplicationCard/ApplicationCard.tsx | 10 +- .../AssessmentResultsTable.tsx | 14 --- .../CandidateDialog/ChangeDateDialog.tsx | 8 +- .../CandidateDialog/ChangeStatusDialog.tsx | 18 +-- .../PoolCandidatesTable.tsx | 10 +- .../PoolCandidatesTable/helpers.tsx | 12 +- apps/web/src/components/PoolCard/PoolCard.tsx | 8 +- .../PoolFilterInput/usePoolFilterOptions.ts | 8 +- .../PoolStatusTable/PoolStatusTable.tsx | 10 +- .../src/components/PoolStatusTable/types.ts | 6 +- .../RecruitmentAvailabilityDialog.tsx | 8 +- .../SearchRequestFilters.tsx | 10 +- .../SearchRequestTable/SearchRequestTable.tsx | 16 +-- .../components/SearchRequestFilterDialog.tsx | 15 ++- .../SearchRequestTable/components/utils.tsx | 13 +- .../ApplicationWelcomePage.tsx | 2 +- apps/web/src/pages/Applications/fragment.ts | 6 +- .../components/ReviewTalentRequestDialog.tsx | 8 +- .../ViewPoolCandidatePage.tsx | 8 +- .../pages/Pools/EditPoolPage/EditPoolPage.tsx | 12 +- .../components/PoolNameSection/Display.tsx | 6 +- .../PoolNameSection/PoolNameSection.tsx | 19 +-- .../components/PoolNameSection/utils.ts | 10 +- .../UpdatePublishedProcessDialog.tsx | 8 +- .../components/PoolFilterDialog.tsx | 20 ++-- .../IndexPoolPage/components/PoolTable.tsx | 17 ++- .../IndexPoolPage/components/helpers.tsx | 24 +++- .../PoolAdvertisementPage.tsx | 18 +-- apps/web/src/pages/Pools/PoolLayout.tsx | 10 +- .../pages/Pools/ViewPoolPage/ViewPoolPage.tsx | 8 +- .../CareerTimelineAndRecruitment.tsx | 3 - .../RequestPage/components/RequestForm.tsx | 39 +++--- .../SearchPage/components/FormFields.tsx | 30 +++-- .../components/SearchForm.stories.tsx | 10 +- .../SearchPage/components/SearchForm.tsx | 28 ++++- .../components/SearchResultCard.tsx | 8 +- .../pages/SearchRequests/SearchPage/utils.ts | 6 +- .../SearchRequestCandidatesTable.tsx | 94 ++++----------- .../components/ViewSearchRequest.tsx | 18 +-- .../AdminUserProfilePage.tsx | 6 +- .../UserInformationPage.tsx | 6 +- .../components/AddToPoolDialog.tsx | 8 +- .../components/NotesSection.tsx | 6 +- .../UserCandidatesTable.tsx | 10 +- .../UserCandidatesTable/helpers.tsx | 4 +- apps/web/src/types/searchRequest.ts | 3 +- apps/web/src/utils/poolUtils.test.ts | 24 ++-- apps/web/src/utils/poolUtils.tsx | 14 +-- apps/web/src/utils/requestUtils.tsx | 12 +- apps/web/src/utils/testData.ts | 2 + .../src/validators/process/classification.ts | 12 +- .../fake-data/src/fakeApplicantFilters.ts | 16 ++- packages/fake-data/src/fakePools.ts | 12 +- packages/fake-data/src/fakeWorkStreams.ts | 20 ++++ packages/fake-data/src/index.ts | 2 + 76 files changed, 704 insertions(+), 450 deletions(-) create mode 100644 api/app/Console/Commands/MigratePoolStream.php create mode 100644 apps/playwright/utils/workStreams.ts create mode 100644 packages/fake-data/src/fakeWorkStreams.ts diff --git a/api/app/Builders/PoolBuilder.php b/api/app/Builders/PoolBuilder.php index d286909b559..88977c963f7 100644 --- a/api/app/Builders/PoolBuilder.php +++ b/api/app/Builders/PoolBuilder.php @@ -137,14 +137,16 @@ public function publishingGroups(?array $publishingGroups): self return $this->whereIn('publishing_group', $publishingGroups); } - public function streams(?array $streams): self + public function whereWorkStreamsIn(?array $streams): self { if (empty($streams)) { return $this; } - return $this->whereIn('stream', $streams); + return $this->whereHas('workStream', function ($query) use ($streams) { + $query->whereIn('id', $streams); + }); } public function whereClassifications(?array $classifications): self @@ -168,7 +170,7 @@ public function whereClassifications(?array $classifications): self /** * Custom sort to handle issues with how laravel aliases * aggregate selects and orderBys for json fields in `lighthouse-php` - * + * The column used in the orderBy is `table_aggregate_column->property` * But is actually aliased to snake case `table_aggregate_columnproperty` */ @@ -184,6 +186,25 @@ public function orderByTeamDisplayName(?array $args): self return $this; } + /** + * Custom sort to handle issues with how laravel aliases + * aggregate selects and orderBys for json fields in `lighthouse-php` + * + * The column used in the orderBy is `table_aggregate_column->property` + * But is actually aliased to snake case `table_aggregate_columnproperty` + */ + public function orderByWorkStreamName(?array $args): self + { + $order = $args['order'] ?? null; + $locale = $args['locale'] ?? null; + + if ($order && $locale) { + return $this->withMax('workStream', 'name->'.$locale)->orderBy('work_stream_max_name'.$locale, $order); + } + + return $this; + } + public function orderByPoolBookmarks(?array $args): self { /** @var \App\Models\User|null */ diff --git a/api/app/Console/Commands/MigratePoolStream.php b/api/app/Console/Commands/MigratePoolStream.php new file mode 100644 index 00000000000..22ffc65f93b --- /dev/null +++ b/api/app/Console/Commands/MigratePoolStream.php @@ -0,0 +1,111 @@ +info('Updating job poster template work streams...'); + $jobPosterTemplatesUpdated = 0; + JobPosterTemplate::whereNotNull('stream')->chunkById(200, function (Collection $jobPosterTemplates) use ($workStreams, &$jobPosterTemplatesUpdated) { + foreach ($jobPosterTemplates as $jobPosterTemplate) { + $stream = $workStreams->where('key', $jobPosterTemplate->stream)->first(); + if ($stream) { + $jobPosterTemplate->work_stream_id = $stream->id; + $jobPosterTemplate->save(); + $jobPosterTemplatesUpdated++; + } else { + $this->error(sprintf('Work stream (%s) not found for job poster template (%s)', + $jobPosterTemplate->stream, + $jobPosterTemplate->id)); + } + } + }); + $this->info("Updated $jobPosterTemplatesUpdated job poster templates"); + + $this->newLine(); + $this->info('Updating pool work streams...'); + $poolsUpdated = 0; + Pool::whereNotNull('stream')->chunkById(200, function (Collection $pools) use ($workStreams, &$poolsUpdated) { + foreach ($pools as $pool) { + $stream = $workStreams->where('key', $pool->stream)->first(); + if ($stream) { + $pool->work_stream_id = $stream->id; + $pool->save(); + $poolsUpdated++; + } else { + $this->error(sprintf('Work stream (%s) not found for pool (%s)', + $pool->stream, + $pool->id)); + } + } + }); + $this->info("Updated $poolsUpdated pools"); + + $applicantFiltersUpdated = 0; + $this->newLine(); + $this->info('Updated applicant filter work streams...'); + ApplicantFilter::whereNotNull('qualified_streams')->chunkById(200, function (Collection $applicantFilters) use ($workStreams, &$applicantFiltersUpdated) { + foreach ($applicantFilters as $applicantFilter) { + $streams = $workStreams->whereIn('key', $applicantFilter->qualified_streams)->pluck('id'); + if ($streams->count()) { + $applicantFilter->workStreams()->sync($streams); + $applicantFiltersUpdated++; + } else { + $this->error(sprintf('Work streams (%s) not found for applicant filter (%s)', + implode(', ', $applicantFilter->qualified_streams), + $applicantFilter->id)); + } + } + }); + $this->info("Updated $applicantFiltersUpdated applicant filters"); + + $this->newLine(); + $this->info('Migration completed'); + $this->table(['Model', 'Affected'], [ + ['JobPosterTemplate', $jobPosterTemplatesUpdated], + ['Pool', $poolsUpdated], + ['ApplicantFilter', $applicantFiltersUpdated], + ]); + + } +} diff --git a/api/app/GraphQL/Queries/CountPoolCandidatesByPool.php b/api/app/GraphQL/Queries/CountPoolCandidatesByPool.php index 3433f3a6aa5..f5a002d8fc7 100644 --- a/api/app/GraphQL/Queries/CountPoolCandidatesByPool.php +++ b/api/app/GraphQL/Queries/CountPoolCandidatesByPool.php @@ -28,8 +28,8 @@ public function __invoke($_, array $args) $query->whereClassifications($filters['qualifiedClassifications']); } - if (array_key_exists('qualifiedStreams', $filters)) { - $query->streams($filters['qualifiedStreams']); + if (array_key_exists('workStreams', $filters)) { + $query->whereWorkStreamsIn($filters['workStreams']); } }); diff --git a/api/app/GraphQL/Validators/PoolIsCompleteValidator.php b/api/app/GraphQL/Validators/PoolIsCompleteValidator.php index dafedbcc457..ef6fd8f468c 100644 --- a/api/app/GraphQL/Validators/PoolIsCompleteValidator.php +++ b/api/app/GraphQL/Validators/PoolIsCompleteValidator.php @@ -30,7 +30,7 @@ public function rules(): array 'name.fr' => ['string'], 'classification_id' => ['required', 'uuid', 'exists:classifications,id'], 'department_id' => ['required', 'uuid', 'exists:departments,id'], - 'stream' => ['required', 'string'], + 'work_stream_id' => ['required', 'uuid', 'exists:work_streams,id'], 'opportunity_length' => ['required', 'string'], // Closing date diff --git a/api/app/Models/Pool.php b/api/app/Models/Pool.php index 02dcea8609d..104f4f67872 100644 --- a/api/app/Models/Pool.php +++ b/api/app/Models/Pool.php @@ -50,6 +50,7 @@ * @property string $team_id * @property string $department_id * @property string $community_id + * @property string $work_stream_id * @property ?string $area_of_selection * @property array $selection_limitations * @property \Illuminate\Support\Carbon $created_at @@ -130,6 +131,7 @@ class Pool extends Model 'process_number', 'department_id', 'community_id', + 'work_stream_id', ]; /** diff --git a/api/app/Models/PoolCandidate.php b/api/app/Models/PoolCandidate.php index 477d46ca04b..1b6dadcb303 100644 --- a/api/app/Models/PoolCandidate.php +++ b/api/app/Models/PoolCandidate.php @@ -310,7 +310,7 @@ public static function scopeQualifiedStreams(Builder $query, ?array $streams): B }) // Now scope for valid pools, according to streams ->whereHas('pool', function ($query) use ($streams) { - $query->whereIn('stream', $streams); + $query->whereWorkStreamsIn($streams); }); return $query; diff --git a/api/app/Models/PoolCandidateSearchRequest.php b/api/app/Models/PoolCandidateSearchRequest.php index a79bb0f06ae..f9215ae4939 100644 --- a/api/app/Models/PoolCandidateSearchRequest.php +++ b/api/app/Models/PoolCandidateSearchRequest.php @@ -146,18 +146,15 @@ public static function scopeSearchRequestStatus(Builder $query, ?array $searchRe return $query; } - public static function scopeStreams(Builder $query, ?array $streams): Builder + public static function scopeWorkStreams(Builder $query, ?array $streams): Builder { if (empty($streams)) { return $query; } - // streams is an array of PoolStream enums - $query->whereHas('applicantFilter', function ($query) use ($streams) { - $query->where(function ($query) use ($streams) { - foreach ($streams as $index => $stream) { - $query->orWhereJsonContains('qualified_streams', $stream); - } + $query->whereHas('applicantFilter', function ($filterQuery) use ($streams) { + $filterQuery->whereHas('workStreams', function ($workStreamQuery) use ($streams) { + $workStreamQuery->whereIn('applicant_filter_work_stream.work_stream_id', $streams); }); }); diff --git a/api/app/Models/User.php b/api/app/Models/User.php index acce75e0aa5..078c5fd8fe8 100644 --- a/api/app/Models/User.php +++ b/api/app/Models/User.php @@ -779,7 +779,7 @@ public static function scopeTalentSearchablePublishingGroup(Builder $query, $arg } if (array_key_exists('qualifiedStreams', $filters)) { - $query->streams($filters['qualifiedStreams']); + $query->whereWorkStreamsIn($filters['qualifiedStreams']); } }); diff --git a/api/database/factories/ApplicantFilterFactory.php b/api/database/factories/ApplicantFilterFactory.php index 645b4b3d231..05382731833 100644 --- a/api/database/factories/ApplicantFilterFactory.php +++ b/api/database/factories/ApplicantFilterFactory.php @@ -4,7 +4,6 @@ use App\Enums\LanguageAbility; use App\Enums\OperationalRequirement; -use App\Enums\PoolStream; use App\Enums\PositionDuration; use App\Enums\WorkRegion; use App\Models\ApplicantFilter; @@ -95,21 +94,35 @@ public function withRelationships(bool $sparse = false) )->get(); $filter->pools()->saveMany($pools); $filter->qualifiedClassifications()->saveMany($pools->flatMap(fn ($pool) => $pool->classification)); - $stream = (empty($pools) || count($pools) === 0) ? $this->faker->randomElements( - array_column(PoolStream::cases(), 'name'), - ) : [$pools[0]->stream]; + $streams = $pools->map(fn ($pool) => $pool->workStream); + $filter->workStreams()->saveMany($streams); $ATIP = Community::where('key', 'atip')->first(); $digital = Community::where('key', 'digital')->first(); - if (in_array(PoolStream::ACCESS_INFORMATION_PRIVACY->name, $stream) && $ATIP?->id) { + $isATIP = $streams->contains(fn ($stream) => $stream->key === 'ACCESS_INFORMATION_PRIVACY'); + + if ($isATIP && $ATIP?->id) { $filter->community_id = $ATIP->id; } elseif ($digital?->id) { $filter->community_id = $digital->id; } - $filter->qualified_streams = $stream; $filter->save(); }); } + + /** + * Create an ApplicantFilter with specific work streams + * + * @var \Illuminate\Support\Collection + * + * @return \Illuminate\Database\Eloquent\Factories\Factory + */ + public function withWorkStreams(array $workStreams) + { + return $this->afterCreating(function (ApplicantFilter $filter) use ($workStreams) { + $filter->workStreams()->saveMany($workStreams); + }); + } } diff --git a/api/database/factories/PoolFactory.php b/api/database/factories/PoolFactory.php index 679d02436ad..c6a446861f2 100644 --- a/api/database/factories/PoolFactory.php +++ b/api/database/factories/PoolFactory.php @@ -188,9 +188,9 @@ public function draft(): Factory // the base state is draft already $hasSpecialNote = $this->faker->boolean(); $isRemote = $this->faker->boolean(); - $workStreamId = WorkStream::inRandomOrder()->first()?->id; - if (! $workStreamId) { - $workStreamId = WorkStream::factory()->create()->id; + $workStream = WorkStream::inRandomOrder()->first(); + if (! $workStream) { + $workStream = WorkStream::factory()->create(); } return [ @@ -207,7 +207,7 @@ public function draft(): Factory 'special_note' => ! $hasSpecialNote ? ['en' => $this->faker->paragraph().' EN', 'fr' => $this->faker->paragraph().' FR'] : null, 'is_remote' => $this->faker->boolean, 'stream' => $this->faker->randomElement(PoolStream::cases())->name, - 'work_stream_id' => $workStreamId, + 'work_stream_id' => $workStream->id, 'process_number' => $this->faker->word(), 'publishing_group' => $this->faker->randomElement(array_column(PublishingGroup::cases(), 'name')), 'opportunity_length' => $this->faker->randomElement(array_column(PoolOpportunityLength::cases(), 'name')), diff --git a/api/graphql/schema.graphql b/api/graphql/schema.graphql index 1d63db08d32..d8eba0e270a 100644 --- a/api/graphql/schema.graphql +++ b/api/graphql/schema.graphql @@ -278,7 +278,7 @@ type Pool implements HasRoleAssignments { @rename(attribute: "security_clearance") language: LocalizedPoolLanguage @rename(attribute: "advertisement_language") status: LocalizedPoolStatus @rename(attribute: "status") - stream: LocalizedPoolStream + workStream: WorkStream @belongsTo processNumber: String @rename(attribute: "process_number") publishingGroup: LocalizedPublishingGroup @rename(attribute: "publishing_group") @@ -590,8 +590,7 @@ type ApplicantFilter { skills: [Skill] @belongsToMany # request creation connects to qualifiedClassifications qualifiedClassifications: [Classification] @belongsToMany # Filters applicants based on the classifications pools they've qualified in. - qualifiedStreams: [LocalizedPoolStream] - @rename(attribute: "qualified_streams") # Filters applicants based on the streams of the pools they've qualified in. + workStreams: [WorkStream!] @belongsToMany # Filters applicants based on the streams of the pools they've qualified in. pools: [Pool] @belongsToMany @canResolved(ability: "view") community: Community @belongsTo # Currently does not affect search, scope needs to be added } @@ -854,7 +853,7 @@ input PoolFilterInput { generalSearch: String @scope name: String @scope team: String @scope - streams: [PoolStream!] @scope + workStreams: [UUID!] @scope(name: "whereWorkStreamsIn") statuses: [PoolStatus!] = [] @scope processNumber: String @scope publishingGroups: [PublishingGroup!] @scope @@ -868,6 +867,11 @@ input PoolTeamDisplayNameOrderByInput { order: SortOrder! } +input PoolWorkStreamNameOrderByInput { + locale: String! + order: SortOrder! +} + input PoolBookmarksOrderByInput { column: String! order: SortOrder! @@ -893,7 +897,7 @@ input ApplicantFilterInput { skillsIntersectional: [IdInput] @scope @pluck(key: "id") # AND filtering skills as opposed to OR qualifiedClassifications: [ClassificationFilterInput] @scope(name: "qualifiedClassifications") # Filters applicants based on the classification of pools they are currently qualified and available in. - qualifiedStreams: [PoolStream] @scope(name: "qualifiedStreams") # Filters applicants based on the stream of pools they are currently qualified and available in. + workStreams: [IdInput] @scope(name: "qualifiedStreams") # Filters applicants based on the stream of pools they are currently qualified and available in. community: IdInput @scope(name: "candidatesInCommunity") @pluck(key: "id") } @@ -921,7 +925,7 @@ input PoolCandidateSearchRequestInput { status: [PoolCandidateSearchStatus] @scope(name: "searchRequestStatus") departments: [ID] @scope classifications: [ID] @scope - streams: [PoolStream] @scope + workStreams: [UUID] @scope fullName: String @scope email: String @scope id: ID @scope @@ -1027,6 +1031,7 @@ type Query { where: PoolFilterInput orderByPoolBookmarks: PoolBookmarksOrderByInput @scope orderByTeamDisplayName: PoolTeamDisplayNameOrderByInput @scope + orderByWorkStreamName: PoolWorkStreamNameOrderByInput @scope orderByColumn: OrderByColumnInput @scope orderBy: _ @orderBy( @@ -1584,7 +1589,7 @@ input CreateApplicantFilterInput { citizenship: CitizenshipStatus armedForcesStatus: ArmedForcesStatus @rename(attribute: "armed_forces_status") qualifiedClassifications: ClassificationBelongsToMany - qualifiedStreams: [PoolStream] @rename(attribute: "qualified_streams") + workStreams: WorkStreamBelongsToMany community: CommunityBelongsTo } @@ -1828,7 +1833,7 @@ input UpdatePoolInput { name: LocalizedStringInput classification: ClassificationBelongsTo department: DepartmentBelongsTo - stream: PoolStream + workStream: WorkStreamBelongsTo processNumber: String @rename(attribute: "process_number") # Closing date closingDate: DateTime diff --git a/api/storage/app/lighthouse-schema.graphql b/api/storage/app/lighthouse-schema.graphql index 97b3f979f44..ce73f43417a 100755 --- a/api/storage/app/lighthouse-schema.graphql +++ b/api/storage/app/lighthouse-schema.graphql @@ -346,6 +346,7 @@ type Query { where: PoolFilterInput orderByPoolBookmarks: PoolBookmarksOrderByInput orderByTeamDisplayName: PoolTeamDisplayNameOrderByInput + orderByWorkStreamName: PoolWorkStreamNameOrderByInput orderByColumn: OrderByColumnInput orderBy: [QueryPoolsPaginatedOrderByRelationOrderByClause!] @@ -738,7 +739,7 @@ type Pool implements HasRoleAssignments { securityClearance: LocalizedSecurityStatus language: LocalizedPoolLanguage status: LocalizedPoolStatus - stream: LocalizedPoolStream + workStream: WorkStream processNumber: String publishingGroup: LocalizedPublishingGroup opportunityLength: LocalizedPoolOpportunityLength @@ -973,7 +974,7 @@ type ApplicantFilter { positionDuration: [PositionDuration] skills: [Skill] qualifiedClassifications: [Classification] - qualifiedStreams: [LocalizedPoolStream] + workStreams: [WorkStream!] pools: [Pool] community: Community } @@ -1206,7 +1207,7 @@ input PoolFilterInput { generalSearch: String name: String team: String - streams: [PoolStream!] + workStreams: [UUID!] statuses: [PoolStatus!] = [] processNumber: String publishingGroups: [PublishingGroup!] @@ -1219,6 +1220,11 @@ input PoolTeamDisplayNameOrderByInput { order: SortOrder! } +input PoolWorkStreamNameOrderByInput { + locale: String! + order: SortOrder! +} + input PoolBookmarksOrderByInput { column: String! order: SortOrder! @@ -1241,7 +1247,7 @@ input ApplicantFilterInput { skills: [IdInput] skillsIntersectional: [IdInput] qualifiedClassifications: [ClassificationFilterInput] - qualifiedStreams: [PoolStream] + workStreams: [IdInput] community: IdInput } @@ -1265,7 +1271,7 @@ input PoolCandidateSearchRequestInput { status: [PoolCandidateSearchStatus] departments: [ID] classifications: [ID] - streams: [PoolStream] + workStreams: [UUID] fullName: String email: String id: ID @@ -1641,7 +1647,7 @@ input CreateApplicantFilterInput { citizenship: CitizenshipStatus armedForcesStatus: ArmedForcesStatus qualifiedClassifications: ClassificationBelongsToMany - qualifiedStreams: [PoolStream] + workStreams: WorkStreamBelongsToMany community: CommunityBelongsTo } @@ -1870,7 +1876,7 @@ input UpdatePoolInput { name: LocalizedStringInput classification: ClassificationBelongsTo department: DepartmentBelongsTo - stream: PoolStream + workStream: WorkStreamBelongsTo processNumber: String closingDate: DateTime closingReason: String diff --git a/api/tests/Feature/ApplicantFilterTest.php b/api/tests/Feature/ApplicantFilterTest.php index e83989dab30..767c2528f06 100644 --- a/api/tests/Feature/ApplicantFilterTest.php +++ b/api/tests/Feature/ApplicantFilterTest.php @@ -4,7 +4,6 @@ use App\Enums\LanguageAbility; use App\Enums\PoolCandidateSearchStatus; -use App\Enums\PoolStream; use App\Facades\Notify; use App\Models\ApplicantFilter; use App\Models\Community; @@ -83,13 +82,13 @@ protected function filterToInput(ApplicantFilter $filter) 'positionDuration' => $filter->position_duration, 'skills' => $filter->skills->map($onlyId)->toArray(), 'pools' => $filter->pools->map($onlyId)->toArray(), + 'workStreams' => $filter->workStreams->map($onlyId)->toArray(), 'qualifiedClassifications' => $filter->qualifiedClassifications->map(function ($classification) { return [ 'group' => $classification->group, 'level' => $classification->level, ]; })->toArray(), - 'qualifiedStreams' => $filter->qualified_streams, 'community' => ['id' => $filter->community->id], ]; } @@ -103,6 +102,9 @@ protected function filterToCreateInput(ApplicantFilter $filter) $input['pools'] = [ 'sync' => $filter->pools->pluck('id')->toArray(), ]; + $input['workStreams'] = [ + 'sync' => $filter->workStreams->pluck('id')->toArray(), + ]; $input['qualifiedClassifications'] = [ 'sync' => $filter->qualifiedClassifications->pluck('id')->toArray(), ]; @@ -284,7 +286,7 @@ public function testQueryRelationships() localized } } - qualifiedStreams { value } + workStreams { id } qualifiedClassifications { id name { @@ -309,7 +311,7 @@ public function testQueryRelationships() $this->assertCount($filter->qualifiedClassifications->count(), $retrievedFilter['qualifiedClassifications']); $this->assertCount($filter->skills->count(), $retrievedFilter['skills']); $this->assertCount($filter->pools->count(), $retrievedFilter['pools']); - $this->assertCount(count($filter->qualified_streams), $retrievedFilter['qualifiedStreams']); + $this->assertCount($filter->workStreams->count(), $retrievedFilter['workStreams']); // Assert that all the content in each collection is correct. foreach ($filter->pools as $pool) { @@ -325,9 +327,11 @@ public function testQueryRelationships() $response->assertJsonFragment(['id' => $skill->id, 'name' => $skill->name]); } - $response->assertJsonFragment(['qualifiedStreams' => [[ - 'value' => $filter->qualified_streams[0], - ]]]); + foreach ($filter->workStreams as $workStream) { + $response->assertJsonFragment([ + 'id' => $workStream->id, + ]); + } $response->assertJsonFragment(['community' => ['id' => $filter->community_id]]); } @@ -437,7 +441,6 @@ public function testFilterCanBeStoredAndRetrievedWithoutChangingResults() 'en' => 'Test Pool EN', 'fr' => 'Test Pool FR', ], - 'stream' => PoolStream::BUSINESS_ADVISORY_SERVICES->name, 'community_id' => $community->id, ]); // Create candidates who may show up in searches @@ -478,7 +481,7 @@ public function testFilterCanBeStoredAndRetrievedWithoutChangingResults() $candidateSkills = $candidateUser->experiences[0]->skills; $filter->skills()->saveMany($candidateSkills->shuffle()->take(3)); $filter->pools()->save($pool); - $filter->qualified_streams = $pool->stream; + $filter->workStreams()->saveMany([$pool->workStream]); $filter->save(); $response = $this->graphQL( /** @lang GraphQL */ @@ -550,7 +553,7 @@ public function testFilterCanBeStoredAndRetrievedWithoutChangingResults() locationPreferences { value } operationalRequirements { value } positionDuration - qualifiedStreams { value } + workStreams { id } qualifiedClassifications { group level @@ -576,7 +579,6 @@ public function testFilterCanBeStoredAndRetrievedWithoutChangingResults() $retrievedFilter['locationPreferences'] = $this->filterEnumToInput($retrievedFilter, 'locationPreferences'); $retrievedFilter['operationalRequirements'] = $this->filterEnumToInput($retrievedFilter, 'operationalRequirements'); - $retrievedFilter['qualifiedStreams'] = $this->filterEnumToInput($retrievedFilter, 'qualifiedStreams'); // Now use the retrieved filter to get the same count $response = $this->graphQL( diff --git a/api/tests/Feature/ApplicantTest.php b/api/tests/Feature/ApplicantTest.php index d6464888513..6c570118bdf 100644 --- a/api/tests/Feature/ApplicantTest.php +++ b/api/tests/Feature/ApplicantTest.php @@ -8,7 +8,6 @@ use App\Enums\LanguageAbility; use App\Enums\OperationalRequirement; use App\Enums\PoolCandidateStatus; -use App\Enums\PoolStream; use App\Enums\PositionDuration; use App\Enums\PublishingGroup; use App\Facades\Notify; @@ -21,6 +20,7 @@ use App\Models\Skill; use App\Models\User; use App\Models\WorkExperience; +use App\Models\WorkStream; use Database\Seeders\RolePermissionSeeder; use Illuminate\Foundation\Testing\RefreshDatabase; use Nuwave\Lighthouse\Testing\MakesGraphQLRequests; @@ -1922,24 +1922,24 @@ public function testClassificationAndStreamsFilter() $targetClassification = Classification::factory()->create(); $excludedClassification = Classification::factory()->create(); - $targetStream = PoolStream::BUSINESS_ADVISORY_SERVICES->name; - $excludedStream = PoolStream::ACCESS_INFORMATION_PRIVACY->name; + $targetStream = WorkStream::factory()->create(); + $excludedStream = WorkStream::factory()->create(); $targetClassificationPool = Pool::factory()->candidatesAvailableInSearch()->create([ 'classification_id' => $targetClassification, - 'stream' => $excludedStream, + 'work_stream_id' => $excludedStream->id, ]); $targetStreamPool = Pool::factory()->candidatesAvailableInSearch()->create([ 'classification_id' => $excludedClassification, - 'stream' => $targetStream, + 'work_stream_id' => $targetStream->id, ]); $targetStreamAndClassificationPool = Pool::factory()->candidatesAvailableInSearch()->create([ 'classification_id' => $targetClassification, - 'stream' => $targetStream, + 'work_stream_id' => $targetStream->id, ]); $excludedPool = Pool::factory()->candidatesAvailableInSearch()->create([ 'classification_id' => $excludedClassification, - 'stream' => $excludedStream, + 'work_stream_id' => $excludedStream->id, ]); $targetUser = User::factory()->create(); @@ -2017,7 +2017,7 @@ public function testClassificationAndStreamsFilter() ->graphQL($query, [ 'where' => [ - 'qualifiedStreams' => [$targetStream], + 'workStreams' => [['id' => $targetStream->id]], ], ] )->assertJson([ @@ -2036,9 +2036,9 @@ public function testClassificationAndStreamsFilter() 'level' => $targetClassification->level, ], ], - 'qualifiedStreams' => [ - $targetStream, - ], + 'workStreams' => [[ + 'id' => $targetStream->id, + ]], ], ] )->assertJson([ diff --git a/api/tests/Feature/CountPoolCandidatesByPoolTest.php b/api/tests/Feature/CountPoolCandidatesByPoolTest.php index 0566e8175bd..d0ba1da5c5c 100644 --- a/api/tests/Feature/CountPoolCandidatesByPoolTest.php +++ b/api/tests/Feature/CountPoolCandidatesByPoolTest.php @@ -5,7 +5,6 @@ use App\Enums\LanguageAbility; use App\Enums\OperationalRequirement; use App\Enums\PoolCandidateStatus; -use App\Enums\PoolStream; use App\Enums\PositionDuration; use App\Enums\PublishingGroup; use App\Enums\WorkRegion; @@ -16,6 +15,7 @@ use App\Models\PoolCandidate; use App\Models\Skill; use App\Models\User; +use App\Models\WorkStream; use Database\Seeders\RolePermissionSeeder; use Illuminate\Foundation\Testing\RefreshDatabase; use Nuwave\Lighthouse\Testing\MakesGraphQLRequests; @@ -771,9 +771,21 @@ public function testAdditionalAvailabilityScopes() 'group' => 'IT', 'level' => 1, ]); + + $unaccosiatedStream = WorkStream::factory()->create(); + $unaccosiatedPool = Pool::factory()->candidatesAvailableInSearch()->published()->create([ + 'work_stream_id' => $unaccosiatedStream->id, + ]); + PoolCandidate::factory()->create([ + 'pool_id' => $unaccosiatedPool->id, + 'pool_candidate_status' => PoolCandidateStatus::QUALIFIED_AVAILABLE->name, + 'expiry_date' => config('constants.far_future_date'), + ]); + + $stream = WorkStream::factory()->create(); $pool = Pool::factory()->candidatesAvailableInSearch()->create([ 'published_at' => config('constants.past_date'), - 'stream' => PoolStream::ACCESS_INFORMATION_PRIVACY->name, + 'work_stream_id' => $stream->id, ]); $user1 = User::factory()->create(); PoolCandidate::factory()->create([ @@ -802,7 +814,7 @@ public function testAdditionalAvailabilityScopes() 'level' => 1, ], ], - 'qualifiedStreams' => [PoolStream::ACCESS_INFORMATION_PRIVACY->name], + 'workStreams' => [['id' => $stream->id]], ], ] )->assertSimilarJson([ diff --git a/api/tests/Feature/PoolCandidateSearchRequestPaginatedTest.php b/api/tests/Feature/PoolCandidateSearchRequestPaginatedTest.php index 060e9571ee3..074695f78d1 100644 --- a/api/tests/Feature/PoolCandidateSearchRequestPaginatedTest.php +++ b/api/tests/Feature/PoolCandidateSearchRequestPaginatedTest.php @@ -3,13 +3,13 @@ namespace Tests\Feature; use App\Enums\PoolCandidateSearchStatus; -use App\Enums\PoolStream; use App\Models\ApplicantFilter; use App\Models\Classification; use App\Models\Community; use App\Models\Department; use App\Models\PoolCandidateSearchRequest; use App\Models\User; +use App\Models\WorkStream; use Database\Seeders\ClassificationSeeder; use Database\Seeders\DepartmentSeeder; use Database\Seeders\RolePermissionSeeder; @@ -299,13 +299,14 @@ public function testSearchRequestClassificationsFiltering(): void public function testSearchRequestStreamsFiltering(): void { - $applicantFilter1 = ApplicantFilter::factory()->create([ - 'qualified_streams' => [PoolStream::SECURITY->name], - ]); - $applicantFilter2 = ApplicantFilter::factory()->create([ - 'qualified_streams' => [PoolStream::BUSINESS_ADVISORY_SERVICES->name], - ]); + $stream1 = WorkStream::factory()->create(); + + $applicantFilter1 = ApplicantFilter::factory()->withWorkStreams([$stream1])->create(); + + $stream2 = WorkStream::factory()->create(); + + $applicantFilter2 = ApplicantFilter::factory()->withWorkStreams([$stream2])->create(); PoolCandidateSearchRequest::factory()->count(1)->create([ 'applicant_filter_id' => $applicantFilter1->id, @@ -316,16 +317,18 @@ public function testSearchRequestStreamsFiltering(): void // streams null results in 3 results $this->actingAs($this->requestResponder, 'api') - ->graphQL($this->searchRequestQuery, ['where' => ['streams' => null]]) + ->graphQL($this->searchRequestQuery, ['where' => ['workStreams' => null]]) ->assertJsonFragment(['count' => 3]); + $unattachedStream = WorkStream::factory()->create(); + // infrastructure passed in returns 0 results $this->actingAs($this->requestResponder, 'api') ->graphQL( $this->searchRequestQuery, [ 'where' => [ - 'streams' => [PoolStream::INFRASTRUCTURE_OPERATIONS->name], + 'workStreams' => [$unattachedStream->id], ], ] ) @@ -337,7 +340,7 @@ public function testSearchRequestStreamsFiltering(): void $this->searchRequestQuery, [ 'where' => [ - 'streams' => [PoolStream::SECURITY->name], + 'workStreams' => [$stream1->id], ], ] ) @@ -349,7 +352,7 @@ public function testSearchRequestStreamsFiltering(): void $this->searchRequestQuery, [ 'where' => [ - 'streams' => [PoolStream::SECURITY->name, PoolStream::INFRASTRUCTURE_OPERATIONS->name], + 'workStreams' => [$stream1->id, $unattachedStream->id], ], ] ) @@ -361,7 +364,7 @@ public function testSearchRequestStreamsFiltering(): void $this->searchRequestQuery, [ 'where' => [ - 'streams' => [PoolStream::SECURITY->name, PoolStream::BUSINESS_ADVISORY_SERVICES->name], + 'workStreams' => [$stream1->id, $stream2->id], ], ] ) diff --git a/api/tests/Feature/PoolTest.php b/api/tests/Feature/PoolTest.php index f0cbc9292fc..bb491d03bf9 100644 --- a/api/tests/Feature/PoolTest.php +++ b/api/tests/Feature/PoolTest.php @@ -3,7 +3,6 @@ namespace Tests\Feature; use App\Enums\PoolStatus; -use App\Enums\PoolStream; use App\Enums\PublishingGroup; use App\Enums\SkillCategory; use App\Models\Classification; @@ -14,6 +13,7 @@ use App\Models\Skill; use App\Models\Team; use App\Models\User; +use App\Models\WorkStream; use Carbon\Carbon; use Database\Helpers\ApiErrorEnums; use Database\Seeders\RolePermissionSeeder; @@ -1159,16 +1159,20 @@ public function testPoolTeamScope(): void */ public function testPoolStreamsScope(): void { - $ATIP = Pool::factory()->published()->create([ - 'stream' => PoolStream::ACCESS_INFORMATION_PRIVACY->name, + $stream1 = WorkStream::factory()->create(); + $stream2 = WorkStream::factory()->create(); + $unassociatedStream = WorkStream::factory()->create(); + + $pool1 = Pool::factory()->published()->create([ + 'work_stream_id' => $stream1->id, ]); - $BAS = Pool::factory()->published()->create([ - 'stream' => PoolStream::BUSINESS_ADVISORY_SERVICES->name, + $pool2 = Pool::factory()->published()->create([ + 'work_stream_id' => $stream2->id, ]); Pool::factory()->published()->create([ - 'stream' => PoolStream::DATABASE_MANAGEMENT->name, + 'work_stream_id' => $unassociatedStream->id, ]); $res = $this->graphQL( @@ -1178,31 +1182,31 @@ public function testPoolStreamsScope(): void poolsPaginated(where: $where) { data { id - stream { value } + workStream { id } } } } ', [ 'where' => [ - 'streams' => [ - PoolStream::ACCESS_INFORMATION_PRIVACY->name, - PoolStream::BUSINESS_ADVISORY_SERVICES->name, + 'workStreams' => [ + $stream1->id, + $stream2->id, ], ], ] )->assertJsonFragment([ 'data' => [ [ - 'id' => $ATIP->id, - 'stream' => [ - 'value' => PoolStream::ACCESS_INFORMATION_PRIVACY->name, + 'id' => $pool1->id, + 'workStream' => [ + 'id' => $stream1->id, ], ], [ - 'id' => $BAS->id, - 'stream' => [ - 'value' => PoolStream::BUSINESS_ADVISORY_SERVICES->name, + 'id' => $pool2->id, + 'workStream' => [ + 'id' => $stream2->id, ], ], ], diff --git a/apps/playwright/tests/admin/pool-candidate.spec.ts b/apps/playwright/tests/admin/pool-candidate.spec.ts index 3e97214a579..e0c88280dd2 100644 --- a/apps/playwright/tests/admin/pool-candidate.spec.ts +++ b/apps/playwright/tests/admin/pool-candidate.spec.ts @@ -224,7 +224,7 @@ test.describe("Pool candidates", () => { await appPage.page.getByRole("button", { name: "Save changes" }).click(); await appPage.waitForGraphqlResponse("PoolCandidate_UpdateNotes"); await expect( - appPage.page.getByRole("button", { name: "Edit notes" }), + appPage.page.getByRole("button", { name: /edit notes/i }), ).toBeVisible(); await expect(appPage.page.getByText(/Notes notes notes/i)).toBeVisible(); }); @@ -371,8 +371,9 @@ test.describe("Pool candidates", () => { await expect( appPage.page.getByRole("button", { name: "Removed", exact: true }), ).toBeHidden(); + await appPage.waitForGraphqlResponse("ReinstateCandidate"); await expect( - appPage.page.getByRole("button", { name: "Record final decision" }), + appPage.page.getByRole("button", { name: /remove candidate/i }), ).toBeVisible(); }); }); diff --git a/apps/playwright/tests/search-workflows.spec.ts b/apps/playwright/tests/search-workflows.spec.ts index f36a64270d5..c34364f6320 100644 --- a/apps/playwright/tests/search-workflows.spec.ts +++ b/apps/playwright/tests/search-workflows.spec.ts @@ -9,6 +9,7 @@ import { PoolCandidateStatus, Skill, SkillCategory, + WorkStream, } from "@gc-digital-talent/graphql"; import { test, expect } from "~/fixtures"; @@ -21,12 +22,14 @@ import { createUserWithRoles, me } from "~/utils/user"; import graphql from "~/utils/graphql"; import { createAndPublishPool } from "~/utils/pools"; import { getClassifications } from "~/utils/classification"; +import { getWorkStreams } from "~/utils/workStreams"; test.describe("Talent search", () => { const uniqueTestId = Date.now().valueOf(); const sub = `playwright.sub.${uniqueTestId}`; const poolName = `Search pool ${uniqueTestId}`; let classification: Classification; + let workStream: WorkStream; let skill: Skill | undefined; const expectNoCandidate = async (page: Page) => { @@ -79,12 +82,16 @@ test.describe("Talent search", () => { const classifications = await getClassifications(adminCtx, {}); classification = classifications[0]; + const workStreams = await getWorkStreams(adminCtx, {}); + workStream = workStreams[0]; + const adminUser = await me(adminCtx, {}); // Accepted pool const createdPool = await createAndPublishPool(adminCtx, { userId: adminUser.id, skillIds: technicalSkill ? [technicalSkill?.id] : undefined, classificationId: classification.id, + workStreamId: workStream.id, name: { en: poolName, fr: `${poolName} (FR)`, @@ -138,7 +145,7 @@ test.describe("Talent search", () => { await expectNoCandidate(appPage.page); await streamFilter.selectOption({ - label: "Business Line Advisory Services", + label: workStream.name?.en ?? "", }); await expect(poolCard).toBeVisible(); @@ -221,7 +228,7 @@ test.describe("Talent search", () => { ).toBeVisible(); await expect( - appPage.page.getByText("Business Line Advisory Services"), + appPage.page.getByText(workStream?.name?.en ?? ""), ).toBeVisible(); await expect( diff --git a/apps/playwright/utils/pools.ts b/apps/playwright/utils/pools.ts index 70936cd7461..dadf4db14da 100644 --- a/apps/playwright/utils/pools.ts +++ b/apps/playwright/utils/pools.ts @@ -7,7 +7,6 @@ import { PoolOpportunityLength, PoolSkill, PoolSkillType, - PoolStream, PublishingGroup, SecurityStatus, SkillCategory, @@ -22,9 +21,9 @@ import { getCommunities } from "./communities"; import { getClassifications } from "./classification"; import { getDepartments } from "./departments"; import { getSkills } from "./skills"; +import { getWorkStreams } from "./workStreams"; const defaultPool: Partial = { - stream: PoolStream.BusinessAdvisoryServices, closingDate: `${FAR_FUTURE_DATE} 00:00:00`, yourImpact: { en: "test impact EN", @@ -222,6 +221,7 @@ interface CreateAndPublishPoolArgs { name?: LocalizedString; classificationId?: string; departmentId?: string; + workStreamId?: string; skillIds?: string[]; input?: UpdatePoolInput; } @@ -239,6 +239,7 @@ export const createAndPublishPool: GraphQLRequestFunc< communityId, classificationId, departmentId, + workStreamId, input, }, ) => { @@ -249,6 +250,12 @@ export const createAndPublishPool: GraphQLRequestFunc< classificationId, departmentId, }).then(async (pool) => { + let workStream = workStreamId; + if (!workStream) { + const workStreams = await getWorkStreams(ctx, {}); + workStream = workStreams[0].id; + } + await updatePool(ctx, { poolId: pool.id, pool: { @@ -258,6 +265,7 @@ export const createAndPublishPool: GraphQLRequestFunc< fr: `Playwright Test Pool FR ${Date.now().valueOf()}`, }, ...input, + workStream: { connect: workStream }, }, }); diff --git a/apps/playwright/utils/workStreams.ts b/apps/playwright/utils/workStreams.ts new file mode 100644 index 00000000000..df041e094a5 --- /dev/null +++ b/apps/playwright/utils/workStreams.ts @@ -0,0 +1,29 @@ +import { WorkStream } from "@gc-digital-talent/graphql"; + +import { GraphQLRequestFunc, GraphQLResponse } from "./graphql"; + +const Test_WorkStreamQueryDocument = /* GraphQL */ ` + query WorkStreams { + workStreams { + id + key + name { + en + fr + } + } + } +`; + +/** + * Get work streams + * + * Get all the work streams directly from the API. + */ +export const getWorkStreams: GraphQLRequestFunc = async (ctx) => { + return ctx + .post(Test_WorkStreamQueryDocument) + .then( + (res: GraphQLResponse<"workStreams", WorkStream[]>) => res.workStreams, + ); +}; diff --git a/apps/web/src/components/ApplicationCard/ApplicationCard.tsx b/apps/web/src/components/ApplicationCard/ApplicationCard.tsx index e97e3cf7aec..5638721289f 100644 --- a/apps/web/src/components/ApplicationCard/ApplicationCard.tsx +++ b/apps/web/src/components/ApplicationCard/ApplicationCard.tsx @@ -46,9 +46,9 @@ export const ApplicationCard_Fragment = graphql(/* GraphQL */ ` pool { id closingDate - stream { - value - label { + workStream { + id + name { en fr } @@ -114,13 +114,13 @@ const ApplicationCard = ({ application.submittedAt, ); const applicationTitle = getShortPoolTitleHtml(intl, { - stream: application.pool.stream, + workStream: application.pool.workStream, name: application.pool.name, publishingGroup: application.pool.publishingGroup, classification: application.pool.classification, }); const applicationTitleString = getShortPoolTitleLabel(intl, { - stream: application.pool.stream, + workStream: application.pool.workStream, name: application.pool.name, publishingGroup: application.pool.publishingGroup, classification: application.pool.classification, diff --git a/apps/web/src/components/AssessmentResultsTable/AssessmentResultsTable.tsx b/apps/web/src/components/AssessmentResultsTable/AssessmentResultsTable.tsx index 52c014ae87d..513c8983946 100644 --- a/apps/web/src/components/AssessmentResultsTable/AssessmentResultsTable.tsx +++ b/apps/web/src/components/AssessmentResultsTable/AssessmentResultsTable.tsx @@ -118,13 +118,6 @@ export const AssessmentResultsTable_Fragment = graphql(/* GraphQL */ ` fr } } - stream { - value - label { - en - fr - } - } name { en fr @@ -134,13 +127,6 @@ export const AssessmentResultsTable_Fragment = graphql(/* GraphQL */ ` group level } - stream { - value - label { - en - fr - } - } assessmentSteps { id title { diff --git a/apps/web/src/components/CandidateDialog/ChangeDateDialog.tsx b/apps/web/src/components/CandidateDialog/ChangeDateDialog.tsx index 57e7cc82d26..9070640da1b 100644 --- a/apps/web/src/components/CandidateDialog/ChangeDateDialog.tsx +++ b/apps/web/src/components/CandidateDialog/ChangeDateDialog.tsx @@ -37,9 +37,9 @@ export const ChangeDateDialog_PoolCandidateFragment = graphql(/* GraphQL */ ` expiryDate pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -177,7 +177,7 @@ const ChangeDateDialog = ({

-{" "} {getShortPoolTitleHtml(intl, { - stream: selectedCandidate.pool.stream, + workStream: selectedCandidate.pool.workStream, name: selectedCandidate.pool.name, publishingGroup: selectedCandidate.pool.publishingGroup, classification: selectedCandidate.pool.classification, diff --git a/apps/web/src/components/CandidateDialog/ChangeStatusDialog.tsx b/apps/web/src/components/CandidateDialog/ChangeStatusDialog.tsx index 9c8a923852a..de7f98b75a2 100644 --- a/apps/web/src/components/CandidateDialog/ChangeStatusDialog.tsx +++ b/apps/web/src/components/CandidateDialog/ChangeStatusDialog.tsx @@ -78,9 +78,9 @@ const ChangeStatusDialog_UserFragment = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -123,9 +123,9 @@ export const ChangeStatusDialog_PoolCandidateFragment = graphql(/* GraphQL */ ` } pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -264,7 +264,7 @@ const ChangeStatusDialogForm = ({ {getShortPoolTitleHtml( intl, { - stream: r.poolCandidate.pool.stream, + workStream: r.poolCandidate.pool.workStream, name: r.poolCandidate.pool.name, publishingGroup: r.poolCandidate.pool.publishingGroup, classification: r.poolCandidate.pool.classification, @@ -412,7 +412,7 @@ const ChangeStatusDialog = ({ { status: getLocalizedName(selectedCandidate.status?.label, intl), poolName: getShortPoolTitleLabel(intl, { - stream: selectedCandidate.pool.stream, + workStream: selectedCandidate.pool.workStream, name: selectedCandidate.pool.name, publishingGroup: selectedCandidate.pool.publishingGroup, classification: selectedCandidate.pool.classification, @@ -453,7 +453,7 @@ const ChangeStatusDialog = ({

-{" "} {getShortPoolTitleHtml(intl, { - stream: selectedCandidate.pool.stream, + workStream: selectedCandidate.pool.workStream, name: selectedCandidate.pool.name, publishingGroup: selectedCandidate.pool.publishingGroup, classification: selectedCandidate.pool.classification, diff --git a/apps/web/src/components/PoolCandidatesTable/PoolCandidatesTable.tsx b/apps/web/src/components/PoolCandidatesTable/PoolCandidatesTable.tsx index f7540789ed9..d675be66be6 100644 --- a/apps/web/src/components/PoolCandidatesTable/PoolCandidatesTable.tsx +++ b/apps/web/src/components/PoolCandidatesTable/PoolCandidatesTable.tsx @@ -195,9 +195,9 @@ const CandidatesTableCandidatesPaginated_Query = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -718,7 +718,7 @@ const PoolCandidatesTable = ({ columnHelper.accessor( ({ poolCandidate: { pool } }) => getFullPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -737,7 +737,7 @@ const PoolCandidatesTable = ({ processCell( { id: pool.id, - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/components/PoolCandidatesTable/helpers.tsx b/apps/web/src/components/PoolCandidatesTable/helpers.tsx index 7a0b8c839d3..3a894b65fba 100644 --- a/apps/web/src/components/PoolCandidatesTable/helpers.tsx +++ b/apps/web/src/components/PoolCandidatesTable/helpers.tsx @@ -13,7 +13,6 @@ import { parseDateTimeUtc } from "@gc-digital-talent/date-helpers"; import { Link, Chip, Spoiler } from "@gc-digital-talent/ui"; import { CandidateExpiryFilter, - PoolStream, PublishingGroup, Maybe, Pool, @@ -100,14 +99,14 @@ export const candidateNameCell = ( }; export const processCell = ( - pool: Pick & { + pool: Pick & { classification?: Maybe>; }, paths: ReturnType, intl: IntlShape, ) => { const poolName = getFullPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -380,7 +379,10 @@ export function transformPoolCandidateSearchInputToFormValues( input?.appliedClassifications ?.filter(notEmpty) .map((c) => `${c.group}-${c.level}`) ?? [], - stream: input?.applicantFilter?.qualifiedStreams?.filter(notEmpty) ?? [], + stream: + input?.applicantFilter?.workStreams + ?.filter(notEmpty) + .map(({ id }) => id) ?? [], languageAbility: input?.applicantFilter?.languageAbility ?? "", workRegion: input?.applicantFilter?.locationPreferences?.filter(notEmpty) ?? [], @@ -427,7 +429,7 @@ export function transformFormValuesToFilterState( languageAbility: data.languageAbility ? stringToEnumLanguage(data.languageAbility) : undefined, - qualifiedStreams: data.stream as PoolStream[], + workStreams: data.stream.map((id) => ({ id })), operationalRequirements: data.operationalRequirement .map((requirement) => { return stringToEnumOperational(requirement); diff --git a/apps/web/src/components/PoolCard/PoolCard.tsx b/apps/web/src/components/PoolCard/PoolCard.tsx index 1a503e0f798..6d2eb882cd4 100644 --- a/apps/web/src/components/PoolCard/PoolCard.tsx +++ b/apps/web/src/components/PoolCard/PoolCard.tsx @@ -41,9 +41,9 @@ import IconLabel from "./IconLabel"; export const PoolCard_Fragment = graphql(/* GraphQL */ ` fragment PoolCard on Pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -258,7 +258,7 @@ const PoolCard = ({ poolQuery, headingLevel = "h3" }: PoolCardProps) => { data-h2-min-height="base(x4.5) p-tablet(auto)" > {getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/components/PoolFilterInput/usePoolFilterOptions.ts b/apps/web/src/components/PoolFilterInput/usePoolFilterOptions.ts index 191cdbc00ce..c53b1f51cb8 100644 --- a/apps/web/src/components/PoolFilterInput/usePoolFilterOptions.ts +++ b/apps/web/src/components/PoolFilterInput/usePoolFilterOptions.ts @@ -32,9 +32,9 @@ const PoolFilter_Query = graphql(/* GraphQL */ ` fr } } - stream { - value - label { + workStream { + id + name { en fr } @@ -85,7 +85,7 @@ const usePoolFilterOptions = ( pools.map((pool) => ({ value: pool.id, label: getShortPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/components/PoolStatusTable/PoolStatusTable.tsx b/apps/web/src/components/PoolStatusTable/PoolStatusTable.tsx index 51ba3363e58..2bdc1044ebb 100644 --- a/apps/web/src/components/PoolStatusTable/PoolStatusTable.tsx +++ b/apps/web/src/components/PoolStatusTable/PoolStatusTable.tsx @@ -65,9 +65,9 @@ const PoolStatusTable_Fragment = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -105,7 +105,7 @@ const PoolStatusTable = ({ userQuery }: PoolStatusTableProps) => { columnHelper.accessor( (row) => getShortPoolTitleLabel(intl, { - stream: row.pool.stream, + workStream: row.pool.workStream, name: row.pool.name, publishingGroup: row.pool.publishingGroup, classification: row.pool.classification, @@ -195,7 +195,7 @@ const PoolStatusTable = ({ userQuery }: PoolStatusTableProps) => { }, ), columnHelper.accessor( - ({ pool: { stream } }) => getLocalizedName(stream?.label, intl), + ({ pool: { workStream } }) => getLocalizedName(workStream?.name, intl), { id: "application", enableHiding: false, diff --git a/apps/web/src/components/PoolStatusTable/types.ts b/apps/web/src/components/PoolStatusTable/types.ts index 4cc0337697e..22fab002a21 100644 --- a/apps/web/src/components/PoolStatusTable/types.ts +++ b/apps/web/src/components/PoolStatusTable/types.ts @@ -24,9 +24,9 @@ const PoolStatusTable_PoolCandidateFragment = graphql(/* GraphQL */ ` fr } } - stream { - value - label { + workStream { + id + name { en fr } diff --git a/apps/web/src/components/RecruitmentAvailabilityDialog/RecruitmentAvailabilityDialog.tsx b/apps/web/src/components/RecruitmentAvailabilityDialog/RecruitmentAvailabilityDialog.tsx index 27ab06e71e0..bf6b126d2c8 100644 --- a/apps/web/src/components/RecruitmentAvailabilityDialog/RecruitmentAvailabilityDialog.tsx +++ b/apps/web/src/components/RecruitmentAvailabilityDialog/RecruitmentAvailabilityDialog.tsx @@ -32,9 +32,9 @@ const RecruitmentAvailabilityDialog_Fragment = graphql(/* GraphQL */ ` suspendedAt pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -77,7 +77,7 @@ const RecruitmentAvailabilityDialog = ({ const [isOpen, setIsOpen] = useState(false); const isSuspended = !!candidate.suspendedAt; const title = poolTitle(intl, { - stream: candidate.pool.stream, + workStream: candidate.pool.workStream, name: candidate.pool.name, publishingGroup: candidate.pool.publishingGroup, classification: candidate.pool.classification, diff --git a/apps/web/src/components/SearchRequestFilters/SearchRequestFilters.tsx b/apps/web/src/components/SearchRequestFilters/SearchRequestFilters.tsx index c1ad42b14c5..afe349201ad 100644 --- a/apps/web/src/components/SearchRequestFilters/SearchRequestFilters.tsx +++ b/apps/web/src/components/SearchRequestFilters/SearchRequestFilters.tsx @@ -111,7 +111,7 @@ const ApplicantFilters = ({ ).map((label) => getLocalizedName(label, intl)); const streams = unpackMaybes( - applicantFilter?.qualifiedStreams?.flatMap((stream) => stream?.label), + applicantFilter?.workStreams?.flatMap((stream) => stream?.name), ).map((label) => getLocalizedName(label, intl)); const communityName: string = @@ -142,7 +142,7 @@ const ApplicantFilters = ({ applicantFilter ? applicantFilter?.pools?.filter(notEmpty)?.map((pool) => getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -277,8 +277,8 @@ const SearchRequestFilters = ({ ? poolCandidateFilter?.pools?.filter(notEmpty) : []; - const streams = pools?.map((pool) => - pool.stream?.label ? getLocalizedName(pool.stream.label, intl) : "", + const streams = pools?.map(({ workStream }) => + getLocalizedName(workStream?.name, intl, true), ); // eslint-disable-next-line deprecation/deprecation @@ -373,7 +373,7 @@ const SearchRequestFilters = ({ pools ? pools.map((pool) => getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/components/SearchRequestTable/SearchRequestTable.tsx b/apps/web/src/components/SearchRequestTable/SearchRequestTable.tsx index 36eeaa89a89..ea2a403af66 100644 --- a/apps/web/src/components/SearchRequestTable/SearchRequestTable.tsx +++ b/apps/web/src/components/SearchRequestTable/SearchRequestTable.tsx @@ -91,7 +91,7 @@ const transformSearchRequestInput = ( status: filterState?.status, departments: filterState?.departments, classifications: filterState?.classifications, - streams: filterState?.streams, + workStreams: filterState?.workStreams, }; }; @@ -125,9 +125,9 @@ const SearchRequestTable_Query = graphql(/* GraphQL */ ` group level } - qualifiedStreams { - value - label { + workStreams { + id + name { en fr } @@ -270,8 +270,8 @@ const SearchRequestTable = ({ title }: SearchRequestTableProps) => { columnHelper.accessor( ({ applicantFilter }) => unpackMaybes( - applicantFilter?.qualifiedStreams?.map((stream) => - getLocalizedName(stream?.label, intl), + applicantFilter?.workStreams?.map((workStream) => + getLocalizedName(workStream?.name, intl), ), ).join(","), { @@ -287,8 +287,8 @@ const SearchRequestTable = ({ title }: SearchRequestTableProps) => { cells.commaList({ list: unpackMaybes( - applicantFilter?.qualifiedStreams?.map((stream) => - getLocalizedName(stream?.label, intl, true), + applicantFilter?.workStreams?.map((workStream) => + getLocalizedName(workStream?.name, intl, true), ), ) ?? [], }), diff --git a/apps/web/src/components/SearchRequestTable/components/SearchRequestFilterDialog.tsx b/apps/web/src/components/SearchRequestTable/components/SearchRequestFilterDialog.tsx index 9afd347ec0b..7e0e8c7f665 100644 --- a/apps/web/src/components/SearchRequestTable/components/SearchRequestFilterDialog.tsx +++ b/apps/web/src/components/SearchRequestTable/components/SearchRequestFilterDialog.tsx @@ -53,9 +53,9 @@ const SearchRequestFilterData_Query = graphql(/* GraphQL */ ` fr } } - streams: localizedEnumStrings(enumName: "PoolStream") { - value - label { + workStreams { + id + name { en fr } @@ -133,11 +133,14 @@ const SearchRequestFilterDialog = ({ }))} /> ({ + value: workStream.id, + label: getLocalizedName(workStream?.name, intl), + }))} /> diff --git a/apps/web/src/components/SearchRequestTable/components/utils.tsx b/apps/web/src/components/SearchRequestTable/components/utils.tsx index c6cd54a68a4..230a4860cf4 100644 --- a/apps/web/src/components/SearchRequestTable/components/utils.tsx +++ b/apps/web/src/components/SearchRequestTable/components/utils.tsx @@ -7,16 +7,13 @@ import { SortOrder, } from "@gc-digital-talent/graphql"; -import { - stringToEnumRequestStatus, - stringToEnumStream, -} from "~/utils/requestUtils"; +import { stringToEnumRequestStatus } from "~/utils/requestUtils"; export interface FormValues { status?: string[]; departments?: string[]; classifications?: string[]; - streams?: string[]; + workStreams?: string[]; } export function transformFormValuesToSearchRequestFilterInput( @@ -30,8 +27,8 @@ export function transformFormValuesToSearchRequestFilterInput( classifications: data.classifications?.length ? data.classifications : undefined, - streams: data.streams?.length - ? data.streams.map(stringToEnumStream).filter(notEmpty) + workStreams: data.workStreams?.length + ? data.workStreams.filter(notEmpty) : undefined, }; } @@ -69,6 +66,6 @@ export function transformSearchRequestFilterInputToFormValues( status: input?.status?.filter(notEmpty) ?? [], departments: input?.departments?.filter(notEmpty) ?? [], classifications: input?.classifications?.filter(notEmpty) ?? [], - streams: input?.streams?.filter(notEmpty) ?? [], + workStreams: input?.workStreams?.filter(notEmpty) ?? [], }; } diff --git a/apps/web/src/pages/Applications/ApplicationWelcomePage/ApplicationWelcomePage.tsx b/apps/web/src/pages/Applications/ApplicationWelcomePage/ApplicationWelcomePage.tsx index a6ef9978ea9..314ca0de7e9 100644 --- a/apps/web/src/pages/Applications/ApplicationWelcomePage/ApplicationWelcomePage.tsx +++ b/apps/web/src/pages/Applications/ApplicationWelcomePage/ApplicationWelcomePage.tsx @@ -71,7 +71,7 @@ const ApplicationWelcome = ({ application }: ApplicationPageProps) => { stepOrdinal: currentStepOrdinal, }); const poolName = getShortPoolTitleHtml(intl, { - stream: application.pool.stream, + workStream: application.pool.workStream, name: application.pool.name, publishingGroup: application.pool.publishingGroup, classification: application.pool.classification, diff --git a/apps/web/src/pages/Applications/fragment.ts b/apps/web/src/pages/Applications/fragment.ts index 0800cce6d63..d0aa7cabf16 100644 --- a/apps/web/src/pages/Applications/fragment.ts +++ b/apps/web/src/pages/Applications/fragment.ts @@ -342,9 +342,9 @@ const Application_PoolCandidateFragment = graphql(/* GraphQL */ ` en fr } - stream { - value - label { + workStream { + id + name { en fr } diff --git a/apps/web/src/pages/Manager/components/ReviewTalentRequestDialog.tsx b/apps/web/src/pages/Manager/components/ReviewTalentRequestDialog.tsx index 1567a325f51..cd592bc0e9f 100644 --- a/apps/web/src/pages/Manager/components/ReviewTalentRequestDialog.tsx +++ b/apps/web/src/pages/Manager/components/ReviewTalentRequestDialog.tsx @@ -55,8 +55,8 @@ const ReviewTalentRequestDialog_Query = graphql(/* GraphQL */ ` group level } - qualifiedStreams { - label { + workStreams { + name { fr en } @@ -126,7 +126,7 @@ const ReviewTalentRequestDialogContent = ({ const classifications = unpackMaybes( request.applicantFilter?.qualifiedClassifications, ); - const workStreams = unpackMaybes(request.applicantFilter?.qualifiedStreams); + const workStreams = unpackMaybes(request.applicantFilter?.workStreams); const equityDescriptions = equitySelectionsToDescriptions( request.applicantFilter?.equity, intl, @@ -189,7 +189,7 @@ const ReviewTalentRequestDialogContent = ({ {workStreams.length > 0 ? deriveSingleString( workStreams, - (stream) => getLocalizedName(stream.label, intl), + (workStream) => getLocalizedName(workStream.name, intl), locale, ) : nullMessage} diff --git a/apps/web/src/pages/PoolCandidates/ViewPoolCandidatePage/ViewPoolCandidatePage.tsx b/apps/web/src/pages/PoolCandidates/ViewPoolCandidatePage/ViewPoolCandidatePage.tsx index c0077a929b7..c07fc1e027a 100644 --- a/apps/web/src/pages/PoolCandidates/ViewPoolCandidatePage/ViewPoolCandidatePage.tsx +++ b/apps/web/src/pages/PoolCandidates/ViewPoolCandidatePage/ViewPoolCandidatePage.tsx @@ -105,9 +105,9 @@ const PoolCandidate_SnapshotQuery = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -168,7 +168,7 @@ export const ViewPoolCandidate = ({ }, { label: getFullPoolTitleLabel(intl, { - stream: poolCandidate.pool.stream, + workStream: poolCandidate.pool.workStream, name: poolCandidate.pool.name, publishingGroup: poolCandidate.pool.publishingGroup, classification: poolCandidate.pool.classification, diff --git a/apps/web/src/pages/Pools/EditPoolPage/EditPoolPage.tsx b/apps/web/src/pages/Pools/EditPoolPage/EditPoolPage.tsx index 287e60f5e1b..f7a3b8c4d1e 100644 --- a/apps/web/src/pages/Pools/EditPoolPage/EditPoolPage.tsx +++ b/apps/web/src/pages/Pools/EditPoolPage/EditPoolPage.tsx @@ -102,9 +102,9 @@ export const EditPool_Fragment = graphql(/* GraphQL */ ` ...EditPoolYourImpact id - stream { - value - label { + workStream { + id + name { en fr } @@ -277,7 +277,7 @@ export const EditPoolForm = ({ areaOfSelection: pool.areaOfSelection, classification: pool.classification, department: pool.department, - stream: pool.stream, + workStream: pool.workStream, name: pool.name, processNumber: pool.processNumber, publishingGroup: pool.publishingGroup, @@ -315,7 +315,7 @@ export const EditPoolForm = ({ areaOfSelection: pool.areaOfSelection, classification: pool.classification, department: pool.department, - stream: pool.stream, + workStream: pool.workStream, name: pool.name, processNumber: pool.processNumber, publishingGroup: pool.publishingGroup, @@ -370,7 +370,7 @@ export const EditPoolForm = ({ educationRequirements: { id: "education-requirements", hasError: educationRequirementIsNull({ - stream: pool.stream, + workStream: pool.workStream, name: pool.name, processNumber: pool.processNumber, publishingGroup: pool.publishingGroup, diff --git a/apps/web/src/pages/Pools/EditPoolPage/components/PoolNameSection/Display.tsx b/apps/web/src/pages/Pools/EditPoolPage/components/PoolNameSection/Display.tsx index a347f6bfa0a..56a463c9bf6 100644 --- a/apps/web/src/pages/Pools/EditPoolPage/components/PoolNameSection/Display.tsx +++ b/apps/web/src/pages/Pools/EditPoolPage/components/PoolNameSection/Display.tsx @@ -29,7 +29,7 @@ const Display = ({ selectionLimitations: poolSelectionLimitations, classification, department, - stream, + workStream, name, processNumber, publishingGroup, @@ -105,10 +105,10 @@ const Display = ({ : notProvided} - {getLocalizedName(stream?.label, intl)} + {getLocalizedName(workStream?.name, intl)} ({ + value: workStream.id, + label: getLocalizedName(workStream?.name, intl), + }), + )} disabled={formDisabled} /> ; classification?: Classification["id"]; department?: Department["id"]; - stream?: PoolStream; + stream?: WorkStream["id"]; specificTitleEn?: LocalizedString["en"]; specificTitleFr?: LocalizedString["fr"]; processNumber?: string; @@ -35,7 +35,7 @@ export const dataToFormValues = (initialData: Pool): FormValues => ({ selectionLimitations: initialData.selectionLimitations?.map((l) => l.value), classification: initialData.classification?.id ?? "", department: initialData.department?.id ?? "", - stream: initialData.stream?.value ?? undefined, + stream: initialData.workStream?.id ?? undefined, specificTitleEn: initialData.name?.en ?? "", specificTitleFr: initialData.name?.fr ?? "", processNumber: initialData.processNumber ?? "", @@ -50,7 +50,7 @@ export type PoolNameSubmitData = Pick< | "classification" | "department" | "name" - | "stream" + | "workStream" | "processNumber" | "publishingGroup" | "opportunityLength" @@ -71,7 +71,7 @@ export const formValuesToSubmitData = ( connect: formValues.department, } : undefined, - stream: formValues.stream ? formValues.stream : undefined, + workStream: formValues.stream ? { connect: formValues.stream } : undefined, name: { en: formValues.specificTitleEn, fr: formValues.specificTitleFr, diff --git a/apps/web/src/pages/Pools/EditPoolPage/components/UpdatePublishedProcessDialog/UpdatePublishedProcessDialog.tsx b/apps/web/src/pages/Pools/EditPoolPage/components/UpdatePublishedProcessDialog/UpdatePublishedProcessDialog.tsx index c0b3b0199cf..ca62d256eb9 100644 --- a/apps/web/src/pages/Pools/EditPoolPage/components/UpdatePublishedProcessDialog/UpdatePublishedProcessDialog.tsx +++ b/apps/web/src/pages/Pools/EditPoolPage/components/UpdatePublishedProcessDialog/UpdatePublishedProcessDialog.tsx @@ -23,9 +23,9 @@ import { PublishedEditableSectionProps } from "../../types"; const UpdatePublishedProcessDialog_Fragment = graphql(/* GraphQL */ ` fragment UpdatePublishedProcessDialog on Pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -64,7 +64,7 @@ const UpdatePublishedProcessDialog = ({ const [isOpen, setIsOpen] = useState(false); const pool = getFragment(UpdatePublishedProcessDialog_Fragment, poolQuery); const title = getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/pages/Pools/IndexPoolPage/components/PoolFilterDialog.tsx b/apps/web/src/pages/Pools/IndexPoolPage/components/PoolFilterDialog.tsx index 3dba45dda12..0b5136fd9dc 100644 --- a/apps/web/src/pages/Pools/IndexPoolPage/components/PoolFilterDialog.tsx +++ b/apps/web/src/pages/Pools/IndexPoolPage/components/PoolFilterDialog.tsx @@ -4,14 +4,13 @@ import { Combobox, localizedEnumToOptions } from "@gc-digital-talent/forms"; import { FragmentType, PoolStatus, - PoolStream, PublishingGroup, Scalars, getFragment, graphql, } from "@gc-digital-talent/graphql"; import { unpackMaybes } from "@gc-digital-talent/helpers"; -import { commonMessages } from "@gc-digital-talent/i18n"; +import { commonMessages, getLocalizedName } from "@gc-digital-talent/i18n"; import FilterDialog, { CommonFilterDialogProps, @@ -22,7 +21,7 @@ export interface FormValues { publishingGroups: PublishingGroup[]; statuses: PoolStatus[]; classifications: Scalars["UUID"]["output"][]; - streams: PoolStream[]; + workStreams: Scalars["UUID"]["output"][]; } const PoolFilterDialogOptions_Fragment = graphql(/* GraphQL */ ` @@ -45,9 +44,9 @@ const PoolFilterDialogOptions_Fragment = graphql(/* GraphQL */ ` fr } } - streams: localizedEnumStrings(enumName: "PoolStream") { - value - label { + workStreams { + id + name { en fr } @@ -92,11 +91,14 @@ const PoolFilterDialog = ({ options={localizedEnumToOptions(data?.statuses, intl)} /> ({ + value: workStream.id, + label: getLocalizedName(workStream.name, intl), + }))} /> { first: paginationState.pageSize, orderByPoolBookmarks: getPoolBookmarkSort(), orderByTeamDisplayName: getTeamDisplayNameSort(sortState, locale), + orderByWorkStreamName: getWorkStreamNameSort(sortState, locale), orderByColumn: getOrderByColumnSort(sortState), orderBy: sortState ? getOrderByClause(sortState) : undefined, }, @@ -299,7 +303,8 @@ const PoolTable = ({ title, initialFilterInput }: PoolTableProps) => { }, }), columnHelper.accessor( - (row) => poolNameAccessor({ name: row.name, stream: row.stream }, intl), + (row) => + poolNameAccessor({ name: row.name, workStream: row.workStream }, intl), { id: "name", header: intl.formatMessage(commonMessages.name), @@ -323,9 +328,9 @@ const PoolTable = ({ title, initialFilterInput }: PoolTableProps) => { classificationCell(pool.classification), }), columnHelper.accessor( - ({ stream }) => getLocalizedName(stream?.label, intl), + ({ workStream }) => getLocalizedName(workStream?.name, intl), { - id: "stream", + id: "workStream", enableColumnFilter: false, header: intl.formatMessage({ defaultMessage: "Stream", diff --git a/apps/web/src/pages/Pools/IndexPoolPage/components/helpers.tsx b/apps/web/src/pages/Pools/IndexPoolPage/components/helpers.tsx index a5918bf69a5..05a98208d70 100644 --- a/apps/web/src/pages/Pools/IndexPoolPage/components/helpers.tsx +++ b/apps/web/src/pages/Pools/IndexPoolPage/components/helpers.tsx @@ -16,6 +16,7 @@ import { PoolBookmarksOrderByInput, PoolFilterInput, PoolTeamDisplayNameOrderByInput, + PoolWorkStreamNameOrderByInput, QueryPoolsPaginatedOrderByClassificationColumn, QueryPoolsPaginatedOrderByRelationOrderByClause, QueryPoolsPaginatedOrderByUserColumn, @@ -32,11 +33,11 @@ import { FormValues } from "./PoolFilterDialog"; import PoolBookmark, { PoolBookmark_Fragment } from "./PoolBookmark"; export function poolNameAccessor( - pool: Pick, + pool: Pick, intl: IntlShape, ) { const name = getLocalizedName(pool.name, intl); - return `${name.toLowerCase()} ${getLocalizedName(pool.stream?.label, intl, true)}`; + return `${name.toLowerCase()} ${getLocalizedName(pool?.workStream?.name, intl, true)}`; } export function viewCell( @@ -173,7 +174,6 @@ export function getOrderByClause( ["id", "id"], ["name", "name"], ["publishingGroup", "publishing_group"], - ["stream", "stream"], ["processNumber", "process_number"], ["ownerName", "FIRST_NAME"], ["ownerEmail", "EMAIL"], @@ -260,6 +260,20 @@ export function getTeamDisplayNameSort( }; } +export function getWorkStreamNameSort( + sortingRules?: SortingState, + locale?: Locales, +): PoolWorkStreamNameOrderByInput | undefined { + const sortingRule = sortingRules?.find((rule) => rule.id === "workStream"); + + if (!sortingRule) return undefined; + + return { + locale: locale ?? "en", + order: sortingRule.desc ? SortOrder.Desc : SortOrder.Asc, + }; +} + export function getOrderByColumnSort( sortingRules?: SortingState, ): OrderByColumnInput | undefined { @@ -296,7 +310,7 @@ export function transformFormValuesToFilterInput( return { publishingGroups: data.publishingGroups, statuses: data.statuses, - streams: data.streams, + workStreams: data.workStreams, classifications: data.classifications.map((classification) => { const [group, level] = classification.split("-"); return { group, level: Number(level) }; @@ -310,7 +324,7 @@ export function transformPoolFilterInputToFormValues( return { publishingGroups: unpackMaybes(input?.publishingGroups), statuses: unpackMaybes(input?.statuses), - streams: unpackMaybes(input?.streams), + workStreams: unpackMaybes(input?.workStreams), classifications: unpackMaybes(input?.classifications).map( (c) => `${c.group}-${c.level}`, ), diff --git a/apps/web/src/pages/Pools/PoolAdvertisementPage/PoolAdvertisementPage.tsx b/apps/web/src/pages/Pools/PoolAdvertisementPage/PoolAdvertisementPage.tsx index b38941eed9e..44c50d36f4c 100644 --- a/apps/web/src/pages/Pools/PoolAdvertisementPage/PoolAdvertisementPage.tsx +++ b/apps/web/src/pages/Pools/PoolAdvertisementPage/PoolAdvertisementPage.tsx @@ -136,9 +136,9 @@ export const PoolAdvertisement_Fragment = graphql(/* GraphQL */ ` en fr } - stream { - value - label { + workStream { + id + name { en fr } @@ -253,9 +253,9 @@ export const PoolAdvertisement_Fragment = graphql(/* GraphQL */ ` en fr } - stream { - value - label { + workStream { + id + name { en fr } @@ -328,13 +328,13 @@ export const PoolPoster = ({ }); } const poolTitle = getShortPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, }); const fullPoolTitle = getFullPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -664,7 +664,7 @@ export const PoolPoster = ({ description: "Label for pool advertisement stream", }) + intl.formatMessage(commonMessages.dividingColon) } - value={getLocalizedName(pool.stream?.label, intl)} + value={getLocalizedName(pool?.workStream?.name, intl)} suffix={ classification?.group === "IT" ? ( { return currentPage?.title; } return getShortPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -127,7 +127,7 @@ const PoolHeader = ({ poolQuery }: PoolHeaderProps) => { id: pool.id, name: pool.name, publishingGroup: pool.publishingGroup, - stream: pool.stream, + workStream: pool.workStream, classification: pool.classification, }); const currentPage = useCurrentPage(pages); diff --git a/apps/web/src/pages/Pools/ViewPoolPage/ViewPoolPage.tsx b/apps/web/src/pages/Pools/ViewPoolPage/ViewPoolPage.tsx index f3faa8e5b34..343855d3d38 100644 --- a/apps/web/src/pages/Pools/ViewPoolPage/ViewPoolPage.tsx +++ b/apps/web/src/pages/Pools/ViewPoolPage/ViewPoolPage.tsx @@ -71,9 +71,9 @@ export const ViewPool_Fragment = graphql(/* GraphQL */ ` } closingDate processNumber - stream { - value - label { + workStream { + id + name { en fr } @@ -123,7 +123,7 @@ export const ViewPool = ({ const { roleAssignments } = useAuthorization(); const pool = getFragment(ViewPool_Fragment, poolQuery); const poolName = getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx b/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx index 9495986d869..5550e064081 100644 --- a/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx +++ b/apps/web/src/pages/Profile/CareerTimelineAndRecruitmentPage/components/CareerTimelineAndRecruitment.tsx @@ -228,9 +228,6 @@ const CareerTimelineApplication_Fragment = graphql(/* GraphQL */ ` publishingGroup { value } - stream { - value - } } } `); diff --git a/apps/web/src/pages/SearchRequests/RequestPage/components/RequestForm.tsx b/apps/web/src/pages/SearchRequests/RequestPage/components/RequestForm.tsx index cf77a476b91..59b2649ba80 100644 --- a/apps/web/src/pages/SearchRequests/RequestPage/components/RequestForm.tsx +++ b/apps/web/src/pages/SearchRequests/RequestPage/components/RequestForm.tsx @@ -43,7 +43,6 @@ import { graphql, FragmentType, getFragment, - PoolStream, } from "@gc-digital-talent/graphql"; import SEO from "~/components/SEO/SEO"; @@ -74,7 +73,7 @@ interface FormValues { qualifiedClassifications?: { sync?: Maybe[]; }; - qualifiedStreams?: ApplicantFilterInput["qualifiedStreams"]; + workStreams?: ApplicantFilterInput["workStreams"]; skills?: { sync?: Maybe[]; }; @@ -131,9 +130,9 @@ const PoolsInFilter_Query = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -177,9 +176,10 @@ const RequestOptions_Query = graphql(/* GraphQL */ ` fr } } - streams: localizedEnumStrings(enumName: "PoolStream") { - value - label { + workStreams { + id + key + name { en fr } @@ -256,9 +256,14 @@ export const RequestForm = ({ values?.positionType === true ? PoolCandidateSearchPositionType.TeamLead : PoolCandidateSearchPositionType.IndividualContributor; - const qualifiedStreams = applicantFilter?.qualifiedStreams; + const qualifiedStreams = applicantFilter?.workStreams; let community = communities?.find((c) => c.key === "digital"); - if (qualifiedStreams?.includes(PoolStream.AccessInformationPrivacy)) { + const ATIPStream = optionsData?.workStreams?.find( + (workStream) => workStream?.key === "ACCESS_INFORMATION_PRIVACY", + ); + if ( + qualifiedStreams?.some((workStream) => workStream?.id === ATIPStream?.id) + ) { community = communities?.find((c) => c.key === "atip"); } @@ -286,7 +291,13 @@ export const RequestForm = ({ equity: applicantFilter?.equity, languageAbility: applicantFilter?.languageAbility, operationalRequirements: applicantFilter?.operationalRequirements, - qualifiedStreams, + workStreams: { + sync: applicantFilter?.workStreams + ? applicantFilter?.workStreams + ?.filter(notEmpty) + .map(({ id }) => id) + : [], + }, community: { connect: community?.id ?? communities[0].id, }, @@ -386,9 +397,9 @@ export const RequestForm = ({ ), ), ), - qualifiedStreams: unpackMaybes( - applicantFilter?.qualifiedStreams?.map((stream) => - enumInputToLocalizedEnum(stream, optionsData?.streams), + workStreams: unpackMaybes(optionsData?.workStreams).filter((workStream) => + applicantFilter?.workStreams?.some( + (filterStream) => filterStream?.id === workStream?.id, ), ), qualifiedClassifications: diff --git a/apps/web/src/pages/SearchRequests/SearchPage/components/FormFields.tsx b/apps/web/src/pages/SearchRequests/SearchPage/components/FormFields.tsx index cf30e3d9059..fd3cfc5e019 100644 --- a/apps/web/src/pages/SearchRequests/SearchPage/components/FormFields.tsx +++ b/apps/web/src/pages/SearchRequests/SearchPage/components/FormFields.tsx @@ -12,9 +12,15 @@ import { commonMessages, errorMessages, getEmploymentEquityGroup, + getLocalizedName, sortWorkRegion, } from "@gc-digital-talent/i18n"; -import { Classification, Skill, graphql } from "@gc-digital-talent/graphql"; +import { + Classification, + Skill, + WorkStream, + graphql, +} from "@gc-digital-talent/graphql"; import { unpackMaybes } from "@gc-digital-talent/helpers"; import { NullSelection } from "~/types/searchRequest"; @@ -29,13 +35,6 @@ import { classificationAriaLabels, classificationLabels } from "../labels"; const SearchRequestOptions_Query = graphql(/* GraphQL */ ` query SearchRequestOptions { - poolStreams: localizedEnumStrings(enumName: "PoolStream") { - value - label { - en - fr - } - } languageAbilities: localizedEnumStrings(enumName: "LanguageAbility") { value label { @@ -56,9 +55,14 @@ const SearchRequestOptions_Query = graphql(/* GraphQL */ ` interface FormFieldsProps { classifications: Pick[]; skills: Skill[]; + workStreams: WorkStream[]; } -const FormFields = ({ classifications, skills }: FormFieldsProps) => { +const FormFields = ({ + classifications, + skills, + workStreams, +}: FormFieldsProps) => { const intl = useIntl(); const [{ data }] = useQuery({ query: SearchRequestOptions_Query, @@ -74,11 +78,15 @@ const FormFields = ({ classifications, skills }: FormFieldsProps) => { ), })); + const workStreamOptions = workStreams.map((workStream) => ({ + value: workStream.id, + label: getLocalizedName(workStream.name, intl), + })); + const languageAbilityOptions = localizedEnumToOptions( data?.languageAbilities, intl, ); - const streamOptions = localizedEnumToOptions(data?.poolStreams, intl); const sortedWorkRegions = sortWorkRegion(unpackMaybes(data?.workRegions)); const workRegionOptions = localizedEnumToOptions(sortedWorkRegions, intl); @@ -133,7 +141,7 @@ const FormFields = ({ classifications, skills }: FormFieldsProps) => { id: "QJ5uDV", description: "Placeholder for stream filter in search form.", })} - options={streamOptions} + options={workStreamOptions} rules={{ required: intl.formatMessage(errorMessages.required), }} diff --git a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.stories.tsx b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.stories.tsx index 8050c4c49b4..fefdf3afa4b 100644 --- a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.stories.tsx +++ b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.stories.tsx @@ -6,13 +6,10 @@ import { fakeClassifications, fakePools, fakeLocalizedEnum, + fakeWorkStreams, } from "@gc-digital-talent/fake-data"; import { MockGraphqlDecorator } from "@gc-digital-talent/storybook-helpers"; -import { - LanguageAbility, - PoolStream, - WorkRegion, -} from "@gc-digital-talent/graphql"; +import { LanguageAbility, WorkRegion } from "@gc-digital-talent/graphql"; import { SearchForm } from "./SearchForm"; @@ -21,6 +18,7 @@ faker.seed(0); const mockPools = fakePools(10); const poolResponse = fakePools(3); const mockClassifications = fakeClassifications(); +const mockWorkStreams = fakeWorkStreams(); const skills = getStaticSkills(); export default { @@ -29,13 +27,13 @@ export default { args: { pools: mockPools, classifications: mockClassifications, + workStreams: mockWorkStreams, skills, }, } as Meta; const SearchRequestOptions = { data: { - poolStreams: fakeLocalizedEnum(PoolStream), languageAbilities: fakeLocalizedEnum(LanguageAbility), workRegions: fakeLocalizedEnum(WorkRegion), }, diff --git a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.tsx b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.tsx index a6d80cf12ec..82622534b2a 100644 --- a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.tsx +++ b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchForm.tsx @@ -17,6 +17,7 @@ import { Classification, ApplicantFilterInput, Skill, + WorkStream, } from "@gc-digital-talent/graphql"; import { commonMessages } from "@gc-digital-talent/i18n"; @@ -44,9 +45,14 @@ const styledCount = (chunks: ReactNode) => ( interface SearchFormProps { classifications: Pick[]; skills: Skill[]; + workStreams: WorkStream[]; } -export const SearchForm = ({ classifications, skills }: SearchFormProps) => { +export const SearchForm = ({ + classifications, + skills, + workStreams, +}: SearchFormProps) => { const intl = useIntl(); const navigate = useNavigate(); const paths = useRoutes(); @@ -145,7 +151,11 @@ export const SearchForm = ({ classifications, skills }: SearchFormProps) => { "Content displayed in the How To area of the hero section of the Search page.", })}

- +
{ const skills = unpackMaybes(data?.skills); const classifications = unpackMaybes(data?.classifications); + const workStreams = unpackMaybes(data?.workStreams); return ( - + ); }; diff --git a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchResultCard.tsx b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchResultCard.tsx index f9ff93c7f09..7c3d5227e37 100644 --- a/apps/web/src/pages/SearchRequests/SearchPage/components/SearchResultCard.tsx +++ b/apps/web/src/pages/SearchRequests/SearchPage/components/SearchResultCard.tsx @@ -22,9 +22,9 @@ const testId = (text: ReactNode) => ( const SearchResultCard_PoolFragment = graphql(/* GraphQL */ ` fragment SearchResultCard_Pool on Pool { id - stream { - value - label { + workStream { + id + name { en fr } @@ -121,7 +121,7 @@ const SearchResultCard = ({ candidateCount, pool }: SearchResultCardProps) => { id={`search_pool_${pool.id}`} > {getShortPoolTitleHtml(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/pages/SearchRequests/SearchPage/utils.ts b/apps/web/src/pages/SearchRequests/SearchPage/utils.ts index a1edeffa2ba..cda3a7fc2e0 100644 --- a/apps/web/src/pages/SearchRequests/SearchPage/utils.ts +++ b/apps/web/src/pages/SearchRequests/SearchPage/utils.ts @@ -104,7 +104,7 @@ export const dataToFormValues = ( data: ApplicantFilterInput, selectedClassifications?: Maybe[]>, ): FormValues => { - const stream = data?.qualifiedStreams?.find(notEmpty); + const stream = data?.workStreams?.find(notEmpty); return { classification: getCurrentClassification(selectedClassifications), @@ -117,7 +117,7 @@ export const dataToFormValues = ( ], educationRequirement: data.hasDiploma ? "has_diploma" : "no_diploma", skills: data.skills?.filter(notEmpty).map((s) => s.id) ?? [], - stream: stream ?? "", + stream: stream?.id ?? "", locationPreferences: data.locationPreferences?.filter(notEmpty) ?? [], operationalRequirements: data.operationalRequirements?.filter(notEmpty) ?? [], @@ -172,6 +172,6 @@ export const formValuesToData = ( ? durationSelectionToEnum(values.employmentDuration) : undefined, locationPreferences: values.locationPreferences ?? [], - qualifiedStreams: values.stream ? [values.stream] : undefined, + workStreams: values.stream ? [{ id: values.stream }] : undefined, }; }; diff --git a/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/SearchRequestCandidatesTable.tsx b/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/SearchRequestCandidatesTable.tsx index b40d79e25b2..1159ffff781 100644 --- a/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/SearchRequestCandidatesTable.tsx +++ b/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/SearchRequestCandidatesTable.tsx @@ -1,24 +1,14 @@ import { useIntl } from "react-intl"; -import omit from "lodash/omit"; -import pick from "lodash/pick"; -import { identity, notEmpty } from "@gc-digital-talent/helpers"; +import { notEmpty, unpackMaybes } from "@gc-digital-talent/helpers"; import { - ApplicantFilterInput, PoolCandidateFilter, ApplicantFilter, - IdInput, - Classification, - ClassificationFilterInput, PoolCandidateSearchInput, PoolCandidateStatus, CandidateSuspendedFilter, CandidateExpiryFilter, } from "@gc-digital-talent/graphql"; -import { - localizedEnumArrayToInput, - localizedEnumToInput, -} from "@gc-digital-talent/i18n"; import PoolCandidatesTable from "~/components/PoolCandidatesTable/PoolCandidatesTable"; import adminMessages from "~/messages/adminMessages"; @@ -33,70 +23,38 @@ function isPoolCandidateFilter( return false; } -function omitIdAndTypename( - x: T, -): Omit { - return omit(x, ["id", "__typename"]) as Omit; -} - -function pickId(x: T): IdInput { - return pick(x, ["id"]); -} - -function classificationToInput(c: Classification): ClassificationFilterInput { - return pick(c, ["group", "level"]); -} - -// Maps each property in ApplicantFilterInput to a function which transforms the matching value from an ApplicantFilter object to the appropriate shape for ApplicantFilterInput. -type MappingType = { - [Property in keyof Omit< - ApplicantFilterInput, - "email" | "name" | "generalSearch" | "skillsIntersectional" - >]: (x: ApplicantFilter[Property]) => ApplicantFilterInput[Property]; -}; - const transformApplicantFilterToPoolCandidateSearchInput = ( applicantFilter: ApplicantFilter, ): PoolCandidateSearchInput => { - // GraphQL will error if an input object includes any unexpected attributes. - // Therefore, transforming ApplicantFilter to ApplicantFilterInput requires omitting any fields not included in the Input type. - const mapping: MappingType = { - equity: omitIdAndTypename, - hasDiploma: identity, - languageAbility: localizedEnumToInput, - locationPreferences: localizedEnumArrayToInput, - operationalRequirements: localizedEnumArrayToInput, - pools: (pools) => pools?.filter(notEmpty).map(pickId), - skills: (skills) => skills?.filter(notEmpty).map(pickId), - positionDuration: identity, - qualifiedStreams: localizedEnumArrayToInput, - community: (community) => (community ? pickId(community) : undefined), - }; - - const emptyFilter: ApplicantFilterInput = {}; - return { - applicantFilter: Object.entries(mapping).reduce( - (applicantFilterInput, filterEntry) => { - const [key, transform] = filterEntry; - const typedKey = key as keyof MappingType; - - // There should be way to get the types to work without using "any", but I'm having trouble. - // I think its safe to fallback on any here because mapping has just been defined, and we can be confident that key and transform line up correctly. - - // eslint-disable-next-line no-param-reassign, @typescript-eslint/no-unsafe-assignment - applicantFilterInput[typedKey] = transform( - // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-argument - applicantFilter[typedKey] as any, - // eslint-disable-next-line @typescript-eslint/no-explicit-any - ) as any; - return applicantFilterInput; + applicantFilter: { + equity: { + isWoman: applicantFilter.equity?.isWoman ?? undefined, + hasDisability: applicantFilter.equity?.hasDisability, + isIndigenous: applicantFilter.equity?.isIndigenous, + isVisibleMinority: applicantFilter.equity?.isVisibleMinority, }, - emptyFilter, - ), + hasDiploma: applicantFilter?.hasDiploma, + positionDuration: applicantFilter?.positionDuration, + languageAbility: applicantFilter.languageAbility?.value, + locationPreferences: applicantFilter?.locationPreferences + ?.filter(notEmpty) + .map((workRegion) => workRegion?.value), + operationalRequirements: applicantFilter?.operationalRequirements + ?.filter(notEmpty) + .map((req) => req?.value), + pools: unpackMaybes(applicantFilter.pools).map(({ id }) => ({ id })), + skills: unpackMaybes(applicantFilter.skills).map(({ id }) => ({ id })), + workStreams: unpackMaybes(applicantFilter.workStreams).map(({ id }) => ({ + id, + })), + community: applicantFilter?.community?.id + ? { id: applicantFilter.community.id } + : undefined, + }, appliedClassifications: applicantFilter.qualifiedClassifications ?.filter(notEmpty) - .map(classificationToInput), + .map(({ group, level }) => ({ group, level })), poolCandidateStatus: [ PoolCandidateStatus.QualifiedAvailable, PoolCandidateStatus.PlacedCasual, diff --git a/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/ViewSearchRequest.tsx b/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/ViewSearchRequest.tsx index 645a85a3256..d378c707e7c 100644 --- a/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/ViewSearchRequest.tsx +++ b/apps/web/src/pages/SearchRequests/ViewSearchRequestPage/components/ViewSearchRequest.tsx @@ -296,9 +296,9 @@ const ViewSearchRequest_SearchRequestFragment = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } @@ -367,9 +367,9 @@ const ViewSearchRequest_SearchRequestFragment = graphql(/* GraphQL */ ` en fr } - stream { - value - label { + workStream { + id + name { en fr } @@ -389,9 +389,9 @@ const ViewSearchRequest_SearchRequestFragment = graphql(/* GraphQL */ ` group level } - qualifiedStreams { - value - label { + workStreams { + id + name { en fr } diff --git a/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx b/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx index eac9c4563eb..175c8620fbf 100644 --- a/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx +++ b/apps/web/src/pages/Users/AdminUserProfilePage/AdminUserProfilePage.tsx @@ -170,9 +170,9 @@ const AdminUserProfileUser_Fragment = graphql(/* GraphQL */ ` group level } - stream { - value - label { + workStream { + id + name { en fr } diff --git a/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx b/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx index 05f0ff27e89..2529f48bd42 100644 --- a/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/UserInformationPage.tsx @@ -172,9 +172,9 @@ export const UserInfo_Fragment = graphql(/* GraphQL */ ` fr } } - stream { - value - label { + workStream { + id + name { en fr } diff --git a/apps/web/src/pages/Users/UserInformationPage/components/AddToPoolDialog.tsx b/apps/web/src/pages/Users/UserInformationPage/components/AddToPoolDialog.tsx index 9c43cee0e63..0ffe36a1d31 100644 --- a/apps/web/src/pages/Users/UserInformationPage/components/AddToPoolDialog.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/components/AddToPoolDialog.tsx @@ -63,9 +63,9 @@ const AvailablePoolsToAddTo_Query = graphql(/* GraphQL */ ` fr } } - stream { - value - label { + workStream { + id + name { en fr } @@ -187,7 +187,7 @@ const AddToPoolDialog = ({ user, poolCandidates }: AddToPoolDialogProps) => { {rejectedRequests.map((rejected) => (
  • {getShortPoolTitleHtml(intl, { - stream: rejected.pool.stream, + workStream: rejected.pool.workStream, name: rejected.pool.name, publishingGroup: rejected.pool.publishingGroup, classification: rejected.pool.classification, diff --git a/apps/web/src/pages/Users/UserInformationPage/components/NotesSection.tsx b/apps/web/src/pages/Users/UserInformationPage/components/NotesSection.tsx index a53cb407833..4149a24e136 100644 --- a/apps/web/src/pages/Users/UserInformationPage/components/NotesSection.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/components/NotesSection.tsx @@ -51,7 +51,7 @@ const NotesSection = ({ user }: BasicUserInformationProps) => { }, { poolName: getShortPoolTitleHtml(intl, { - stream: candidate.pool.stream, + workStream: candidate.pool.workStream, name: candidate.pool.name, publishingGroup: candidate.pool.publishingGroup, classification: candidate.pool.classification, @@ -72,7 +72,7 @@ const NotesSection = ({ user }: BasicUserInformationProps) => { }, { poolName: getShortPoolTitleHtml(intl, { - stream: candidate.pool.stream, + workStream: candidate.pool.workStream, name: candidate.pool.name, publishingGroup: candidate.pool.publishingGroup, classification: candidate.pool.classification, @@ -122,7 +122,7 @@ const NotesSection = ({ user }: BasicUserInformationProps) => { }, { poolName: getShortPoolTitleHtml(intl, { - stream: candidate.pool.stream, + workStream: candidate.pool.workStream, name: candidate.pool.name, publishingGroup: candidate.pool.publishingGroup, classification: candidate.pool.classification, diff --git a/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/UserCandidatesTable.tsx b/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/UserCandidatesTable.tsx index 5a39a38b230..3456f0cfef7 100644 --- a/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/UserCandidatesTable.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/UserCandidatesTable.tsx @@ -84,9 +84,9 @@ const UserCandidatesTableRow_Fragment = graphql(/* GraphQL */ ` en fr } - stream { - value - label { + workStream { + id + name { en fr } @@ -188,7 +188,7 @@ const UserCandidatesTable = ({ columnHelper.accessor( ({ pool }) => getFullPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, @@ -205,7 +205,7 @@ const UserCandidatesTable = ({ processCell( { id: pool.id, - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/helpers.tsx b/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/helpers.tsx index 8fab98faa67..d7cf5b2252c 100644 --- a/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/helpers.tsx +++ b/apps/web/src/pages/Users/UserInformationPage/components/UserCandidatesTable/helpers.tsx @@ -106,14 +106,14 @@ export const priorityCell = ( }; export const processCell = ( - pool: Pick & { + pool: Pick & { classification?: Maybe>; }, paths: ReturnType, intl: IntlShape, ) => { const poolName = getFullPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/types/searchRequest.ts b/apps/web/src/types/searchRequest.ts index e0dce73437c..b50502b6648 100644 --- a/apps/web/src/types/searchRequest.ts +++ b/apps/web/src/types/searchRequest.ts @@ -2,7 +2,6 @@ import { Scalars, ApplicantFilterInput, LanguageAbility, - PoolStream, UserPoolFilterInput, Classification, } from "@gc-digital-talent/graphql"; @@ -16,7 +15,7 @@ export type FormValues = Pick< languageAbility: LanguageAbility | typeof NullSelection; employmentDuration: string; classification: string | undefined; - stream: PoolStream | ""; + stream?: string; skills: string[] | undefined; employmentEquity: string[] | undefined; educationRequirement: "has_diploma" | "no_diploma"; diff --git a/apps/web/src/utils/poolUtils.test.ts b/apps/web/src/utils/poolUtils.test.ts index c124bc6bf69..d6ec7b066a6 100644 --- a/apps/web/src/utils/poolUtils.test.ts +++ b/apps/web/src/utils/poolUtils.test.ts @@ -4,11 +4,7 @@ import { createIntl, createIntlCache } from "react-intl"; -import { - fakeClassifications, - toLocalizedEnum, -} from "@gc-digital-talent/fake-data"; -import { PoolStream } from "@gc-digital-talent/graphql"; +import { fakeClassifications } from "@gc-digital-talent/fake-data"; import { formattedPoolPosterTitle } from "./poolUtils"; @@ -24,7 +20,13 @@ describe("poolUtils tests", () => { const baseInputs = { title: "Web Developer", classification: fakeClassifications()[0], - stream: toLocalizedEnum(PoolStream.SoftwareSolutions), + workStream: { + id: "uuid", + name: { + en: "Software solutions EN", + fr: "Software solutions FR", + }, + }, intl, }; test("should combine title, classification and stream if all are provided", () => { @@ -70,13 +72,13 @@ describe("poolUtils tests", () => { expect( formattedPoolPosterTitle({ ...baseInputs, - stream: null, + workStream: null, }).label, ).toBe("Web Developer (IT-01)"); expect( formattedPoolPosterTitle({ ...baseInputs, - stream: undefined, + workStream: undefined, }).label, ).toBe("Web Developer (IT-01)"); }); @@ -85,14 +87,14 @@ describe("poolUtils tests", () => { formattedPoolPosterTitle({ ...baseInputs, classification: undefined, - stream: null, + workStream: null, }).label, ).toBe("Web Developer"); expect( formattedPoolPosterTitle({ ...baseInputs, classification: null, - stream: undefined, + workStream: undefined, }).label, ).toBe("Web Developer"); }); @@ -101,7 +103,7 @@ describe("poolUtils tests", () => { formattedPoolPosterTitle({ title: "", classification: null, - stream: null, + workStream: null, intl, }).label, ).toBe(""); diff --git a/apps/web/src/utils/poolUtils.tsx b/apps/web/src/utils/poolUtils.tsx index 5bf1d60ab40..7d61597aa7a 100644 --- a/apps/web/src/utils/poolUtils.tsx +++ b/apps/web/src/utils/poolUtils.tsx @@ -23,8 +23,8 @@ import { Maybe, Classification, Pool, - LocalizedPoolStream, LocalizedPoolStatus, + WorkStream, } from "@gc-digital-talent/graphql"; import { PageNavInfo } from "~/types/pages"; @@ -90,7 +90,7 @@ export const formatClassificationString = ({ interface formattedPoolPosterTitleProps { title: Maybe | undefined; classification: Maybe> | undefined; - stream?: Maybe; + workStream?: Maybe; short?: boolean; intl: IntlShape; } @@ -98,14 +98,14 @@ interface formattedPoolPosterTitleProps { export const formattedPoolPosterTitle = ({ title, classification, - stream, + workStream, short, intl, }: formattedPoolPosterTitleProps): { html: ReactNode; label: string; } => { - const streamString = stream ? getLocalizedName(stream.label, intl) : ""; + const streamString = getLocalizedName(workStream?.name, intl, true); const groupAndLevel = classification ? formatClassificationString(classification) : ""; @@ -146,7 +146,7 @@ interface PoolTitleOptions { } type PoolTitle = Maybe< - Pick & { + Pick & { classification?: Maybe>; } >; @@ -183,7 +183,7 @@ export const poolTitle = ( const formattedTitle = formattedPoolPosterTitle({ title: specificTitle, classification: pool?.classification, - stream: pool?.stream, + workStream: pool?.workStream, short: options?.short, intl, }); @@ -232,7 +232,7 @@ export const useAdminPoolPages = ( ) => { const paths = useRoutes(); const poolName = getFullPoolTitleLabel(intl, { - stream: pool.stream, + workStream: pool.workStream, name: pool.name, publishingGroup: pool.publishingGroup, classification: pool.classification, diff --git a/apps/web/src/utils/requestUtils.tsx b/apps/web/src/utils/requestUtils.tsx index dd92032be06..43604304624 100644 --- a/apps/web/src/utils/requestUtils.tsx +++ b/apps/web/src/utils/requestUtils.tsx @@ -1,7 +1,4 @@ -import { - PoolCandidateSearchStatus, - PoolStream, -} from "@gc-digital-talent/graphql"; +import { PoolCandidateSearchStatus } from "@gc-digital-talent/graphql"; export function stringToEnumRequestStatus( selection: string, @@ -15,10 +12,3 @@ export function stringToEnumRequestStatus( } return undefined; } - -export function stringToEnumStream(selection: string): PoolStream | undefined { - if (Object.values(PoolStream).includes(selection as PoolStream)) { - return selection as PoolStream; - } - return undefined; -} diff --git a/apps/web/src/utils/testData.ts b/apps/web/src/utils/testData.ts index a130cdcb287..b860ca46d69 100644 --- a/apps/web/src/utils/testData.ts +++ b/apps/web/src/utils/testData.ts @@ -8,6 +8,7 @@ import { fakePools, fakeSkillFamilies, fakeSkills, + fakeWorkStreams, toLocalizedEnum, } from "@gc-digital-talent/fake-data"; import { @@ -31,6 +32,7 @@ const fakePool = fakePools( fakeSkills(20, fakeSkillFamilies(6)), fakeClassifications(), fakeDepartments(), + fakeWorkStreams(), 3, )[0]; // make three assessment steps which assess all the pool skills diff --git a/apps/web/src/validators/process/classification.ts b/apps/web/src/validators/process/classification.ts index 269ce3ec28e..9a58d09b491 100644 --- a/apps/web/src/validators/process/classification.ts +++ b/apps/web/src/validators/process/classification.ts @@ -5,16 +5,16 @@ import { Pool } from "@gc-digital-talent/graphql"; Note: The pool.classification should not be null, therefore it doesn't need to checked */ export function isInNullState({ - stream, + workStream, name, processNumber, publishingGroup, }: Pick< Pool, - "stream" | "name" | "processNumber" | "publishingGroup" + "workStream" | "name" | "processNumber" | "publishingGroup" >): boolean { return !!( - !stream && + !workStream && !name?.en && !name?.fr && !processNumber && @@ -27,7 +27,7 @@ export function hasEmptyRequiredFields({ areaOfSelection, classification, department, - stream, + workStream, name, processNumber, publishingGroup, @@ -37,7 +37,7 @@ export function hasEmptyRequiredFields({ | "areaOfSelection" | "classification" | "department" - | "stream" + | "workStream" | "name" | "processNumber" | "publishingGroup" @@ -47,7 +47,7 @@ export function hasEmptyRequiredFields({ !areaOfSelection?.value || !classification || !department || - !stream || + !workStream || !name?.en || !name?.fr || !processNumber || diff --git a/packages/fake-data/src/fakeApplicantFilters.ts b/packages/fake-data/src/fakeApplicantFilters.ts index 5bbae1a35c6..34334220491 100644 --- a/packages/fake-data/src/fakeApplicantFilters.ts +++ b/packages/fake-data/src/fakeApplicantFilters.ts @@ -8,17 +8,19 @@ import { LanguageAbility, PositionDuration, Skill, - PoolStream, + WorkStream, } from "@gc-digital-talent/graphql"; import fakeSkills from "./fakeSkills"; import fakePools from "./fakePools"; +import fakeWorkStreams from "./fakeWorkStreams"; import toLocalizedEnum from "./fakeLocalizedEnum"; const generateApplicantFilters = ( operationalRequirements: OperationalRequirement[], pools: Pool[], skills: Skill[], + workStreams: WorkStream[], ): ApplicantFilter => { return { __typename: "ApplicantFilter", @@ -47,9 +49,7 @@ const generateApplicantFilters = ( Object.values(PositionDuration), ), skills, - qualifiedStreams: faker.helpers - .arrayElements(Object.values(PoolStream), 1) - .map((stream) => toLocalizedEnum(stream)), + workStreams: faker.helpers.arrayElements(workStreams, 1), }; }; @@ -60,9 +60,15 @@ export default (): ApplicantFilter[] => { ); const pools = fakePools(); const skills = fakeSkills(5); + const workStreams = fakeWorkStreams(); faker.seed(0); // repeatable results return Array.from({ length: 20 }, () => - generateApplicantFilters(operationalRequirements, pools, skills), + generateApplicantFilters( + operationalRequirements, + pools, + skills, + workStreams, + ), ); }; diff --git a/packages/fake-data/src/fakePools.ts b/packages/fake-data/src/fakePools.ts index 4b0aaa7f2a3..266e2de9e82 100644 --- a/packages/fake-data/src/fakePools.ts +++ b/packages/fake-data/src/fakePools.ts @@ -13,7 +13,6 @@ import { PoolLanguage, User, UserPublicProfile, - PoolStream, PublishingGroup, SecurityStatus, Skill, @@ -27,6 +26,7 @@ import { PoolOpportunityLength, PoolAreaOfSelection, PoolSelectionLimitation, + WorkStream, } from "@gc-digital-talent/graphql"; import fakeScreeningQuestions from "./fakeScreeningQuestions"; @@ -39,12 +39,14 @@ import toLocalizedString from "./fakeLocalizedString"; import fakeAssessmentSteps from "./fakeAssessmentSteps"; import fakeDepartments from "./fakeDepartments"; import toLocalizedEnum from "./fakeLocalizedEnum"; +import fakeWorkStreams from "./fakeWorkStreams"; const generatePool = ( users: User[], skills: Skill[], classifications: Classification[], departments: Department[], + workStreams: WorkStream[], englishName = "", frenchName = "", essentialSkillCount = -1, @@ -106,10 +108,8 @@ const generatePool = ( }, classification: faker.helpers.arrayElement(classifications), department: faker.helpers.arrayElement(departments), + workStream: faker.helpers.arrayElement(workStreams), keyTasks: toLocalizedString(faker.lorem.paragraphs()), - stream: toLocalizedEnum( - faker.helpers.arrayElement(Object.values(PoolStream)), - ), processNumber: faker.helpers.maybe(() => faker.lorem.word()), publishingGroup: faker.helpers.maybe(() => toLocalizedEnum( @@ -168,6 +168,7 @@ export default ( skills = fakeSkills(100, fakeSkillFamilies(6)), classifications = fakeClassifications(), departments = fakeDepartments(), + workStreams = fakeWorkStreams(), essentialSkillCount = -1, ): Pool[] => { const users = fakeUsers(); @@ -180,6 +181,7 @@ export default ( skills, classifications, departments, + workStreams, "CMO", "CMO", essentialSkillCount, @@ -191,6 +193,7 @@ export default ( skills, classifications, departments, + workStreams, "IT Apprenticeship Program for Indigenous Peoples", "Programme d’apprentissage en TI pour les personnes autochtones", essentialSkillCount, @@ -202,6 +205,7 @@ export default ( skills, classifications, departments, + workStreams, "", "", essentialSkillCount, diff --git a/packages/fake-data/src/fakeWorkStreams.ts b/packages/fake-data/src/fakeWorkStreams.ts new file mode 100644 index 00000000000..f1ee4350e81 --- /dev/null +++ b/packages/fake-data/src/fakeWorkStreams.ts @@ -0,0 +1,20 @@ +import { faker } from "@faker-js/faker/locale/en"; + +import { WorkStream } from "@gc-digital-talent/graphql"; + +import toLocalizedString from "./fakeLocalizedString"; + +const generateWorkStream = (): WorkStream => { + return { + id: faker.string.uuid(), + key: faker.helpers.slugify(faker.string.sample()), + name: toLocalizedString(faker.company.catchPhrase()), + plainLanguageName: toLocalizedString(faker.company.catchPhrase()), + }; +}; + +export default (numToGenerate = 10): WorkStream[] => { + faker.seed(0); // repeatable results + + return Array.from({ length: numToGenerate }, () => generateWorkStream()); +}; diff --git a/packages/fake-data/src/index.ts b/packages/fake-data/src/index.ts index 95a5cf3b4c0..8865e20000d 100644 --- a/packages/fake-data/src/index.ts +++ b/packages/fake-data/src/index.ts @@ -16,6 +16,7 @@ import fakeSkills, { getStaticSkills } from "./fakeSkills"; import fakeTeams from "./fakeTeams"; import fakeUsers, { fakeApplicants, fakeUser } from "./fakeUsers"; import fakeUserSkills from "./fakeUserSkills"; +import fakeWorkStreams from "./fakeWorkStreams"; // Faker Generated Data export { @@ -38,6 +39,7 @@ export { fakeUsers, fakeUser, fakeUserSkills, + fakeWorkStreams, fakeLocalizedEnum, toLocalizedEnum, }; From c8736f468123cf65cb7b9e2366ea7194891714b2 Mon Sep 17 00:00:00 2001 From: Vachan <40485260+vd1992@users.noreply.github.com> Date: Fri, 13 Dec 2024 14:34:02 -0700 Subject: [PATCH 07/31] [Accessibility] Fix accordion focus overlap with actions (#12231) * Accordion styling changes * feedback --- .../ui/src/components/Accordion/Accordion.tsx | 75 ++++++++++++++++--- 1 file changed, 64 insertions(+), 11 deletions(-) diff --git a/packages/ui/src/components/Accordion/Accordion.tsx b/packages/ui/src/components/Accordion/Accordion.tsx index ab133f52c60..49ad0dca2fa 100644 --- a/packages/ui/src/components/Accordion/Accordion.tsx +++ b/packages/ui/src/components/Accordion/Accordion.tsx @@ -196,7 +196,7 @@ const Trigger = forwardRef< className="Accordion__Trigger" data-h2-align-items="base(flex-start)" data-h2-background-color="base(transparent) base:focus-visible(focus)" - data-h2-color="base(black) base:focus-visible(black) base:children[.Accordion__Subtitle](black.light) base:focus-visible:children[.Accordion__Subtitle](black) base:children[.Accordion__Chevron](black.light) base:focus-visible:children[.Accordion__Chevron](black)" + data-h2-color="base(black) base:focus-visible(black) base:children[.Accordion__Subtitle](black.light) base:all:focus-visible:children[*](black) base:children[.Accordion__Chevron](black.light) base:focus-visible:children[.Accordion__Chevron](black)" data-h2-cursor="base(pointer)" data-h2-display="base(flex)" data-h2-flex-wrap="base(wrap) p-tablet(nowrap)" @@ -290,25 +290,33 @@ interface AccordionMetaDataProps { } const MetaData = ({ metadata }: AccordionMetaDataProps) => { + const metadataLength = metadata.length; + const separatorSpan = ( + + • + + ); + return (
    - {metadata.map(({ type, color, href, children, onClick, key }) => { + {metadata.map(({ type, color, href, children, onClick, key }, index) => { switch (type) { case "text": - return ( + return index + 1 === metadataLength ? ( { > {children} + ) : ( + <> + + {children} + + {separatorSpan} + ); case "chip": - return ( + return index + 1 === metadataLength ? ( {children} + ) : ( + <> + + {children} + + {separatorSpan} + ); case "button": - return ( + return index + 1 === metadataLength ? ( + ) : ( + <> + + {separatorSpan} + ); case "link": - return ( + return index + 1 === metadataLength ? ( { > {children} + ) : ( + <> + + {children} + + {separatorSpan} + ); default: return null; From e56d5d9907d75a72d5237a7d4681993af6edf716 Mon Sep 17 00:00:00 2001 From: Matt Date: Mon, 16 Dec 2024 10:25:45 -0500 Subject: [PATCH 08/31] [Copy] Fixes null state instructions and associated button labels (#12281) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Change infinitif to impératif * Change infinitif to impératif * Replace action buttons Create a singularNoun with Create singularNoun for consistency * Differentiate between Add member * Fix incorrect message * Fix Instructions for adding process membership to a user * Replace Add new member with Add member * Fix label * Replace Add a new skill with Add skill * Update snapshot * Fix message * Update snapshot * Update to use impératif tense --- apps/web/src/components/SkillBrowser/utils.ts | 4 +- .../src/lang/__snapshots__/lang.test.ts.snap | 34 ++++-- apps/web/src/lang/fr.json | 108 ++++++++++-------- .../components/ClassificationTable.tsx | 4 +- .../components/AddCommunityMemberDialog.tsx | 18 +-- .../CommunityTable/CommunityTable.tsx | 6 +- .../components/AssetSkillsSection.tsx | 4 +- .../components/EssentialSkillsSection.tsx | 4 +- .../components/AddPoolMembershipDialog.tsx | 20 ++-- .../Teams/TeamMembersPage/TeamMembersPage.tsx | 4 +- .../components/AddTeamMemberDialog.tsx | 26 +++-- .../components/TrainingOpportunitiesTable.tsx | 4 +- .../components/ProcessRoleTable.tsx | 5 +- 13 files changed, 138 insertions(+), 103 deletions(-) diff --git a/apps/web/src/components/SkillBrowser/utils.ts b/apps/web/src/components/SkillBrowser/utils.ts index 34a3abb46c5..2203c2dc37c 100644 --- a/apps/web/src/components/SkillBrowser/utils.ts +++ b/apps/web/src/components/SkillBrowser/utils.ts @@ -79,8 +79,8 @@ export const getSkillBrowserDialogMessages: GetSkillBrowserDialogMessages = ({ return { ...defaults, trigger: intl.formatMessage({ - defaultMessage: "Add a new skill", - id: "ZYqWBR", + defaultMessage: "Add skill", + id: "RSUgO3", description: "Button text to open the skill dialog and add a skill", }), title: intl.formatMessage({ diff --git a/apps/web/src/lang/__snapshots__/lang.test.ts.snap b/apps/web/src/lang/__snapshots__/lang.test.ts.snap index c355564e1e3..f15a9a99f1c 100644 --- a/apps/web/src/lang/__snapshots__/lang.test.ts.snap +++ b/apps/web/src/lang/__snapshots__/lang.test.ts.snap @@ -45,10 +45,10 @@ exports[`message files should have no changes to duplicate strings 1`] = ` ] }, { - "en": "Create a community", + "en": "Add member", "fr": [ - "Créez une collectivité", - "Créer une collectivité" + "Ajouter un membre", + "Ajoutez un membre" ] }, { @@ -87,17 +87,24 @@ exports[`message files should have no changes to duplicate strings 1`] = ` ] }, { - "en": "Add team role", + "en": "Create training opportunity", "fr": [ - "Ajouter un rôle dans l'équipe", - "Ajoutez un rôle dans l'équipe" + "Créez une possibilité de formation", + "Créez la possibilité de formation" ] }, { - "en": "Create a training opportunity", + "en": "Create community", "fr": [ - "Créez une possibilité de formation", - "Créer une possibilité de formation" + "Créez une collectivité", + "Créer une collectivité" + ] + }, + { + "en": "Add team role", + "fr": [ + "Ajouter un rôle dans l'équipe", + "Ajoutez un rôle dans l'équipe" ] }, { @@ -138,10 +145,17 @@ exports[`message files should have no changes to duplicate strings 1`] = ` { "en": "Create skill family", "fr": [ - "Créer un groupe de compétences", + "Créez un groupe de compétences", "Créez le groupe de compétences" ] }, + { + "en": "Create process", + "fr": [ + "Créer un processus", + "Créez un processus" + ] + }, { "en": "Remove from community", "fr": [ diff --git a/apps/web/src/lang/fr.json b/apps/web/src/lang/fr.json index 5d7f4acce3e..886bda0f87d 100644 --- a/apps/web/src/lang/fr.json +++ b/apps/web/src/lang/fr.json @@ -119,10 +119,6 @@ "defaultMessage": "Voir les critères d’admissibilité", "description": "Button text for program eligibility criteria" }, - "+e2nr6": { - "defaultMessage": "Ajouter un nouveau membre", - "description": "Label for the add member to team form" - }, "+fIw4g": { "defaultMessage": "Modifier ces renseignements pour {title}", "description": "Text label for button to edit employment equity category in profile." @@ -420,7 +416,7 @@ "description": "Third paragraph for pool closing date dialog" }, "07sCDh": { - "defaultMessage": "Utilisez le bouton « Créer un processus » pour commencer.", + "defaultMessage": "Utilisez le bouton « Créez un processus » pour commencer.", "description": "Instructions for adding a process item." }, "08IbRz": { @@ -560,7 +556,7 @@ "description": "Simplified status label for a complete process advertisement or assessment" }, "0jwdac": { - "defaultMessage": "Utilisez le bouton « Créer une compétence » pour commencer.", + "defaultMessage": "Utilisez le bouton « Créez une compétence » pour commencer.", "description": "Instructions for adding a skill item." }, "0k6j4V": { @@ -764,7 +760,7 @@ "description": "Role with group, HTML" }, "1xI8uo": { - "defaultMessage": "Utilisez le bouton « Ajoutez un rôle dans l’équipe » pour commencer.", + "defaultMessage": "Utilisez le bouton « Ajoutez un rôle individuel » pour commencer.", "description": "Instructions for adding a role to a user." }, "2/fjEP": { @@ -924,7 +920,7 @@ "description": "Paragraph 3, importance of self-declaration" }, "2m5USi": { - "defaultMessage": "Utilisez le bouton « Modifier » pour commencer. ", + "defaultMessage": "Utilisez le bouton « Modifiez » pour commencer. ", "description": "Null message on sections for edit pool page." }, "2n0e2i": { @@ -1388,7 +1384,7 @@ "description": "Filename for skills CSV file download" }, "4ujx9e": { - "defaultMessage": "Utilisez le bouton « Créer un groupe de compétences » pour commencer.", + "defaultMessage": "Utilisez le bouton « Créez un groupe de compétences » pour commencer.", "description": "Instructions for adding a skill family item." }, "4v9y7U": { @@ -2343,6 +2339,10 @@ "defaultMessage": "Oh! on dirait que vous avez été trop vite!", "description": "Application step skipped page title" }, + "9vluO2": { + "defaultMessage": "Ajouter un membre", + "description": "Header for the add member to team form" + }, "A+4huJ": { "defaultMessage": "Mettre à jour ou supprimer une expérience dans votre chronologie de carrière", "description": "Display text for edit experience form in breadcrumbs" @@ -2611,10 +2611,6 @@ "defaultMessage": "Compte et confidentialité", "description": "Link to the 'Account and privacy' page" }, - "BRd2Xw": { - "defaultMessage": "Créez une collectivité", - "description": "Text to create a community" - }, "BSSYnh": { "defaultMessage": "Nous voulons vous informer qu’entre-temps, des mises à jour sont apportées à ce site qui permettront aux personnes autochtones qui souhaitent se joindre au Programme d’apprentissage en TI de présenter une demande en ligne.", "description": "Second paragraph for apply now dialog" @@ -2908,7 +2904,7 @@ "description": "Description of how many items are being displayed out of the total value" }, "Cu+CH3": { - "defaultMessage": "Utilisez le bouton « Créer une collectivité » pour commencer.", + "defaultMessage": "Utilisez le bouton « Créez une collectivité » pour commencer.", "description": "Instructions for adding a community item" }, "CuHYqt": { @@ -3120,7 +3116,7 @@ "description": "Assessment summary" }, "DrR/rp": { - "defaultMessage": "Utilisez le bouton « Ajouter un membre » pour commencer.", + "defaultMessage": "Utilisez le bouton « Ajoutez un membre » pour commencer.", "description": "Instructions for adding a member to a community." }, "Ds7ONS": { @@ -3371,6 +3367,10 @@ "defaultMessage": "Publié", "description": "Title displayed on the Pool table published at column" }, + "FBx3Q4": { + "defaultMessage": "Ajoutez un membre", + "description": "Label for the add member to team form" + }, "FDGVHB": { "defaultMessage": "L’information des ministères ne s’avère pas nécessaire; réduction des exigences en matière de collecte pour les ministères", "description": "third 2024 key change rationale to the directive" @@ -3599,6 +3599,10 @@ "defaultMessage": "Niveau", "description": "Label displayed on classification level input" }, + "GMbOaT": { + "defaultMessage": "Utilisez le bouton « Ajoutez une compétence » pour commencer.", + "description": "Null message description for asset skills table." + }, "GMglI5": { "defaultMessage": "Adresse courriel validée", "description": "The email address has been verified to be owned by user" @@ -3895,6 +3899,10 @@ "defaultMessage": "{title} à {organization}", "description": "Title at organization, HTML" }, + "IHyNL8": { + "defaultMessage": "Ajouter un membre", + "description": "Title for the add member to community form" + }, "II4+N3": { "defaultMessage": "Au moyen de la liste des compétences du processus de recrutement, sélectionnez les compétences que vous prévoyez évaluer au moyen de cette méthode d’évaluation.", "description": "description of 'skill selection' section of the 'assessment details' dialog" @@ -4107,6 +4115,10 @@ "defaultMessage": "Employé(e)s (de même niveau)", "description": "Combined eligibility string for 'employees only' and 'at-level only'" }, + "JCZlxS": { + "defaultMessage": "Utilisez le bouton « Ajoutez un rôle dans le processus » pour commencer.", + "description": "Instructions for adding process membership to a user." + }, "JDQvla": { "defaultMessage": "Ces notes sont disponibles pour tous les gestionnaires de ce bassin, mais pas pour les candidats.", "description": "Description of pool candidate notes field" @@ -4311,6 +4323,10 @@ "defaultMessage": "Date d’expiration (heure locale)", "description": "A date at which data will expire, in the local time zone" }, + "KHmf+e": { + "defaultMessage": "Utilisez le bouton « Créez une classification » pour commencer.", + "description": "Instructions for adding a classification item." + }, "KJDxM1": { "defaultMessage": "État du processus", "description": "Title for card for actions related to changing the status of a process" @@ -4799,6 +4815,10 @@ "defaultMessage": "Processus", "description": "Title for the index pool page" }, + "MkUz+j": { + "defaultMessage": "Ajoutez un membre", + "description": "Label for the add member to community form (action)" + }, "MmTWYV": { "defaultMessage": "Exigences linguistiques pour les personnes candidates", "description": "Link to second language evaluations PSC info page on language requirements dialog" @@ -5371,6 +5391,10 @@ "defaultMessage": "Grade en économie, en sociologie ou en statistique", "description": "Heading for the `just education` option for EC education requirements" }, + "PdkgWB": { + "defaultMessage": "Créez une possibilité de formation", + "description": "Title for link to page to create a training opportunity (imperative in French)" + }, "Pe1kas": { "defaultMessage": "Notes sur les demandes", "description": "Label displayed on the search request form request notes field." @@ -5411,6 +5435,10 @@ "defaultMessage": "Vous avez été supprimé des résultats de la recherche.", "description": "Alert displayed to the user when application card dialog submits successfully." }, + "PrTwov": { + "defaultMessage": "Créez une collectivité", + "description": "Text to create a community (action)" + }, "Ps929U": { "defaultMessage": "Erreur : La mise à jour de votre autodéclaration a échoué", "description": "Message displayed to user after self-declaration fails to be updated." @@ -5775,6 +5803,10 @@ "defaultMessage": "{label} navigation de page", "description": "Label for the table pagination" }, + "RSUgO3": { + "defaultMessage": "Ajoutez une compétence", + "description": "Button text to open the skill dialog and add a skill" + }, "RSkJQR": { "defaultMessage": "Brève description de l'équipe (en français)", "description": "Label for team description in French language" @@ -5867,10 +5899,6 @@ "defaultMessage": "Il manque des données requises par le gouvernement", "description": "Error message displayed when a users government information is incomplete" }, - "RtX9oA": { - "defaultMessage": "Créez une possibilité de formation", - "description": "Title for link to page to create a training opportunity (imperative in French)" - }, "RvB1GT": { "defaultMessage": "Relations de travail", "description": "Label for _labour relations_ option in _authorities involved_ fieldset in the _digital services contracting questionnaire_" @@ -6011,10 +6039,6 @@ "defaultMessage": "Le gouvernement du Canada s’est engagé à financer le perfectionnement de ses spécialistes des TI. Grâce au Fonds de formation et de perfectionnement de la collectivité de la TI, les membres du personnel du groupe TI, représentés par l’Institut professionnel de la fonction publique du Canada (IPFPC), ont désormais accès à une plus vaste gamme de possibilités d’apprentissage afin d’acquérir des compétences et les approfondir.", "description": "First paragraph describing investing in future talent" }, - "SfbDLA": { - "defaultMessage": "Utilisez le bouton « Ajouter un nouveau membre » pour commencer.", - "description": "Instructions for adding a member to a team." - }, "SfhT1q": { "defaultMessage": "Acquérir de l'expérience en matière d'embauche", "description": "Title to get hiring experience" @@ -6211,10 +6235,6 @@ "defaultMessage": "Solutions logicielles de la TI", "description": "Title for the 'software solutions' IT work stream" }, - "Tl2FNA": { - "defaultMessage": "Utilisez le bouton « Créez une classification » pour commencer.", - "description": "Instructions for adding a classification item." - }, "TomxAe": { "defaultMessage": "{time} le {date}", "description": "A datetime formatted as a certain time on a certain date" @@ -6546,6 +6566,10 @@ "defaultMessage": "Nous aimerions maintenant savoir si vous êtes actuellement employé(e) du gouvernement du Canada. Nous recueillons cette information parce qu'elle nous aide à comprendre, de manière globale, comment les compétences numériques sont réparties entre les différents ministères.", "description": "Message after main heading in employee information page - paragraph 1." }, + "VaToft": { + "defaultMessage": "Utilisez le bouton « Ajoutez une compétence » pour commencer.", + "description": "Null message description for essential skills table." + }, "VaVo2t": { "defaultMessage": "Erreur : Échec de la création du ministère", "description": "Message displayed to user after department fails to get created." @@ -6750,10 +6774,6 @@ "defaultMessage": "Droit de priorité", "description": "Label for applicant's priority entitlement status" }, - "Wd9+xg": { - "defaultMessage": "Utilisez le bouton « Ajouter une nouvelle compétence » pour commencer.", - "description": "Null message description for asset skills table." - }, "WdBt7s": { "defaultMessage": "Approvisionnement", "description": "Label for _procurement_ option in _authorities involved_ fieldset in the _digital services contracting questionnaire_" @@ -7386,10 +7406,6 @@ "defaultMessage": "Erreur lors du chargement des candidats", "description": "Error message when pool candidates could not be loaded" }, - "ZYqWBR": { - "defaultMessage": "Ajouter une nouvelle compétence", - "description": "Button text to open the skill dialog and add a skill" - }, "ZZpC8s": { "defaultMessage": "Voir les définitions pour « {skillName} »", "description": "Accordion title for skill header on screening decision dialog." @@ -9150,10 +9166,6 @@ "defaultMessage": "Affinez les résultats de votre tableau à l'aide des filtres suivants.", "description": "Candidate search filter dialog: subtitle" }, - "hryX4G": { - "defaultMessage": "Utilisez le bouton « Ajoutez des rôles dans des processus  » pour commencer.", - "description": "Instructions for adding process membership to a user." - }, "htxH4r": { "defaultMessage": "Retourner à votre tableau de bord", "description": "Link text to navigate to the profile and applications page" @@ -10035,7 +10047,7 @@ "description": "Link to a page to view all the requests" }, "mKrj0x": { - "defaultMessage": "Sauvegarder et ajouter un membre", + "defaultMessage": "Sauvegardez et ajoutez un membre", "description": "Label for add member to a community form" }, "mKzQwr": { @@ -10431,7 +10443,7 @@ "description": "Button label to return to the skills table" }, "oDDr9J": { - "defaultMessage": "Créer un groupe de compétences", + "defaultMessage": "Créez un groupe de compétences", "description": "Heading displayed above the Create Skill family form." }, "oES0/4": { @@ -11134,6 +11146,10 @@ "defaultMessage": "Le processus a été archivé avec succès!", "description": "Message affiché à l'utilisateur après l'archivage du bassin" }, + "s57oJd": { + "defaultMessage": "Utilisez le bouton « Ajoutez un membre » pour commencer.", + "description": "Instructions for adding a member to a team." + }, "s5hTYo": { "defaultMessage": "Rôles dans l'équipe", "description": "Label for the input to select role of a team role" @@ -11562,10 +11578,6 @@ "defaultMessage": "Type d'emploi", "description": "Label for the employment type radio group" }, - "uiuMqi": { - "defaultMessage": "Utilisez le bouton « Ajouter une nouvelle compétence » pour commencer.", - "description": "Null message description for essential skills table." - }, "ukcsxj": { "defaultMessage": "Si vous choisissez de vous autodéclarer dans votre profil et que vous postulez un emploi chez Talents numériques du GC, ces renseignements peuvent être utilisés à n’importe quelle étape du processus d’emploi, de la demande initiale à l’évaluation en passant par la nomination. Les gestionnaires d’embauche ou l’équipe de recrutement de Talents numériques du GC pourraient utiliser ces renseignements", "description": "Paragraph 2, how self-declaration data is used" @@ -11810,10 +11822,6 @@ "defaultMessage": "Cette demande a été vérifiée, {expiryDate}.", "description": "Message when a claim as been verified" }, - "wBMn5c": { - "defaultMessage": "Ajouter un membre", - "description": "Label for the add member to community form" - }, "wBU9X4": { "defaultMessage": "Français seulement", "description": "Filter by option on instructor training page." @@ -11855,7 +11863,7 @@ "description": "Placeholder displayed on the skill form families field." }, "wP9+aN": { - "defaultMessage": "Créer un processus", + "defaultMessage": "Créez un processus", "description": "Heading displayed above the Create process form." }, "wPpvvm": { diff --git a/apps/web/src/pages/Classifications/components/ClassificationTable.tsx b/apps/web/src/pages/Classifications/components/ClassificationTable.tsx index 03853d50f05..64e9e630ba2 100644 --- a/apps/web/src/pages/Classifications/components/ClassificationTable.tsx +++ b/apps/web/src/pages/Classifications/components/ClassificationTable.tsx @@ -146,8 +146,8 @@ export const ClassificationTable = ({ nullMessage={{ description: intl.formatMessage({ defaultMessage: - 'Use the "Create Classification" button to get started.', - id: "Tl2FNA", + 'Use the "Create classification" button to get started.', + id: "KHmf+e", description: "Instructions for adding a classification item.", }), }} diff --git a/apps/web/src/pages/Communities/CommunityMembersPage/components/AddCommunityMemberDialog.tsx b/apps/web/src/pages/Communities/CommunityMembersPage/components/AddCommunityMemberDialog.tsx index 49428a56441..a6ba8a5d81c 100644 --- a/apps/web/src/pages/Communities/CommunityMembersPage/components/AddCommunityMemberDialog.tsx +++ b/apps/web/src/pages/Communities/CommunityMembersPage/components/AddCommunityMemberDialog.tsx @@ -120,17 +120,15 @@ AddCommunityMemberDialogProps) => { ), })); - const label = intl.formatMessage({ - defaultMessage: "Add member", - id: "wBMn5c", - description: "Label for the add member to community form", - }); - return ( @@ -148,7 +146,11 @@ AddCommunityMemberDialogProps) => { }, )} > - {label} + {intl.formatMessage({ + defaultMessage: "Add member", + id: "IHyNL8", + description: "Title for the add member to community form", + })} diff --git a/apps/web/src/pages/Communities/IndexCommunityPage/components/CommunityTable/CommunityTable.tsx b/apps/web/src/pages/Communities/IndexCommunityPage/components/CommunityTable/CommunityTable.tsx index d53b45b3a6d..c413989354c 100644 --- a/apps/web/src/pages/Communities/IndexCommunityPage/components/CommunityTable/CommunityTable.tsx +++ b/apps/web/src/pages/Communities/IndexCommunityPage/components/CommunityTable/CommunityTable.tsx @@ -130,9 +130,9 @@ export const CommunityTable = ({ linkProps: { href: paths.communityCreate(), label: intl.formatMessage({ - defaultMessage: "Create a community", - id: "BRd2Xw", - description: "Text to create a community", + defaultMessage: "Create community", + id: "PrTwov", + description: "Text to create a community (action)", }), from: currentUrl, }, diff --git a/apps/web/src/pages/Pools/EditPoolPage/components/AssetSkillsSection.tsx b/apps/web/src/pages/Pools/EditPoolPage/components/AssetSkillsSection.tsx index 20996979008..4af8cbf5f53 100644 --- a/apps/web/src/pages/Pools/EditPoolPage/components/AssetSkillsSection.tsx +++ b/apps/web/src/pages/Pools/EditPoolPage/components/AssetSkillsSection.tsx @@ -131,8 +131,8 @@ const AssetSkillsSection = ({ description: "Null message title for asset skills table.", }), description: intl.formatMessage({ - defaultMessage: `Use the "Add a new skill" button to get started.`, - id: "Wd9+xg", + defaultMessage: `Use the "Add skill" button to get started.`, + id: "GMbOaT", description: "Null message description for asset skills table.", }), }} diff --git a/apps/web/src/pages/Pools/EditPoolPage/components/EssentialSkillsSection.tsx b/apps/web/src/pages/Pools/EditPoolPage/components/EssentialSkillsSection.tsx index df0cfe21859..03c431e9969 100644 --- a/apps/web/src/pages/Pools/EditPoolPage/components/EssentialSkillsSection.tsx +++ b/apps/web/src/pages/Pools/EditPoolPage/components/EssentialSkillsSection.tsx @@ -131,8 +131,8 @@ const EssentialSkillsSection = ({ description: "Null message title for essential skills table.", }), description: intl.formatMessage({ - defaultMessage: `Use the "Add a new skill" button to get started.`, - id: "uiuMqi", + defaultMessage: `Use the "Add skill" button to get started.`, + id: "VaToft", description: "Null message description for essential skills table.", }), }} diff --git a/apps/web/src/pages/Pools/ManageAccessPage/components/AddPoolMembershipDialog.tsx b/apps/web/src/pages/Pools/ManageAccessPage/components/AddPoolMembershipDialog.tsx index eec30f69f0b..6f8f3227f7a 100644 --- a/apps/web/src/pages/Pools/ManageAccessPage/components/AddPoolMembershipDialog.tsx +++ b/apps/web/src/pages/Pools/ManageAccessPage/components/AddPoolMembershipDialog.tsx @@ -120,21 +120,25 @@ const AddPoolMembershipDialog = ({ ), })); - const label = intl.formatMessage({ - defaultMessage: "Add member", - id: "wBMn5c", - description: "Label for the add member to community form", - }); - return ( - {label} + + {intl.formatMessage({ + defaultMessage: "Add member", + id: "IHyNL8", + description: "Title for the add member to community form", + })} +
    diff --git a/apps/web/src/pages/Teams/TeamMembersPage/TeamMembersPage.tsx b/apps/web/src/pages/Teams/TeamMembersPage/TeamMembersPage.tsx index d63a480996a..db46cdd2c74 100644 --- a/apps/web/src/pages/Teams/TeamMembersPage/TeamMembersPage.tsx +++ b/apps/web/src/pages/Teams/TeamMembersPage/TeamMembersPage.tsx @@ -124,8 +124,8 @@ const TeamMembers = ({ teamQuery }: TeamMembersProps) => { }, nullMessage: { description: intl.formatMessage({ - defaultMessage: 'Use the "Add new member" button to get started.', - id: "SfbDLA", + defaultMessage: 'Use the "Add member" button to get started.', + id: "s57oJd", description: "Instructions for adding a member to a team.", }), }, diff --git a/apps/web/src/pages/Teams/TeamMembersPage/components/AddTeamMemberDialog.tsx b/apps/web/src/pages/Teams/TeamMembersPage/components/AddTeamMemberDialog.tsx index 38da31c72fa..e3e6f2bf5eb 100644 --- a/apps/web/src/pages/Teams/TeamMembersPage/components/AddTeamMemberDialog.tsx +++ b/apps/web/src/pages/Teams/TeamMembersPage/components/AddTeamMemberDialog.tsx @@ -113,21 +113,25 @@ AddTeamMemberDialogProps) => { ), })); - const label = intl.formatMessage({ - defaultMessage: "Add new member", - id: "+e2nr6", - description: "Label for the add member to team form", - }); - return ( - {label} + + {intl.formatMessage({ + defaultMessage: "Add member", + id: "9vluO2", + description: "Header for the add member to team form", + })} +

    {intl.formatMessage({ @@ -206,7 +210,11 @@ AddTeamMemberDialogProps) => {