ente/lib/ui/home_widget.dart

226 lines
7 KiB
Dart
Raw Normal View History

2020-05-05 12:56:24 +00:00
import 'dart:async';
2020-04-27 15:11:29 +00:00
import 'package:flutter/cupertino.dart';
2020-04-14 15:36:18 +00:00
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
2020-04-14 15:36:18 +00:00
import 'package:flutter/widgets.dart';
import 'package:photos/core/configuration.dart';
2020-05-04 20:03:06 +00:00
import 'package:photos/core/event_bus.dart';
import 'package:photos/events/local_photos_updated_event.dart';
2020-10-26 11:18:00 +00:00
import 'package:photos/events/tab_changed_event.dart';
import 'package:photos/models/filters/important_items_filter.dart';
2020-06-19 23:03:26 +00:00
import 'package:photos/models/file.dart';
2020-10-03 17:58:26 +00:00
import 'package:photos/repositories/file_repository.dart';
import 'package:photos/models/selected_files.dart';
2020-10-03 17:58:26 +00:00
import 'package:photos/services/sync_service.dart';
import 'package:photos/ui/collections_gallery_widget.dart';
2020-05-05 15:50:36 +00:00
import 'package:photos/ui/gallery.dart';
2020-05-01 18:20:12 +00:00
import 'package:photos/ui/gallery_app_bar_widget.dart';
2020-06-15 19:57:48 +00:00
import 'package:photos/ui/loading_photos_widget.dart';
2020-07-15 20:48:46 +00:00
import 'package:photos/ui/loading_widget.dart';
2020-07-21 10:25:19 +00:00
import 'package:photos/ui/memories_widget.dart';
2020-06-02 11:32:35 +00:00
import 'package:photos/ui/search_page.dart';
2020-10-03 17:56:18 +00:00
import 'package:photos/services/user_service.dart';
import 'package:photos/ui/shared_collections_gallery.dart';
2020-05-02 17:12:03 +00:00
import 'package:photos/utils/logging_util.dart';
import 'package:shake/shake.dart';
import 'package:logging/logging.dart';
import 'package:uni_links/uni_links.dart';
2020-04-14 15:36:18 +00:00
class HomeWidget extends StatefulWidget {
final String title;
const HomeWidget(this.title, {Key key}) : super(key: key);
@override
State<StatefulWidget> createState() => _HomeWidgetState();
}
class _HomeWidgetState extends State<HomeWidget> {
2020-05-05 15:50:36 +00:00
static final importantItemsFilter = ImportantItemsFilter();
2020-05-04 20:10:11 +00:00
final _logger = Logger("HomeWidgetState");
final _sharedCollectionGallery = SharedCollectionGallery();
final _deviceFolderGalleryWidget = CollectionsGalleryWidget();
final _selectedFiles = SelectedFiles();
final _memoriesWidget = MemoriesWidget();
2020-06-15 19:57:48 +00:00
2020-05-04 20:10:11 +00:00
ShakeDetector _detector;
int _selectedNavBarItem = 0;
2020-06-15 19:57:48 +00:00
StreamSubscription<LocalPhotosUpdatedEvent>
_localPhotosUpdatedEventSubscription;
2020-10-26 11:18:00 +00:00
StreamSubscription<TabChangedEvent> _tabChangedEventSubscription;
2020-04-14 15:36:18 +00:00
2020-05-02 17:12:03 +00:00
@override
void initState() {
2020-05-05 12:56:24 +00:00
_detector = ShakeDetector.autoStart(
2020-05-04 15:45:47 +00:00
shakeThresholdGravity: 3,
2020-05-02 17:21:50 +00:00
onPhoneShake: () {
2020-05-04 20:10:11 +00:00
_logger.info("Emailing logs");
2020-05-02 17:21:50 +00:00
LoggingUtil.instance.emailLogs();
});
2020-06-15 19:57:48 +00:00
_localPhotosUpdatedEventSubscription =
Bus.instance.on<LocalPhotosUpdatedEvent>().listen((event) {
setState(() {});
});
2020-10-26 11:18:00 +00:00
_tabChangedEventSubscription =
Bus.instance.on<TabChangedEvent>().listen((event) {
setState(() {
_selectedNavBarItem = event.selectedIndex;
});
});
_initDeepLinks();
2020-05-04 20:03:06 +00:00
super.initState();
2020-05-02 17:12:03 +00:00
}
2020-04-14 15:36:18 +00:00
@override
Widget build(BuildContext context) {
2020-04-27 11:13:35 +00:00
return Scaffold(
appBar: GalleryAppBarWidget(
2020-05-17 14:23:37 +00:00
GalleryAppBarType.homepage,
2020-04-27 11:13:35 +00:00
widget.title,
_selectedFiles,
2020-05-17 12:39:38 +00:00
"/",
2020-04-27 11:13:35 +00:00
),
bottomNavigationBar: _buildBottomNavigationBar(),
body: IndexedStack(
children: <Widget>[
2020-10-03 17:58:26 +00:00
SyncService.instance.hasScannedDisk()
2020-07-15 20:48:46 +00:00
? _getMainGalleryWidget()
: LoadingPhotosWidget(),
_deviceFolderGalleryWidget,
_sharedCollectionGallery,
],
index: _selectedNavBarItem,
),
2020-06-02 11:32:35 +00:00
floatingActionButton: FloatingActionButton(
onPressed: () {
Navigator.of(context).push(
MaterialPageRoute(
builder: (BuildContext context) {
return SearchPage();
},
),
);
},
2020-06-02 11:40:34 +00:00
child: Icon(
Icons.search,
size: 28,
),
2020-06-02 11:32:35 +00:00
elevation: 1,
2020-06-02 11:40:34 +00:00
backgroundColor: Colors.black38,
2020-08-25 00:54:32 +00:00
foregroundColor: Theme.of(context).accentColor,
2020-06-02 11:32:35 +00:00
),
);
}
Future<bool> _initDeepLinks() async {
// Platform messages may fail, so we use a try/catch PlatformException.
try {
String initialLink = await getInitialLink();
// Parse the link and warn the user, if it is not correct,
// but keep in mind it could be `null`.
if (initialLink != null) {
_logger.info("Initial link received: " + initialLink);
_getCredentials(context, initialLink);
return true;
} else {
_logger.info("No initial link received.");
}
} on PlatformException {
// Handle exception by warning the user their action did not succeed
// return?
_logger.severe("PlatformException thrown while getting initial link");
}
// Attach a listener to the stream
getLinksStream().listen((String link) {
_logger.info("Link received: " + link);
_getCredentials(context, link);
}, onError: (err) {
_logger.severe(err);
});
return false;
}
void _getCredentials(BuildContext context, String link) {
if (Configuration.instance.hasConfiguredAccount()) {
return;
}
final ott = Uri.parse(link).queryParameters["ott"];
_logger.info("Ott: " + ott);
2020-10-03 17:56:18 +00:00
UserService.instance.getCredentials(context, ott);
}
2020-06-15 19:57:48 +00:00
Widget _getMainGalleryWidget() {
2020-07-15 20:48:46 +00:00
return FutureBuilder(
future: FileRepository.instance.loadFiles().then((files) {
return _getFilteredPhotos(files);
}),
builder: (context, snapshot) {
if (snapshot.hasData) {
return Gallery(
syncLoader: () {
return _getFilteredPhotos(FileRepository.instance.files);
},
reloadEvent: Bus.instance.on<LocalPhotosUpdatedEvent>(),
2020-10-03 17:58:26 +00:00
onRefresh: SyncService.instance.sync,
2020-07-15 20:48:46 +00:00
tagPrefix: "home_gallery",
selectedFiles: _selectedFiles,
headerWidget: _memoriesWidget,
2020-07-15 20:48:46 +00:00
);
} else if (snapshot.hasError) {
return Center(child: Text(snapshot.error.toString()));
} else {
return loadWidget;
}
},
2020-06-06 14:24:00 +00:00
);
}
BottomNavigationBar _buildBottomNavigationBar() {
return BottomNavigationBar(
items: const <BottomNavigationBarItem>[
BottomNavigationBarItem(
2020-06-15 19:59:18 +00:00
icon: Icon(Icons.photo_library),
label: "Photos",
2020-04-14 15:36:18 +00:00
),
BottomNavigationBarItem(
icon: Icon(Icons.folder_special),
label: "Collections",
),
BottomNavigationBarItem(
icon: Icon(Icons.folder_shared),
label: "Shared",
),
],
currentIndex: _selectedNavBarItem,
2020-08-25 00:54:32 +00:00
selectedItemColor: Theme.of(context).accentColor,
onTap: (index) {
setState(() {
_selectedNavBarItem = index;
});
},
);
}
2020-05-02 17:12:03 +00:00
2020-06-19 23:03:26 +00:00
List<File> _getFilteredPhotos(List<File> unfilteredFiles) {
_logger.info("Filtering " + unfilteredFiles.length.toString());
2020-06-19 23:03:26 +00:00
final List<File> filteredPhotos = List<File>();
for (File file in unfilteredFiles) {
if (importantItemsFilter.shouldInclude(file)) {
filteredPhotos.add(file);
2020-05-05 15:50:36 +00:00
}
}
2020-07-12 21:45:50 +00:00
_logger.info("Filtered down to " + filteredPhotos.length.toString());
2020-05-05 15:50:36 +00:00
return filteredPhotos;
}
2020-05-02 17:12:03 +00:00
@override
void dispose() {
2020-05-04 20:10:11 +00:00
_detector.stopListening();
2020-06-15 19:57:48 +00:00
_localPhotosUpdatedEventSubscription.cancel();
2020-10-26 11:18:00 +00:00
_tabChangedEventSubscription.cancel();
2020-05-02 17:12:03 +00:00
super.dispose();
}
2020-04-14 15:36:18 +00:00
}