[ios] Implement the Recently Deleted screen to restore deleted categories #7978
Labels
No labels
Accessibility
Accessibility
Address
Address
Android
Android
Android Auto
Android Auto
Android Automotive (AAOS)
Android Automotive (AAOS)
API
API
AppGallery
AppGallery
AppStore
AppStore
Battery and Performance
Battery and Performance
Blocker
Blocker
Bookmarks and Tracks
Bookmarks and Tracks
Borders
Borders
Bug
Bug
Build
Build
CarPlay
CarPlay
Classificator
Classificator
Community
Community
Core
Core
CrashReports
CrashReports
Cycling
Cycling
Desktop
Desktop
DevEx
DevEx
DevOps
DevOps
dev_sandbox
dev_sandbox
Directions
Directions
Documentation
Documentation
Downloader
Downloader
Drape
Drape
Driving
Driving
Duplicate
Duplicate
Editor
Editor
Elevation
Elevation
Enhancement
Enhancement
Epic
Epic
External Map Datasets
External Map Datasets
F-Droid
F-Droid
Fonts
Fonts
Frequently User Reported
Frequently User Reported
Fund
Fund
Generator
Generator
Good first issue
Good first issue
Google Play
Google Play
GPS
GPS
GSoC
GSoC
iCloud
iCloud
Icons
Icons
iOS
iOS
Legal
Legal
Linux Desktop
Linux Desktop
Linux packaging
Linux packaging
Linux Phone
Linux Phone
Mac OS
Mac OS
Map Data
Map Data
Metro
Metro
Navigation
Navigation
Need Feedback
Need Feedback
Night Mode
Night Mode
NLnet 2024-06-281
NLnet 2024-06-281
No Feature Parity
No Feature Parity
Opening Hours
Opening Hours
Outdoors
Outdoors
POI Info
POI Info
Privacy
Privacy
Public Transport
Public Transport
Raw Idea
Raw Idea
Refactoring
Refactoring
Regional
Regional
Regression
Regression
Releases
Releases
RoboTest
RoboTest
Route Planning
Route Planning
Routing
Routing
Ruler
Ruler
Search
Search
Security
Security
Styles
Styles
Tests
Tests
Track Recording
Track Recording
Translations
Translations
TTS
TTS
UI
UI
UX
UX
Walk Navigation
Walk Navigation
Watches
Watches
Web
Web
Wikipedia
Wikipedia
Windows
Windows
Won't fix
Won't fix
World Map
World Map
No milestone
No project
No assignees
3 participants
Due date
No due date set.
Dependencies
No dependencies set.
Reference: organicmaps/organicmaps-tmp#7978
Loading…
Add table
Reference in a new issue
No description provided.
Delete branch "ios/restore-catergories-from-the-recently-deleted"
Deleting a branch is permanent. Although the deleted branch may continue to exist for a short time before it actually gets removed, it CANNOT be undone in most cases. Continue?
Based on: organicmaps/organicmaps#8719
Closes (for iOS) organicmaps/organicmaps#1026
This PR contains the default implementation of the Recently Deleted Screen in iOS to restore deleted categories.
Key concepts:
- when the user deletes a category it will be removed from the categories list and the related file will be marked with the.deleted
extension- when the user deletes a file from theRecently Deleted
list it should be totally removed from the file system- when the user restores a category from theRecently Deleted
list the.deleted
ext will be removed and the file will be loaded into the bookmarks categories list-Recently deleted
button is visible only when there is nothing to restore (not implemented)- when the user deletes the categoryfile.kml
and creates (or import) a new category with the same name the newly created file will have thefile1.kml
name to avoid conflicts with the "old"file.kml
+file.kml.deleted
. This allows to recover thefile.kml.deleted
without renaming.Trash
directory to deleted filesiCloud sync manager
cannot directly remove the file. Instead, it only uses the trash mechanism to delete the file. It will help to avoid destructive behavior and unexpected bugs - it will be possible to recover the file if smth happens.Todo:
iOS:
- [x] iCloud sync support- [x] Recover the current file by a swipe- [x] Delete file with swiperecently deleted
button when there is nothing to delete- [x] Fix removing animations for cells (optional)Core:
recently deleted
list is emptyIt would be better to get an array of elements with:category_path
,category_name
, anddeletion_date
(done)Files App (as a reference):

