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