Define OrgSelectionService interface and data models
epic-organization-selection-service-labels-task-001 — Define the abstract interface for OrgSelectionService including method signatures for loadOrganizations, selectOrganization, getSelectedOrganization, and clearSelection. Define the Organization data model with all required fields (id, name, labels map, branding config). Establish the contract that downstream consumers depend on.
Acceptance Criteria
Technical Requirements
Implementation Notes
File locations: lib/domain/services/org_selection_service.dart (interface), lib/domain/models/organization.dart (data class), lib/domain/models/org_branding_config.dart (branding sub-model). Use Dart's built-in const constructors where possible for Organization and OrgBrandingConfig. For the labels map, type it as Map
The labels map on Organization is the raw data from Supabase — OrgLabelsProvider (task-002) is the typed access layer on top.
Testing Requirements
No direct tests for the abstract interface. Test the Organization data model: verify equality by id (two organizations with same id but different name are equal), verify copyWith produces a new object with the updated field, verify fromJson/toJson round-trips if JSON serialization is added. Use flutter_test with plain unit tests — no mocks needed for pure data classes.
The OrgLabelsProvider must be fully initialized before any UI widget renders organization-specific text. If the service triggers label initialization asynchronously and the UI builds before the labels stream emits, widgets will briefly display raw label keys instead of human-readable text, which constitutes a WCAG 2.2 AA failure for screen readers that announce whatever text is present at render time.
Mitigation & Contingency
Mitigation: Design the OrgSelectionService.select() method to await OrgLabelsProvider initialization before completing its Future — callers receive a resolved future only after labels are ready. Use an AsyncValue loading state to gate UI rendering via Riverpod's when() pattern.
Contingency: If awaiting initialization causes perceivable UI lag, implement an optimistic render with a skeleton loading state that is announced by the live-region-announcer as 'Loading organization content' so screen readers do not announce raw keys.
If the label definitions loaded from the backend for each organization do not match the label key registry used in the UI, widgets will fall back to raw keys silently. This is particularly harmful for Blindeforbundet users relying on VoiceOver, where a raw key announced by a screen reader is incomprehensible.
Mitigation & Contingency
Mitigation: Define a compile-time label key registry (enum or const strings) and assert at OrgLabelsProvider initialization that all required keys are present in the loaded map. Log a structured warning for any missing key and substitute a human-readable fallback string rather than the raw key.
Contingency: If a label schema mismatch reaches production, the OrgLabelsProvider fallback mechanism ensures users see a reasonable English default rather than a raw key. A backend label patch can be deployed without an app release.