sojorn/sojorn_app/lib/models/user_settings.dart
Patrick Britton 3c4680bdd7 Initial commit: Complete threaded conversation system with inline replies
**Major Features Added:**
- **Inline Reply System**: Replace compose screen with inline reply boxes
- **Thread Navigation**: Parent/child navigation with jump functionality
- **Chain Flow UI**: Reply counts, expand/collapse animations, visual hierarchy
- **Enhanced Animations**: Smooth transitions, hover effects, micro-interactions

 **Frontend Changes:**
- **ThreadedCommentWidget**: Complete rewrite with animations and navigation
- **ThreadNode Model**: Added parent references and descendant counting
- **ThreadedConversationScreen**: Integrated navigation handlers
- **PostDetailScreen**: Replaced with threaded conversation view
- **ComposeScreen**: Added reply indicators and context
- **PostActions**: Fixed visibility checks for chain buttons

 **Backend Changes:**
- **API Route**: Added /posts/:id/thread endpoint
- **Post Repository**: Include allow_chain and visibility fields in feed
- **Thread Handler**: Support for fetching post chains

 **UI/UX Improvements:**
- **Reply Context**: Clear indication when replying to specific posts
- **Character Counting**: 500 character limit with live counter
- **Visual Hierarchy**: Depth-based indentation and styling
- **Smooth Animations**: SizeTransition, FadeTransition, hover states
- **Chain Navigation**: Parent/child buttons with visual feedback

 **Technical Enhancements:**
- **Animation Controllers**: Proper lifecycle management
- **State Management**: Clean separation of concerns
- **Navigation Callbacks**: Reusable navigation system
- **Error Handling**: Graceful fallbacks and user feedback

This creates a Reddit-style threaded conversation experience with smooth
animations, inline replies, and intuitive navigation between posts in a chain.
2026-01-30 07:40:19 -06:00

60 lines
1.6 KiB
Dart

class UserSettings {
final String userId;
final int? defaultPostTtl;
const UserSettings({
required this.userId,
this.defaultPostTtl,
});
factory UserSettings.fromJson(Map<String, dynamic> json) {
return UserSettings(
userId: json['user_id'] as String,
defaultPostTtl: _parseIntervalHours(json['default_post_ttl']),
);
}
Map<String, dynamic> toJson() {
return {
'user_id': userId,
'default_post_ttl': defaultPostTtl != null ? '${defaultPostTtl} hours' : null,
};
}
UserSettings copyWith({
int? defaultPostTtl,
}) {
return UserSettings(
userId: userId,
defaultPostTtl: defaultPostTtl ?? this.defaultPostTtl,
);
}
static int? _parseIntervalHours(dynamic value) {
if (value == null) return null;
if (value is int) return value;
if (value is double) return value.round();
if (value is String) {
final trimmed = value.trim();
if (trimmed.isEmpty) return null;
final dayMatch = RegExp(r'(\\d+)\\s+day').firstMatch(trimmed);
final hourMatch = RegExp(r'(\\d+)\\s+hour').firstMatch(trimmed);
final timeMatch = RegExp(r'(\\d{1,2}):(\\d{2}):(\\d{2})').firstMatch(trimmed);
var totalHours = 0;
if (dayMatch != null) {
totalHours += (int.tryParse(dayMatch.group(1) ?? '') ?? 0) * 24;
}
if (timeMatch != null) {
totalHours += int.tryParse(timeMatch.group(1) ?? '') ?? 0;
} else if (hourMatch != null) {
totalHours += int.tryParse(hourMatch.group(1) ?? '') ?? 0;
}
return totalHours == 0 ? null : totalHours;
}
return null;
}
}