diff --git a/assets/icons/cart.svg b/assets/icons/cart.svg new file mode 100644 index 0000000..7b846df --- /dev/null +++ b/assets/icons/cart.svg @@ -0,0 +1,4 @@ + + + + diff --git a/assets/icons/settings.svg b/assets/icons/settings.svg new file mode 100644 index 0000000..f6634dd --- /dev/null +++ b/assets/icons/settings.svg @@ -0,0 +1,3 @@ + + + diff --git a/assets/icons/start-new-search.svg b/assets/icons/start-new-search.svg new file mode 100644 index 0000000..4aad3c7 --- /dev/null +++ b/assets/icons/start-new-search.svg @@ -0,0 +1,3 @@ + + + diff --git a/assets/icons/trash.svg b/assets/icons/trash.svg new file mode 100644 index 0000000..5726808 --- /dev/null +++ b/assets/icons/trash.svg @@ -0,0 +1,3 @@ + + + diff --git a/assets/icons/wishlists.svg b/assets/icons/wishlists.svg new file mode 100644 index 0000000..fb2c609 --- /dev/null +++ b/assets/icons/wishlists.svg @@ -0,0 +1,4 @@ + + + + diff --git a/lib/main.dart b/lib/main.dart index 1deb065..8e6d29e 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -1,158 +1,134 @@ -import 'dart:convert'; import 'package:flutter/material.dart'; -import 'package:graphql/client.dart'; -import 'package:shopping_assistant_mobile_client/network/api_client.dart'; +import 'package:flutter_svg/flutter_svg.dart'; +import 'package:shopping_assistant_mobile_client/screens/wishlists.dart'; void main() { runApp(const MyApp()); } -class MyApp extends StatelessWidget { +class MyApp extends StatefulWidget { const MyApp({super.key}); - // This widget is the root of your application. + static const List _pageNameOptions = [ + 'Wishlists', + 'New Chat', + 'Settings', + ]; + + static const List _widgetOptions = [ + WishlistsScreen(), + Text(''), + Text(''), + ]; + + static const Color _selectedColor = Color.fromRGBO(36, 36, 36, 1); + static const Color _unselectedColor = Color.fromRGBO(144, 144, 144, 1); + + @override + State createState() => _MyAppState(); +} + +class _MyAppState extends State { + int _selectedIndex = 0; + + void _onItemTapped(int index) { + setState(() { + _selectedIndex = index; + }); + } + @override Widget build(BuildContext context) { return MaterialApp( - title: 'Flutter Demo', theme: ThemeData( - // This is the theme of your application. - // - // TRY THIS: Try running your application with "flutter run". You'll see - // the application has a blue toolbar. Then, without quitting the app, - // try changing the seedColor in the colorScheme below to Colors.green - // and then invoke "hot reload" (save your changes or press the "hot - // reload" button in a Flutter-supported IDE, or press "r" if you used - // the command line to start the app). - // - // Notice that the counter didn't reset back to zero; the application - // state is not lost during the reload. To reset the state, use hot - // restart instead. - // - // This works for code too, not just values: Most code changes can be - // tested with just a hot reload. - colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple), useMaterial3: true, - ), - home: const MyHomePage(title: 'Flutter Demo Home Page'), - ); - } -} - -class MyHomePage extends StatefulWidget { - const MyHomePage({super.key, required this.title}); - - // This widget is the home page of your application. It is stateful, meaning - // that it has a State object (defined below) that contains fields that affect - // how it looks. - - // This class is the configuration for the state. It holds the values (in this - // case the title) provided by the parent (in this case the App widget) and - // used by the build method of the State. Fields in a Widget subclass are - // always marked "final". - - final String title; - - @override - State createState() => _MyHomePageState(); -} - -class _MyHomePageState extends State { - int _counter = 0; - - var client = ApiClient(); - Future _incrementCounter() async { - setState(() { - // This call to setState tells the Flutter framework that something has - // changed in this State, which causes it to rerun the build method below - // so that the display can reflect the updated values. If we changed - // _counter without calling setState(), then the build method would not be - // called again, and so nothing would appear to happen. - _counter++; - }); - - const String startPersonalWishlistMutations = r''' - mutation startPersonalWishlist($dto: WishlistCreateDtoInput!) { - startPersonalWishlist(dto: $dto) { - createdById, id, name, type - } - } - '''; - - MutationOptions mutationOptions = MutationOptions( - document: gql(startPersonalWishlistMutations), - variables: const { - 'dto': { - 'firstMessageText': 'Gaming mechanical keyboard', - 'type': 'Product' - }, - } - ); - - var result = await client.mutate(mutationOptions); - print(jsonEncode(result)); - - var wishlistId = result?['startPersonalWishlist']['id']; - var sseStream = client.getServerSentEventStream( - 'api/productssearch/search/$wishlistId', - {'text': 'silent wireless mouse'}); - await for (var chunk in sseStream) { - print('${chunk.event}: ${chunk.data}'); - } - } - - @override - Widget build(BuildContext context) { - // This method is rerun every time setState is called, for instance as done - // by the _incrementCounter method above. - // - // The Flutter framework has been optimized to make rerunning build methods - // fast, so that you can just rebuild anything that needs updating rather - // than having to individually change instances of widgets. - return Scaffold( - appBar: AppBar( - // TRY THIS: Try changing the color here to a specific color (to - // Colors.amber, perhaps?) and trigger a hot reload to see the AppBar - // change color while the other colors stay the same. - backgroundColor: Theme.of(context).colorScheme.inversePrimary, - // Here we take the value from the MyHomePage object that was created by - // the App.build method, and use it to set our appbar title. - title: Text(widget.title), - ), - body: Center( - // Center is a layout widget. It takes a single child and positions it - // in the middle of the parent. - child: Column( - // Column is also a layout widget. It takes a list of children and - // arranges them vertically. By default, it sizes itself to fit its - // children horizontally, and tries to be as tall as its parent. - // - // Column has various properties to control how it sizes itself and - // how it positions its children. Here we use mainAxisAlignment to - // center the children vertically; the main axis here is the vertical - // axis because Columns are vertical (the cross axis would be - // horizontal). - // - // TRY THIS: Invoke "debug painting" (choose the "Toggle Debug Paint" - // action in the IDE, or press "p" in the console), to see the - // wireframe for each widget. - mainAxisAlignment: MainAxisAlignment.center, - children: [ - const Text( - 'You have pushed the button this many times:', - ), - Text( - '$_counter', - style: Theme.of(context).textTheme.headlineMedium, - ), - ], + appBarTheme: AppBarTheme(), + textTheme: TextTheme( + bodyMedium: TextStyle( + fontSize: 16, + fontWeight: FontWeight.w500, + ), + ), + ), + home: Scaffold( + appBar: AppBar( + title: Text(MyApp._pageNameOptions[_selectedIndex]), + centerTitle: true, + bottom: PreferredSize( + preferredSize: const Size.fromHeight(1), + child: Container( + color: Color.fromRGBO(234, 234, 234, 1), + height: 1, + ), + ), + ), + body: MyApp._widgetOptions[_selectedIndex], + bottomNavigationBar: BottomNavigationBar( + items: [ + BottomNavigationBarItem( + icon: SvgPicture.asset( + 'assets/icons/wishlists.svg', + color: _selectedIndex == 0 + ? MyApp._selectedColor + : MyApp._unselectedColor, + ), + label: 'Wishlists', + ), + BottomNavigationBarItem( + icon: SvgPicture.asset( + 'assets/icons/start-new-search.svg', + color: _selectedIndex == 1 + ? MyApp._selectedColor + : MyApp._unselectedColor, + ), + label: 'New Chat', + ), + BottomNavigationBarItem( + icon: SvgPicture.asset( + 'assets/icons/settings.svg', + color: _selectedIndex == 2 + ? MyApp._selectedColor + : MyApp._unselectedColor, + ), + label: 'Settings', + ), + ], + selectedItemColor: MyApp._selectedColor, + unselectedItemColor: MyApp._unselectedColor, + selectedFontSize: 14, + unselectedFontSize: 14, + currentIndex: _selectedIndex, + onTap: _onItemTapped, ), ), - floatingActionButton: FloatingActionButton( - onPressed: _incrementCounter, - tooltip: 'Increment', - child: const Icon(Icons.add), - ), // This trailing comma makes auto-formatting nicer for build methods. ); } } + +// Use to seed wishlists for new user + +// const String startPersonalWishlistMutations = r''' +// mutation startPersonalWishlist($dto: WishlistCreateDtoInput!) { +// startPersonalWishlist(dto: $dto) { +// createdById, id, name, type +// } +// } +// '''; +// +// MutationOptions mutationOptions = MutationOptions( +// document: gql(startPersonalWishlistMutations), +// variables: const { +// 'dto': { +// 'firstMessageText': 'Gaming mechanical keyboard', +// 'type': 'Product' +// }, +// }); +// +// var client = ApiClient(); +// // for (var i = 0; i < 5; i++) { +// // client +// // .mutate(mutationOptions) +// // .then((result) => print(jsonEncode(result))); +// // sleep(Duration(milliseconds: 100)); +// // } +// diff --git a/lib/models/wishlist.dart b/lib/models/wishlist.dart new file mode 100644 index 0000000..c967bd5 --- /dev/null +++ b/lib/models/wishlist.dart @@ -0,0 +1,21 @@ +class Wishlist { + Wishlist( + {required this.id, + required this.name, + required this.type, + required this.createdById}); + + String id; + + String name; + + String type; + + String createdById; + + Wishlist.fromJson(Map json) + : id = json['id'] as String, + name = json['name'] as String, + type = json['type'] as String, + createdById = json['createdById'] as String; +} diff --git a/lib/network/authentication_service.dart b/lib/network/authentication_service.dart index be65417..e1c195c 100644 --- a/lib/network/authentication_service.dart +++ b/lib/network/authentication_service.dart @@ -12,21 +12,18 @@ class AuthenticationService { late SharedPreferences prefs; - AuthenticationService() { - SharedPreferences.getInstance().then((result) => {prefs = result}); - } - Future getAccessToken() async { + prefs = await SharedPreferences.getInstance(); var accessToken = prefs.getString('accessToken'); var refreshToken = prefs.getString('refreshToken'); if (accessToken == null && refreshToken != null) { print('WTF??'); } else if (accessToken == null && refreshToken == null) { - accessToken = await accessGuest(); + accessToken = await _accessGuest(); print('Got new access token $accessToken'); } else if (JwtDecoder.isExpired(accessToken!)) { - accessToken = await refreshAccessToken(); + accessToken = await _refreshAccessToken(); print('Refreshed access token $accessToken'); } @@ -35,6 +32,7 @@ class AuthenticationService { } Future login(String? email, String? phone, String password) async { + prefs = await SharedPreferences.getInstance(); const String loginQuery = r''' mutation Login($login: AccessUserModelInput!) { login(login: $login) { @@ -68,7 +66,8 @@ class AuthenticationService { prefs.setString('refreshToken', refreshToken); } - Future accessGuest() async { + Future _accessGuest() async { + prefs = await SharedPreferences.getInstance(); String? guestId = prefs.getString('guestId'); guestId ??= const Uuid().v4(); prefs.setString('guestId', guestId); @@ -106,7 +105,7 @@ class AuthenticationService { return accessToken; } - Future refreshAccessToken() async { + Future _refreshAccessToken() async { var accessToken = prefs.getString('accessToken'); var refreshToken = prefs.getString('refreshToken'); diff --git a/lib/screens/wishlists.dart b/lib/screens/wishlists.dart new file mode 100644 index 0000000..b8ed8d9 --- /dev/null +++ b/lib/screens/wishlists.dart @@ -0,0 +1,284 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_svg/svg.dart'; +import 'package:graphql/client.dart'; +import 'package:shopping_assistant_mobile_client/models/wishlist.dart'; +import 'package:shopping_assistant_mobile_client/network/api_client.dart'; + +class WishlistsScreen extends StatefulWidget { + const WishlistsScreen({super.key}); + + @override + State createState() => _WishlistsScreenState(); +} + +class _WishlistsScreenState extends State { + var client = ApiClient(); + + late Future _wishlistsFuture; + late List _wishlists; + + @override + void initState() { + super.initState(); + _wishlistsFuture = _fetchWishlistPage(); + } + + Future _fetchWishlistPage() async { + const String personalWishlistsPageQuery = r''' + query personalWishlistsPage($pageNumber: Int!, $pageSize: Int!) { + personalWishlistsPage(pageNumber: $pageNumber, pageSize: $pageSize) { + items { + id, name, type, createdById, + }, + hasNextPage, hasPreviousPage, pageNumber, pageSize, totalItems, totalPages, + } + } + '''; + + QueryOptions queryOptions = QueryOptions( + document: gql(personalWishlistsPageQuery), + variables: const { + 'pageNumber': 1, + 'pageSize': 200, + }); + + var result = await client.query(queryOptions); + + _wishlists = List>.from( + result?['personalWishlistsPage']['items']) + .map((e) => Wishlist.fromJson(e)) + .toList(); + + return; + } + + void _deleteWishlist(Wishlist wishlist) async { + const String deletePersonalWishlistMutation = r''' + mutation deletePersonalWishlist($wishlistId: String!) { + deletePersonalWishlist(wishlistId: $wishlistId) { + } + } + '''; + + MutationOptions mutationOptions = MutationOptions( + document: gql(deletePersonalWishlistMutation), + variables: { + 'wishlistId': wishlist.id, + }); + + var result = await client.mutate(mutationOptions); + + setState(() { + _wishlists.remove(wishlist); + }); + } + + @override + Widget build(BuildContext context) { + return FutureBuilder( + future: _wishlistsFuture, + builder: (context, snapshot) { + if (snapshot.hasError) { + return Center( + child: Text('Error: ${snapshot.error}'), + ); + } else if (snapshot.connectionState == ConnectionState.done) { + // Data loaded successfully, display the widget + return Container( + color: Colors.white, + child: _wishlists.length == 0 + ? Center( + child: Text('No wishlists found'), + ) + : ListView.builder( + padding: EdgeInsets.symmetric(vertical: 30), + itemCount: _wishlists.length, + itemBuilder: (context, index) { + return WishlistItem( + wishlist: _wishlists[index], + onDelete: () => _deleteWishlist(_wishlists[index]), + ); + }, + ), + ); + } + + return Center( + child: CircularProgressIndicator(), + ); + }, + ); + } +} + +class WishlistItem extends StatefulWidget { + WishlistItem({ + super.key, + required Wishlist wishlist, + required Function() onDelete, + }) : _wishlist = wishlist, + _onDelete = onDelete; + + final Wishlist _wishlist; + final Function() _onDelete; + + @override + State createState() => _WishlistItemState(); +} + +class _WishlistItemState extends State { + double _xOffset = 0; + double _rightBorderRadius = 10.0; + + bool _isDeleting = false; + + void _transformLeft() { + setState(() { + _xOffset = -70; + _rightBorderRadius = 0; + }); + } + + void _transformRight() { + setState(() { + _xOffset = 0; + _rightBorderRadius = 10; + }); + } + + void _onDelete() async { + setState(() { + _isDeleting = true; + }); + + await widget._onDelete(); + + setState(() { + _isDeleting = false; + }); + _transformRight(); + } + + @override + Widget build(BuildContext context) { + return Container( + height: 54, + margin: EdgeInsets.only( + bottom: 10, + left: 20, + right: 20, + ), + child: Stack( + children: [ + Positioned( + child: GestureDetector( + onTap: () => _onDelete(), + child: Container( + margin: EdgeInsets.symmetric( + horizontal: .25, + vertical: .25, + ), + decoration: BoxDecoration( + color: Color.fromRGBO(0, 82, 204, 1), + borderRadius: BorderRadius.circular(10), + ), + alignment: AlignmentDirectional.centerEnd, + child: _isDeleting + ? Container( + margin: EdgeInsets.only( + right: 25, + ), + width: 20, + height: 20, + child: CircularProgressIndicator( + color: Colors.white, + strokeWidth: 2, + ), + ) + : Container( + padding: EdgeInsets.symmetric( + horizontal: 25, + vertical: 17, + ), + child: SvgPicture.asset( + 'assets/icons/trash.svg', + color: Colors.white, + width: 20, + ), + ), + ), + ), + ), + Positioned( + child: GestureDetector( + onTap: () => print(Navigator.push( + context, + MaterialPageRoute( + builder: (context) => + Text('Chat ' + widget._wishlist.id)))), + onHorizontalDragUpdate: (DragUpdateDetails details) => { + if (details.delta.dx < -1) + {_transformLeft()} + else if (details.delta.dx > 1) + {_transformRight()} + }, + child: AnimatedContainer( + transform: Matrix4.translationValues(_xOffset, 0, 0), + duration: Duration(milliseconds: 250), + curve: Curves.easeInOut, + decoration: BoxDecoration( + color: Color.fromRGBO(234, 234, 234, 1), + borderRadius: BorderRadius.only( + topLeft: Radius.circular(10), + bottomLeft: Radius.circular(10), + topRight: Radius.circular(_rightBorderRadius), + bottomRight: Radius.circular(_rightBorderRadius), + ), + ), + alignment: AlignmentDirectional.centerStart, + padding: EdgeInsets.only( + left: 15, + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Expanded( + child: Text( + widget._wishlist.name, + style: TextStyle( + fontSize: 16, + fontWeight: FontWeight.w500, + ), + overflow: TextOverflow.ellipsis, + maxLines: 1, + ), + ), + GestureDetector( + onTap: () => print(Navigator.push( + context, + MaterialPageRoute( + builder: (context) => + Text('Cart ' + widget._wishlist.id)))), + behavior: HitTestBehavior.opaque, + child: Container( + padding: EdgeInsets.symmetric( + horizontal: 17, + vertical: 17, + ), + child: SvgPicture.asset( + 'assets/icons/cart.svg', + color: Color.fromRGBO(32, 32, 32, 1), + width: 20, + ), + ), + ), + ], + ), + ), + ), + ), + ], + ), + ); + } +} diff --git a/pubspec.lock b/pubspec.lock index 41b9539..ccd3ceb 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -134,6 +134,14 @@ packages: url: "https://pub.dev" source: hosted version: "2.0.3" + flutter_svg: + dependency: "direct main" + description: + name: flutter_svg + sha256: d39e7f95621fc84376bc0f7d504f05c3a41488c562f4a8ad410569127507402c + url: "https://pub.dev" + source: hosted + version: "2.0.9" flutter_test: dependency: "direct dev" description: flutter @@ -312,6 +320,14 @@ packages: url: "https://pub.dev" source: hosted version: "1.8.3" + path_parsing: + dependency: transitive + description: + name: path_parsing + sha256: e3e67b1629e6f7e8100b367d3db6ba6af4b1f0bb80f64db18ef1fbabd2fa9ccf + url: "https://pub.dev" + source: hosted + version: "1.0.1" path_provider: dependency: transitive description: @@ -517,6 +533,30 @@ packages: url: "https://pub.dev" source: hosted version: "3.0.7" + vector_graphics: + dependency: transitive + description: + name: vector_graphics + sha256: "0f0c746dd2d6254a0057218ff980fc7f5670fd0fcf5e4db38a490d31eed4ad43" + url: "https://pub.dev" + source: hosted + version: "1.1.9+1" + vector_graphics_codec: + dependency: transitive + description: + name: vector_graphics_codec + sha256: "0edf6d630d1bfd5589114138ed8fada3234deacc37966bec033d3047c29248b7" + url: "https://pub.dev" + source: hosted + version: "1.1.9+1" + vector_graphics_compiler: + dependency: transitive + description: + name: vector_graphics_compiler + sha256: d24333727332d9bd20990f1483af4e09abdb9b1fc7c3db940b56ab5c42790c26 + url: "https://pub.dev" + source: hosted + version: "1.1.9+1" vector_math: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index 0840073..358522f 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -41,6 +41,7 @@ dependencies: shared_preferences: ^2.2.2 uuid: ^3.0.7 graphql_flutter: ^5.2.0-beta.6 + flutter_svg: ^2.0.9 dev_dependencies: flutter_test: @@ -65,9 +66,12 @@ flutter: uses-material-design: true # To add assets to your application, add an assets section, like this: - # assets: - # - images/a_dot_burr.jpeg - # - images/a_dot_ham.jpeg + assets: + - assets/icons/cart.svg + - assets/icons/trash.svg + - assets/icons/wishlists.svg + - assets/icons/start-new-search.svg + - assets/icons/settings.svg # An image asset can refer to one or more resolution-specific "variants", see # https://flutter.dev/assets-and-images/#resolution-aware diff --git a/test/widget_test.dart b/test/widget_test.dart index 7aca564..3e97501 100644 --- a/test/widget_test.dart +++ b/test/widget_test.dart @@ -1,30 +1,30 @@ -// This is a basic Flutter widget test. +// // This is a basic Flutter widget test. +// // +// // To perform an interaction with a widget in your test, use the WidgetTester +// // utility in the flutter_test package. For example, you can send tap and scroll +// // gestures. You can also use WidgetTester to find child widgets in the widget +// // tree, read text, and verify that the values of widget properties are correct. // -// To perform an interaction with a widget in your test, use the WidgetTester -// utility in the flutter_test package. For example, you can send tap and scroll -// gestures. You can also use WidgetTester to find child widgets in the widget -// tree, read text, and verify that the values of widget properties are correct. - -import 'package:flutter/material.dart'; -import 'package:flutter_test/flutter_test.dart'; - -import 'package:shopping_assistant_mobile_client/main.dart'; - -void main() { - testWidgets('Counter increments smoke test', (WidgetTester tester) async { - // Build our app and trigger a frame. - await tester.pumpWidget(const MyApp()); - - // Verify that our counter starts at 0. - expect(find.text('0'), findsOneWidget); - expect(find.text('1'), findsNothing); - - // Tap the '+' icon and trigger a frame. - await tester.tap(find.byIcon(Icons.add)); - await tester.pump(); - - // Verify that our counter has incremented. - expect(find.text('0'), findsNothing); - expect(find.text('1'), findsOneWidget); - }); -} +// import 'package:flutter/material.dart'; +// import 'package:flutter_test/flutter_test.dart'; +// +// import 'package:shopping_assistant_mobile_client/main.dart'; +// +// void main() { +// testWidgets('Counter increments smoke test', (WidgetTester tester) async { +// // Build our app and trigger a frame. +// await tester.pumpWidget(const MyApp()); +// +// // Verify that our counter starts at 0. +// expect(find.text('0'), findsOneWidget); +// expect(find.text('1'), findsNothing); +// +// // Tap the '+' icon and trigger a frame. +// await tester.tap(find.byIcon(Icons.add)); +// await tester.pump(); +// +// // Verify that our counter has incremented. +// expect(find.text('0'), findsNothing); +// expect(find.text('1'), findsOneWidget); +// }); +// }