getting spaces and rooms from api {null checks}

This commit is contained in:
Mohammad Salameh
2024-03-13 13:52:22 +03:00
parent 0f3cc453ce
commit 024f15728b
26 changed files with 390 additions and 266 deletions

View File

@ -16,6 +16,13 @@ class NavCubit extends Cubit<NavState> {
NavCubit() : super(NavInitial()); NavCubit() : super(NavInitial());
static NavCubit of(context) => BlocProvider.of<NavCubit>(context); static NavCubit of(context) => BlocProvider.of<NavCubit>(context);
//functoin to do the important work when the user logs out
static clear() {
pageIndex = 0;
pageController.jumpToPage(0);
}
static int pageIndex = 0; static int pageIndex = 0;
static Map<String, List<Widget>> appBarActions = { static Map<String, List<Widget>> appBarActions = {
@ -139,11 +146,10 @@ class NavCubit extends Cubit<NavState> {
const MenuView(), const MenuView(),
]; ];
final PageController pageController = PageController(); static final PageController pageController = PageController();
void updatePageIndex(int index) { void updatePageIndex(int index) {
pageIndex = index; pageIndex = index;
print('index: $index');
pageController.animateToPage(index, pageController.animateToPage(index,
duration: const Duration(milliseconds: 150), curve: Curves.easeIn); duration: const Duration(milliseconds: 150), curve: Curves.easeIn);
emit(NavChangePage()); emit(NavChangePage());

View File

@ -10,7 +10,11 @@ part 'spaces_state.dart';
class SpacesCubit extends Cubit<SpacesState> { class SpacesCubit extends Cubit<SpacesState> {
SpacesCubit() : super(SpacesInitial()) { SpacesCubit() : super(SpacesInitial()) {
fetchSpaces(); fetchSpaces().then((value) {
if (selectedSpace != null) {
fetchRooms(selectedSpace!);
}
});
} }
static SpacesCubit get(context) => BlocProvider.of(context); static SpacesCubit get(context) => BlocProvider.of(context);
@ -83,12 +87,24 @@ class SpacesCubit extends Cubit<SpacesState> {
emit(SpacesLoading()); emit(SpacesLoading());
try { try {
spaces = await SpacesAPI.getSpaces(); spaces = await SpacesAPI.getSpaces();
spaces.isNotEmpty ? selectSpace(spaces.first) : null;
emit(SpacesLoaded(spaces)); emit(SpacesLoaded(spaces));
} on DioException catch (e) { } on DioException catch (e) {
emit(SpacesError(e.message ?? "Something went wrong")); emit(SpacesError(ServerFailure.fromDioError(e).errMessage));
throw ServerFailure.fromDioError(e); }
} catch (e) { }
emit(SpacesError(e.toString()));
fetchRooms(SpaceModel space) async {
emit(SpaceRoomsLoading());
try {
space.rooms = await SpacesAPI.getRooms(space.id!);
if (space.rooms != null) {
emit(SpaceRoomsLoaded(space.rooms!));
} else {
emit(SpaceRoomsError("No rooms found"));
}
} on DioException catch (e) {
emit(SpacesError(ServerFailure.fromDioError(e).errMessage));
} }
} }
} }

View File

@ -18,6 +18,18 @@ class SpacesError extends SpacesState {
SpacesError(this.errMessage); SpacesError(this.errMessage);
} }
class SpaceRoomsLoading extends SpacesLoading {}
class SpaceRoomsLoaded extends SpacesLoading {
final List<RoomModel> rooms;
SpaceRoomsLoaded(this.rooms);
}
class SpaceRoomsError extends SpacesError {
SpaceRoomsError(super.errMessage);
}
class SpacesSelected extends SpacesState { class SpacesSelected extends SpacesState {
final SpaceModel space; final SpaceModel space;

View File

@ -3,7 +3,7 @@ import 'package:syncrow_app/features/devices/model/room_model.dart';
class SpaceModel { class SpaceModel {
final int? id; final int? id;
final String? name; final String? name;
final List<RoomModel>? rooms; late List<RoomModel>? rooms;
SpaceModel({ SpaceModel({
required this.id, required this.id,

View File

@ -6,6 +6,7 @@ import 'package:syncrow_app/features/app_layout/bloc/spaces_cubit.dart';
import 'package:syncrow_app/features/app_layout/view/widgets/app_body.dart'; import 'package:syncrow_app/features/app_layout/view/widgets/app_body.dart';
import 'package:syncrow_app/features/app_layout/view/widgets/default_app_bar.dart'; import 'package:syncrow_app/features/app_layout/view/widgets/default_app_bar.dart';
import 'package:syncrow_app/features/app_layout/view/widgets/default_nav_bar.dart'; import 'package:syncrow_app/features/app_layout/view/widgets/default_nav_bar.dart';
import 'package:syncrow_app/navigation/routing_constants.dart';
import 'package:syncrow_app/utils/resource_manager/color_manager.dart'; import 'package:syncrow_app/utils/resource_manager/color_manager.dart';
class AppLayout extends StatelessWidget { class AppLayout extends StatelessWidget {
@ -13,7 +14,21 @@ class AppLayout extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocBuilder<NavCubit, NavState>( return BlocProvider(
create: (context) => SpacesCubit(),
child: BlocListener<SpacesCubit, SpacesState>(
listener: (context, state) {
if (state is SpacesError) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(state.errMessage),
),
);
Navigator.of(context)
.popUntil((route) => route.settings.name == Routes.authLogin);
}
},
child: BlocBuilder<NavCubit, NavState>(
builder: (context, state) { builder: (context, state) {
return AnnotatedRegion( return AnnotatedRegion(
value: SystemUiOverlayStyle( value: SystemUiOverlayStyle(
@ -27,16 +42,23 @@ class AppLayout extends StatelessWidget {
backgroundColor: ColorsManager.backgroundColor, backgroundColor: ColorsManager.backgroundColor,
extendBodyBehindAppBar: true, extendBodyBehindAppBar: true,
extendBody: true, extendBody: true,
appBar: state is SpacesLoaded ? DefaultAppBar(context) : null, appBar:
state is! SpacesLoading || state is! SpaceRoomsLoading
? const DefaultAppBar()
: null,
body: const AppBody(), body: const AppBody(),
bottomNavigationBar: bottomNavigationBar:
state is SpacesLoaded ? const DefaultNavBar() : null, state is! SpacesLoading || state is! SpaceRoomsLoading
? const DefaultNavBar()
: null,
); );
}, },
), ),
), ),
); );
}, },
),
),
); );
} }
} }

