2024-08-19 19:36:01 +08:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_markdown/flutter_markdown.dart';
|
2024-08-21 00:48:51 +08:00
|
|
|
import 'package:flutter_svg/svg.dart';
|
2024-08-19 19:36:01 +08:00
|
|
|
import 'package:get/get.dart';
|
|
|
|
import 'package:solian/providers/link_expander.dart';
|
2024-09-10 21:53:05 +08:00
|
|
|
import 'package:solian/widgets/auto_cache_image.dart';
|
2024-08-19 19:36:01 +08:00
|
|
|
import 'package:url_launcher/url_launcher_string.dart';
|
|
|
|
|
|
|
|
class LinkExpansion extends StatelessWidget {
|
|
|
|
final String content;
|
|
|
|
|
|
|
|
const LinkExpansion({super.key, required this.content});
|
|
|
|
|
2024-08-19 19:56:44 +08:00
|
|
|
Widget _buildImage(String url, {double? width, double? height}) {
|
2024-08-21 00:48:51 +08:00
|
|
|
if (url.endsWith('svg')) {
|
|
|
|
return SvgPicture.network(url, width: width, height: height);
|
|
|
|
}
|
2024-09-10 21:53:05 +08:00
|
|
|
return AutoCacheImage(
|
|
|
|
url,
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
);
|
2024-08-19 19:56:44 +08:00
|
|
|
}
|
|
|
|
|
2024-08-19 19:36:01 +08:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final linkRegex = RegExp(
|
2024-08-21 10:06:05 +08:00
|
|
|
r'(?<!\()(?:(?:https?):\/\/|www\.)(?:[-_a-z0-9]+\.)*(?:[-a-z0-9]+\.[-a-z0-9]+)[^\s<]*[^\s<?!.,:*_~]',
|
2024-08-19 19:36:01 +08:00
|
|
|
);
|
|
|
|
final matches = linkRegex.allMatches(content);
|
|
|
|
if (matches.isEmpty) {
|
2024-09-07 17:48:07 +08:00
|
|
|
return const SizedBox.shrink();
|
2024-08-19 19:36:01 +08:00
|
|
|
}
|
|
|
|
|
2024-09-02 23:11:40 +08:00
|
|
|
final LinkExpandProvider expandController = Get.find();
|
2024-08-19 19:36:01 +08:00
|
|
|
|
2024-08-19 20:13:08 +08:00
|
|
|
return Wrap(
|
2024-08-19 19:36:01 +08:00
|
|
|
children: matches.map((x) {
|
2024-08-19 20:13:08 +08:00
|
|
|
return Container(
|
|
|
|
constraints: BoxConstraints(
|
|
|
|
maxWidth: matches.length == 1 ? 480 : 340,
|
|
|
|
),
|
|
|
|
child: FutureBuilder(
|
|
|
|
future: expandController.expandLink(x.group(0)!),
|
|
|
|
builder: (context, snapshot) {
|
|
|
|
if (!snapshot.hasData) {
|
2024-09-07 17:48:07 +08:00
|
|
|
return const SizedBox.shrink();
|
2024-08-19 20:13:08 +08:00
|
|
|
}
|
2024-08-19 19:36:01 +08:00
|
|
|
|
2024-08-19 20:13:08 +08:00
|
|
|
final isRichDescription = [
|
2024-08-21 00:48:51 +08:00
|
|
|
'solsynth.dev',
|
2024-08-19 20:13:08 +08:00
|
|
|
].contains(Uri.parse(snapshot.data!.url).host);
|
2024-08-19 19:36:01 +08:00
|
|
|
|
2024-08-19 20:13:08 +08:00
|
|
|
return GestureDetector(
|
|
|
|
child: Card(
|
|
|
|
child: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
children: [
|
|
|
|
if ([
|
|
|
|
(snapshot.data!.icon?.isNotEmpty ?? false),
|
|
|
|
snapshot.data!.siteName != null
|
|
|
|
].any((x) => x))
|
|
|
|
Row(
|
|
|
|
children: [
|
|
|
|
if (snapshot.data!.icon?.isNotEmpty ?? false)
|
|
|
|
ClipRRect(
|
|
|
|
borderRadius: const BorderRadius.all(
|
|
|
|
Radius.circular(8),
|
|
|
|
),
|
|
|
|
child: _buildImage(
|
|
|
|
snapshot.data!.icon!,
|
|
|
|
width: 32,
|
|
|
|
height: 32,
|
|
|
|
),
|
|
|
|
).paddingOnly(right: 8),
|
|
|
|
if (snapshot.data!.siteName != null)
|
2024-09-21 22:10:59 +08:00
|
|
|
Expanded(
|
|
|
|
child: Text(
|
|
|
|
snapshot.data!.siteName!,
|
|
|
|
style: Theme.of(context).textTheme.labelLarge,
|
|
|
|
maxLines: 1,
|
|
|
|
overflow: TextOverflow.ellipsis,
|
|
|
|
),
|
2024-08-19 19:36:01 +08:00
|
|
|
),
|
2024-08-19 20:13:08 +08:00
|
|
|
],
|
|
|
|
).paddingOnly(
|
|
|
|
bottom: (snapshot.data!.icon?.isNotEmpty ?? false)
|
|
|
|
? 8
|
|
|
|
: 4,
|
2024-08-19 19:36:01 +08:00
|
|
|
),
|
2024-08-19 20:13:08 +08:00
|
|
|
if (snapshot.data!.image != null &&
|
|
|
|
(snapshot.data!.image?.startsWith('http') ?? false))
|
|
|
|
ClipRRect(
|
|
|
|
borderRadius: const BorderRadius.all(
|
|
|
|
Radius.circular(8),
|
|
|
|
),
|
|
|
|
child: _buildImage(
|
|
|
|
snapshot.data!.image!,
|
|
|
|
),
|
|
|
|
).paddingOnly(bottom: 8),
|
2024-08-19 19:36:01 +08:00
|
|
|
Text(
|
2024-08-19 20:13:08 +08:00
|
|
|
snapshot.data!.title ?? 'No Title',
|
|
|
|
maxLines: 1,
|
|
|
|
overflow: TextOverflow.fade,
|
|
|
|
style: Theme.of(context).textTheme.bodyLarge,
|
2024-08-19 19:36:01 +08:00
|
|
|
),
|
2024-08-19 20:13:08 +08:00
|
|
|
if (snapshot.data!.description != null &&
|
|
|
|
isRichDescription)
|
|
|
|
MarkdownBody(data: snapshot.data!.description!)
|
|
|
|
else if (snapshot.data!.description != null)
|
|
|
|
Text(
|
|
|
|
snapshot.data!.description!,
|
|
|
|
maxLines: 3,
|
|
|
|
overflow: TextOverflow.ellipsis,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
).paddingAll(12),
|
|
|
|
),
|
|
|
|
onTap: () {
|
|
|
|
launchUrlString(x.group(0)!);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
2024-08-19 19:36:01 +08:00
|
|
|
);
|
|
|
|
}).toList(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|