sojorn/sojorn_app/lib/models/comment.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

78 lines
1.9 KiB
Dart

import 'profile.dart';
/// Comment status enum
enum CommentStatus {
active('active'),
flagged('flagged'),
removed('removed');
final String value;
const CommentStatus(this.value);
static CommentStatus fromString(String value) {
return CommentStatus.values.firstWhere(
(status) => status.value == value,
orElse: () => CommentStatus.active,
);
}
}
/// Comment model matching backend comments table
class Comment {
final String id;
final String postId;
final String authorId;
final String body;
final CommentStatus status;
final DateTime createdAt;
final DateTime? updatedAt;
// Relations
final Profile? author;
final int? voteCount;
Comment({
required this.id,
required this.postId,
required this.authorId,
required this.body,
required this.status,
required this.createdAt,
this.updatedAt,
this.author,
this.voteCount,
});
factory Comment.fromJson(Map<String, dynamic> json) {
return Comment(
id: json['id'] as String,
postId: json['post_id'] as String,
authorId: json['author_id'] as String,
body: json['body'] as String,
status: CommentStatus.fromString(json['status'] as String),
createdAt: DateTime.parse(json['created_at'] as String),
updatedAt: json['updated_at'] != null
? DateTime.parse(json['updated_at'] as String)
: null,
author: json['author'] != null
? Profile.fromJson(json['author'] as Map<String, dynamic>)
: null,
voteCount: json['vote_count'] as int?,
);
}
Map<String, dynamic> toJson() {
return {
'id': id,
'post_id': postId,
'author_id': authorId,
'body': body,
'status': status.value,
'created_at': createdAt.toIso8601String(),
'updated_at': updatedAt?.toIso8601String(),
'author': author?.toJson(),
'vote_count': voteCount,
};
}
}