2023-05-12 17:28:36 +00:00
import ' dart:async ' ;
import ' dart:convert ' ;
import ' package:flutter/material.dart ' ;
import ' package:drifter/domain_models/domain_models.dart ' ;
2023-05-30 01:09:23 +00:00
import ' package:drifter/models/models.dart ' ;
2023-05-12 17:28:36 +00:00
import ' package:drifter/theme/app_colors.dart ' ;
import ' package:drifter/pages/home_screen/widgets/message_ok_button_widget.dart ' ;
import ' package:drifter/pages/home_screen/widgets/message_text_button_widget.dart ' ;
import ' package:drifter/pages/home_screen/widgets/message_text_form_field_widget.dart ' ;
import ' package:drifter/pages/profile_screen/widgets/message_snack_bar.dart ' ;
import ' package:nostr_tools/nostr_tools.dart ' ;
class HomeScreen extends StatefulWidget {
const HomeScreen ( { super . key } ) ;
@ override
State < HomeScreen > createState ( ) = > _HomeScreenState ( ) ;
}
class _HomeScreenState extends State < HomeScreen > {
bool _isConnected = false ;
final List < Event > _events = [ ] ;
final Map < String , Metadata > _metaDatas = { } ;
late Stream < Event > _stream ;
final _controller = StreamController < Event > ( ) ;
bool _isNotePublishing = false ;
@ override
void initState ( ) {
_initStream ( ) ;
super . initState ( ) ;
}
Future < Stream < Event > > _connectToRelay ( ) async {
final stream = await Relay . relay . connect ( ) ;
Relay . relay . on ( ( event ) {
if ( event = = RelayEvent . connect ) {
setState ( ( ) = > _isConnected = true ) ;
} else if ( event = = RelayEvent . error ) {
setState ( ( ) = > _isConnected = false ) ;
}
} ) ;
Relay . relay . sub ( [
Filter (
kinds: [ 1 ] ,
limit: 100 ,
t: [ ' nostr ' ] ,
)
] ) ;
return stream
. where ( ( message ) = > message . type = = ' EVENT ' )
. map ( ( message ) = > message . message ) ;
}
void _initStream ( ) async {
_stream = await _connectToRelay ( ) ;
_stream . listen ( ( message ) {
final event = message ;
if ( event . kind = = 1 ) {
setState ( ( ) = > _events . add ( event ) ) ;
Relay . relay . sub ( [
Filter ( kinds: [ 0 ] , authors: [ event . pubkey ] )
] ) ;
} else if ( event . kind = = 0 ) {
final metadata = Metadata . fromJson ( jsonDecode ( event . content ) ) ;
setState ( ( ) = > _metaDatas [ event . pubkey ] = metadata ) ;
}
_controller . add ( event ) ;
} ) ;
}
// The _resubscribeStream() method clears the _events and _metaData scollection after a 1 second delay
Future < void > _resubscribeStream ( ) async {
await Future . delayed ( const Duration ( seconds: 1 ) , ( ) {
setState ( ( ) {
_events . clear ( ) ;
_metaDatas . clear ( ) ;
} ) ;
// _initStream() method, responsible for initializing and subscribing to the stream, to reconnect and re-subscribe to the filter.
_initStream ( ) ;
} ) ;
}
@ override
Widget build ( BuildContext context ) {
return Scaffold (
2023-05-30 21:56:26 +00:00
backgroundColor: AppColors . mainBackground ,
2023-05-12 17:28:36 +00:00
body: RefreshIndicator (
onRefresh: ( ) async {
await _resubscribeStream ( ) ;
} ,
child: StreamBuilder (
stream: _controller . stream ,
builder: ( context , snapshot ) {
// Inside the builder callback, the snapshot object contains the most recent event from the thread.
// If snapshot.hasData is true, there is data to display. In this case, ListView.builder is returned, which displays a list of NoostCard widgets.
if ( snapshot . hasData ) {
return ListView . builder (
// The itemCount property of ListView.builder is set to _events.length, , which is the number of events in the _events list.
itemCount: _events . length ,
itemBuilder: ( context , index ) {
final event = _events [ index ] ;
final metadata = _metaDatas [ event . pubkey ] ;
// For each event, a Noost object is created that encapsulates the details of the event, including id, avatarUrl, name,username, time, content и pubkey.
// _metaDatas, you can map the event public key to the author's metadata.
final domain = Domain (
noteId: event . id ,
avatarUrl: metadata ? . picture ? ?
' https://robohash.org/ ${ event . pubkey } ' ,
name: metadata ? . name ? ? ' Anon ' ,
username: metadata ? . displayName ? ?
( metadata ? . display_name ? ? ' Anon ' ) ,
time: TimeAgo . format ( event . created_at ) ,
content: event . content ,
pubkey: event . pubkey ,
) ;
return DomainCard ( domain: domain ) ;
} ,
) ;
} else if ( snapshot . connectionState = = ConnectionState . waiting ) {
return const Center ( child: Text ( ' Loading.... ' ) ) ;
} else if ( snapshot . hasError ) {
return Center ( child: Text ( ' Error: ${ snapshot . error } ' ) ) ;
}
return const CenteredCircularProgressIndicator ( ) ;
} ,
) ,
) ,
floatingActionButton: Keys . keysExist
? CreatePost (
// The publishNote function is called when the user launches the "Noost!" button in the dialog box.
publishNote: ( note ) {
setState ( ( ) = > _isNotePublishing = true ) ;
// EventApi Creates an instance of the class defined in the nostr_tools package.
final eventApi = EventApi ( ) ;
// The finishEvent method of the EventApi class is called with the Event object and the _privateKey variable.
// finishEvent will set the event id with the event hash and sign the event with the given _privateKey.
final event = eventApi . finishEvent (
// This creates a new instance of the Event class with certain properties, such as:
Event (
kind: 1 ,
tags: [
[ ' t ' , ' nostr ' ]
] ,
content: note ! ,
created_at: DateTime . now ( ) . millisecondsSinceEpoch ~ / 1000 ,
) ,
Keys . publicKey ,
) ;
if ( eventApi . verifySignature ( event ) ) {
try {
// If the signature is verified, the _relay method is called for the object to publish the event.
Relay . relay . publish ( event ) ;
// After the _resubscribeStream event is published, a method is called that will probably update the stream or subscription to reflect the recently published event.
_resubscribeStream ( ) ;
// Show SnackBar to display a message that the note has been successfully published.
ScaffoldMessenger . of ( context ) . showSnackBar (
MessageSnackBar (
label: ' Congratulations! Noost Published! ' ) ,
) ;
} catch ( _ ) {
// If an error occurs during the publishing process (e.g., an exception is caught), SnackBar displays a warning instead.
ScaffoldMessenger . of ( context ) . showSnackBar ( MessageSnackBar (
label: ' Oops! Something went wrong! ' ,
isWarning: true ,
) ) ;
}
}
setState ( ( ) = > _isNotePublishing = false ) ;
Navigator . pop ( context ) ;
} ,
isNotePublishing: _isNotePublishing ,
)
:
// If _keysExist is false, then an empty widget is displayed, which means that the FAB will not be visible. Container()
Container ( ) ,
) ;
}
}
class TimeAgo {
static String format ( int timestamp ) {
DateTime dateTime = DateTime . fromMillisecondsSinceEpoch ( timestamp * 1000 ) ;
Duration difference = DateTime . now ( ) . difference ( dateTime ) ;
String timeAgo = ' ' ;
if ( difference . inDays > 0 ) {
2023-05-30 21:56:26 +00:00
timeAgo = ' ${ difference . inDays } d ' ;
// '${difference.inDays} ${difference.inDays == 1 ? 'day' : 'days'} ago';
2023-05-12 17:28:36 +00:00
} else if ( difference . inHours > 0 ) {
2023-05-30 21:56:26 +00:00
timeAgo = ' ${ difference . inHours } h ' ;
// '${difference.inHours} ${difference.inHours == 1 ? 'hour' : 'hours'} ago';
2023-05-12 17:28:36 +00:00
} else if ( difference . inMinutes > 0 ) {
2023-05-30 21:56:26 +00:00
timeAgo = ' ${ difference . inMinutes } m ' ;
// '${difference.inMinutes} ${difference.inMinutes == 1 ? 'minute' : 'minutes'} ago';
2023-05-12 17:28:36 +00:00
} else {
2023-05-30 21:56:26 +00:00
timeAgo = ' ${ difference . inMinutes } m ' ;
2023-05-12 17:28:36 +00:00
}
return timeAgo ;
}
}
class DomainCard extends StatelessWidget {
const DomainCard ( {
super . key ,
required this . domain ,
} ) ;
final Domain domain ;
List < String > ? extractImage ( String text ) {
final RegExp exp = RegExp (
r"(http(s?):)([/|.|\w|\s|-])*\.(?:jpg|gif|png|jpeg)" ,
caseSensitive: false ,
multiLine: true ,
) ;
final Iterable < Match > matches = exp . allMatches ( text ) ;
final List < String > imageLinks =
matches . map ( ( match ) = > match . group ( 0 ) ! ) . toList ( ) ;
return imageLinks . isNotEmpty ? imageLinks : null ;
}
@ override
Widget build ( BuildContext context ) {
final List < String > ? imageLinks = extractImage ( domain . content ) ;
return Container (
2023-05-30 01:09:23 +00:00
margin: const EdgeInsets . all ( 4 ) ,
2023-05-12 17:28:36 +00:00
decoration: BoxDecoration (
2023-05-30 21:56:26 +00:00
color: AppColors . postBg ,
2023-05-30 01:09:23 +00:00
borderRadius: BorderRadius . circular ( 8 ) ,
2023-05-12 17:28:36 +00:00
) ,
child: Column (
children: [
2023-05-30 21:56:26 +00:00
// Row(
// children: [
// CircleAvatar(
// radius: 25,
// backgroundImage: FadeInImage(
// placeholder: const NetworkImage(
// 'https://i.ibb.co/mJkxDkb/satoshi.png'),
// image: NetworkImage(domain.avatarUrl),
// ).image,
// ),
// Container(
// width: 300,
// child: Row(
// children: [
// Text(domain.name),
// SizedBox(
// width: 4,
// ),
// Text('@${domain.username.toLowerCase()}',
// style: TextStyle(color: AppColors.postUserName)),
// Expanded(child: SizedBox()),
// Text('${domain.time}',
// style: TextStyle(color: AppColors.postUserName)),
// ],
// ),
// )
// ],
// ),
2023-05-12 17:28:36 +00:00
ListTile (
2023-05-30 21:56:26 +00:00
contentPadding: EdgeInsets . only ( top: 16 , right: 16 , left: 16 ) ,
2023-05-12 17:28:36 +00:00
leading: CircleAvatar (
2023-05-30 21:56:26 +00:00
radius: 25 ,
2023-05-12 17:28:36 +00:00
backgroundImage: FadeInImage (
placeholder:
const NetworkImage ( ' https://i.ibb.co/mJkxDkb/satoshi.png ' ) ,
image: NetworkImage ( domain . avatarUrl ) ,
) . image ,
) ,
2023-05-30 21:56:26 +00:00
title: Row (
children: [
Text ( domain . name ) ,
SizedBox (
width: 4 ,
) ,
Expanded (
child: Container (
child: Text ( ' @ ${ domain . username . toLowerCase ( ) } ' ,
overflow: TextOverflow . ellipsis ,
style: TextStyle ( color: AppColors . postUserName ) ) ,
) ,
) ,
Text ( ' ${ domain . time } ' ,
style: TextStyle ( color: AppColors . postUserName ) ) ,
] ,
) ,
trailing:
const Icon ( Icons . more_horiz , color: AppColors . postMoreIcon ) ,
2023-05-12 17:28:36 +00:00
) ,
Padding (
2023-05-30 21:56:26 +00:00
padding:
const EdgeInsets . only ( top: 12 , right: 16 , bottom: 14 , left: 78 ) ,
2023-05-12 17:28:36 +00:00
child: Text ( domain . content ,
2023-05-30 21:56:26 +00:00
style: const TextStyle ( color: AppColors . postBodyText ) ) ,
2023-05-12 17:28:36 +00:00
) ,
if ( imageLinks ! = null & & imageLinks . isNotEmpty )
2023-05-30 21:56:26 +00:00
Padding (
padding: const EdgeInsets . only (
top: 12 , right: 16 , bottom: 14 , left: 78 ) ,
2023-05-12 17:28:36 +00:00
child: Stack (
children: [
const Placeholder (
fallbackHeight: 200 ,
color: Colors . transparent ,
) ,
Center (
child: FadeInImage (
placeholder: const NetworkImage (
2023-05-30 21:56:26 +00:00
' https://media.tenor.com/On7kvXhzml4AAAAj/loading-gif.gif ' ,
2023-05-12 17:28:36 +00:00
) ,
image: NetworkImage ( imageLinks . first ) ,
fit: BoxFit . cover ,
) ,
) ,
] ,
) ,
) ,
] ,
) ,
) ;
}
}
class CenteredCircularProgressIndicator extends StatelessWidget {
const CenteredCircularProgressIndicator ( { Key ? key } ) : super ( key: key ) ;
@ override
Widget build ( BuildContext context ) {
return const Center (
child: CircularProgressIndicator ( ) ,
) ;
}
}
class CreatePost extends StatefulWidget {
const CreatePost ( {
Key ? key ,
required this . publishNote ,
required this . isNotePublishing ,
} ) : super ( key: key ) ;
final Function ( String ? ) publishNote ;
final bool isNotePublishing ;
@ override
State < CreatePost > createState ( ) = > _CreatePostState ( ) ;
}
class _CreatePostState extends State < CreatePost > {
final _noteController = TextEditingController ( ) ;
final GlobalKey < FormFieldState > _formKey = GlobalKey < FormFieldState > ( ) ;
@ override
Widget build ( BuildContext context ) {
return FloatingActionButton (
backgroundColor: Colors . deepPurpleAccent ,
tooltip: ' Create a new post ' ,
elevation: 2 ,
highlightElevation: 4 ,
foregroundColor: Colors . white ,
child: Stack (
alignment: Alignment . center ,
children: [
Container (
width: 60 ,
height: 60 ,
decoration: const BoxDecoration (
shape: BoxShape . circle , color: AppColors . mainDarkBlue ) ,
) ,
const Icon (
Icons . add ,
color: Colors . white ,
) ,
] ,
) ,
onPressed: ( ) async {
_noteController . clear ( ) ;
await showDialog (
barrierDismissible: false ,
context: context ,
builder: ( ( context ) {
return Dialog (
shape: RoundedRectangleBorder (
borderRadius: BorderRadius . circular ( 12 ) ,
) ,
child: Container (
constraints: const BoxConstraints ( maxWidth: 600 ) ,
decoration: BoxDecoration (
borderRadius: BorderRadius . circular ( 12 ) ,
color: Colors . white ,
) ,
child: Column (
mainAxisSize: MainAxisSize . min ,
children: [
Container (
padding: const EdgeInsets . symmetric ( vertical: 24 ) ,
decoration: BoxDecoration (
borderRadius: BorderRadius . circular ( 12 ) ,
2023-05-17 23:16:20 +00:00
color: AppColors . background ,
2023-05-12 17:28:36 +00:00
) ,
child: const Center (
child: Text (
' Create a Noost ' ,
style: TextStyle (
fontSize: 24 ,
fontWeight: FontWeight . bold ,
color: Colors . white ,
) ,
) ,
) ,
) ,
const SizedBox ( height: 24 ) ,
Padding (
padding: const EdgeInsets . symmetric ( horizontal: 24 ) ,
child: MessageTextFormField (
maxLines: 5 ,
hintText: ' Type your Noost here... ' ,
controller: _noteController ,
formKey: _formKey ,
validator: ( value ) {
if ( value = = null | | value . trim ( ) . isEmpty ) {
return ' Please enter your note. ' ;
}
return null ;
} ,
) ,
) ,
const SizedBox ( height: 24 ) ,
widget . isNotePublishing
? const CenteredCircularProgressIndicator ( )
: Row (
mainAxisAlignment: MainAxisAlignment . end ,
children: [
MessageTextButton (
onPressed: ( ) {
Navigator . pop ( context ) ;
} ,
label: ' Cancel ' ,
) ,
const SizedBox ( width: 16 ) ,
MessageOkButton (
onPressed: ( ) {
if ( _formKey . currentState ! . validate ( ) ) {
widget . publishNote (
_noteController . text . trim ( ) ) ;
} else {
_formKey . currentState ? . setState ( ( ) { } ) ;
}
} ,
label: ' Noost! ' ,
) ,
const SizedBox ( width: 24 ) ,
] ,
)
] ,
) ,
) ,
) ;
} ) ,
) ;
} ,
) ;
}
}