Results:
The PR can be split into two, and the second one can depend on the first one. Then we can review and fix them separately, focusing on the first one.
This PR can be the second one )
This is a static method that can be called directly, without adding it into the FileWriter (class members are not used/needed). A separate commit/PR can also remove the unnecessary wrapper
FileWriter::DeleteFileX
Already defined in common.xcconfig
Already defined in common.xcconfig
Already defined in common.xcconfig
Already defined in common-debug.xcconfig
Already defined in common.xcconfig
Why is this call needed?
Is a .Trash or any other folder for deleted files really needed? Why deleted files can't be in the same bookmarks directory, together with non-deleted files?
@ -404,6 +404,53 @@ void BookmarkManager::ResetRecentlyDeletedBookmark()
m_recentlyDeletedBookmark.reset();
}
A detailed comment is needed on why is this call required. Is there any way to avoid using it?
@ -2498,2 +2543,2 @@
if (deleteFile)
FileWriter::DeleteFileX(it->second->GetFileName());
auto const & filePath = it->second->GetFileName();
if (permanently)
gettimeofday and utime are not available on Windows.
Why is it a teardown?
Where the restriction to delete the last list is defined? In the iOS/Android code?
Yes in ios... and spread across the view controller... at least 2 times.
There is no restrictions in the core to delete the last group.
I will add a separate commit to make with a small refactoring by moving this check into the viewModel
The
.Trash
dir is needed to handle situations when the file was directly deleted from the Files app/ mac.In this case, we faced up with the absence of the file in the icloud local container and should decide:
In this case, we inspect the cloud's trash to check that file was deleted (because files deleted manually from the Files app are moved to trash).
If there is no file in the cloud's trash the file will be recreated.
When we create a file in the cloud its
lastModified
date will be equal to thecreationDate
and will not be equal to the source local file'slastModified
.In this case, we will have situation:
It is important to keep the
lastModified
time the same as the default c++move
do (FileManager behavior is different).@ -404,6 +404,53 @@ void BookmarkManager::ResetRecentlyDeletedBookmark()
m_recentlyDeletedBookmark.reset();
}
We should track the
delete
/recover
action.Situation:
device 1:
device 2:
Expectation: the file should be recovered.
But we have:
on cloud:
file.kml.deleted (
lastModified
0001)on device 1:
file.kml (
lastModified
0001)how to decide what is the latest file version?
Updating
lastModified
allows to keep track the file's lifecycle.The implementation without using
lastModified
does not allow to track it.The way to avoid this: create the versionig system or some kind of transactions journal.
What are the alternative approaches and their pros/cons? E.g.
Won't the file updation be triggered anyway before the modification time is reset to the original file's time?
I've tested this behaviour - no, this update will be triggered only once because the writing operation is synchronous in the fileCoordinator's scope.
the approach of storing the flags has a huge disadvantage - all the files should be downloaded first and parsed by the core before we can only check the flags/attributes. It will impact performance and mobile data consumption a lot.
This approach will allow the tracking of the file's history of changes and not rely on the file's content itself.
pros:
lastModifiedDate
attr directlycons:
When the icloud's metadata query starts to work and sends the NSMetadataItems list to the app we should decide for every item - download and update the current file or not.
Usually, we download all the files (because all of them should be synced) and it's not a blocker.
But in case of updating the local file with the cloud's version (or vice versa) we should:
It may be ok process a lightweight file, but if it will be >5-10mb?
On every bookmark/color update?
Notification comes from the icloud's metadata query asynchronously and sometimes very frequently.
We cannot parse files "only once" and forget when we use >1 devices. As we see the users actively enable/disable sync and use multiple devices and this additional parsing of the cloud's contents will slow down the app performance.
We should rely on the lightweight information (file's metadata that is already composed by the cloud or custom journal to make a decision of how to process the file.
It's not straightforward that the remote version is newer.
There are a lot of cases when at the same time:
When the user starts the app or enables sync in the settings we do not have the "previous" state of the local/cloud contents but only the current and compare files without knowing "what the action triggers the change".
The cloud content may be updated from the other devices.
Maybe I'm not fully understanding the process. Let's discuss a simple case:
4... on the 2nd device
5. The local file will be updated from the cloud's version (if there are no version conflicts)
All is correct.
The updation process is quite simple if the current files exist on both devices (except the situation when the sync was disabled and the file was changed on both devices).
If this simple case works, then why storing a "deleted" flag inside a file won't work? Isn't it the same write operation as "add a bookmark" or "delete a bookmark"?
The design from Aug 3 organicmaps/organicmaps#7978 (comment) LGTM.
That's a cool idea!
Please create a separate issue or PR for this bug.
Agree, thanks for explaining your arguments!
Also agree. It would be great to integrate multi selection in the main list in a natural way too, I think there was an issue about it, right?
@organicmaps/translations Please review and check if "Lists" is translated in the same way as here for consistency:
Apple uses Recover word in Files app. We use Restore on accidentally deleted bookmarks on the map. Should the existing Restore translation for bookmarks on the map be replaced with Recover? It can be done later in a separate PR.
ditto?
ditto?
ditto?
Will this work?
Why it should be run on UI thread? ReloadBookmark already uses necessary threads internally.
@vng is there a check/optimization that if we're already running on the same thread as RunTask is receiving, then the code is run immediately, without putting it into a queue? If no, let's create an issue for it.
@ -404,6 +404,53 @@ void BookmarkManager::ResetRecentlyDeletedBookmark()
m_recentlyDeletedBookmark.reset();
}
size_t BookmarkManager::GetRecentlyDeletedCategoriesCount() const
Is this method covered by any test? It should be used by UI to quickly retrieve the count without expensive parsing of all deleted categories.
@ -407,0 +411,4 @@
return files.size();
}
BookmarkManager::KMLDataCollectionPtr BookmarkManager::GetRecentlyDeletedCategories()
Is this method called synchronously from the main thread? It may cause a huge delay for large deleted lists in such a case. Please also make sure that it's only called when necessary (when Recently Deleted UI is displayed).
@ -407,0 +417,4 @@
[](kml::FileData const &)
{
return true;
});
Will this work?
@ -407,0 +428,4 @@
void BookmarkManager::RecoverRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & filePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
This check should not be needed as we run the task below on file thread anyway. @vng right?
@ -407,0 +442,4 @@
void BookmarkManager::DeleteRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & deletedFilePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
This call may contain a lot of file delete operations. Would it be better to remove the thread checker line and run it on the async File thread instead of the UI thread? It doesn't return anything. But I assume that the caller expects that after returning to the main bookmarks screen files are already deleted, and an updated count in Recently Deleted is displayed. Correct?
@ -2499,1 +2543,3 @@
FileWriter::DeleteFileX(it->second->GetFileName());
auto const & filePath = it->second->GetFileName();
if (permanently)
{
Should it be
to avoid unnecessary allocation/copy?
Let's require an explicit
permanently
specification for callers for safety.Can it be const?
nit: an alternative approach may leverage something like
ForEachDeletedCategory(TLambda &&)
, but let's leave it for a separate discussion later.he
is good.Look good for French ^^
@ -2499,1 +2543,3 @@
FileWriter::DeleteFileX(it->second->GetFileName());
auto const & filePath = it->second->GetFileName();
if (permanently)
{
How much $
does it save?
@ -407,0 +442,4 @@
void BookmarkManager::DeleteRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & deletedFilePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
As far as I can see, all deletion operations in the existing code are currently performed in the UI thread.
@ -404,6 +404,53 @@ void BookmarkManager::ResetRecentlyDeletedBookmark()
m_recentlyDeletedBookmark.reset();
}
size_t BookmarkManager::GetRecentlyDeletedCategoriesCount() const
I fixed the current cpp test and added checks for the GetRecentlyDeletedCategoriesCount
@ -407,0 +417,4 @@
[](kml::FileData const &)
{
return true;
});
No. The method was designed to work with the

