mirror of
https://github.com/nostrlabs-io/zap-stream-flutter.git
synced 2025-06-15 19:48:23 +00:00
feat: category / hashtag pages
- with links form stream info closes #12
This commit is contained in:
BIN
assets/category/art.jpeg
Normal file
BIN
assets/category/art.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 660 KiB |
BIN
assets/category/gaming.jpeg
Normal file
BIN
assets/category/gaming.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 333 KiB |
BIN
assets/category/irl.jpeg
Normal file
BIN
assets/category/irl.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 521 KiB |
BIN
assets/category/music.jpeg
Normal file
BIN
assets/category/music.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 300 KiB |
BIN
assets/category/talk.jpeg
Normal file
BIN
assets/category/talk.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 402 KiB |
@ -6,6 +6,8 @@ import 'package:ndk/ndk.dart';
|
|||||||
import 'package:ndk_amber/ndk_amber.dart';
|
import 'package:ndk_amber/ndk_amber.dart';
|
||||||
import 'package:ndk_objectbox/ndk_objectbox.dart';
|
import 'package:ndk_objectbox/ndk_objectbox.dart';
|
||||||
import 'package:ndk_rust_verifier/ndk_rust_verifier.dart';
|
import 'package:ndk_rust_verifier/ndk_rust_verifier.dart';
|
||||||
|
import 'package:zap_stream_flutter/pages/category.dart';
|
||||||
|
import 'package:zap_stream_flutter/pages/hashtag.dart';
|
||||||
import 'package:zap_stream_flutter/pages/login.dart';
|
import 'package:zap_stream_flutter/pages/login.dart';
|
||||||
import 'package:zap_stream_flutter/pages/login_input.dart';
|
import 'package:zap_stream_flutter/pages/login_input.dart';
|
||||||
import 'package:zap_stream_flutter/pages/new_account.dart';
|
import 'package:zap_stream_flutter/pages/new_account.dart';
|
||||||
@ -42,6 +44,7 @@ const defaultRelays = [
|
|||||||
"wss://relay.damus.io",
|
"wss://relay.damus.io",
|
||||||
"wss://relay.primal.net",
|
"wss://relay.primal.net",
|
||||||
"wss://relay.snort.social",
|
"wss://relay.snort.social",
|
||||||
|
"wss://relay.fountain.fm",
|
||||||
];
|
];
|
||||||
const searchRelays = ["wss://relay.nostr.band", "wss://search.nos.today"];
|
const searchRelays = ["wss://relay.nostr.band", "wss://search.nos.today"];
|
||||||
|
|
||||||
@ -150,6 +153,21 @@ Future<void> main() async {
|
|||||||
return ProfilePage(pubkey: state.pathParameters["id"]!);
|
return ProfilePage(pubkey: state.pathParameters["id"]!);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
GoRoute(
|
||||||
|
path: "/t/:id",
|
||||||
|
builder: (context, state) {
|
||||||
|
return HashtagPage(tag: state.pathParameters["id"]!);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
GoRoute(
|
||||||
|
path: "/category/:id",
|
||||||
|
builder: (context, state) {
|
||||||
|
return CategoryPage(
|
||||||
|
category: state.pathParameters["id"]!,
|
||||||
|
info: state.extra as GameInfo?,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
110
lib/pages/category.dart
Normal file
110
lib/pages/category.dart
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/widgets.dart';
|
||||||
|
import 'package:ndk/ndk.dart';
|
||||||
|
import 'package:zap_stream_flutter/imgproxy.dart';
|
||||||
|
import 'package:zap_stream_flutter/rx_filter.dart';
|
||||||
|
import 'package:zap_stream_flutter/theme.dart';
|
||||||
|
import 'package:zap_stream_flutter/utils.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/category_top_zapped.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/header.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/pill.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/stream_grid.dart';
|
||||||
|
|
||||||
|
class CategoryPage extends StatelessWidget {
|
||||||
|
final String category;
|
||||||
|
final GameInfo? info;
|
||||||
|
|
||||||
|
const CategoryPage({super.key, required this.category, required this.info});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return SingleChildScrollView(
|
||||||
|
child: Container(
|
||||||
|
margin: EdgeInsets.all(5.0),
|
||||||
|
child: Column(
|
||||||
|
spacing: 16,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
HeaderWidget(),
|
||||||
|
Row(
|
||||||
|
spacing: 8,
|
||||||
|
children: [
|
||||||
|
if (info?.coverImage != null)
|
||||||
|
Container(
|
||||||
|
clipBehavior: Clip.antiAlias,
|
||||||
|
constraints: BoxConstraints(maxHeight: 200),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
borderRadius: DEFAULT_BR,
|
||||||
|
color: LAYER_1,
|
||||||
|
),
|
||||||
|
child:
|
||||||
|
info!.coverImage!.startsWith("assets/")
|
||||||
|
? Image.asset(
|
||||||
|
info!.coverImage!,
|
||||||
|
fit: BoxFit.contain,
|
||||||
|
)
|
||||||
|
: CachedNetworkImage(
|
||||||
|
imageUrl: proxyImg(context, info!.coverImage!),
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
spacing: 8,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
info?.name ?? category,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 24,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (info?.genres != null)
|
||||||
|
Row(
|
||||||
|
spacing: 4,
|
||||||
|
children:
|
||||||
|
info!.genres
|
||||||
|
.map(
|
||||||
|
(g) => PillWidget(
|
||||||
|
color: LAYER_1,
|
||||||
|
child: Text(
|
||||||
|
g,
|
||||||
|
style: TextStyle(
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(),
|
||||||
|
),
|
||||||
|
CategoryTopZapped(tag: category),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
RxFilter<Nip01Event>(
|
||||||
|
Key("category-page:$category"),
|
||||||
|
filters: [
|
||||||
|
Filter(
|
||||||
|
kinds: [30_311],
|
||||||
|
limit: 100,
|
||||||
|
tTags: [category.toLowerCase()],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
builder: (ctx, state) {
|
||||||
|
if (state == null) {
|
||||||
|
return SizedBox.shrink();
|
||||||
|
} else {
|
||||||
|
return StreamGrid(events: state);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
45
lib/pages/hashtag.dart
Normal file
45
lib/pages/hashtag.dart
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:ndk/ndk.dart';
|
||||||
|
import 'package:zap_stream_flutter/rx_filter.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/category_top_zapped.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/header.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/stream_grid.dart';
|
||||||
|
|
||||||
|
class HashtagPage extends StatelessWidget {
|
||||||
|
final String tag;
|
||||||
|
|
||||||
|
const HashtagPage({super.key, required this.tag});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return SingleChildScrollView(
|
||||||
|
child: Container(
|
||||||
|
margin: EdgeInsets.all(5.0),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
HeaderWidget(),
|
||||||
|
Text(
|
||||||
|
"#$tag",
|
||||||
|
style: TextStyle(fontSize: 30, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
CategoryTopZapped(tag: tag),
|
||||||
|
RxFilter<Nip01Event>(
|
||||||
|
Key("tags-page:$tag"),
|
||||||
|
filters: [
|
||||||
|
Filter(kinds: [30_311], limit: 50, tTags: [tag.toLowerCase()]),
|
||||||
|
],
|
||||||
|
builder: (ctx, state) {
|
||||||
|
if (state == null) {
|
||||||
|
return SizedBox.shrink();
|
||||||
|
} else {
|
||||||
|
return StreamGrid(events: state);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -13,6 +13,7 @@ class HomePage extends StatelessWidget {
|
|||||||
child: Container(
|
child: Container(
|
||||||
margin: EdgeInsets.all(5.0),
|
margin: EdgeInsets.all(5.0),
|
||||||
child: Column(
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
HeaderWidget(),
|
HeaderWidget(),
|
||||||
RxFilter<Nip01Event>(
|
RxFilter<Nip01Event>(
|
||||||
@ -28,7 +29,6 @@ class HomePage extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
//other stuff..
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
146
lib/utils.dart
146
lib/utils.dart
@ -3,6 +3,7 @@ import 'dart:convert';
|
|||||||
import 'package:bech32/bech32.dart';
|
import 'package:bech32/bech32.dart';
|
||||||
import 'package:collection/collection.dart';
|
import 'package:collection/collection.dart';
|
||||||
import 'package:convert/convert.dart';
|
import 'package:convert/convert.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:ndk/ndk.dart';
|
import 'package:ndk/ndk.dart';
|
||||||
import 'package:ndk/shared/nips/nip19/nip19.dart';
|
import 'package:ndk/shared/nips/nip19/nip19.dart';
|
||||||
@ -88,16 +89,16 @@ class StreamInfo {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class GameInfo {
|
class GameInfo {
|
||||||
String id;
|
final String id;
|
||||||
String name;
|
final String name;
|
||||||
List<String> genres;
|
final List<String> genres;
|
||||||
String className;
|
final String? coverImage;
|
||||||
|
|
||||||
GameInfo({
|
const GameInfo({
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.name,
|
required this.name,
|
||||||
required this.genres,
|
required this.genres,
|
||||||
required this.className,
|
required this.coverImage,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -196,32 +197,32 @@ StreamInfo extractStreamInfo(Nip01Event ev) {
|
|||||||
({GameInfo? gameInfo, String? gameId}) extractGameTag(List<String> tags) {
|
({GameInfo? gameInfo, String? gameId}) extractGameTag(List<String> tags) {
|
||||||
final gameId = tags.firstWhereOrNull((a) => gameTagFormat.hasMatch(a));
|
final gameId = tags.firstWhereOrNull((a) => gameTagFormat.hasMatch(a));
|
||||||
|
|
||||||
final internalGame = AllCategories.firstWhereOrNull(
|
final internalGame = allCategories.firstWhereOrNull(
|
||||||
(a) => gameId == 'internal:${a.id}',
|
(a) => gameId == 'internal:${a.id}',
|
||||||
);
|
);
|
||||||
if (internalGame != null) {
|
if (internalGame != null) {
|
||||||
return (
|
return (
|
||||||
gameInfo: GameInfo(
|
gameInfo: GameInfo(
|
||||||
id: internalGame.id,
|
id: "internal:${internalGame.id}",
|
||||||
name: internalGame.name,
|
name: internalGame.name,
|
||||||
genres: internalGame.tags,
|
genres: internalGame.tags,
|
||||||
className: internalGame.className,
|
coverImage: internalGame.coverImage,
|
||||||
),
|
),
|
||||||
gameId: gameId,
|
gameId: gameId,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
final lowerTags = tags.map((t) => t.toLowerCase());
|
final lowerTags = tags.map((t) => t.toLowerCase());
|
||||||
final taggedCategory = AllCategories.firstWhereOrNull(
|
final taggedCategory = allCategories.firstWhereOrNull(
|
||||||
(a) => a.tags.any(lowerTags.contains),
|
(a) => a.tags.any(lowerTags.contains),
|
||||||
);
|
);
|
||||||
if (taggedCategory != null) {
|
if (taggedCategory != null) {
|
||||||
return (
|
return (
|
||||||
gameInfo: GameInfo(
|
gameInfo: GameInfo(
|
||||||
id: taggedCategory.id,
|
id: "internal:${taggedCategory.id}",
|
||||||
name: taggedCategory.name,
|
name: taggedCategory.name,
|
||||||
genres: taggedCategory.tags,
|
genres: taggedCategory.tags,
|
||||||
className: taggedCategory.className,
|
coverImage: taggedCategory.coverImage,
|
||||||
),
|
),
|
||||||
gameId: gameId,
|
gameId: gameId,
|
||||||
);
|
);
|
||||||
@ -237,23 +238,83 @@ String getHost(Nip01Event ev) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class Category {
|
class Category {
|
||||||
String id;
|
final String id;
|
||||||
String name;
|
final String name;
|
||||||
List<String> tags;
|
final IconData icon;
|
||||||
String className;
|
final List<String> tags;
|
||||||
|
final int order;
|
||||||
|
final String? coverImage;
|
||||||
|
|
||||||
Category({
|
const Category({
|
||||||
required this.id,
|
required this.id,
|
||||||
required this.name,
|
required this.name,
|
||||||
|
required this.icon,
|
||||||
required this.tags,
|
required this.tags,
|
||||||
required this.className,
|
required this.order,
|
||||||
|
this.coverImage,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Category> AllCategories = []; // Implement as needed
|
const List<Category> allCategories = [
|
||||||
|
Category(
|
||||||
|
id: "irl",
|
||||||
|
name: "IRL",
|
||||||
|
icon: Icons.face,
|
||||||
|
tags: ["irl"],
|
||||||
|
order: 0,
|
||||||
|
coverImage: "assets/category/irl.jpeg",
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "gaming",
|
||||||
|
name: "Gaming",
|
||||||
|
icon: Icons.gamepad,
|
||||||
|
tags: ["gaming"],
|
||||||
|
order: 0,
|
||||||
|
coverImage: "assets/category/gaming.jpeg",
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "music",
|
||||||
|
name: "Music",
|
||||||
|
icon: Icons.note,
|
||||||
|
tags: ["music", "raido"],
|
||||||
|
order: 0,
|
||||||
|
coverImage: "assets/category/music.jpeg",
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "talk",
|
||||||
|
name: "Talk",
|
||||||
|
icon: Icons.mic,
|
||||||
|
tags: ["talk", "just-chatting"],
|
||||||
|
order: 0,
|
||||||
|
coverImage: "assets/category/talk.jpeg",
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "art",
|
||||||
|
name: "Art",
|
||||||
|
icon: Icons.brush,
|
||||||
|
tags: ["art"],
|
||||||
|
order: 0,
|
||||||
|
coverImage: "assets/category/art.jpeg",
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "gambling",
|
||||||
|
name: "Gambling",
|
||||||
|
icon: Icons.casino,
|
||||||
|
tags: ["gambling", "casino", "slots"],
|
||||||
|
order: 1,
|
||||||
|
),
|
||||||
|
Category(
|
||||||
|
id: "science-and-technology",
|
||||||
|
name: "Science & Technology",
|
||||||
|
icon: Icons.casino,
|
||||||
|
tags: ["science", "tech", "technology"],
|
||||||
|
order: 1,
|
||||||
|
),
|
||||||
|
];
|
||||||
|
|
||||||
String formatSats(int n) {
|
String formatSats(int n, {int? maxDigits}) {
|
||||||
final fmt = NumberFormat();
|
final fmt = NumberFormat();
|
||||||
|
fmt.maximumFractionDigits = maxDigits ?? 2;
|
||||||
if (n >= 1000000) {
|
if (n >= 1000000) {
|
||||||
return "${fmt.format(n / 1000000)}M";
|
return "${fmt.format(n / 1000000)}M";
|
||||||
} else if (n >= 1500) {
|
} else if (n >= 1500) {
|
||||||
@ -270,6 +331,49 @@ String zapSum(List<Nip01Event> zaps) {
|
|||||||
return formatSats(total);
|
return formatSats(total);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class TopZaps {
|
||||||
|
final int sum;
|
||||||
|
final List<ZapReceipt> zaps;
|
||||||
|
|
||||||
|
const TopZaps({required this.sum, required this.zaps});
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, TopZaps> topZapSender(Iterable<ZapReceipt> zaps) {
|
||||||
|
return Map.fromEntries(
|
||||||
|
zaps
|
||||||
|
.where((e) => e.sender != null)
|
||||||
|
.groupListsBy((v) => v.sender!)
|
||||||
|
.entries
|
||||||
|
.map(
|
||||||
|
(v) => MapEntry(
|
||||||
|
v.key,
|
||||||
|
TopZaps(
|
||||||
|
sum: v.value.fold(0, (acc, v) => acc + (v.amountSats ?? 0)),
|
||||||
|
zaps: v.value,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<String, TopZaps> topZapReceiver(Iterable<ZapReceipt> zaps) {
|
||||||
|
return Map.fromEntries(
|
||||||
|
zaps
|
||||||
|
.where((e) => e.recipient != null)
|
||||||
|
.groupListsBy((v) => v.recipient!)
|
||||||
|
.entries
|
||||||
|
.map(
|
||||||
|
(v) => MapEntry(
|
||||||
|
v.key,
|
||||||
|
TopZaps(
|
||||||
|
sum: v.value.fold(0, (acc, v) => acc + (v.amountSats ?? 0)),
|
||||||
|
zaps: v.value,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
String bech32ToHex(String bech32) {
|
String bech32ToHex(String bech32) {
|
||||||
final decoder = Bech32Decoder();
|
final decoder = Bech32Decoder();
|
||||||
final data = decoder.convert(bech32, 10_000);
|
final data = decoder.convert(bech32, 10_000);
|
||||||
@ -365,4 +469,4 @@ String encodeBech32TLV(String hrp, List<TLV> tlvs) {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
throw FormatException('Failed to encode Bech32 or TLV: $e');
|
throw FormatException('Failed to encode Bech32 or TLV: $e');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
109
lib/widgets/category_top_zapped.dart
Normal file
109
lib/widgets/category_top_zapped.dart
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
import 'package:collection/collection.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:ndk/ndk.dart';
|
||||||
|
import 'package:zap_stream_flutter/main.dart';
|
||||||
|
import 'package:zap_stream_flutter/rx_filter.dart';
|
||||||
|
import 'package:zap_stream_flutter/theme.dart';
|
||||||
|
import 'package:zap_stream_flutter/utils.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/avatar.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/profile.dart';
|
||||||
|
|
||||||
|
class CategoryTopZapped extends StatelessWidget {
|
||||||
|
final String tag;
|
||||||
|
final int? limit;
|
||||||
|
|
||||||
|
const CategoryTopZapped({super.key, required this.tag, this.limit});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
spacing: 8,
|
||||||
|
children: [
|
||||||
|
RichText(
|
||||||
|
text: TextSpan(
|
||||||
|
children: [
|
||||||
|
WidgetSpan(
|
||||||
|
child: Icon(Icons.bolt, color: ZAP_1),
|
||||||
|
alignment: PlaceholderAlignment.middle,
|
||||||
|
),
|
||||||
|
TextSpan(
|
||||||
|
text: " Most Zapped Streamers",
|
||||||
|
style: TextStyle(color: LAYER_4, fontWeight: FontWeight.w500),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
SingleChildScrollView(
|
||||||
|
scrollDirection: Axis.horizontal,
|
||||||
|
primary: false,
|
||||||
|
child: FutureBuilder(
|
||||||
|
future:
|
||||||
|
ndk.requests
|
||||||
|
.query(
|
||||||
|
filters: [
|
||||||
|
Filter(
|
||||||
|
kinds: [30_311],
|
||||||
|
limit: 100,
|
||||||
|
tTags: [tag.toLowerCase()],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.future,
|
||||||
|
builder: (context, state) {
|
||||||
|
final aTags =
|
||||||
|
(state.data ?? [])
|
||||||
|
.map((e) => "30311:${e.pubKey}:${e.getDtag()}")
|
||||||
|
.toList();
|
||||||
|
return RxFilter<Nip01Event>(
|
||||||
|
Key("top-zapped:$tag:${aTags.length}"),
|
||||||
|
filters: [
|
||||||
|
Filter(kinds: [9735], aTags: aTags),
|
||||||
|
],
|
||||||
|
builder: (context, zaps) {
|
||||||
|
final parsedZaps =
|
||||||
|
zaps?.map((e) => ZapReceipt.fromEvent(e)) ?? [];
|
||||||
|
final topZapped = topZapReceiver(parsedZaps).entries
|
||||||
|
.sortedBy((e) => e.value.sum)
|
||||||
|
.reversed
|
||||||
|
.take(limit ?? 5);
|
||||||
|
|
||||||
|
return Row(
|
||||||
|
spacing: 16,
|
||||||
|
children:
|
||||||
|
topZapped
|
||||||
|
.map(
|
||||||
|
(e) => Row(
|
||||||
|
spacing: 8,
|
||||||
|
children: [
|
||||||
|
AvatarWidget.pubkey(e.key),
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment:
|
||||||
|
CrossAxisAlignment.start,
|
||||||
|
spacing: 2,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
formatSats(e.value.sum, maxDigits: 0),
|
||||||
|
style: TextStyle(
|
||||||
|
color: ZAP_1,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ProfileNameWidget.pubkey(e.key),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
.toList(),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
25
lib/widgets/game_info.dart
Normal file
25
lib/widgets/game_info.dart
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import 'package:flutter/widgets.dart';
|
||||||
|
import 'package:go_router/go_router.dart';
|
||||||
|
import 'package:zap_stream_flutter/theme.dart';
|
||||||
|
import 'package:zap_stream_flutter/utils.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/pill.dart';
|
||||||
|
|
||||||
|
class GameInfoWidget extends StatelessWidget {
|
||||||
|
final GameInfo info;
|
||||||
|
|
||||||
|
const GameInfoWidget({super.key, required this.info});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return PillWidget(
|
||||||
|
color: LAYER_2,
|
||||||
|
onTap: () {
|
||||||
|
context.push("/category/${Uri.encodeComponent(info.id)}", extra: info);
|
||||||
|
},
|
||||||
|
child: Text(
|
||||||
|
info.name,
|
||||||
|
style: TextStyle(color: PRIMARY_1, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -3,12 +3,18 @@ import 'package:flutter/material.dart';
|
|||||||
class PillWidget extends StatelessWidget {
|
class PillWidget extends StatelessWidget {
|
||||||
final Widget child;
|
final Widget child;
|
||||||
final Color? color;
|
final Color? color;
|
||||||
|
final void Function()? onTap;
|
||||||
|
|
||||||
const PillWidget({super.key, required this.child, required this.color});
|
const PillWidget({
|
||||||
|
super.key,
|
||||||
|
required this.child,
|
||||||
|
required this.color,
|
||||||
|
this.onTap,
|
||||||
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Container(
|
final inner = Container(
|
||||||
padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4),
|
padding: EdgeInsets.symmetric(horizontal: 8, vertical: 4),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
borderRadius: BorderRadius.all(Radius.circular(8)),
|
borderRadius: BorderRadius.all(Radius.circular(8)),
|
||||||
@ -16,5 +22,6 @@ class PillWidget extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
child: child,
|
child: child,
|
||||||
);
|
);
|
||||||
|
return onTap != null ? GestureDetector(onTap: onTap, child: inner) : inner;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -25,6 +25,11 @@ class StreamGrid extends StatelessWidget {
|
|||||||
events
|
events
|
||||||
.map((e) => StreamEvent(e))
|
.map((e) => StreamEvent(e))
|
||||||
.where((e) => e.info.stream?.contains(".m3u8") ?? false)
|
.where((e) => e.info.stream?.contains(".m3u8") ?? false)
|
||||||
|
.where(
|
||||||
|
(e) =>
|
||||||
|
(e.info.starts ?? e.event.createdAt) <=
|
||||||
|
(DateTime.now().millisecondsSinceEpoch / 1000),
|
||||||
|
)
|
||||||
.sortedBy((a) => a.info.starts ?? a.event.createdAt)
|
.sortedBy((a) => a.info.starts ?? a.event.createdAt)
|
||||||
.reversed;
|
.reversed;
|
||||||
final live = streams.where((s) => s.info.status == StreamStatus.live);
|
final live = streams.where((s) => s.info.status == StreamStatus.live);
|
||||||
|
@ -1,8 +1,12 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:go_router/go_router.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:zap_stream_flutter/theme.dart';
|
import 'package:zap_stream_flutter/theme.dart';
|
||||||
import 'package:zap_stream_flutter/utils.dart';
|
import 'package:zap_stream_flutter/utils.dart';
|
||||||
import 'package:zap_stream_flutter/widgets/button_follow.dart';
|
import 'package:zap_stream_flutter/widgets/button_follow.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/game_info.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/nostr_text.dart';
|
||||||
|
import 'package:zap_stream_flutter/widgets/pill.dart';
|
||||||
import 'package:zap_stream_flutter/widgets/profile.dart';
|
import 'package:zap_stream_flutter/widgets/profile.dart';
|
||||||
import 'package:zap_stream_flutter/widgets/stream_cards.dart';
|
import 'package:zap_stream_flutter/widgets/stream_cards.dart';
|
||||||
|
|
||||||
@ -45,7 +49,36 @@ class StreamInfoWidget extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
if (stream.info.summary?.isNotEmpty ?? false)
|
if (stream.info.summary?.isNotEmpty ?? false)
|
||||||
Text(stream.info.summary!),
|
Text.rich(
|
||||||
|
TextSpan(
|
||||||
|
children: textToSpans(
|
||||||
|
stream.info.summary!,
|
||||||
|
[],
|
||||||
|
stream.info.host,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
if (stream.info.tags.isNotEmpty || stream.info.gameInfo != null)
|
||||||
|
Row(
|
||||||
|
spacing: 2,
|
||||||
|
children: [
|
||||||
|
if (stream.info.gameInfo != null)
|
||||||
|
GameInfoWidget(info: stream.info.gameInfo!),
|
||||||
|
...stream.info.tags.map(
|
||||||
|
(t) => PillWidget(
|
||||||
|
color: LAYER_2,
|
||||||
|
onTap: () {
|
||||||
|
context.push("/t/${Uri.encodeComponent(t)}");
|
||||||
|
},
|
||||||
|
child: Text(
|
||||||
|
t,
|
||||||
|
style: TextStyle(fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
StreamCardsWidget(stream: stream),
|
StreamCardsWidget(stream: stream),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
@ -60,4 +60,5 @@ flutter:
|
|||||||
uses-material-design: true
|
uses-material-design: true
|
||||||
assets:
|
assets:
|
||||||
- "assets/svg/"
|
- "assets/svg/"
|
||||||
- "assets/logo.png"
|
- "assets/logo.png"
|
||||||
|
- "assets/category/"
|
Reference in New Issue
Block a user