diff --git a/lib/blocs/drive_create/drive_create_cubit.dart b/lib/blocs/drive_create/drive_create_cubit.dart index e6e6960717..66bb94e1e7 100644 --- a/lib/blocs/drive_create/drive_create_cubit.dart +++ b/lib/blocs/drive_create/drive_create_cubit.dart @@ -98,6 +98,7 @@ class DriveCreateCubit extends Cubit { id: drive.rootFolderId, driveId: drive.id, name: driveName, + isHidden: false, ); final rootFolderDataItem = await _arweave.prepareEntityDataItem( diff --git a/lib/blocs/drive_detail/drive_detail_cubit.dart b/lib/blocs/drive_detail/drive_detail_cubit.dart index 31bff1bf94..81ae57ebef 100644 --- a/lib/blocs/drive_detail/drive_detail_cubit.dart +++ b/lib/blocs/drive_detail/drive_detail_cubit.dart @@ -38,7 +38,7 @@ class DriveDetailCubit extends Cubit { bool _forceDisableMultiselect = false; - bool _refreshSelectedItem = false; + final bool _refreshSelectedItem = true; bool _showHiddenFiles = false; @@ -159,8 +159,10 @@ class DriveDetailCubit extends Cubit { ); if (index >= 0) { + final item = folderContents.files[index]; + _selectedItem = DriveDataTableItemMapper.toFileDataTableItem( - folderContents.files[index], + item, _selectedItem!.index, _selectedItem!.isOwner, ); @@ -170,8 +172,10 @@ class DriveDetailCubit extends Cubit { (element) => element.id == _selectedItem!.id, ); if (index >= 0) { + final item = folderContents.subfolders[index]; + _selectedItem = DriveDataTableItemMapper.fromFolderEntry( - folderContents.subfolders[index], + item, _selectedItem!.index, _selectedItem!.isOwner, ); @@ -184,8 +188,6 @@ class DriveDetailCubit extends Cubit { _selectedItem!.isOwner, ); } - - _refreshSelectedItem = false; } final currentFolderContents = parseEntitiesToDatatableItem( @@ -408,22 +410,12 @@ class DriveDetailCubit extends Cubit { ); } - void refreshDriveDataTable({ - bool reComputeFolderContents = false, - }) { - _refreshSelectedItem = true; + void refreshDriveDataTable() { + // _refreshSelectedItem = true; if (state is DriveDetailLoadSuccess) { final state = this.state as DriveDetailLoadSuccess; - if (reComputeFolderContents) { - openFolder( - path: state.folderInView.folder.path, - contentOrderBy: state.contentOrderBy, - contentOrderingMode: state.contentOrderingMode, - ); - } else { - emit(state.copyWith()); - } + emit(state.copyWith()); } } diff --git a/lib/blocs/drive_detail/drive_detail_state.dart b/lib/blocs/drive_detail/drive_detail_state.dart index 5e5828c668..efae0f3f69 100644 --- a/lib/blocs/drive_detail/drive_detail_state.dart +++ b/lib/blocs/drive_detail/drive_detail_state.dart @@ -51,7 +51,7 @@ class DriveDetailLoadSuccess extends DriveDetailState { this.hasFoldersSelected = false, this.selectedFilePreviewUrl, required this.driveIsEmpty, - this.selectedItem, + required this.selectedItem, required this.currentFolderContents, required this.isShowingHiddenFiles, }); @@ -113,6 +113,7 @@ class DriveDetailLoadSuccess extends DriveDetailState { _equatableBust, driveIsEmpty, multiselect, + selectedItem, ]; SelectedItem? maybeSelectedItem() => selectedItems.isNotEmpty ? selectedItems.first : null; diff --git a/lib/blocs/folder_create/folder_create_cubit.dart b/lib/blocs/folder_create/folder_create_cubit.dart index f88bc163dd..03210608b1 100644 --- a/lib/blocs/folder_create/folder_create_cubit.dart +++ b/lib/blocs/folder_create/folder_create_cubit.dart @@ -76,6 +76,7 @@ class FolderCreateCubit extends Cubit { driveId: targetFolder.driveId, parentFolderId: targetFolder.id, name: folderName, + isHidden: false, ); if (_turboUploadService.useTurboUpload) { final folderDataItem = await _arweave.prepareEntityDataItem( diff --git a/lib/blocs/hide/hide_bloc.dart b/lib/blocs/hide/hide_bloc.dart index 05e8a3a3cf..443a86ed22 100644 --- a/lib/blocs/hide/hide_bloc.dart +++ b/lib/blocs/hide/hide_bloc.dart @@ -72,6 +72,7 @@ class HideBloc extends Bloc { on(_onConfirmUploadEvent); on(_onSelectUploadMethodEvent); on(_refreshTurboBalance); + on(_onErrorEvent); } bool get _useTurboUpload => @@ -318,6 +319,8 @@ class HideBloc extends Bloc { ); final folderEntity = newFolder.asEntity(); + logger.d('Unhiding folder with JSON: ${folderEntity.toJson()}'); + final driveKey = await _driveDao.getDriveKey( event.driveId, profile.cipherKey, @@ -418,6 +421,13 @@ class HideBloc extends Bloc { ); } + void _onErrorEvent( + ErrorEvent event, + Emitter emit, + ) { + emit(FailureHideState(hideAction: event.hideAction)); + } + void _computeIsFreeThanksToTurbo() { final allowedDataItemSizeForTurbo = _appConfig.allowedDataItemSizeForTurbo; final forceNoFreeThanksToTurbo = _appConfig.forceNoFreeThanksToTurbo; @@ -553,4 +563,14 @@ class HideBloc extends Bloc { totalSize: _totalSize, ); } + + @override + void onError(Object error, StackTrace stackTrace) { + add(ErrorEvent( + error: error, + stackTrace: stackTrace, + hideAction: state.hideAction, + )); + super.onError(error, stackTrace); + } } diff --git a/lib/blocs/hide/hide_event.dart b/lib/blocs/hide/hide_event.dart index ef81ea9633..7e5049a3fe 100644 --- a/lib/blocs/hide/hide_event.dart +++ b/lib/blocs/hide/hide_event.dart @@ -1,3 +1,4 @@ +import 'package:ardrive/blocs/hide/hide_state.dart'; import 'package:ardrive/blocs/upload/upload_cubit.dart'; import 'package:ardrive_utils/ardrive_utils.dart'; import 'package:equatable/equatable.dart'; @@ -82,3 +83,21 @@ class RefreshTurboBalanceEvent extends HideEvent { @override List get props => []; } + +class ErrorEvent extends HideEvent { + final Object error; + final StackTrace stackTrace; + final HideAction hideAction; + + const ErrorEvent({ + required this.error, + required this.stackTrace, + required this.hideAction, + }); + + @override + List get props => [ + error, + stackTrace, + ]; +} diff --git a/lib/blocs/hide/hide_state.dart b/lib/blocs/hide/hide_state.dart index 4cc6e4e476..588a9d9fb4 100644 --- a/lib/blocs/hide/hide_state.dart +++ b/lib/blocs/hide/hide_state.dart @@ -24,7 +24,7 @@ class UploadingHideState extends HideState { }); @override - List get props => []; + List get props => []; } class PreparingAndSigningHideState extends HideState { @@ -33,7 +33,7 @@ class PreparingAndSigningHideState extends HideState { }); @override - List get props => []; + List get props => []; } class ConfirmingHideState extends HideState { @@ -127,19 +127,16 @@ class SuccessHideState extends HideState { }); @override - List get props => []; + List get props => []; } class FailureHideState extends HideState { - final String message; - const FailureHideState({ - required this.message, required super.hideAction, }); @override - List get props => [message]; + List get props => []; } enum HideAction { diff --git a/lib/blocs/upload/upload_cubit.dart b/lib/blocs/upload/upload_cubit.dart index e4562b609d..b8954bdcae 100644 --- a/lib/blocs/upload/upload_cubit.dart +++ b/lib/blocs/upload/upload_cubit.dart @@ -802,6 +802,7 @@ class UploadCubit extends Cubit { id: metadata.id, name: metadata.name, parentFolderId: metadata.parentFolderId, + isHidden: false, ); if (metadata.metadataTxId == null) { diff --git a/lib/blocs/upload/upload_handles/folder_data_item_upload_handle.dart b/lib/blocs/upload/upload_handles/folder_data_item_upload_handle.dart index dd646f274d..e5ada1875b 100644 --- a/lib/blocs/upload/upload_handles/folder_data_item_upload_handle.dart +++ b/lib/blocs/upload/upload_handles/folder_data_item_upload_handle.dart @@ -25,6 +25,7 @@ class FolderDataItemUploadHandle implements UploadHandle, DataItemHandle { driveId: targetDriveId, parentFolderId: folder.parentFolderId, name: folder.name, + isHidden: false, ).toJson(), ).codeUnits.length; @@ -58,6 +59,7 @@ class FolderDataItemUploadHandle implements UploadHandle, DataItemHandle { driveId: targetDriveId, parentFolderId: folder.parentFolderId, name: folder.name, + isHidden: false, ); folderEntityTx = await arweave.prepareEntityDataItem( diff --git a/lib/components/details_panel.dart b/lib/components/details_panel.dart index 6c4a13aaec..4986dd6967 100644 --- a/lib/components/details_panel.dart +++ b/lib/components/details_panel.dart @@ -1,9 +1,12 @@ import 'package:ardrive/authentication/ardrive_auth.dart'; import 'package:ardrive/blocs/fs_entry_preview/fs_entry_preview_cubit.dart'; +import 'package:ardrive/blocs/hide/hide_bloc.dart'; +import 'package:ardrive/blocs/hide/hide_event.dart'; import 'package:ardrive/components/app_version_widget.dart'; import 'package:ardrive/components/components.dart'; import 'package:ardrive/components/dotted_line.dart'; import 'package:ardrive/components/drive_rename_form.dart'; +import 'package:ardrive/components/hide_dialog.dart'; import 'package:ardrive/components/pin_indicator.dart'; import 'package:ardrive/components/sizes.dart'; import 'package:ardrive/components/truncated_address.dart'; @@ -38,7 +41,6 @@ class DetailsPanel extends StatefulWidget { const DetailsPanel({ super.key, required this.item, - required this.maybeSelectedItem, required this.drivePrivacy, this.revisions, this.fileKey, @@ -50,7 +52,6 @@ class DetailsPanel extends StatefulWidget { }); final ArDriveDataTableItem item; - final SelectedItem? maybeSelectedItem; final Privacy drivePrivacy; final List? revisions; final SecretKey? fileKey; @@ -68,7 +69,8 @@ class _DetailsPanelState extends State { @override Widget build(BuildContext context) { return MultiBlocProvider( - // Specify a key to ensure a new cubit is provided when the folder/file id changes. + // Specify a key to ensure a new cubit is provided when the folder/file id + // changes. key: ValueKey( '${widget.item.driveId}${widget.item.id}${widget.item.name}', ), @@ -225,8 +227,15 @@ class _DetailsPanelState extends State { ScreenTypeLayout.builder( desktop: (context) => Column( children: [ - DetailsPanelToolbar( - item: widget.item, + BlocBuilder( + builder: (context, driveDetailState) { + final driveDetailLoadSuccess = + driveDetailState as DriveDetailLoadSuccess; + return DetailsPanelToolbar( + item: widget.item, + driveDetailLoadSuccess: driveDetailLoadSuccess, + ); + }, ), const SizedBox( height: 24, @@ -798,10 +807,10 @@ class _DetailsPanelState extends State { title = appLocalizationsOf(context).folderWasMoved; break; case RevisionAction.hide: - title = 'This folder was hidden'; + title = appLocalizationsOf(context).folderWasHidden; break; case RevisionAction.unhide: - title = 'This folder was unhidden'; + title = appLocalizationsOf(context).folderWasUnhidden; break; default: title = appLocalizationsOf(context).folderWasModified; @@ -828,10 +837,10 @@ class _DetailsPanelState extends State { .driveWasRenamed(revision.name); break; case RevisionAction.hide: - title = 'This drive was hidden'; + title = appLocalizationsOf(context).driveWasHidden; break; case RevisionAction.unhide: - title = 'This drive was unhidden'; + title = appLocalizationsOf(context).driveWasUnhidden; break; default: title = appLocalizationsOf(context).driveWasModified; @@ -892,10 +901,10 @@ class _DetailsPanelState extends State { ); break; case RevisionAction.hide: - title = 'This file was hidden'; + title = appLocalizationsOf(context).fileWasHidden; break; case RevisionAction.unhide: - title = 'This file was unhidden'; + title = appLocalizationsOf(context).fileWasUnhidden; break; default: title = appLocalizationsOf(context).fileWasModified; @@ -1158,15 +1167,15 @@ class DetailsPanelToolbar extends StatelessWidget { const DetailsPanelToolbar({ super.key, required this.item, + required this.driveDetailLoadSuccess, }); final ArDriveDataTableItem item; + final DriveDetailLoadSuccess driveDetailLoadSuccess; @override Widget build(BuildContext context) { - final drive = - (context.read().state as DriveDetailLoadSuccess) - .currentDrive; + final drive = driveDetailLoadSuccess.currentDrive; return Container( padding: const EdgeInsets.symmetric(vertical: 12), @@ -1266,6 +1275,61 @@ class DetailsPanelToolbar extends StatelessWidget { promptToMove(context, driveId: drive.id, selectedItems: [item]); }, ), + if (item.isOwner) + _buildActionIcon( + tooltip: item.isHidden + ? appLocalizationsOf(context).unhide + : appLocalizationsOf(context).hide, + icon: item.isHidden + ? ArDriveIcons.eyeClosed(size: defaultIconSize) + : ArDriveIcons.eyeOpen(size: defaultIconSize), + onTap: () { + final hideBloc = context.read(); + final driveDetailCubit = context.read(); + + if (item is FileDataTableItem) { + if (item.isHidden) { + hideBloc.add(UnhideFileEvent( + driveId: item.driveId, + fileId: item.id, + )); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); + } else { + hideBloc.add(HideFileEvent( + driveId: item.driveId, + fileId: item.id, + )); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); + } + } else if (item is FolderDataTableItem) { + if (item.isHidden) { + hideBloc.add(UnhideFolderEvent( + driveId: item.driveId, + folderId: item.id, + )); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); + } else { + hideBloc.add(HideFolderEvent( + driveId: item.driveId, + folderId: item.id, + )); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); + } + } + }, + ), const Spacer(), _buildActionIcon( tooltip: appLocalizationsOf(context).close, diff --git a/lib/components/hide_dialog.dart b/lib/components/hide_dialog.dart index 507123625c..69d25a9f80 100644 --- a/lib/components/hide_dialog.dart +++ b/lib/components/hide_dialog.dart @@ -1,22 +1,31 @@ +import 'package:ardrive/blocs/drive_detail/drive_detail_cubit.dart'; import 'package:ardrive/blocs/hide/hide_bloc.dart'; import 'package:ardrive/blocs/hide/hide_event.dart'; import 'package:ardrive/blocs/hide/hide_state.dart'; +import 'package:ardrive/utils/app_localizations_wrapper.dart'; +import 'package:ardrive/utils/logger/logger.dart'; import 'package:ardrive_ui/ardrive_ui.dart'; import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; Future promptToHide( - BuildContext context, -) async { + BuildContext context, { + required DriveDetailCubit driveDetailCubit, +}) async { return showAnimatedDialog( context, barrierDismissible: false, - content: const HideDialog(), + content: HideDialog(driveDetailCubit: driveDetailCubit), ); } class HideDialog extends StatelessWidget { - const HideDialog({super.key}); + final DriveDetailCubit _driveDetailCubit; + + const HideDialog({ + super.key, + required DriveDetailCubit driveDetailCubit, + }) : _driveDetailCubit = driveDetailCubit; @override Widget build(BuildContext context) { @@ -24,33 +33,65 @@ class HideDialog extends StatelessWidget { listener: (context, state) { if (state is SuccessHideState) { Navigator.of(context).pop(); + logger.d('Successfully hid/unhid entity'); } else if (state is ConfirmingHideState) { + _driveDetailCubit.refreshDriveDataTable(); context.read().add(const ConfirmUploadEvent()); } }, builder: (context, state) { return ArDriveStandardModal( - title: _buildTitle(state.hideAction), - content: _buildContent(), + title: _buildTitle(context, state), + content: _buildContent(context, state), + actions: _buildActions(context, state), ); }, ); } - String _buildTitle(HideAction hideAction) { + String _buildTitle(BuildContext context, HideState state) { + final hideAction = state.hideAction; + if (state is FailureHideState) { + switch (hideAction) { + case HideAction.hideFile: + return appLocalizationsOf(context).failedToHideFile; + case HideAction.hideFolder: + return appLocalizationsOf(context).failedToHideFolder; + case HideAction.unhideFile: + return appLocalizationsOf(context).failedToUnhideFile; + case HideAction.unhideFolder: + return appLocalizationsOf(context).failedToUnhideFolder; + } + } + switch (hideAction) { case HideAction.hideFile: - return 'Hiding file'; // TODO: localize + return appLocalizationsOf(context).hidingFile; case HideAction.hideFolder: - return 'Hiding folder'; // TODO: localize + return appLocalizationsOf(context).hidingFolder; case HideAction.unhideFile: - return 'Unhiding file'; // TODO: localize + return appLocalizationsOf(context).unhidingFile; case HideAction.unhideFolder: - return 'Unhiding folder'; // TODO: localize + return appLocalizationsOf(context).unhidingFolder; } } - Widget _buildContent() { + Widget _buildContent(BuildContext context, HideState state) { + if (state is FailureHideState) { + final hideAction = state.hideAction; + + switch (hideAction) { + case HideAction.hideFile: + return Text(appLocalizationsOf(context).failedToHideFile); + case HideAction.hideFolder: + return Text(appLocalizationsOf(context).failedToHideFolder); + case HideAction.unhideFile: + return Text(appLocalizationsOf(context).failedToUnhideFile); + case HideAction.unhideFolder: + return Text(appLocalizationsOf(context).failedToUnhideFolder); + } + } + return const Column( children: [ Center( @@ -59,4 +100,22 @@ class HideDialog extends StatelessWidget { ], ); } + + List? _buildActions( + BuildContext context, + HideState state, + ) { + if (state is FailureHideState) { + return [ + ModalAction( + action: () { + Navigator.of(context).pop(); + }, + title: appLocalizationsOf(context).close, + ), + ]; + } else { + return null; + } + } } diff --git a/lib/l10n/app_en.arb b/lib/l10n/app_en.arb index da99ab2113..49745ad176 100644 --- a/lib/l10n/app_en.arb +++ b/lib/l10n/app_en.arb @@ -585,6 +585,10 @@ } } }, + "driveWasHidden": "This drive was hidden", + "@driveWasHidden": { + "description": "This drive was hidden." + }, "driveWasModified": "This drive was modified.", "@driveWasModified": { "description": "Drive activity (journal) entry: modified" @@ -599,6 +603,10 @@ } } }, + "driveWasUnhidden": "This drive was unhidden", + "@driveWasUnhidden": { + "description": "This drive was unhidden." + }, "duplicateFiles": "{numberOfDuplicateFiles, plural, zero{} one{A duplicate file found} other{Duplicate files found}}", "@duplicateFiles": { "description": "Count of conflicting items", @@ -733,6 +741,22 @@ "@failedToCreatePin": { "description": "The pin could not be created" }, + "failedToHideFile": "Failed to hide file", + "@failedToHideFile": { + "description": "Failed to hide file" + }, + "failedToHideFileTryAgain": "Failed to hide file, please try again", + "@failedToHideFileTryAgain": { + "description": "Failed to hide file, try again." + }, + "failedToHideFolder": "Failed to hide folder", + "@failedToHideFolder": { + "description": "Failed to hide folder" + }, + "failedToHideFolderTryAgain": "Failed to hide folder, please try again", + "@failedToHideFolderTryAgain": { + "description": "Failed to hide folder, try again." + }, "failedToRetrieveFileInfromation": "Failed to retrieve file information", "@failedToRetrieveFileInfromation": { "description": "Explains that there was an error while retrieving the file infromation" @@ -741,6 +765,22 @@ "@failedToSyncDrive": { "description": "The app wasn't able to sync the drive" }, + "failedToUnhideFile": "Failed to unhide file", + "@failedToUnhideFile": { + "description": "Failed to unhide file" + }, + "failedToUnhideFileTryAgain": "Failed to unhide file, please try again", + "@failedToUnhideFileTryAgain": { + "description": "Failed to unhide file, try again." + }, + "failedToUnhideFolder": "Failed to unhide folder", + "@failedToUnhideFolder": { + "description": "Failed to unhide folder" + }, + "failedToUnhideFolderTryAgain": "Failed to unhide folder, please try again", + "@failedToUnhideFolderTryAgain": { + "description": "Failed to unhide folder, try again." + }, "feedbackContent": "Would you be willing to leave feedback on your experience using the app?", "@feedbackContent": { "description": "Asks for feedback" @@ -873,6 +913,10 @@ } } }, + "fileWasHidden": "This file was hidden", + "@fileWasHidden": { + "description": "This file was hidden." + }, "fileWasModified": "This file was modified.", "@fileWasModified": { "description": "File activity (journal): modified" @@ -895,6 +939,10 @@ } } }, + "fileWasUnhidden": "This file was unhidden", + "@fileWasUnhidden": { + "description": "This file was unhidden." + }, "finishingThingsUp": "Finishing things up!", "@finishingThingsUp": { "description": "Finishing things up!" @@ -961,6 +1009,10 @@ } } }, + "folderWasHidden": "This folder was hidden", + "@folderWasHidden": { + "description": "This folder was hidden." + }, "folderWasModified": "This folder was modified.", "@folderWasModified": { "description": "Folder activity (journal): modified" @@ -979,6 +1031,10 @@ } } }, + "folderWasUnhidden": "This folder was unhidden", + "@folderWasUnhidden": { + "description": "This folder was unhidden." + }, "forgetWallet": "Forget wallet and change profile", "@forgetWallet": { "description": "Use a different wallet by forgetting the current one" @@ -1035,6 +1091,18 @@ }, "helpCenter": "Help Center", "@helpCenter": {}, + "hide": "Hide", + "@hide": { + "description": "Hide" + }, + "hidingFile": "Hiding file", + "@hidingFile": { + "description": "Hiding file" + }, + "hidingFolder": "Hiding folder", + "@hidingFolder": { + "description": "Hiding folder" + }, "howAreConversionsDetermined": "How are conversions determined?", "@howAreConversionsDetermined": {}, "howDoesKeyfileLoginWork": "How do keyfile and seed phrase login work?", @@ -1926,6 +1994,18 @@ "@unableToFetchEstimateAtThisTime": {}, "unableToUpdateQuote": "Unable to update quote. Please try again.", "@unableToUpdateQuote": {}, + "unhide": "Unhide", + "@unhide": { + "description": "Unhide" + }, + "unhidingFile": "Unhiding file", + "@unhidingFile": { + "description": "Unhiding file" + }, + "unhidingFolder": "Unhiding folder", + "@unhidingFolder": { + "description": "Unhiding folder" + }, "unit": "Unit", "@unit": {}, "units": "Units", diff --git a/lib/pages/app_router_delegate.dart b/lib/pages/app_router_delegate.dart index 8a7e4fef9b..9406dcf639 100644 --- a/lib/pages/app_router_delegate.dart +++ b/lib/pages/app_router_delegate.dart @@ -192,14 +192,16 @@ class AppRouterDelegate extends RouterDelegate child: MultiBlocListener( listeners: [ BlocListener( - listener: (context, state) { - if (state is DriveDetailLoadSuccess) { - driveId = state.currentDrive.id; - driveFolderId = state.folderInView.folder.id; + listener: (context, driveDetailCubitState) { + if (driveDetailCubitState is DriveDetailLoadSuccess) { + driveId = driveDetailCubitState.currentDrive.id; + driveFolderId = + driveDetailCubitState.folderInView.folder.id; //Can be null at the root folder of the drive notifyListeners(); - } else if (state is DriveDetailLoadNotFound) { + } else if (driveDetailCubitState + is DriveDetailLoadNotFound) { // Do not prompt the user to attach an unfound drive if they are logging out. final profileCubit = context.read(); diff --git a/lib/pages/drive_detail/components/drive_detail_data_list.dart b/lib/pages/drive_detail/components/drive_detail_data_list.dart index fa2c7e9a94..4df7be28d9 100644 --- a/lib/pages/drive_detail/components/drive_detail_data_list.dart +++ b/lib/pages/drive_detail/components/drive_detail_data_list.dart @@ -11,6 +11,7 @@ Widget _buildDataList( state.currentDrive, isMultiselecting: state.multiselect, isShowingHiddenFiles: state.isShowingHiddenFiles, + selectedItem: state.selectedItem, ); } @@ -66,32 +67,20 @@ class FolderDataTableItem extends ArDriveDataTableItem { final bool isGhostFolder; FolderDataTableItem({ - required String driveId, + required super.driveId, required String folderId, - required String name, - required DateTime lastUpdated, - required DateTime dateCreated, - required String contentType, - required String path, - String? fileStatusFromTransactions, + required super.name, + required super.lastUpdated, + required super.dateCreated, + required super.contentType, + required super.path, + super.fileStatusFromTransactions, + required super.isHidden, + required super.index, + required super.isOwner, this.parentFolderId, this.isGhostFolder = false, - required super.isHidden, - required int index, - required bool isOwner, - }) : super( - driveId: driveId, - path: path, - id: folderId, - name: name, - size: null, - lastUpdated: lastUpdated, - dateCreated: dateCreated, - contentType: contentType, - fileStatusFromTransactions: fileStatusFromTransactions, - index: index, - isOwner: isOwner, - ); + }) : super(id: folderId); @override List get props => [id, name, isHidden]; @@ -108,38 +97,26 @@ class FileDataTableItem extends ArDriveDataTableItem { final String? pinnedDataOwnerAddress; FileDataTableItem({ + required super.driveId, + required super.lastUpdated, + required super.name, + required super.size, + required super.dateCreated, + required super.contentType, + required super.path, + required super.isHidden, + super.fileStatusFromTransactions, + required super.index, + required super.isOwner, required this.fileId, - required String driveId, required this.parentFolderId, required this.dataTxId, - required DateTime lastUpdated, required this.lastModifiedDate, required this.metadataTx, required this.dataTx, required this.pinnedDataOwnerAddress, - required String name, - required int size, - required DateTime dateCreated, - required String contentType, - required String path, - required super.isHidden, - String? fileStatusFromTransactions, this.bundledIn, - required int index, - required bool isOwner, - }) : super( - path: path, - driveId: driveId, - id: fileId, - name: name, - size: size, - lastUpdated: lastUpdated, - dateCreated: dateCreated, - contentType: contentType, - fileStatusFromTransactions: fileStatusFromTransactions, - index: index, - isOwner: isOwner, - ); + }) : super(id: fileId); @override List get props => [fileId, name, isHidden]; @@ -152,6 +129,7 @@ Widget _buildDataListContent( Drive drive, { required bool isMultiselecting, required bool isShowingHiddenFiles, + required ArDriveDataTableItem? selectedItem, }) { final List filteredItems; if (isShowingHiddenFiles) { @@ -263,15 +241,15 @@ Widget _buildDataListContent( } else if (row is FileDataTableItem) { if (row.id == cubit.selectedItem?.id) { cubit.toggleSelectedItemDetails(); - return; + } else { + cubit.selectDataItem(row); } - cubit.selectDataItem(row); } }, ); }, rows: filteredItems, - selectedRow: context.watch().selectedItem, + selectedRow: selectedItem, ); }); } diff --git a/lib/pages/drive_detail/components/drive_explorer_item_tile.dart b/lib/pages/drive_detail/components/drive_explorer_item_tile.dart index 4e7ed7459d..f3e0474a9d 100644 --- a/lib/pages/drive_detail/components/drive_explorer_item_tile.dart +++ b/lib/pages/drive_detail/components/drive_explorer_item_tile.dart @@ -319,19 +319,26 @@ class _DriveExplorerItemTileTrailingState ArDriveDropdownItem( onClick: () { final hideBloc = context.read(); + final driveDetailCubit = context.read(); if (item.isHidden) { hideBloc.add(UnhideFolderEvent( driveId: widget.drive.id, folderId: item.id, )); - promptToHide(context); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); } else { hideBloc.add(HideFolderEvent( driveId: widget.drive.id, folderId: item.id, )); - promptToHide(context); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); } }, content: _buildItem( @@ -436,19 +443,26 @@ class _DriveExplorerItemTileTrailingState ArDriveDropdownItem( onClick: () { final hideBloc = context.read(); + final driveDetailCubit = context.read(); if (item.isHidden) { hideBloc.add(UnhideFileEvent( driveId: widget.drive.id, fileId: item.id, )); - promptToHide(context); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); } else { hideBloc.add(HideFileEvent( driveId: widget.drive.id, fileId: item.id, )); - promptToHide(context); + promptToHide( + context, + driveDetailCubit: driveDetailCubit, + ); } }, content: _buildItem( diff --git a/lib/pages/drive_detail/drive_detail_page.dart b/lib/pages/drive_detail/drive_detail_page.dart index 63dd6b00e7..99c66d3043 100644 --- a/lib/pages/drive_detail/drive_detail_page.dart +++ b/lib/pages/drive_detail/drive_detail_page.dart @@ -526,20 +526,13 @@ class _DriveDetailPageState extends State { driveDetailState, context), ), child: driveDetailState.showSelectedItemDetails && - context - .read() - .selectedItem != - null + driveDetailState.selectedItem != null ? DetailsPanel( currentDrive: driveDetailState.currentDrive, isSharePage: false, drivePrivacy: driveDetailState.currentDrive.privacy, - maybeSelectedItem: - driveDetailState.maybeSelectedItem(), - item: context - .read() - .selectedItem!, + item: driveDetailState.selectedItem!, onNextImageNavigation: () { context .read() @@ -591,13 +584,13 @@ class _DriveDetailPageState extends State { } Widget _mobileView( - DriveDetailLoadSuccess state, + DriveDetailLoadSuccess driveDetailLoadSuccessState, bool hasSubfolders, bool hasFiles, ) { - final items = state.currentFolderContents; + final items = driveDetailLoadSuccessState.currentFolderContents; - if (state.showSelectedItemDetails && + if (driveDetailLoadSuccessState.showSelectedItemDetails && context.read().selectedItem != null) { return Material( child: WillPopScope( @@ -606,11 +599,10 @@ class _DriveDetailPageState extends State { return false; }, child: DetailsPanel( - currentDrive: state.currentDrive, + currentDrive: driveDetailLoadSuccessState.currentDrive, isSharePage: false, - drivePrivacy: state.currentDrive.privacy, - maybeSelectedItem: state.maybeSelectedItem(), - item: context.read().selectedItem!, + drivePrivacy: driveDetailLoadSuccessState.currentDrive.privacy, + item: driveDetailLoadSuccessState.selectedItem!, onNextImageNavigation: () { context.read().selectNextImage(); }, @@ -632,7 +624,7 @@ class _DriveDetailPageState extends State { .withOpacity(0.5), drawer: const AppSideBar(), appBar: MobileAppBar( - leading: (state.showSelectedItemDetails && + leading: (driveDetailLoadSuccessState.showSelectedItemDetails && context.read().selectedItem != null) ? ArDriveIconButton( icon: ArDriveIcons.arrowLeft(), @@ -658,7 +650,7 @@ class _DriveDetailPageState extends State { }, ), body: _mobileViewContent( - state, + driveDetailLoadSuccessState, hasSubfolders, hasFiles, items, diff --git a/lib/pages/shared_file/shared_file_page.dart b/lib/pages/shared_file/shared_file_page.dart index b9903cbed6..15abfe2085 100644 --- a/lib/pages/shared_file/shared_file_page.dart +++ b/lib/pages/shared_file/shared_file_page.dart @@ -52,7 +52,6 @@ class SharedFilePage extends StatelessWidget { false, ), isSharePage: true, - maybeSelectedItem: null, fileKey: state.fileKey, revisions: state.fileRevisions, drivePrivacy: state.fileKey != null ? 'private' : 'public', diff --git a/test/models/entity_version_tag_test.dart b/test/models/entity_version_tag_test.dart index ad42ba9934..e1817aea4c 100644 --- a/test/models/entity_version_tag_test.dart +++ b/test/models/entity_version_tag_test.dart @@ -164,6 +164,7 @@ void main() { driveId: driveId, parentFolderId: rootFolderId, name: testEntityName, + isHidden: false, ); AppPlatform.setMockPlatform(platform: SystemPlatform.unknown);