View File

@ -28,10 +28,10 @@ class AppBarHomeDropdown extends StatelessWidget {
underline: const SizedBox.shrink(), underline: const SizedBox.shrink(),
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
borderRadius: BorderRadius.circular(20), borderRadius: BorderRadius.circular(20),
value: SpacesCubit.get(context).selectedSpace, value: SpacesCubit.get(context).selectedSpace!.id,
items: SpacesCubit.spaces.map((space) { items: SpacesCubit.spaces.map((space) {
return DropdownMenuItem( return DropdownMenuItem(
value: space, value: space.id,
child: SizedBox( child: SizedBox(
width: 100, width: 100,
child: Row( child: Row(
@ -50,7 +50,7 @@ class AppBarHomeDropdown extends StatelessWidget {
const SizedBox(width: 5), const SizedBox(width: 5),
Expanded( Expanded(
child: BodyMedium( child: BodyMedium(
text: space.name ?? "", text: space.name ?? "??",
style: context.bodyMedium.copyWith( style: context.bodyMedium.copyWith(
fontSize: 15, fontSize: 15,
color: ColorsManager.textPrimaryColor, color: ColorsManager.textPrimaryColor,
@ -64,7 +64,10 @@ class AppBarHomeDropdown extends StatelessWidget {
); );
}).toList(), }).toList(),
onChanged: (value) { onChanged: (value) {
SpacesCubit.get(context).selectSpace(value!); if (value != null) {
SpacesCubit.get(context).selectSpace(SpacesCubit.spaces
.firstWhere((element) => element.id == value));
}
}, },
), ),
); );

View File

@ -37,16 +37,15 @@ class AppBody extends StatelessWidget {
} }
}, },
builder: (context, state) { builder: (context, state) {
return state is! SpacesLoading return state is! SpacesLoading || state is! SpaceRoomsLoading
? PageView( ? PageView(
physics: const NeverScrollableScrollPhysics(), physics: const NeverScrollableScrollPhysics(),
controller: NavCubit.of(context).pageController, controller: NavCubit.pageController,
children: NavCubit.of(context).pages, children: NavCubit.of(context).pages,
) )
: const Center(child: CircularProgressIndicator()); : const Center(child: CircularProgressIndicator());
}, },
) ),
// NavCubit.of(context).currentPage,
); );
}, },
); );

