mirror of
https://github.com/SoPat712/RUSwipeShare.git
synced 2025-08-21 19:08:46 -04:00
Profile Page
This commit is contained in:
@@ -1,10 +1,11 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutterfire_ui/auth.dart';
|
||||||
import 'package:persistent_bottom_nav_bar/persistent_tab_view.dart';
|
import 'package:persistent_bottom_nav_bar/persistent_tab_view.dart';
|
||||||
|
import 'profile_screen_custom.dart';
|
||||||
import 'main_screen.dart';
|
import 'main_screen.dart';
|
||||||
|
|
||||||
class HomeScreen extends StatefulWidget {
|
class HomeScreen extends StatefulWidget {
|
||||||
const HomeScreen({ Key? key }) : super(key: key);
|
const HomeScreen({Key? key}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
_HomeScreenState createState() => _HomeScreenState();
|
_HomeScreenState createState() => _HomeScreenState();
|
||||||
@@ -12,25 +13,21 @@ class HomeScreen extends StatefulWidget {
|
|||||||
|
|
||||||
class _HomeScreenState extends State<HomeScreen> {
|
class _HomeScreenState extends State<HomeScreen> {
|
||||||
late PersistentTabController _controller;
|
late PersistentTabController _controller;
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_controller = PersistentTabController();
|
_controller = PersistentTabController();
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Widget> _buildScreens() => [
|
List<Widget> _buildScreens() => [
|
||||||
const MainScreen(),
|
const MainScreen(),
|
||||||
const MainScreen(),
|
const MainScreen(),
|
||||||
const MainScreen(),
|
const ProfileScreenCustom(),
|
||||||
const MainScreen(),
|
const MainScreen(),
|
||||||
];
|
];
|
||||||
|
|
||||||
List<PersistentBottomNavBarItem> _navBarsItems() => [
|
List<PersistentBottomNavBarItem> _navBarsItems() => [
|
||||||
PersistentBottomNavBarItem(
|
PersistentBottomNavBarItem(icon: const Icon(Icons.shopping_cart), title: "Buy", activeColorPrimary: Colors.blue, inactiveColorPrimary: Colors.grey, inactiveColorSecondary: Colors.purple),
|
||||||
icon: const Icon(Icons.shopping_cart),
|
|
||||||
title: "Buy",
|
|
||||||
activeColorPrimary: Colors.blue,
|
|
||||||
inactiveColorPrimary: Colors.grey,
|
|
||||||
inactiveColorSecondary: Colors.purple),
|
|
||||||
PersistentBottomNavBarItem(
|
PersistentBottomNavBarItem(
|
||||||
icon: const Icon(Icons.attach_money),
|
icon: const Icon(Icons.attach_money),
|
||||||
title: "Sell",
|
title: "Sell",
|
||||||
@@ -63,32 +60,34 @@ class _HomeScreenState extends State<HomeScreen> {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return PersistentTabView(
|
return PersistentTabView(
|
||||||
context,
|
context,
|
||||||
controller: _controller,
|
controller: _controller,
|
||||||
screens: _buildScreens(),
|
screens: _buildScreens(),
|
||||||
items: _navBarsItems(),
|
items: _navBarsItems(),
|
||||||
confineInSafeArea: true,
|
confineInSafeArea: true,
|
||||||
backgroundColor: Colors.white, // Default is Colors.white.
|
backgroundColor: Colors.white, // Default is Colors.white.
|
||||||
handleAndroidBackButtonPress: true, // Default is true.
|
handleAndroidBackButtonPress: true, // Default is true.
|
||||||
resizeToAvoidBottomInset: true, // This needs to be true if you want to move up the screen when keyboard appears. Default is true.
|
resizeToAvoidBottomInset: true, // This needs to be true if you want to move up the screen when keyboard appears. Default is true.
|
||||||
stateManagement: true, // Default is true.
|
stateManagement: true, // Default is true.
|
||||||
hideNavigationBarWhenKeyboardShows: true, // Recommended to set 'resizeToAvoidBottomInset' as true while using this argument. Default is true.
|
hideNavigationBarWhenKeyboardShows: true, // Recommended to set 'resizeToAvoidBottomInset' as true while using this argument. Default is true.
|
||||||
decoration: NavBarDecoration(
|
decoration: NavBarDecoration(
|
||||||
borderRadius: BorderRadius.circular(10.0),
|
borderRadius: BorderRadius.circular(10.0),
|
||||||
colorBehindNavBar: Colors.white,
|
colorBehindNavBar: Colors.white,
|
||||||
),
|
),
|
||||||
popAllScreensOnTapOfSelectedTab: true,
|
popAllScreensOnTapOfSelectedTab: true,
|
||||||
popActionScreens: PopActionScreensType.all,
|
popActionScreens: PopActionScreensType.all,
|
||||||
itemAnimationProperties: const ItemAnimationProperties( // Navigation Bar's items animation properties.
|
itemAnimationProperties: const ItemAnimationProperties(
|
||||||
duration: Duration(milliseconds: 200),
|
// Navigation Bar's items animation properties.
|
||||||
curve: Curves.ease,
|
duration: Duration(milliseconds: 200),
|
||||||
),
|
curve: Curves.ease,
|
||||||
screenTransitionAnimation: const ScreenTransitionAnimation( // Screen transition animation on change of selected tab.
|
),
|
||||||
animateTabTransition: true,
|
screenTransitionAnimation: const ScreenTransitionAnimation(
|
||||||
curve: Curves.ease,
|
// Screen transition animation on change of selected tab.
|
||||||
duration: Duration(milliseconds: 200),
|
animateTabTransition: true,
|
||||||
),
|
curve: Curves.ease,
|
||||||
navBarStyle: NavBarStyle.style13, // Choose the nav bar style with this property.
|
duration: Duration(milliseconds: 200),
|
||||||
|
),
|
||||||
|
navBarStyle: NavBarStyle.style13, // Choose the nav bar style with this property.
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
295
lib/profile_screen_custom.dart
Normal file
295
lib/profile_screen_custom.dart
Normal file
@@ -0,0 +1,295 @@
|
|||||||
|
// Copyright 2022, the Chromium project authors. Please see the AUTHORS file
|
||||||
|
// for details. All rights reserved. Use of this source code is governed by a
|
||||||
|
// BSD-style license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
import 'package:firebase_auth/firebase_auth.dart' show ActionCodeSettings, FirebaseAuth, FirebaseAuthException, User;
|
||||||
|
import 'package:flutter/cupertino.dart' hide Title;
|
||||||
|
import 'package:flutterfire_ui/i10n.dart';
|
||||||
|
import 'package:flutter/material.dart' hide Title;
|
||||||
|
import 'package:flutterfire_ui/auth.dart';
|
||||||
|
import 'package:flutterfire_ui/src/auth/widgets/internal/loading_button.dart';
|
||||||
|
|
||||||
|
import 'package:flutterfire_ui/src/auth/widgets/internal/universal_button.dart';
|
||||||
|
|
||||||
|
import 'package:flutterfire_ui/src/auth/screens/internal/multi_provider_screen.dart';
|
||||||
|
|
||||||
|
import 'package:flutterfire_ui/src/auth/widgets/internal/rebuild_scope.dart';
|
||||||
|
import 'package:flutterfire_ui/src/auth/widgets/internal/subtitle.dart';
|
||||||
|
import 'package:flutterfire_ui/src/auth/widgets/internal/universal_icon_button.dart';
|
||||||
|
|
||||||
|
class EditButton extends StatelessWidget {
|
||||||
|
final bool isEditing;
|
||||||
|
final VoidCallback? onPressed;
|
||||||
|
|
||||||
|
const EditButton({
|
||||||
|
Key? key,
|
||||||
|
required this.isEditing,
|
||||||
|
this.onPressed,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
|
||||||
|
return UniversalIconButton(
|
||||||
|
materialIcon: isEditing ? Icons.check : Icons.edit,
|
||||||
|
cupertinoIcon: isEditing ? CupertinoIcons.check_mark : CupertinoIcons.pen,
|
||||||
|
color: theme.colorScheme.secondary,
|
||||||
|
onPressed: () {
|
||||||
|
onPressed?.call();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class EmailVerificationBadge extends StatefulWidget {
|
||||||
|
final FirebaseAuth auth;
|
||||||
|
final ActionCodeSettings? actionCodeSettings;
|
||||||
|
const EmailVerificationBadge({
|
||||||
|
Key? key,
|
||||||
|
required this.auth,
|
||||||
|
this.actionCodeSettings,
|
||||||
|
}) : super(key: key);
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<EmailVerificationBadge> createState() => _EmailVerificationBadgeState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _EmailVerificationBadgeState extends State<EmailVerificationBadge> {
|
||||||
|
late final service = EmailVerificationService(widget.auth)
|
||||||
|
..addListener(() {
|
||||||
|
setState(() {});
|
||||||
|
})
|
||||||
|
..reload();
|
||||||
|
|
||||||
|
EmailVerificationState get state => service.state;
|
||||||
|
|
||||||
|
User get user {
|
||||||
|
return widget.auth.currentUser!;
|
||||||
|
}
|
||||||
|
|
||||||
|
TargetPlatform get platform {
|
||||||
|
return Theme.of(context).platform;
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
if (state == EmailVerificationState.dismissed || state == EmailVerificationState.unresolved || state == EmailVerificationState.verified) {
|
||||||
|
return const SizedBox.shrink();
|
||||||
|
}
|
||||||
|
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 16),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
Container(
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Colors.yellow,
|
||||||
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(12),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
Subtitle(
|
||||||
|
text: state == EmailVerificationState.sent || state == EmailVerificationState.pending ? 'Verification email sent' : 'Email is not verified',
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
if (state == EmailVerificationState.pending) ...[
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
const Text(
|
||||||
|
'Please check your email and click the link to verify your email address.',
|
||||||
|
),
|
||||||
|
]
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
if (state == EmailVerificationState.pending)
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: const [
|
||||||
|
LoadingIndicator(size: 16, borderWidth: 0.5),
|
||||||
|
SizedBox(width: 16),
|
||||||
|
Text('Waiting for email verification'),
|
||||||
|
],
|
||||||
|
)
|
||||||
|
else
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
|
children: [
|
||||||
|
if (state != EmailVerificationState.sent && state != EmailVerificationState.sending)
|
||||||
|
UniversalButton(
|
||||||
|
variant: ButtonVariant.text,
|
||||||
|
color: Theme.of(context).colorScheme.error,
|
||||||
|
text: 'Dismiss',
|
||||||
|
onPressed: () {
|
||||||
|
setState(service.dismiss);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
if (state != EmailVerificationState.sent)
|
||||||
|
LoadingButton(
|
||||||
|
isLoading: state == EmailVerificationState.sending,
|
||||||
|
label: 'Send verification email',
|
||||||
|
onTap: () {
|
||||||
|
service.sendVerificationEmail(
|
||||||
|
platform,
|
||||||
|
widget.actionCodeSettings,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
)
|
||||||
|
else
|
||||||
|
UniversalButton(
|
||||||
|
variant: ButtonVariant.text,
|
||||||
|
text: 'Ok',
|
||||||
|
onPressed: () {
|
||||||
|
setState(service.dismiss);
|
||||||
|
},
|
||||||
|
)
|
||||||
|
],
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ProfileScreenCustom extends MultiProviderScreen {
|
||||||
|
final List<Widget> children;
|
||||||
|
final Color? avatarPlaceholderColor;
|
||||||
|
final ShapeBorder? avatarShape;
|
||||||
|
final double? avatarSize;
|
||||||
|
final List<FlutterFireUIAction>? actions;
|
||||||
|
final AppBar? appBar;
|
||||||
|
final CupertinoNavigationBar? cupertinoNavigationBar;
|
||||||
|
final ActionCodeSettings? actionCodeSettings;
|
||||||
|
final Set<FlutterFireUIStyle>? styles;
|
||||||
|
|
||||||
|
const ProfileScreenCustom({
|
||||||
|
Key? key,
|
||||||
|
FirebaseAuth? auth,
|
||||||
|
List<ProviderConfiguration>? providerConfigs,
|
||||||
|
this.avatarPlaceholderColor,
|
||||||
|
this.avatarShape,
|
||||||
|
this.avatarSize,
|
||||||
|
this.children = const [],
|
||||||
|
this.actions,
|
||||||
|
this.appBar,
|
||||||
|
this.cupertinoNavigationBar,
|
||||||
|
this.actionCodeSettings,
|
||||||
|
this.styles,
|
||||||
|
}) : super(key: key, providerConfigs: providerConfigs, auth: auth);
|
||||||
|
|
||||||
|
Future<bool> _reauthenticate(BuildContext context) {
|
||||||
|
return showReauthenticateDialog(
|
||||||
|
context: context,
|
||||||
|
providerConfigs: providerConfigs,
|
||||||
|
auth: auth,
|
||||||
|
onSignedIn: () => Navigator.of(context).pop(true),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<ProviderConfiguration> getLinkedProviders(User user) {
|
||||||
|
return providerConfigs.where((config) => user.isProviderLinked(config.providerId)).toList();
|
||||||
|
}
|
||||||
|
|
||||||
|
List<ProviderConfiguration> getAvailableProviders(User user) {
|
||||||
|
return providerConfigs.where((config) => !user.isProviderLinked(config.providerId)).toList();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return FlutterFireUITheme(
|
||||||
|
styles: styles ?? const {},
|
||||||
|
child: Builder(builder: buildPage),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget buildPage(BuildContext context) {
|
||||||
|
final isCupertino = CupertinoUserInterfaceLevel.maybeOf(context) != null;
|
||||||
|
final providersScopeKey = RebuildScopeKey();
|
||||||
|
final emailVerificationScopeKey = RebuildScopeKey();
|
||||||
|
|
||||||
|
final user = auth.currentUser!;
|
||||||
|
|
||||||
|
final content = Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
|
children: [
|
||||||
|
Align(
|
||||||
|
child: UserAvatar(
|
||||||
|
auth: auth,
|
||||||
|
placeholderColor: avatarPlaceholderColor,
|
||||||
|
shape: avatarShape,
|
||||||
|
size: avatarSize,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Align(child: EditableUserDisplayName(auth: auth)),
|
||||||
|
if (!user.emailVerified) ...[
|
||||||
|
RebuildScope(
|
||||||
|
builder: (context) {
|
||||||
|
if (user.emailVerified) {
|
||||||
|
return const SizedBox.shrink();
|
||||||
|
}
|
||||||
|
|
||||||
|
return EmailVerificationBadge(
|
||||||
|
auth: auth,
|
||||||
|
actionCodeSettings: actionCodeSettings,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
scopeKey: emailVerificationScopeKey,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
...children,
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
SignOutButton(
|
||||||
|
auth: auth,
|
||||||
|
variant: ButtonVariant.outlined,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
final body = Padding(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Center(
|
||||||
|
child: LayoutBuilder(
|
||||||
|
builder: (context, constraints) {
|
||||||
|
if (constraints.maxWidth > 500) {
|
||||||
|
return ConstrainedBox(
|
||||||
|
constraints: const BoxConstraints(maxWidth: 500),
|
||||||
|
child: content,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
Widget child = SafeArea(child: SingleChildScrollView(child: body));
|
||||||
|
|
||||||
|
if (isCupertino) {
|
||||||
|
child = CupertinoPageScaffold(
|
||||||
|
navigationBar: cupertinoNavigationBar,
|
||||||
|
child: SafeArea(
|
||||||
|
child: SingleChildScrollView(child: child),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
child = Scaffold(
|
||||||
|
appBar: appBar,
|
||||||
|
body: SafeArea(
|
||||||
|
child: SingleChildScrollView(child: body),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return FlutterFireUIActions(
|
||||||
|
actions: actions ?? const [],
|
||||||
|
child: child,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user