feat: category / hashtag pages

- with links form stream info
closes #12
This commit is contained in:
2025-05-15 14:12:37 +01:00
parent 54a61322cf
commit 52953a4c16
16 changed files with 483 additions and 26 deletions

BIN
assets/category/art.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 660 KiB

BIN
assets/category/gaming.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 333 KiB

BIN
assets/category/irl.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 521 KiB

BIN
assets/category/music.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 300 KiB

BIN
assets/category/talk.jpeg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 402 KiB

View File

@ -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
View 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
View 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);
}
},
),
],
),
),
);
}
}

View File

@ -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..
], ],
), ),
), ),

View File

@ -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');
} }
} }

View 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(),
);
},
);
},
),
),
],
);
}
}

View 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),
),
);
}
}

View File

@ -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;
} }
} }

View File

@ -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);

View File

@ -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),
], ],
), ),

View File

@ -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/"