View File

@ -4,9 +4,7 @@ import 'package:syncrow_app/features/app_layout/bloc/nav_cubit.dart';
import 'package:syncrow_app/utils/resource_manager/constants.dart'; import 'package:syncrow_app/utils/resource_manager/constants.dart';
class DefaultAppBar extends StatelessWidget implements PreferredSizeWidget { class DefaultAppBar extends StatelessWidget implements PreferredSizeWidget {
const DefaultAppBar(this.context, {super.key}); const DefaultAppBar({super.key});
final BuildContext context;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {

View File

@ -3,6 +3,7 @@ import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:syncrow_app/features/app_layout/bloc/nav_cubit.dart';
import 'package:syncrow_app/features/auth/model/login_with_email_model.dart'; import 'package:syncrow_app/features/auth/model/login_with_email_model.dart';
import 'package:syncrow_app/features/auth/model/token.dart'; import 'package:syncrow_app/features/auth/model/token.dart';
import 'package:syncrow_app/features/auth/model/user_model.dart'; import 'package:syncrow_app/features/auth/model/user_model.dart';
@ -20,7 +21,7 @@ class AuthCubit extends Cubit<AuthState> {
TextEditingController passwordController = TextEditingController(); TextEditingController passwordController = TextEditingController();
bool isPasswordVisible = false; bool isPasswordVisible = false;
GlobalKey<FormState> formKey = GlobalKey<FormState>(); static GlobalKey<FormState> formKey = GlobalKey<FormState>();
void changePasswordVisibility() { void changePasswordVisibility() {
isPasswordVisible = !isPasswordVisible; isPasswordVisible = !isPasswordVisible;
@ -54,15 +55,16 @@ class AuthCubit extends Cubit<AuthState> {
FlutterSecureStorage storage = const FlutterSecureStorage(); FlutterSecureStorage storage = const FlutterSecureStorage();
await storage.write( await storage.write(
key: Token.loginAccessTokenKey, value: token.accessToken); key: Token.loginAccessTokenKey, value: token.accessToken);
const FlutterSecureStorage().write(
key: UserModel.userUuidKey,
value: Token.decodeToken(token.accessToken)['uuid'].toString());
user = UserModel.fromToken(token); user = UserModel.fromToken(token);
emit(AuthSuccess()); emit(AuthSuccess());
} else { } else {
emit(AuthError('Something went wrong')); emit(AuthError('Something went wrong'));
} }
} on DioException catch (e) { } on DioException catch (e) {
emit(AuthError(e.message ?? "Something went wrong")); emit(AuthError(ServerFailure.fromDioError(e).errMessage));
throw ServerFailure.fromDioError(e);
} }
} }
@ -71,9 +73,10 @@ class AuthCubit extends Cubit<AuthState> {
try { try {
FlutterSecureStorage storage = const FlutterSecureStorage(); FlutterSecureStorage storage = const FlutterSecureStorage();
await storage.delete(key: Token.loginAccessTokenKey); await storage.delete(key: Token.loginAccessTokenKey);
NavCubit.clear();
emit(AuthLoggedOut()); emit(AuthLoggedOut());
} on DioException catch (e) { } on DioException catch (e) {
throw ServerFailure.fromDioError(e); emit(AuthError(ServerFailure.fromDioError(e).errMessage));
} }
} }
} }

View File

