Files
Eventify-frontend/lib/features/events/models/event_models.dart
Sicherhaven 1badeff966 feat: add complete review/rating system for events
New feature: Users can view, submit, and interact with event reviews.

Components added:
- ReviewModel, ReviewStatsModel, ReviewListResponse (models)
- ReviewService with getReviews, submitReview, markHelpful, flagReview
- StarRatingInput (interactive 5-star picker with labels)
- StarDisplay (read-only fractional star display)
- ReviewSummary (average rating + distribution bars)
- ReviewForm (star picker + comment field + submit/update)
- ReviewCard (avatar, timestamp, expandable comment, helpful/flag)
- ReviewSection (main container with pagination and state mgmt)

Integration:
- Added to LearnMoreScreen (both mobile and desktop layouts)
- Review API endpoints point to app.eventifyplus.com Node.js backend
- EventModel updated with averageRating/reviewCount fields

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-30 18:04:37 +05:30

162 lines
5.0 KiB
Dart

// lib/features/events/models/event_models.dart
import '../../../core/api/api_endpoints.dart';
class EventTypeModel {
final int id;
final String name;
final String? iconUrl;
EventTypeModel({required this.id, required this.name, this.iconUrl});
/// Resolve a relative media path (e.g. `/media/...`) to a full URL.
static String? _resolveMediaUrl(String? raw) {
if (raw == null || raw.isEmpty) return null;
if (raw.startsWith('http://') || raw.startsWith('https://')) return raw;
return '${ApiEndpoints.mediaBaseUrl}$raw';
}
factory EventTypeModel.fromJson(Map<String, dynamic> j) {
return EventTypeModel(
id: j['id'] as int,
name: (j['event_type'] ?? j['name'] ?? '') as String,
iconUrl: _resolveMediaUrl((j['event_type_icon'] ?? j['icon_url']) as String?),
);
}
}
class EventImageModel {
final bool isPrimary;
final String image;
EventImageModel({required this.isPrimary, required this.image});
factory EventImageModel.fromJson(Map<String, dynamic> j) {
return EventImageModel(
isPrimary: j['is_primary'] == true,
image: EventTypeModel._resolveMediaUrl(j['image'] as String?) ?? '',
);
}
}
class EventModel {
final int id;
final String name;
final String? title;
final String? description;
final String startDate; // YYYY-MM-DD
final String endDate;
final String? startTime;
final String? endTime;
final String? pincode;
final String? place;
final bool isBookable;
final int? eventTypeId;
final String? thumbImg;
final List<EventImageModel> images;
// NEW fields mapped from backend
final String? importantInformation;
final String? venueName;
final String? eventStatus;
final String? cancelledReason;
// Geo / location fields
final double? latitude;
final double? longitude;
final String? locationName;
// Structured important info list [{title, value}, ...]
final List<Map<String, String>> importantInfo;
// Review stats (populated when backend includes them)
final double? averageRating;
final int? reviewCount;
EventModel({
required this.id,
required this.name,
this.title,
this.description,
required this.startDate,
required this.endDate,
this.startTime,
this.endTime,
this.pincode,
this.place,
this.isBookable = true,
this.eventTypeId,
this.thumbImg,
this.images = const [],
this.importantInformation,
this.venueName,
this.eventStatus,
this.cancelledReason,
this.latitude,
this.longitude,
this.locationName,
this.importantInfo = const [],
this.averageRating,
this.reviewCount,
});
/// Safely parse a double from backend (may arrive as String or num)
static double? _parseDouble(dynamic raw) {
if (raw == null) return null;
if (raw is num) return raw.toDouble();
if (raw is String) return double.tryParse(raw);
return null;
}
/// Safely parse important_info from backend (list of {title, value} maps)
static List<Map<String, String>> _parseImportantInfo(dynamic raw) {
if (raw is List) {
return raw.map<Map<String, String>>((e) {
if (e is Map) {
return {
'title': (e['title'] ?? '').toString(),
'value': (e['value'] ?? '').toString(),
};
}
return {'title': '', 'value': e.toString()};
}).toList();
}
return [];
}
factory EventModel.fromJson(Map<String, dynamic> j) {
final imgs = <EventImageModel>[];
if (j['images'] is List) {
for (final im in j['images']) {
if (im is Map<String, dynamic>) imgs.add(EventImageModel.fromJson(im));
}
}
return EventModel(
id: j['id'] is int ? j['id'] as int : int.parse(j['id'].toString()),
name: (j['name'] ?? '') as String,
title: j['title'] as String?,
description: j['description'] as String?,
startDate: (j['start_date'] ?? '') as String,
endDate: (j['end_date'] ?? '') as String,
startTime: j['start_time'] as String?,
endTime: j['end_time'] as String?,
pincode: j['pincode'] as String?,
place: (j['place'] ?? j['venue_name']) as String?,
isBookable: j['is_bookable'] == null ? true : (j['is_bookable'] == true || j['is_bookable'].toString().toLowerCase() == 'true'),
eventTypeId: j['event_type'] is int ? j['event_type'] as int : (j['event_type'] != null ? int.tryParse(j['event_type'].toString()) : null),
thumbImg: EventTypeModel._resolveMediaUrl(j['thumb_img'] as String?),
images: imgs,
importantInformation: j['important_information'] as String?,
venueName: j['venue_name'] as String?,
eventStatus: j['event_status'] as String?,
cancelledReason: j['cancelled_reason'] as String?,
latitude: _parseDouble(j['latitude']),
longitude: _parseDouble(j['longitude']),
locationName: j['location_name'] as String?,
importantInfo: _parseImportantInfo(j['important_info']),
averageRating: (j['average_rating'] as num?)?.toDouble(),
reviewCount: (j['review_count'] as num?)?.toInt(),
);
}
}