opt: AppBar渐变背景
This commit is contained in:
@ -3,6 +3,7 @@ import 'dart:async';
|
|||||||
import 'package:custom_sliding_segmented_control/custom_sliding_segmented_control.dart';
|
import 'package:custom_sliding_segmented_control/custom_sliding_segmented_control.dart';
|
||||||
import 'package:easy_debounce/easy_throttle.dart';
|
import 'package:easy_debounce/easy_throttle.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:hive/hive.dart';
|
import 'package:hive/hive.dart';
|
||||||
import 'package:pilipala/common/skeleton/dynamic_card.dart';
|
import 'package:pilipala/common/skeleton/dynamic_card.dart';
|
||||||
@ -77,10 +78,14 @@ class _DynamicsPageState extends State<DynamicsPage>
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
super.build(context);
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
scrolledUnderElevation: 0,
|
scrolledUnderElevation: 0,
|
||||||
titleSpacing: 0,
|
backgroundColor: Colors.transparent,
|
||||||
|
systemOverlayStyle: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? SystemUiOverlayStyle.light
|
||||||
|
: SystemUiOverlayStyle.dark,
|
||||||
title: SizedBox(
|
title: SizedBox(
|
||||||
height: 34,
|
height: 34,
|
||||||
child: Stack(
|
child: Stack(
|
||||||
|
@ -46,104 +46,66 @@ class _HomePageState extends State<HomePage>
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
super.build(context);
|
||||||
Brightness currentBrightness = MediaQuery.of(context).platformBrightness;
|
|
||||||
// 设置状态栏图标的亮度
|
|
||||||
if (_homeController.enableGradientBg) {
|
|
||||||
SystemChrome.setSystemUIOverlayStyle(SystemUiOverlayStyle(
|
|
||||||
statusBarIconBrightness: currentBrightness == Brightness.light
|
|
||||||
? Brightness.dark
|
|
||||||
: Brightness.light,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
extendBody: true,
|
extendBody: true,
|
||||||
extendBodyBehindAppBar: true,
|
extendBodyBehindAppBar: true,
|
||||||
appBar: _homeController.enableGradientBg
|
backgroundColor: Colors.transparent,
|
||||||
? null
|
appBar: AppBar(
|
||||||
: AppBar(toolbarHeight: 0, elevation: 0),
|
toolbarHeight: 0,
|
||||||
body: Stack(
|
elevation: 0,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
systemOverlayStyle: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? SystemUiOverlayStyle.light
|
||||||
|
: SystemUiOverlayStyle.dark,
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
children: [
|
children: [
|
||||||
// gradient background
|
CustomAppBar(
|
||||||
if (_homeController.enableGradientBg) ...[
|
stream: _homeController.hideSearchBar
|
||||||
Align(
|
? stream
|
||||||
alignment: Alignment.topLeft,
|
: StreamController<bool>.broadcast().stream,
|
||||||
child: Opacity(
|
ctr: _homeController,
|
||||||
opacity: 0.6,
|
callback: showUserBottomSheet,
|
||||||
child: Container(
|
),
|
||||||
width: MediaQuery.of(context).size.width,
|
if (_homeController.tabs.length > 1) ...[
|
||||||
height: MediaQuery.of(context).size.height,
|
if (_homeController.enableGradientBg) ...[
|
||||||
decoration: BoxDecoration(
|
const CustomTabs(),
|
||||||
gradient: LinearGradient(
|
] else ...[
|
||||||
colors: [
|
Container(
|
||||||
Theme.of(context)
|
width: double.infinity,
|
||||||
.colorScheme
|
height: 42,
|
||||||
.primary
|
padding: const EdgeInsets.only(top: 4),
|
||||||
.withOpacity(0.4),
|
child: Align(
|
||||||
Theme.of(context)
|
alignment: Alignment.center,
|
||||||
.colorScheme
|
child: TabBar(
|
||||||
.surface
|
controller: _homeController.tabController,
|
||||||
.withOpacity(0.5),
|
tabs: [
|
||||||
Theme.of(context).colorScheme.surface
|
for (var i in _homeController.tabs) Tab(text: i['label'])
|
||||||
],
|
],
|
||||||
begin: Alignment.topCenter,
|
isScrollable: true,
|
||||||
end: Alignment.bottomCenter,
|
dividerColor: Colors.transparent,
|
||||||
stops: const [0.1, 0.3, 0.5]),
|
enableFeedback: true,
|
||||||
|
splashBorderRadius: BorderRadius.circular(10),
|
||||||
|
tabAlignment: TabAlignment.center,
|
||||||
|
onTap: (value) {
|
||||||
|
feedBack();
|
||||||
|
if (_homeController.initialIndex.value == value) {
|
||||||
|
_homeController.tabsCtrList[value]().animateToTop();
|
||||||
|
}
|
||||||
|
_homeController.initialIndex.value = value;
|
||||||
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
],
|
|
||||||
Column(
|
|
||||||
children: [
|
|
||||||
CustomAppBar(
|
|
||||||
stream: _homeController.hideSearchBar
|
|
||||||
? stream
|
|
||||||
: StreamController<bool>.broadcast().stream,
|
|
||||||
ctr: _homeController,
|
|
||||||
callback: showUserBottomSheet,
|
|
||||||
),
|
|
||||||
if (_homeController.tabs.length > 1) ...[
|
|
||||||
if (_homeController.enableGradientBg) ...[
|
|
||||||
const CustomTabs(),
|
|
||||||
] else ...[
|
|
||||||
const SizedBox(height: 4),
|
|
||||||
SizedBox(
|
|
||||||
width: double.infinity,
|
|
||||||
height: 42,
|
|
||||||
child: Align(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: TabBar(
|
|
||||||
controller: _homeController.tabController,
|
|
||||||
tabs: [
|
|
||||||
for (var i in _homeController.tabs)
|
|
||||||
Tab(text: i['label'])
|
|
||||||
],
|
|
||||||
isScrollable: true,
|
|
||||||
dividerColor: Colors.transparent,
|
|
||||||
enableFeedback: true,
|
|
||||||
splashBorderRadius: BorderRadius.circular(10),
|
|
||||||
tabAlignment: TabAlignment.center,
|
|
||||||
onTap: (value) {
|
|
||||||
feedBack();
|
|
||||||
if (_homeController.initialIndex.value == value) {
|
|
||||||
_homeController.tabsCtrList[value]().animateToTop();
|
|
||||||
}
|
|
||||||
_homeController.initialIndex.value = value;
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
] else ...[
|
|
||||||
const SizedBox(height: 6),
|
|
||||||
],
|
|
||||||
Expanded(
|
|
||||||
child: TabBarView(
|
|
||||||
controller: _homeController.tabController,
|
|
||||||
children: _homeController.tabsPageList,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
|
] else ...[
|
||||||
|
const SizedBox(height: 6),
|
||||||
|
],
|
||||||
|
Expanded(
|
||||||
|
child: TabBarView(
|
||||||
|
controller: _homeController.tabController,
|
||||||
|
children: _homeController.tabsPageList,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@ -280,7 +242,10 @@ class DefaultUser extends StatelessWidget {
|
|||||||
style: ButtonStyle(
|
style: ButtonStyle(
|
||||||
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
padding: MaterialStateProperty.all(EdgeInsets.zero),
|
||||||
backgroundColor: MaterialStateProperty.resolveWith((states) {
|
backgroundColor: MaterialStateProperty.resolveWith((states) {
|
||||||
return Theme.of(context).colorScheme.onInverseSurface;
|
return Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSecondaryContainer
|
||||||
|
.withOpacity(0.05);
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
onPressed: () => callback?.call(),
|
onPressed: () => callback?.call(),
|
||||||
@ -317,7 +282,7 @@ class _CustomTabsState extends State<CustomTabs> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Container(
|
return Container(
|
||||||
height: 44,
|
height: 44,
|
||||||
margin: const EdgeInsets.only(top: 4),
|
margin: const EdgeInsets.only(top: 8),
|
||||||
child: Obx(
|
child: Obx(
|
||||||
() => ListView.separated(
|
() => ListView.separated(
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 14.0),
|
padding: const EdgeInsets.symmetric(horizontal: 14.0),
|
||||||
|
@ -26,6 +26,7 @@ class MainController extends GetxController {
|
|||||||
Box userInfoCache = GStrorage.userInfo;
|
Box userInfoCache = GStrorage.userInfo;
|
||||||
RxBool userLogin = false.obs;
|
RxBool userLogin = false.obs;
|
||||||
late Rx<DynamicBadgeMode> dynamicBadgeType = DynamicBadgeMode.number.obs;
|
late Rx<DynamicBadgeMode> dynamicBadgeType = DynamicBadgeMode.number.obs;
|
||||||
|
late bool enableGradientBg;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
@ -44,6 +45,8 @@ class MainController extends GetxController {
|
|||||||
if (dynamicBadgeType.value != DynamicBadgeMode.hidden) {
|
if (dynamicBadgeType.value != DynamicBadgeMode.hidden) {
|
||||||
getUnreadDynamic();
|
getUnreadDynamic();
|
||||||
}
|
}
|
||||||
|
enableGradientBg =
|
||||||
|
setting.get(SettingBoxKey.enableGradientBg, defaultValue: true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void onBackPressed(BuildContext context) {
|
void onBackPressed(BuildContext context) {
|
||||||
|
@ -117,14 +117,47 @@ class _MainAppState extends State<MainApp> with SingleTickerProviderStateMixin {
|
|||||||
},
|
},
|
||||||
child: Scaffold(
|
child: Scaffold(
|
||||||
extendBody: true,
|
extendBody: true,
|
||||||
body: PageView(
|
body: Stack(
|
||||||
physics: const NeverScrollableScrollPhysics(),
|
children: [
|
||||||
controller: _mainController.pageController,
|
if (_mainController.enableGradientBg)
|
||||||
onPageChanged: (index) {
|
Align(
|
||||||
_mainController.selectedIndex = index;
|
alignment: Alignment.topLeft,
|
||||||
setState(() {});
|
child: Opacity(
|
||||||
},
|
opacity: 0.6,
|
||||||
children: _mainController.pages,
|
child: Container(
|
||||||
|
width: MediaQuery.of(context).size.width,
|
||||||
|
// height: MediaQuery.of(context).size.height,
|
||||||
|
height: MediaQuery.of(context).padding.top + 400,
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
gradient: LinearGradient(
|
||||||
|
colors: [
|
||||||
|
Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.primary
|
||||||
|
.withOpacity(0.6),
|
||||||
|
Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.surface
|
||||||
|
.withOpacity(0.3),
|
||||||
|
Theme.of(context).colorScheme.surface
|
||||||
|
],
|
||||||
|
begin: Alignment.topCenter,
|
||||||
|
end: Alignment.bottomCenter,
|
||||||
|
stops: const [0.1, 0.8, 1]),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
PageView(
|
||||||
|
physics: const NeverScrollableScrollPhysics(),
|
||||||
|
controller: _mainController.pageController,
|
||||||
|
onPageChanged: (index) {
|
||||||
|
_mainController.selectedIndex = index;
|
||||||
|
setState(() {});
|
||||||
|
},
|
||||||
|
children: _mainController.pages,
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
bottomNavigationBar: _mainController.navigationBars.length > 1
|
bottomNavigationBar: _mainController.navigationBars.length > 1
|
||||||
? StreamBuilder(
|
? StreamBuilder(
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:get/get.dart';
|
import 'package:get/get.dart';
|
||||||
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
import 'package:pilipala/common/widgets/network_img_layer.dart';
|
||||||
import 'package:pilipala/models/user/fav_folder.dart';
|
import 'package:pilipala/models/user/fav_folder.dart';
|
||||||
import 'package:pilipala/pages/media/index.dart';
|
import 'package:pilipala/pages/media/index.dart';
|
||||||
import 'package:pilipala/utils/main_stream.dart';
|
|
||||||
import 'package:pilipala/utils/utils.dart';
|
import 'package:pilipala/utils/utils.dart';
|
||||||
|
|
||||||
class MediaPage extends StatefulWidget {
|
class MediaPage extends StatefulWidget {
|
||||||
@ -46,7 +46,16 @@ class _MediaPageState extends State<MediaPage>
|
|||||||
super.build(context);
|
super.build(context);
|
||||||
Color primary = Theme.of(context).colorScheme.primary;
|
Color primary = Theme.of(context).colorScheme.primary;
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(toolbarHeight: 30),
|
backgroundColor: Colors.transparent,
|
||||||
|
appBar: AppBar(
|
||||||
|
elevation: 0,
|
||||||
|
scrolledUnderElevation: 0,
|
||||||
|
toolbarHeight: 30,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
systemOverlayStyle: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? SystemUiOverlayStyle.light
|
||||||
|
: SystemUiOverlayStyle.dark,
|
||||||
|
),
|
||||||
body: SingleChildScrollView(
|
body: SingleChildScrollView(
|
||||||
controller: mediaController.scrollController,
|
controller: mediaController.scrollController,
|
||||||
child: Column(
|
child: Column(
|
||||||
|
@ -17,13 +17,10 @@ class RankController extends GetxController with GetTickerProviderStateMixin {
|
|||||||
Box setting = GStrorage.setting;
|
Box setting = GStrorage.setting;
|
||||||
late final StreamController<bool> searchBarStream =
|
late final StreamController<bool> searchBarStream =
|
||||||
StreamController<bool>.broadcast();
|
StreamController<bool>.broadcast();
|
||||||
late bool enableGradientBg;
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
super.onInit();
|
super.onInit();
|
||||||
enableGradientBg =
|
|
||||||
setting.get(SettingBoxKey.enableGradientBg, defaultValue: true);
|
|
||||||
// 进行tabs配置
|
// 进行tabs配置
|
||||||
setTabConfig();
|
setTabConfig();
|
||||||
}
|
}
|
||||||
|
@ -31,94 +31,56 @@ class _RankPageState extends State<RankPage>
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
super.build(context);
|
super.build(context);
|
||||||
Brightness currentBrightness = MediaQuery.of(context).platformBrightness;
|
|
||||||
// 设置状态栏图标的亮度
|
|
||||||
if (_rankController.enableGradientBg) {
|
|
||||||
SystemChrome.setSystemUIOverlayStyle(SystemUiOverlayStyle(
|
|
||||||
statusBarIconBrightness: currentBrightness == Brightness.light
|
|
||||||
? Brightness.dark
|
|
||||||
: Brightness.light,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
extendBody: true,
|
extendBody: true,
|
||||||
extendBodyBehindAppBar: false,
|
extendBodyBehindAppBar: true,
|
||||||
appBar: _rankController.enableGradientBg
|
backgroundColor: Colors.transparent,
|
||||||
? null
|
appBar: AppBar(
|
||||||
: AppBar(toolbarHeight: 0, elevation: 0),
|
toolbarHeight: 0,
|
||||||
body: Stack(
|
elevation: 0,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
systemOverlayStyle: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? SystemUiOverlayStyle.light
|
||||||
|
: SystemUiOverlayStyle.dark,
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
children: [
|
children: [
|
||||||
// gradient background
|
const CustomAppBar(),
|
||||||
if (_rankController.enableGradientBg) ...[
|
if (_rankController.tabs.length > 1) ...[
|
||||||
Align(
|
const SizedBox(height: 4),
|
||||||
alignment: Alignment.topLeft,
|
SizedBox(
|
||||||
child: Opacity(
|
width: double.infinity,
|
||||||
opacity: 0.6,
|
height: 42,
|
||||||
child: Container(
|
child: Align(
|
||||||
width: MediaQuery.of(context).size.width,
|
alignment: Alignment.center,
|
||||||
height: MediaQuery.of(context).size.height,
|
child: TabBar(
|
||||||
decoration: BoxDecoration(
|
controller: _rankController.tabController,
|
||||||
gradient: LinearGradient(
|
tabs: [
|
||||||
colors: [
|
for (var i in _rankController.tabs) Tab(text: i['label'])
|
||||||
Theme.of(context)
|
],
|
||||||
.colorScheme
|
isScrollable: true,
|
||||||
.primary
|
dividerColor: Colors.transparent,
|
||||||
.withOpacity(0.9),
|
enableFeedback: true,
|
||||||
Theme.of(context)
|
splashBorderRadius: BorderRadius.circular(10),
|
||||||
.colorScheme
|
tabAlignment: TabAlignment.center,
|
||||||
.primary
|
onTap: (value) {
|
||||||
.withOpacity(0.5),
|
feedBack();
|
||||||
Theme.of(context).colorScheme.surface
|
if (_rankController.initialIndex.value == value) {
|
||||||
],
|
_rankController.tabsCtrList[value].animateToTop();
|
||||||
begin: Alignment.topLeft,
|
}
|
||||||
end: Alignment.bottomRight,
|
_rankController.initialIndex.value = value;
|
||||||
stops: const [0, 0.0034, 0.34]),
|
},
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
] else ...[
|
||||||
|
const SizedBox(height: 6),
|
||||||
],
|
],
|
||||||
Column(
|
Expanded(
|
||||||
children: [
|
child: TabBarView(
|
||||||
const CustomAppBar(),
|
controller: _rankController.tabController,
|
||||||
if (_rankController.tabs.length > 1) ...[
|
children: _rankController.tabsPageList,
|
||||||
const SizedBox(height: 4),
|
),
|
||||||
SizedBox(
|
|
||||||
width: double.infinity,
|
|
||||||
height: 42,
|
|
||||||
child: Align(
|
|
||||||
alignment: Alignment.center,
|
|
||||||
child: TabBar(
|
|
||||||
controller: _rankController.tabController,
|
|
||||||
tabs: [
|
|
||||||
for (var i in _rankController.tabs)
|
|
||||||
Tab(text: i['label'])
|
|
||||||
],
|
|
||||||
isScrollable: true,
|
|
||||||
dividerColor: Colors.transparent,
|
|
||||||
enableFeedback: true,
|
|
||||||
splashBorderRadius: BorderRadius.circular(10),
|
|
||||||
tabAlignment: TabAlignment.center,
|
|
||||||
onTap: (value) {
|
|
||||||
feedBack();
|
|
||||||
if (_rankController.initialIndex.value == value) {
|
|
||||||
_rankController.tabsCtrList[value].animateToTop();
|
|
||||||
}
|
|
||||||
_rankController.initialIndex.value = value;
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
] else ...[
|
|
||||||
const SizedBox(height: 6),
|
|
||||||
],
|
|
||||||
Expanded(
|
|
||||||
child: TabBarView(
|
|
||||||
controller: _rankController.tabController,
|
|
||||||
children: _rankController.tabsPageList,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
@ -4,6 +4,7 @@ import 'dart:ui';
|
|||||||
|
|
||||||
import 'package:extended_nested_scroll_view/extended_nested_scroll_view.dart';
|
import 'package:extended_nested_scroll_view/extended_nested_scroll_view.dart';
|
||||||
import 'package:floating/floating.dart';
|
import 'package:floating/floating.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
||||||
import 'package:flutter_svg/svg.dart';
|
import 'package:flutter_svg/svg.dart';
|
||||||
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
||||||
|
Reference in New Issue
Block a user