@ -10,9 +10,26 @@ class Token {
final String accessToken; final String accessToken;
final String refreshToken; final String refreshToken;
//{
// "email": "test@test.com",
// "userId": 3,
// "uuid": "563e22d2-cb30-46d3-8c48-fa7d762342f0",
// "sessionId": "f76aa067-c915-4921-b04d-9fbc71c4965a",
// "iat": 1710137435,
// "exp": 1710137735
// }
final String sessionId;
final int iat;
final int exp;
Token.emptyConstructor() Token.emptyConstructor()
: accessToken = '', : accessToken = '',
refreshToken = ''; refreshToken = '',
sessionId = '',
iat = 0,
exp = 0;
bool get accessTokenIsNotEmpty => accessToken.isNotEmpty; bool get accessTokenIsNotEmpty => accessToken.isNotEmpty;
@ -23,9 +40,16 @@ class Token {
Token( Token(
this.accessToken, this.accessToken,
this.refreshToken, this.refreshToken,
this.sessionId,
this.iat,
this.exp,
); );
Token.refreshToken(this.refreshToken) : accessToken = ''; Token.refreshToken(this.refreshToken)
: accessToken = '',
sessionId = '',
iat = 0,
exp = 0;
factory Token.fromJson(Map<String, dynamic> json) { factory Token.fromJson(Map<String, dynamic> json) {
//save token to secure storage //save token to secure storage
@ -34,15 +58,16 @@ class Token {
key: loginAccessTokenKey, value: json[loginAccessTokenKey] ?? ''); key: loginAccessTokenKey, value: json[loginAccessTokenKey] ?? '');
//create token object ? //create token object ?
return Token( return Token(json[loginAccessTokenKey] ?? '',
json[loginAccessTokenKey] ?? '', json[loginRefreshTokenKey] ?? ''); json[loginRefreshTokenKey] ?? '', '', 0, 0);
} }
Map<String, String> toJson() => {loginRefreshTokenKey: refreshToken}; Map<String, String> refreshTokenToJson() =>
{loginRefreshTokenKey: refreshToken};
Map<String, String> accessTokenToJson() => {loginAccessTokenKey: accessToken}; Map<String, String> accessTokenToJson() => {loginAccessTokenKey: accessToken};
Map<String, dynamic> decodeToken() { static Map<String, dynamic> decodeToken(String accessToken) {
final parts = accessToken.split('.'); final parts = accessToken.split('.');
if (parts.length != 3) { if (parts.length != 3) {
throw Exception('invalid access token'); throw Exception('invalid access token');

View File

@ -1,6 +1,7 @@
import 'package:syncrow_app/features/auth/model/token.dart'; import 'package:syncrow_app/features/auth/model/token.dart';
class UserModel { class UserModel {
static String userUuidKey = 'userUuid';
final String? uuid; final String? uuid;
final String? email; final String? email;
final String? name; final String? name;
@ -36,13 +37,9 @@ class UserModel {
//uuid to json //uuid to json
Map<String, dynamic> uuIdAsJson() => {
'userUuid': uuid,
};
//from token //from token
factory UserModel.fromToken(Token token) { factory UserModel.fromToken(Token token) {
Map<String, dynamic> tempJson = token.decodeToken(); Map<String, dynamic> tempJson = Token.decodeToken(token.accessToken);
return UserModel( return UserModel(
uuid: tempJson['uuid'].toString(), uuid: tempJson['uuid'].toString(),

View File

@ -1,46 +0,0 @@
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart';
import 'package:syncrow_app/features/shared_widgets/default_button.dart';
class LoginButton extends StatelessWidget {
const LoginButton({
super.key,
});
@override
Widget build(BuildContext context) {
return BlocBuilder<AuthCubit, AuthState>(
builder: (context, state) {
return Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Expanded(
child: DefaultButton(
isDone: state is AuthSuccess,
isLoading: state is AuthLoading,
customButtonStyle: ButtonStyle(
backgroundColor: MaterialStateProperty.all(
Colors.black.withOpacity(.25),
),
foregroundColor: MaterialStateProperty.all(
Colors.white,
),
),
child: const Text(
'Login',
),
onPressed: () {
if (AuthCubit.get(context).formKey.currentState!.validate()) {
AuthCubit.get(context).login();
FocusScope.of(context).unfocus();
}
},
),
),
],
);
},
);
}
}

View File

@ -1,6 +1,8 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart'; import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/forget_password.dart';
import 'package:syncrow_app/features/shared_widgets/default_button.dart';
import 'package:syncrow_app/features/shared_widgets/text_widgets/body_medium.dart'; import 'package:syncrow_app/features/shared_widgets/text_widgets/body_medium.dart';
import 'package:syncrow_app/utils/resource_manager/styles_manager.dart'; import 'package:syncrow_app/utils/resource_manager/styles_manager.dart';
@ -11,10 +13,11 @@ class LoginForm extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var formKey = GlobalKey<FormState>();
return BlocBuilder<AuthCubit, AuthState>( return BlocBuilder<AuthCubit, AuthState>(
builder: (context, state) { builder: (context, state) {
return Form( return Form(
key: AuthCubit.get(context).formKey, key: formKey,
child: SingleChildScrollView( child: SingleChildScrollView(
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
@ -40,7 +43,7 @@ class LoginForm extends StatelessWidget {
return null; return null;
}, },
onTapOutside: (event) { onTapOutside: (event) {
AuthCubit.get(context).formKey.currentState!.validate(); formKey.currentState!.validate();
}, },
decoration: defaultInputDecoration(context, decoration: defaultInputDecoration(context,
hint: "Example@email.com"), hint: "Example@email.com"),
@ -65,12 +68,44 @@ class LoginForm extends StatelessWidget {
return null; return null;
}, },
onTapOutside: (event) { onTapOutside: (event) {
AuthCubit.get(context).formKey.currentState!.validate(); formKey.currentState!.validate();
}, },
obscureText: !AuthCubit.get(context).isPasswordVisible, obscureText: !AuthCubit.get(context).isPasswordVisible,
decoration: defaultInputDecoration(context, decoration: defaultInputDecoration(context,
hint: "At least 8 characters"), hint: "At least 8 characters"),
), ),
const SizedBox(height: 10),
// const LoginUserAgreement(),
const ForgetPassword(),
const SizedBox(height: 10),
Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Expanded(
child: DefaultButton(
isDone: state is AuthSuccess,
isLoading: state is AuthLoading,
customButtonStyle: ButtonStyle(
backgroundColor: MaterialStateProperty.all(
Colors.black.withOpacity(.25),
),
foregroundColor: MaterialStateProperty.all(
Colors.white,
),
),
child: const Text(
'Login',
),
onPressed: () {
if (formKey.currentState!.validate()) {
AuthCubit.get(context).login();
FocusScope.of(context).unfocus();
}
},
),
),
],
)
], ],
), ),
), ),

View File

@ -3,8 +3,6 @@ import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_svg/flutter_svg.dart'; import 'package:flutter_svg/flutter_svg.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart'; import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/dont_have_an_account.dart'; import 'package:syncrow_app/features/auth/view/widgets/login/dont_have_an_account.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/forget_password.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/login_button.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/login_divider.dart'; import 'package:syncrow_app/features/auth/view/widgets/login/login_divider.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/login_form.dart'; import 'package:syncrow_app/features/auth/view/widgets/login/login_form.dart';
import 'package:syncrow_app/features/auth/view/widgets/login/login_with_google_facebook.dart'; import 'package:syncrow_app/features/auth/view/widgets/login/login_with_google_facebook.dart';
@ -94,11 +92,6 @@ class LoginView extends StatelessWidget {
height: 20, height: 20,
), ),
const LoginForm(), const LoginForm(),
const SizedBox(height: 10),
// const LoginUserAgreement(),
const ForgetPassword(),
const SizedBox(height: 10),
const LoginButton(),
const LoginDivider(), const LoginDivider(),
const LoginWithGoogleFacebook(), const LoginWithGoogleFacebook(),
const DontHaveAnAccount(), const DontHaveAnAccount(),

View File

@ -1,10 +1,10 @@
import 'package:syncrow_app/features/devices/model/device_category_model.dart'; import 'package:syncrow_app/features/devices/model/device_category_model.dart';
class RoomModel { class RoomModel {
final String id; final int? id;
final String name; final String? name;
final List<DevicesCategoryModel> categories; final List<DevicesCategoryModel>? categories;
RoomModel({ RoomModel({
required this.id, required this.id,
@ -22,8 +22,8 @@ class RoomModel {
factory RoomModel.fromJson(Map<String, dynamic> json) { factory RoomModel.fromJson(Map<String, dynamic> json) {
return RoomModel( return RoomModel(
id: json['id'], id: json['roomId'],
name: json['name'], name: json['roomName'],
categories: json['devices'], categories: json['devices'],
); );
} }

View File

@ -19,12 +19,17 @@ class DevicesViewBody extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocProvider( return BlocProvider(
create: (context) => DevicesCubit(), create: (context) => DevicesCubit(),
child: BlocBuilder<DevicesCubit, DevicesState>( child: BlocBuilder<SpacesCubit, SpacesState>(
builder: (context, state) { builder: (context, state) {
return BlocBuilder<DevicesCubit, DevicesState>(
builder: (context, state) {
print(
"length : ${SpacesCubit.get(context).selectedSpace!.rooms!.length}");
//TODO : move to NavigationCubit //TODO : move to NavigationCubit
return state is DevicesLoading if (state is DevicesLoading) {
? const Center(child: CircularProgressIndicator()) return const Center(child: CircularProgressIndicator());
: Padding( } else {
return Padding(
padding: EdgeInsets.only( padding: EdgeInsets.only(
top: Constants.appBarHeight, top: Constants.appBarHeight,
bottom: Constants.bottomNavBarHeight, bottom: Constants.bottomNavBarHeight,
@ -46,6 +51,11 @@ class DevicesViewBody extends StatelessWidget {
}, },
children: [ children: [
const WizardPage(), const WizardPage(),
if (SpacesCubit.get(context).selectedSpace != null)
if (SpacesCubit.get(context)
.selectedSpace!
.rooms !=
null)
...SpacesCubit.get(context) ...SpacesCubit.get(context)
.selectedSpace! .selectedSpace!
.rooms! .rooms!
@ -66,7 +76,11 @@ class DevicesViewBody extends StatelessWidget {
child: SmoothPageIndicator( child: SmoothPageIndicator(
controller: controller:
SpacesCubit.get(context).devicesPageController, SpacesCubit.get(context).devicesPageController,
count: 3, count: SpacesCubit.get(context)
.selectedSpace!
.rooms!
.length +
1,
effect: const WormEffect( effect: const WormEffect(
paintStyle: PaintingStyle.stroke, paintStyle: PaintingStyle.stroke,
dotHeight: 8, dotHeight: 8,
@ -77,6 +91,9 @@ class DevicesViewBody extends StatelessWidget {
], ],
), ),
); );
}
},
);
}, },
), ),
); );

View File

@ -14,7 +14,7 @@ class RoomPage extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
List<DeviceModel> devices = []; List<DeviceModel> devices = [];
for (var category in room.categories) { for (var category in room.categories ?? []) {
devices.addAll(category.devices); devices.addAll(category.devices);
} }
return Padding( return Padding(

View File

@ -40,6 +40,8 @@ class RoomsSlider extends StatelessWidget {
), ),
), ),
), ),
if (SpacesCubit.get(context).selectedSpace != null)
if (SpacesCubit.get(context).selectedSpace!.rooms != null)
...SpacesCubit.get(context).selectedSpace!.rooms!.map( ...SpacesCubit.get(context).selectedSpace!.rooms!.map(
(room) => Padding( (room) => Padding(
padding: const EdgeInsets.symmetric(horizontal: 15), padding: const EdgeInsets.symmetric(horizontal: 15),
@ -52,10 +54,11 @@ class RoomsSlider extends StatelessWidget {
.indexOf(room)); .indexOf(room));
}, },
child: TitleMedium( child: TitleMedium(
text: room.name, text: room.name!,
style: context.titleMedium.copyWith( style: context.titleMedium.copyWith(
fontSize: 25, fontSize: 25,
color: SpacesCubit.get(context).selectedRoom == room color: SpacesCubit.get(context).selectedRoom ==
room
? ColorsManager.textPrimaryColor ? ColorsManager.textPrimaryColor
: ColorsManager.textPrimaryColor : ColorsManager.textPrimaryColor
.withOpacity(.2), .withOpacity(.2),

View File

@ -10,17 +10,20 @@ class SplashView extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
//TODO remove this delay return FutureBuilder(
Future.value().then((value) async { future: getTokenAndValidate(context),
var isLoggedIn = await const FlutterSecureStorage() builder: (context, snapshot) {
.read(key: Token.loginAccessTokenKey) != if (snapshot.hasData) {
null; if (snapshot.data ?? false) {
if (isLoggedIn) { Future.delayed(const Duration(seconds: 1), () {
Navigator.pushReplacementNamed(context, Routes.homeRoute); Navigator.pushReplacementNamed(context, Routes.homeRoute);
} else {
Navigator.pushReplacementNamed(context, Routes.authLogin);
}
}); });
} else {
Future.delayed(const Duration(seconds: 1), () {
Navigator.pushReplacementNamed(context, Routes.authLogin);
});
}
}
return Scaffold( return Scaffold(
body: Stack( body: Stack(
alignment: Alignment.center, alignment: Alignment.center,
@ -55,5 +58,29 @@ class SplashView extends StatelessWidget {
], ],
), ),
); );
},
);
}
Future<bool> getTokenAndValidate(BuildContext context) async {
return await const FlutterSecureStorage()
.read(key: Token.loginAccessTokenKey)
.then((value) {
if (value == null) {
return false;
}
print("Decoding token Started");
var tokenData = Token.decodeToken(value ?? "");
print("checking token data");
if (tokenData.containsKey('exp')) {
var exp = tokenData['exp'] ?? 0;
var currentTime = DateTime.now().millisecondsSinceEpoch ~/ 1000;
print('time: $currentTime exp: $exp');
return currentTime < exp;
} else {
return false;
}
});
} }
} }

View File

@ -16,6 +16,12 @@ class Assets {
static const String iconsDevices = 'assets/icons/Devices.svg'; static const String iconsDevices = 'assets/icons/Devices.svg';
static const String iconsDevicesFill = 'assets/icons/Devices-fill.svg'; static const String iconsDevicesFill = 'assets/icons/Devices-fill.svg';
static const String iconsDoorLock = 'assets/icons/doorLock.svg'; static const String iconsDoorLock = 'assets/icons/doorLock.svg';
static const String iconsDoorLockLinkage = 'assets/icons/DoorLockLinkage.svg';
static const String iconsDoorLockLock = 'assets/icons/DoorLockLock.svg';
static const String iconsDoorLockMembers = 'assets/icons/DoorLockMembers.svg';
static const String iconsDoorLockPassword =
'assets/icons/DoorLockPassword.svg';
static const String iconsDoorLockRecords = 'assets/icons/DoorLockRecords.svg';
static const String iconsFacebook = 'assets/icons/Facebook.svg'; static const String iconsFacebook = 'assets/icons/Facebook.svg';
static const String iconsFan0 = 'assets/icons/fan-0.svg'; static const String iconsFan0 = 'assets/icons/fan-0.svg';
static const String iconsFan1 = 'assets/icons/fan-1.svg'; static const String iconsFan1 = 'assets/icons/fan-1.svg';

View File

@ -1,7 +1,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:syncrow_app/features/app_layout/bloc/nav_cubit.dart'; import 'package:syncrow_app/features/app_layout/bloc/nav_cubit.dart';
import 'package:syncrow_app/features/app_layout/bloc/spaces_cubit.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart'; import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart';
import 'package:syncrow_app/utils/resource_manager/color_manager.dart'; import 'package:syncrow_app/utils/resource_manager/color_manager.dart';
import 'package:syncrow_app/utils/resource_manager/constants.dart'; import 'package:syncrow_app/utils/resource_manager/constants.dart';
@ -30,9 +29,6 @@ class MyApp extends StatelessWidget {
BlocProvider( BlocProvider(
create: (context) => NavCubit(), create: (context) => NavCubit(),
), ),
BlocProvider(
create: (context) => SpacesCubit(),
),
BlocProvider( BlocProvider(
create: (context) => DevicesCubit(), create: (context) => DevicesCubit(),
), ),

View File

@ -12,4 +12,5 @@ abstract class ApiEndpoints {
// Spaces // Spaces
static const String spaces = '$baseUrl/home'; static const String spaces = '$baseUrl/home';
static const String rooms = '$baseUrl/room';
} }

View File

@ -1,4 +1,3 @@
import 'package:flutter/cupertino.dart';
import 'package:syncrow_app/features/auth/model/login_with_email_model.dart'; import 'package:syncrow_app/features/auth/model/login_with_email_model.dart';
import 'package:syncrow_app/features/auth/model/token.dart'; import 'package:syncrow_app/features/auth/model/token.dart';
import 'package:syncrow_app/features/auth/model/verify_code.dart'; import 'package:syncrow_app/features/auth/model/verify_code.dart';
@ -11,21 +10,22 @@ class AuthenticationAPI {
path: ApiEndpoints.verifyOtp, path: ApiEndpoints.verifyOtp,
body: data.toJson(), body: data.toJson(),
showServerMessage: false, showServerMessage: false,
expectedResponseModel: (json) { expectedResponseModel: (json) => Token.fromJson(json));
Token token = Token.fromJson(json);
return token;
});
return response; return response;
} }
static Future<Token> loginWithEmail( static Future<Token> loginWithEmail(
{required LoginWithEmailModel model}) async { {required LoginWithEmailModel model}) async {
try {
final response = await HTTPService().post( final response = await HTTPService().post(
path: ApiEndpoints.login, path: ApiEndpoints.login,
body: model.toJson(), body: model.toJson(),
showServerMessage: false, showServerMessage: false,
expectedResponseModel: (json) => Token.fromJson(json['data'])); expectedResponseModel: (json) => Token.fromJson(json['data']));
debugPrint("response: $response"); // debugPrint("response: $response");
return response; return response;
} catch (e) {
rethrow;
}
} }
} }

View File

@ -1,5 +1,6 @@
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:syncrow_app/features/auth/model/token.dart';
class HTTPInterceptor extends InterceptorsWrapper { class HTTPInterceptor extends InterceptorsWrapper {
// @override // @override
@ -11,9 +12,9 @@ class HTTPInterceptor extends InterceptorsWrapper {
@override @override
void onRequest( void onRequest(
RequestOptions options, RequestInterceptorHandler handler) async { RequestOptions options, RequestInterceptorHandler handler) async {
//pass the token from the flutter secure storage to the request header var storage = FlutterSecureStorage();
var token = await storage.read(key: Token.loginAccessTokenKey);
options.headers['Authorization'] = 'Bearer ${AuthCubit.token.accessToken}'; options.headers['Authorization'] = 'Bearer $token';
super.onRequest(options, handler); super.onRequest(options, handler);
} }

View File

@ -17,8 +17,8 @@ class HTTPService {
baseUrl: ApiEndpoints.baseUrl, baseUrl: ApiEndpoints.baseUrl,
receiveDataWhenStatusError: true, receiveDataWhenStatusError: true,
followRedirects: false, followRedirects: false,
connectTimeout: const Duration(seconds: 5), connectTimeout: const Duration(seconds: 30),
receiveTimeout: const Duration(seconds: 5), receiveTimeout: const Duration(seconds: 30),
), ),
); );
@ -62,8 +62,8 @@ class HTTPService {
queryParameters: queryParameters, queryParameters: queryParameters,
options: options, options: options,
); );
debugPrint("status code is ${response.statusCode}"); // debugPrint("status code is ${response.statusCode}");
debugPrint("response data is ${response.data}"); // debugPrint("response data is ${response.data}");
return expectedResponseModel(response.data); return expectedResponseModel(response.data);
} catch (error) { } catch (error) {
rethrow; rethrow;

View File

@ -1,26 +1,19 @@
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
import 'package:syncrow_app/features/app_layout/model/space_model.dart'; import 'package:syncrow_app/features/app_layout/model/space_model.dart';
import 'package:syncrow_app/features/auth/bloc/auth_cubit.dart'; import 'package:syncrow_app/features/auth/model/user_model.dart';
import 'package:syncrow_app/features/devices/model/room_model.dart';
import 'package:syncrow_app/services/api/api_links_endpoints.dart'; import 'package:syncrow_app/services/api/api_links_endpoints.dart';
import 'package:syncrow_app/services/api/http_service.dart'; import 'package:syncrow_app/services/api/http_service.dart';
class SpacesAPI { class SpacesAPI {
// static Future<Token> loginWithEmail(
// {required LoginWithEmailModel model}) async {
// final response = await HTTPService().post(
// path: ApiEndpoints.login,
// body: model.toJson(),
// showServerMessage: false,
// expectedResponseModel: (json) {
// Token token = Token.fromJson(json['data']);
// return token;
// });
// debugPrint("response: $response");
// return response;
// }
static Future<List<SpaceModel>> getSpaces() async { static Future<List<SpaceModel>> getSpaces() async {
var uuid =
await const FlutterSecureStorage().read(key: UserModel.userUuidKey);
final response = await HTTPService().get( final response = await HTTPService().get(
path: "${ApiEndpoints.spaces}/${AuthCubit.user!.uuid}", path: ApiEndpoints.spaces,
queryParameters: {
"userUuid": uuid,
},
showServerMessage: false, showServerMessage: false,
expectedResponseModel: (json) { expectedResponseModel: (json) {
List<SpaceModel> spaces = []; List<SpaceModel> spaces = [];
@ -32,4 +25,21 @@ class SpacesAPI {
); );
return response; return response;
} }
//get rooms by space id
static Future<List<RoomModel>> getRooms(int spaceId) async {
final response = await HTTPService().get(
path: ApiEndpoints.rooms,
queryParameters: {"homeId": spaceId},
showServerMessage: false,
expectedResponseModel: (json) {
List<RoomModel> rooms = [];
for (var room in json) {
rooms.add(RoomModel.fromJson(room));
}
return rooms;
},
);
return response;
}
} }