Initialer Commit: Projekt Start

This commit is contained in:
Tim Leikauf
2026-01-03 15:24:36 +01:00
commit 3773f94303
168 changed files with 228080 additions and 0 deletions

View File

@@ -0,0 +1,34 @@
import 'package:flutter/material.dart';
class LoadingWidget extends StatelessWidget {
final String? message;
const LoadingWidget({
super.key,
this.message,
});
@override
Widget build(BuildContext context) {
return Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const CircularProgressIndicator(
valueColor: AlwaysStoppedAnimation<Color>(Color(0xFF8B6F47)),
),
if (message != null) ...[
const SizedBox(height: 16),
Text(
message!,
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
color: Colors.grey[600],
),
),
],
],
),
);
}
}

View File

@@ -0,0 +1,137 @@
import 'package:flutter/material.dart';
import 'package:cached_network_image/cached_network_image.dart';
import '../models/product.dart';
class ProductCard extends StatelessWidget {
final Product product;
final VoidCallback onTap;
const ProductCard({
super.key,
required this.product,
required this.onTap,
});
@override
Widget build(BuildContext context) {
return Card(
clipBehavior: Clip.antiAlias,
child: InkWell(
onTap: onTap,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Produktbild
Expanded(
flex: 3,
child: Stack(
children: [
if (product.imageUrl != null)
CachedNetworkImage(
imageUrl: product.imageUrl!,
width: double.infinity,
fit: BoxFit.cover,
placeholder: (context, url) => Container(
color: Colors.grey[300],
child: const Center(
child: CircularProgressIndicator(),
),
),
errorWidget: (context, url, error) => Container(
color: Colors.grey[300],
child: const Icon(
Icons.image_not_supported,
size: 48,
color: Colors.grey,
),
),
)
else
Container(
color: Colors.grey[300],
child: const Icon(
Icons.image_not_supported,
size: 48,
color: Colors.grey,
),
),
// Sale Badge
if (product.isOnSale)
Positioned(
top: 8,
right: 8,
child: Container(
padding: const EdgeInsets.symmetric(
horizontal: 8,
vertical: 4,
),
decoration: BoxDecoration(
color: Colors.red,
borderRadius: BorderRadius.circular(4),
),
child: Text(
'Sale',
style: Theme.of(context).textTheme.bodySmall?.copyWith(
color: Colors.white,
fontWeight: FontWeight.bold,
),
),
),
),
],
),
),
// Produktinfo
Expanded(
flex: 2,
child: Padding(
padding: const EdgeInsets.all(8),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Produktname
Text(
product.name,
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
fontWeight: FontWeight.bold,
),
maxLines: 2,
overflow: TextOverflow.ellipsis,
),
const Spacer(),
// Preis
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
if (product.isOnSale) ...[
Text(
'${product.regularPrice}',
style: Theme.of(context).textTheme.bodySmall?.copyWith(
decoration: TextDecoration.lineThrough,
color: Colors.grey,
),
),
],
Text(
'${product.price}',
style: Theme.of(context).textTheme.titleSmall?.copyWith(
color: const Color(0xFF8B6F47),
fontWeight: FontWeight.bold,
),
),
],
),
],
),
),
),
],
),
),
);
}
}

View File

@@ -0,0 +1,52 @@
import 'package:flutter/material.dart';
class RetryWidget extends StatelessWidget {
final String message;
final VoidCallback onRetry;
final IconData? icon;
const RetryWidget({
super.key,
required this.message,
required this.onRetry,
this.icon,
});
@override
Widget build(BuildContext context) {
return Center(
child: Padding(
padding: const EdgeInsets.all(24),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
icon ?? Icons.error_outline,
size: 64,
color: Colors.grey[400],
),
const SizedBox(height: 16),
Text(
message,
style: Theme.of(context).textTheme.titleMedium?.copyWith(
color: Colors.grey[600],
),
textAlign: TextAlign.center,
),
const SizedBox(height: 24),
ElevatedButton.icon(
onPressed: onRetry,
icon: const Icon(Icons.refresh),
label: const Text('Erneut versuchen'),
style: ElevatedButton.styleFrom(
backgroundColor: const Color(0xFF8B6F47),
foregroundColor: Colors.white,
),
),
],
),
),
);
}
}

View File