using BookmarksChecker = std::function<bool(kml::FileData const &)>;
that should returns bool:
@ -407,0 +411,4 @@
return files.size();
}
BookmarkManager::KMLDataCollectionPtr BookmarkManager::GetRecentlyDeletedCategories()
Yes
This is why the
GetRecentlyDeletedCategoriesCount
was added.The
GetRecentlyDeletedCategories
was called only when theRecentlyDeletedCategoriesViewController
is opening and during the it's updates (when the category was deleted by the cloud for example).@biodranik the Reload and Load methods check the thread first so they should be called from that main (maybe because to work safe with the m_bookmarkLoadingQueue)
No, because the LoadBookmarks cannot.
@ -407,0 +442,4 @@
void BookmarkManager::DeleteRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & deletedFilePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
The main idea is to call this method synchronously on the main and Not pass the completion handler -> update the state immediately without waiting for deletion ansync finish.
Deletion is quite fast and it is better to rely on the actual directory content.
For example:
delete all
@ -407,0 +428,4 @@
void BookmarkManager::RecoverRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & filePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
Thanks!
Thanks! Semantically it is not the scanning the folder but it works as expected (and faster).
This will require changing the Android's code...
It seems like all translations is correct
Could you please provide more details on why you believe this change is necessary?
@ -407,0 +442,4 @@
void BookmarkManager::DeleteRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & deletedFilePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
Enhancing the error handling for I/O operations in the Storage thread across the code base is a good task to be tackled separately. This PR follows the existing approach.
Fixed!
@ -407,0 +442,4 @@
void BookmarkManager::DeleteRecentlyDeletedCategoriesAtPaths(std::vector<std::string> const & deletedFilePaths)
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
I understand the reasoning for the synchronous call.
Won't the android linter/runtime checker complain for file operations made on the UI thread in this case?
@ -407,0 +417,4 @@
[](kml::FileData const &)
{
return true;
});
Sorry for the typo, of course I meant an empty std::function/lambda, which compiles:
auto collection = LoadBookmarks(GetTrashDirectory(), kKmlExtension, KmlFileType::Text, {});
This approach looks like a heavy overhead:
If it's ok to delete files from the main thread, then it should be also ok to move files from the main thread.
RecoverRecentlyDeletedCategoriesAtPaths
is called from the main thread, right? Can step (1) and (3) be removed in this case?nit: spaces are inserted automatically in our logging/assert macros.
@ -2497,3 +2542,2 @@
if (deleteFile)
FileWriter::DeleteFileX(it->second->GetFileName());
auto const & filePath = it->second->GetFileName();
nit: avoid unnecessary copy
@ -2499,1 +2543,3 @@
FileWriter::DeleteFileX(it->second->GetFileName());
auto const & filePath = it->second->GetFileName();
if (permanently)
{
It's not about $
. It's about clarity and clear code. And understanding what's going on.
Actually, LoadBookmarks method should be marked as const, because it doesn't modify any members. Then GetRecentlyDeletedCategories() can also be properly marked as const. Otherwise, reading it, there is an assumption that it changes the internal state, while it's not.
@rtsisyk to clearly specify the needed behavior in all places of usage.
Thanks everyone for your time and patience. Let's proceed with the release.