use db::sqlx; use model::users::UserAccessibilityModel; use serde::{Deserialize, Serialize}; use session::Session; use crate::{AppService, error::AppError, session_user}; #[derive(Debug, Clone, Serialize, Deserialize, utoipa::ToSchema)] pub struct UserAccessibilityConfig { pub reduce_motion: bool, pub high_contrast: bool, pub screen_reader_optimized: bool, pub font_scale_percent: i32, pub color_blind_mode: Option, } #[derive(Debug, Clone, Deserialize, utoipa::ToSchema)] pub struct UpdateUserAccessibilityConfig { pub reduce_motion: Option, pub high_contrast: Option, pub screen_reader_optimized: Option, pub font_scale_percent: Option, pub color_blind_mode: Option>, } impl AppService { pub async fn user_update_accessibility_config( &self, ctx: &Session, params: UpdateUserAccessibilityConfig, ) -> Result { let user_uid = session_user(ctx)?; let mut config = self.user_accessibility_config(user_uid).await?; if let Some(reduce_motion) = params.reduce_motion { config.reduce_motion = reduce_motion; } if let Some(high_contrast) = params.high_contrast { config.high_contrast = high_contrast; } if let Some(screen_reader_optimized) = params.screen_reader_optimized { config.screen_reader_optimized = screen_reader_optimized; } if let Some(font_scale_percent) = params.font_scale_percent { config.font_scale_percent = font_scale_percent.clamp(50, 200); } if let Some(color_blind_mode) = params.color_blind_mode { config.color_blind_mode = color_blind_mode; } let now = chrono::Utc::now(); sqlx::query( "INSERT INTO user_accessibility \ (\"user\", reduce_motion, high_contrast, screen_reader_optimized, font_scale_percent, color_blind_mode, created_at, updated_at) \ VALUES ($1, $2, $3, $4, $5, $6, $7, $7) \ ON CONFLICT (\"user\") DO UPDATE SET \ reduce_motion = EXCLUDED.reduce_motion, high_contrast = EXCLUDED.high_contrast, \ screen_reader_optimized = EXCLUDED.screen_reader_optimized, font_scale_percent = EXCLUDED.font_scale_percent, \ color_blind_mode = EXCLUDED.color_blind_mode, updated_at = EXCLUDED.updated_at", ) .bind(user_uid) .bind(config.reduce_motion) .bind(config.high_contrast) .bind(config.screen_reader_optimized) .bind(config.font_scale_percent) .bind(&config.color_blind_mode) .bind(now) .execute(self.db.writer()) .await .map_err(|e| AppError::DatabaseError(e.to_string()))?; Ok(config) } pub(crate) async fn user_accessibility_config( &self, user_uid: uuid::Uuid, ) -> Result { let row = sqlx::query_as::<_, UserAccessibilityModel>( "SELECT \"user\", reduce_motion, high_contrast, screen_reader_optimized, font_scale_percent, color_blind_mode, created_at, updated_at \ FROM user_accessibility WHERE \"user\" = $1", ) .bind(user_uid) .fetch_optional(self.db.reader()) .await .map_err(|e| AppError::DatabaseError(e.to_string()))?; Ok(row.map(Into::into).unwrap_or_default()) } } impl Default for UserAccessibilityConfig { fn default() -> Self { Self { reduce_motion: false, high_contrast: false, screen_reader_optimized: false, font_scale_percent: 100, color_blind_mode: None, } } } impl From for UserAccessibilityConfig { fn from(value: UserAccessibilityModel) -> Self { Self { reduce_motion: value.reduce_motion, high_contrast: value.high_contrast, screen_reader_optimized: value.screen_reader_optimized, font_scale_percent: value.font_scale_percent, color_blind_mode: value.color_blind_mode, } } }