@@ -0,0 +1,219 @@
import 'package:flutter/material.dart';
import 'package:intl/intl.dart';
import 'package:cached_network_image/cached_network_image.dart';
import '../models/review.dart';
import '../services/woocommerce_service.dart';
import '../widgets/loading_widget.dart';
import '../utils/error_handler.dart';
class ReviewsWidget extends StatefulWidget {
final int productId;
const ReviewsWidget({
super.key,
required this.productId,
});
@override
State<ReviewsWidget> createState() => _ReviewsWidgetState();
}
class _ReviewsWidgetState extends State<ReviewsWidget> {
final WooCommerceService _wooCommerceService = WooCommerceService();
List<Review> _reviews = [];
ProductRating? _rating;
bool _isLoading = true;
String? _error;
@override
void initState() {
super.initState();
_loadReviews();
}
Future<void> _loadReviews() async {
setState(() {
_isLoading = true;
_error = null;
});
try {
final reviews = await _wooCommerceService.getProductReviews(widget.productId);
final rating = await _wooCommerceService.getProductRating(widget.productId);
setState(() {
_reviews = reviews;
_rating = rating;
_isLoading = false;
});
} catch (e) {
setState(() {
_error = e.toString();
_isLoading = false;
});
}
}
@override
Widget build(BuildContext context) {
if (_isLoading) {
return const LoadingWidget();
}
if (_error != null) {
return Center(
child: Column(
children: [
Text('Fehler beim Laden der Bewertungen'),
ElevatedButton(
onPressed: _loadReviews,
child: const Text('Erneut versuchen'),
),
],
),
);
}
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (_rating != null) _buildRatingSummary(_rating!),
const SizedBox(height: 16),
Text(
'Bewertungen (${_reviews.length})',
style: Theme.of(context).textTheme.titleLarge?.copyWith(
fontWeight: FontWeight.bold,
),
),
const SizedBox(height: 12),
if (_reviews.isEmpty)
const Padding(
padding: EdgeInsets.all(16.0),
child: Text('Noch keine Bewertungen'),
)
else
..._reviews.map((review) => _buildReviewCard(review)),
],
);
}
Widget _buildRatingSummary(ProductRating rating) {
return Container(
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: Colors.grey[100],
borderRadius: BorderRadius.circular(12),
),
child: Row(
children: [
Column(
children: [
Text(
rating.averageRating.toStringAsFixed(1),
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
fontWeight: FontWeight.bold,
color: const Color(0xFF8B6F47),
),
),
_buildStars(rating.averageRating),
Text(
'${rating.ratingCount} Bewertungen',
style: Theme.of(context).textTheme.bodySmall,
),
],
),
],
),
);
}
Widget _buildStars(double rating) {
return Row(
mainAxisSize: MainAxisSize.min,
children: List.generate(5, (index) {
return Icon(
index < rating.floor()
? Icons.star
: index < rating
? Icons.star_half
: Icons.star_border,
color: Colors.amber,
size: 20,
);
}),
);
}
Widget _buildReviewCard(Review review) {
final dateFormat = DateFormat('dd.MM.yyyy');
return Card(
margin: const EdgeInsets.only(bottom: 12),
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
if (review.reviewerAvatarUrl != null)
CircleAvatar(
radius: 20,
backgroundImage: CachedNetworkImageProvider(review.reviewerAvatarUrl!),
)
else
CircleAvatar(
radius: 20,
child: Text(review.reviewer[0].toUpperCase()),
),
const SizedBox(width: 12),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
review.reviewer,
style: Theme.of(context).textTheme.titleSmall?.copyWith(
fontWeight: FontWeight.bold,
),
),
Text(
dateFormat.format(review.dateCreated),
style: Theme.of(context).textTheme.bodySmall?.copyWith(
color: Colors.grey[600],
),
),
],
),
),
_buildStars(review.rating.toDouble()),
],
),
const SizedBox(height: 8),
Text(
review.review,
style: Theme.of(context).textTheme.bodyMedium,
),
if (review.verified)
Padding(
padding: const EdgeInsets.only(top: 8),
child: Row(
children: [
Icon(Icons.verified, size: 16, color: Colors.green),
const SizedBox(width: 4),
Text(
'Verifizierter Kauf',
style: Theme.of(context).textTheme.bodySmall?.copyWith(
color: Colors.green,
),
),
],
),
),
],
),
),
);
}
}