**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.
60 lines
1.6 KiB
Dart
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;
|
|
}
|
|
}
|