[core] Implement products configuration #9695
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
4 participants
Due date
No due date set.
Dependencies
No dependencies set.
Reference: organicmaps/organicmaps-tmp#9695
Loading…
Add table
Reference in a new issue
No description provided.
Delete branch "add-products-config"
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?
This PR adds support for the product's config to the core.
productsConfig
the file will be deleted)Timeout are different for the debug and release schemes.
Details below.
@ -38,7 +43,13 @@ std::optional<MetaConfig> ParseMetaConfig(std::string const & jsonStr)
outMetaConfig.m_settings[key] = valueStr;
}
We are doing unnecessary transformations here:
@ -76,32 +77,8 @@ public:
void EnterForeground();
void EnterBackground();
first launch or sessions count?
Make ProductSettings like singleton and remove these functions from settings. Will use separate ProductSettings::Instance()
ProductSettings::Instance().Update(metaConfig.m_productsConfig); // passing a json string
@ -3316,0 +3361,4 @@
}
void Framework::DidCloseProductsPopup(ProductsPopupCloseReason reason) const
{
ProductSettings::Instance().Get()
We can store actual already parsed instance do avoid parsing every time we show PP:
@ -38,7 +43,13 @@ std::optional<MetaConfig> ParseMetaConfig(std::string const & jsonStr)
outMetaConfig.m_settings[key] = valueStr;
}
Fixed! Thanks
@ -3316,0 +3361,4 @@
}
void Framework::DidCloseProductsPopup(ProductsPopupCloseReason reason) const
{
Fixed! Thanks
@ -76,32 +77,8 @@ public:
void EnterForeground();
void EnterBackground();
Fixed! Thanks
@ -76,32 +77,8 @@ public:
void EnterForeground();
void EnterBackground();
@ -758,0 +774,4 @@
private:
bool ShouldShowProducts() const;
uint32_t GetTimeoutForReason(ProductsPopupCloseReason reason) const;
std::string_view ToString(ProductsPopupCloseReason reason) const;
nit: Move just above the DidCloseProductsPopup function.
nit: { from new string
nit: A true constexpr is
char const kProductsConfig[] = "productsConfig";
@ -0,0 +2,4 @@
#include <string>
#include <vector>
#include <optional>
#include <mutex>
@ -425,18 +426,16 @@ void UsageStats::EnterBackground()
m_ss.SetValue(m_sessions, ToString(m_sessionsCount));
}
Sorry, I think that it is a valid situation:
Better to declare constants in both cases, e.g:
Thx. Fixed!
@ -425,18 +426,16 @@ void UsageStats::EnterBackground()
m_ss.SetValue(m_sessions, ToString(m_sessionsCount));
}
Thx!
Did review of Framework::ShouldShowProducts()
uint64_t popupCloseTime = 0;
uint64_t productSelectTime = 0;
popupCloseTime == 0
is not needed after valid initialization.@ -23,6 +23,7 @@ using namespace std;
std::string_view kMeasurementUnits = "Units";
Probably last 2 options are needed only in framework.cpp, no?
popupCloseTime == 0 and productSelectTime == 0 are cases when there is no saved user action (the prompt has never been closed by any reason) and in this case we should always show the promp
I suggest the explicitly set the 0 if there are no such fields in settings
@ -23,6 +23,7 @@ using namespace std;
std::string_view kMeasurementUnits = "Units";
Yes. But I suppose that it is better to store all the setting keys in
settings
to avoid accidental key duplication and unnecessary string literal usage. It will guarantee key uniquenessYes, and if
popupCloseTime == 0
,popupCloseTime + kLastProductCloseTimeout < now
is true.One condition is enough here.
@ -23,6 +23,7 @@ using namespace std;
std::string_view kMeasurementUnits = "Units";
We already have a dozen keys declared in framework.cpp
From my perspective, the more local scope they have - the better.
I think that 2 hours is too much. 30 minutes max, IMO.
I did a quick check, my OM Beta total foreground now is 3570 < 1 hour. I'm developer and active tester :)
Did uninstall and install Beta 2 weeks ago probably.
1 week of my developer-tester metrics:
US_FirstLaunch : 1731609060
US_LastBackground : 1732198335
US_SessionsCount : 105
US_TotalForeground : 3570
Updated to 15 min!
@ -3316,0 +3346,4 @@
auto const now = base::SecondsSinceEpoch();
auto const timeout = GetTimeoutForReason(FromString(productCloseReason));
bool const timeoutExpired = popupCloseTime + timeout < now;
Let's take out the function:
And call it before IsPointCoveredByDownloadedMaps
nit:
*m_infoGetter
@ -3316,0 +3346,4 @@
auto const now = base::SecondsSinceEpoch();
auto const timeout = GetTimeoutForReason(FromString(productCloseReason));
bool const timeoutExpired = popupCloseTime + timeout < now;
Nice idea! Thanks!
The "Remind Later" and close "x" behavior should be different.
Now both lead to a 2 weeks delay till the next popup.
IMO its too long for "Remind Later", maybe a week?
People will choose the "x" button to close the popup if their reason is different from other available options (donate, remind later, already donated): don't want to donate, no money, etc.
There is no point to display the popup again soon to users who e.g. doesn't want to donate, but it could lead to unwanted irritation.
So its better to change the "x" close option to a long delay like it was in the initial version - maybe same as Already Donated 180 days.
Could we make a longer lead-in delay for brand new users? As proposed in organicmaps/organicmaps#8638
E.g. if its the first app start then delay the popup for a month?
The 15 in foreground is about 2-3 days of average usage. @vng has made a test somewhere in comments...
@ -3316,0 +3378,4 @@
uint32_t constexpr kProductSelectTimeout = 20;
uint32_t constexpr kRemindMeLaterTimeout = 5;
#else
uint32_t constexpr kPopupCloseTimeout = 60 * 60 * 24 * 30; // 30 days
why do we need to store the exact donation option a user has chosen?
seems like its not used anywhere
15 minutes is very little in terms of getting to know the app.
In terms of the foreground time it should be several hours.
@ -3316,0 +3378,4 @@
uint32_t constexpr kProductSelectTimeout = 20;
uint32_t constexpr kRemindMeLaterTimeout = 5;
#else
uint32_t constexpr kPopupCloseTimeout = 60 * 60 * 24 * 30; // 30 days
btw I assume a user could change the donation option (amount, recurring or one-time) after the button press? i.e. while on the stripe page already?
organicmaps/organicmaps#9695
With this branch organicmaps/organicmaps#9710
you can get US_* log entries (when the app starts) to check your usage. Several hours in foreground is way too much.
@ -3316,0 +3378,4 @@
uint32_t constexpr kProductSelectTimeout = 20;
uint32_t constexpr kRemindMeLaterTimeout = 5;
#else
uint32_t constexpr kPopupCloseTimeout = 60 * 60 * 24 * 30; // 30 days
Yes, he can do anything after the donate page, but we save what button prompted him to go.
The same 15min foreground time could differ wildly in terms of the exploration of app's features.
E.g. you're an experienced user and tend to be very efficient when you beta-test or use the app, but for a new user 15min could be just enough to download some maps and browse around a little. Or its just 15min of a drive (using the navigation feature) and not much else.
I think 15min won't be enough even for a weekend trip for most new users.
Let's make it an hour at least? (could translate to somewhere between a week and a month of "real time", which looks reasonable)
It applies for new users only, of course.
For existing users there is no need to wait 15min even, e.g. make it 3min (as we're sure existing users have had the app installed at least 2 weeks before according to our update cycle).
Looks great!
Please set Later to 3 days.
The opinion poll in OM dev chats showed that weighted average preference is 2.4 months (
(0.5 * 3 + 1 * 3 + 3 * 4 + 6 * 2) / 12 = 2.38
) (based on votes of 12 developers).I guess we can round it to 2.5 months / 75 days?
I propose:
Strange logic now here.
We should save kPlacePageProductsPopupCloseTime (ok now) and last ProductsPopupCloseReason instead of kPlacePageProductRemindMeLaterTime or kPlacePageProductSelectionTime.
Then according to the last saved ProductsPopupCloseReason we select a timeout to check in ShouldShowProducts.
Thanks for running the poll. There are no illusions that this new donation reminder may be bothersome. Reminders are meant to be a somehow annoying - that is what they are for. Ask people how often they want to be annoyed, and the option to not add this reminder at all would easily win any poll by a large margin.
The proposed UI is fairly moderate and far from being as disruptive as a paywall. This particular constant in the code is for "X" (Close) button that plays "Remind me later" function. 2.5 months, or 75 days, is an eternity in terms of an app's lifecycle. Most people probably do not even remember the app they used after just a few months. Two weeks is probably the maximum reasonable value here to make this reminder functional. People who donated can press "I've donated already" button to hide the message for 6 months.
Did you forget there is a separate "Remind Later" button which is set to 3 days (as per your own suggestion organicmaps/organicmaps#9695) ? :)
The "X" (close) button is for other reasons, e.g. "Don't want to donate".
But please let's not repeat ourselves with the same arguments.
Developers have different opinions on this topic and we couldn't come to an agreement, that's why we had a poll.
Fixed! I moved the Timeout selection to a separate method based on the reason and stored the reason in the settings.
We should return some value here in case of possible bad value. Lets say kPopupCloseTimeout.
nit: I'd prefer like-string serialization here, but ok ..
nit: int productCloseReason;
yeap it'd better in case the list of reason changes later