import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/material.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:gap/gap.dart'; import 'package:google_fonts/google_fonts.dart'; import 'package:marquee/marquee.dart'; import 'package:provider/provider.dart'; import 'package:responsive_framework/responsive_framework.dart'; import 'package:styled_widget/styled_widget.dart'; import 'package:surface/types/link.dart'; import 'package:surface/widgets/universal_image.dart'; import 'package:url_launcher/url_launcher_string.dart'; import '../providers/link_preview.dart'; class LinkPreviewWidget extends StatefulWidget { final String text; const LinkPreviewWidget({super.key, required this.text}); @override State createState() => _LinkPreviewWidgetState(); } class _LinkPreviewWidgetState extends State { final List _links = List.empty(growable: true); Future _getLinkMeta() async { final linkRegex = RegExp(r'https?:\/\/[^\s/$.?#].[^\s]*'); final links = linkRegex.allMatches(widget.text).map((e) => e.group(0)).toSet(); final lp = context.read(); final List> futures = links.where((e) => e != null).map((e) => lp.getLinkMeta(e!)).toList(); final results = await Future.wait(futures); _links.addAll(results.where((e) => e != null).map((e) => e!).toList()); if (_links.isNotEmpty && mounted) setState(() {}); } @override void initState() { super.initState(); _getLinkMeta(); } @override Widget build(BuildContext context) { if (_links.isEmpty) return const SizedBox.shrink(); return Wrap( spacing: 8, runSpacing: 8, children: _links.map((e) => _LinkPreviewEntry(meta: e)).toList(), ); } } class _LinkPreviewEntry extends StatelessWidget { final SnLinkMeta meta; const _LinkPreviewEntry({ super.key, required this.meta, }); @override Widget build(BuildContext context) { return Container( constraints: BoxConstraints( maxWidth: ResponsiveBreakpoints.of(context).smallerOrEqualTo(MOBILE) ? double.infinity : 480, ), child: GestureDetector( child: Card( child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ if (meta.image != null) Container( margin: const EdgeInsets.only(bottom: 4), color: Theme.of(context).colorScheme.surfaceContainer, child: AspectRatio( aspectRatio: 16 / 9, child: ClipRRect( child: AutoResizeUniversalImage( meta.image!, fit: BoxFit.contain, ), ), ), ), SizedBox( height: 48, child: Row( crossAxisAlignment: CrossAxisAlignment.center, children: [ if (meta.icon?.isNotEmpty ?? false) StyledWidget( meta.icon!.endsWith('.svg') ? SvgPicture.network(meta.icon!) : UniversalImage( meta.icon!, width: 36, height: 36, cacheHeight: 36, cacheWidth: 36, ), ).padding(all: 4, right: 16), Expanded( child: Column( crossAxisAlignment: CrossAxisAlignment.start, mainAxisAlignment: MainAxisAlignment.center, children: [ SizedBox( height: 24, child: ((meta.title?.length ?? 0) > 40) ? Marquee( text: meta.title ?? 'unknown'.tr(), style: TextStyle(fontSize: 17, height: 1), scrollAxis: Axis.horizontal, showFadingOnlyWhenScrolling: true, pauseAfterRound: const Duration(seconds: 3), ) : Text( meta.title ?? 'unknown'.tr(), style: TextStyle(fontSize: 17, height: 1), maxLines: 1, overflow: TextOverflow.ellipsis, ), ), if (meta.siteName != null) Text( meta.siteName!, style: TextStyle(fontSize: 13, height: 0.9), ).fontSize(11), ], ), ), const Gap(6), ], ).padding(horizontal: 16), ), if (meta.description != null) Text( meta.description!, maxLines: 3, overflow: TextOverflow.ellipsis, ).padding(horizontal: 16, bottom: 8), Text( meta.url, style: GoogleFonts.roboto(fontSize: 11, height: 0.9), maxLines: 1, overflow: TextOverflow.ellipsis, ).opacity(0.75).padding(horizontal: 16), const Gap(4), Text( 'poweredBy'.tr(args: ['HyperNet.Reader']), style: GoogleFonts.roboto(fontSize: 11, height: 0.9), ).opacity(0.75).padding(horizontal: 16), const Gap(16), ], ), ), onTap: () { launchUrlString(meta.url, mode: LaunchMode.externalApplication); }, ), ); } }