Skip to content

Commit d45eb5e

Browse files
authored
feat(onboarding): Present Terms to users upgrading from v1 (#21124)
We now show the onboarding intro requesting the user to accept the Terms of Use & Privacy Policy with the new button "Explore the new Status" if the user had installed any version of Status older than the one from this PR and had at least one profile. Fixes #21113 status-go PR status-im/status-go#5766 In practice, this means: - Users coming from Status v1 who had at least one profile will see the modified onboarding intro screen and will need to accept the terms to proceed. - Users who already installed v2 and are upgrading to this PR build (devs & QAs mostly) and who had at least one profile will also see the modified intro screen and will need to accept the terms to proceed. Areas that may be impacted - Onboarding Steps to test: The criteria used during development: 1. Given that user Alice had installed v1 and had one or more profiles. 2. When she installs v2 and opens it, she sees the new onboarding intro and must agree to the terms to enable the button "Explore the new Status". 3. After pressing the button, she can login as usual in any of her profiles. 1. Given that user Alice already upgraded from v1 and accepted the terms. 2. When she reopens the app she does not need to accept terms again and can immediately sign-in with any of her profiles. 1. Given that user Alice already upgraded from v1 and accepted the terms. 2. When she deletes all profiles, she sees the onboarding intro for users who have not upgraded, i.e. she has to agree to terms and she sees the usual two buttons "Create profile" and "Sync or recover profile". 1. Given that user Alice never installed Status. 2. When she installs v2, she sees the normal onboarding intro screen, where she has to accept the terms and she sees two buttons "Create profile" and "Sync or recover profile". 3. When she reopens the app, she doesn't see anymore the screen to accept terms.
1 parent b1c9077 commit d45eb5e

File tree

12 files changed

+189
-79
lines changed

12 files changed

+189
-79
lines changed

modules/react-native-status/android/src/main/java/im/status/ethereum/module/AccountManager.kt

+5
Original file line numberDiff line numberDiff line change
@@ -253,6 +253,11 @@ class AccountManager(private val reactContext: ReactApplicationContext) : ReactC
253253
utils.executeRunnableStatusGoMethod({ Statusgo.initializeApplication(request) }, callback)
254254
}
255255

256+
@ReactMethod
257+
private fun acceptTerms(callback: Callback) {
258+
Log.d(TAG, "acceptTerms")
259+
utils.executeRunnableStatusGoMethod({ Statusgo.acceptTerms() }, callback)
260+
}
256261

257262
@ReactMethod
258263
fun logout() {

modules/react-native-status/ios/RCTStatus/AccountManager.m

+8
Original file line numberDiff line numberDiff line change
@@ -204,6 +204,14 @@ -(NSString *) prepareDirAndUpdateConfig:(NSString *)config
204204
callback(@[result]);
205205
}
206206

207+
RCT_EXPORT_METHOD(acceptTerms:(RCTResponseSenderBlock)callback) {
208+
#if DEBUG
209+
NSLog(@"acceptTerms() method called");
210+
#endif
211+
NSString *result = StatusgoAcceptTerms();
212+
callback(@[result]);
213+
}
214+
207215
RCT_EXPORT_METHOD(openAccounts:(RCTResponseSenderBlock)callback) {
208216
#if DEBUG
209217
NSLog(@"OpenAccounts() method called");

modules/react-native-status/nodejs/status.cpp

+21
Original file line numberDiff line numberDiff line change
@@ -761,6 +761,26 @@ void _Logout(const FunctionCallbackInfo<Value>& args) {
761761

762762
}
763763

764+
void _AcceptTerms(const FunctionCallbackInfo<Value>& args) {
765+
Isolate* isolate = args.GetIsolate();
766+
767+
if (args.Length() != 0) {
768+
// Throw an Error that is passed back to JavaScript
769+
isolate->ThrowException(Exception::TypeError(
770+
String::NewFromUtf8Literal(isolate, "Wrong number of arguments for AcceptTerms")));
771+
return;
772+
}
773+
774+
// Check the argument types
775+
776+
// Call exported Go function, which returns a C string
777+
char *c = AcceptTerms();
778+
779+
Local<String> ret = String::NewFromUtf8(isolate, c).ToLocalChecked();
780+
args.GetReturnValue().Set(ret);
781+
delete c;
782+
}
783+
764784
void _HashMessage(const FunctionCallbackInfo<Value>& args) {
765785
Isolate* isolate = args.GetIsolate();
766786
Local<Context> context = isolate->GetCurrentContext();
@@ -1998,6 +2018,7 @@ void init(Local<Object> exports) {
19982018
NODE_SET_METHOD(exports, "multiAccountStoreAccount", _MultiAccountStoreAccount);
19992019
NODE_SET_METHOD(exports, "initKeystore", _InitKeystore);
20002020
NODE_SET_METHOD(exports, "initializeApplication", _InitializeApplication);
2021+
NODE_SET_METHOD(exports, "acceptTerms", _AcceptTerms);
20012022
NODE_SET_METHOD(exports, "fleets", _Fleets);
20022023
NODE_SET_METHOD(exports, "stopCPUProfiling", _StopCPUProfiling);
20032024
NODE_SET_METHOD(exports, "encodeTransfer", _EncodeTransfer);

src/native_module/core.cljs

+6
Original file line numberDiff line numberDiff line change
@@ -79,6 +79,12 @@
7979
(types/clj->json request)
8080
#(callback (types/json->clj %))))
8181

82+
(defn accept-terms
83+
([]
84+
(native-utils/promisify-native-module-call accept-terms))
85+
([callback]
86+
(.acceptTerms ^js (account-manager) callback)))
87+
8288
(defn prepare-dir-and-update-config
8389
[key-uid config callback]
8490
(log/debug "[native-module] prepare-dir-and-update-config")

src/status_im/contexts/onboarding/intro/view.cljs

+83-57
Original file line numberDiff line numberDiff line change
@@ -11,66 +11,92 @@
1111
[utils.i18n :as i18n]
1212
[utils.re-frame :as rf]))
1313

14+
(defn- show-terms-of-use
15+
[]
16+
(rf/dispatch [:show-bottom-sheet {:content terms/terms-of-use :shell? true}]))
17+
18+
(defn- show-privacy-policy
19+
[]
20+
(rf/dispatch [:show-bottom-sheet {:content privacy/privacy-statement :shell? true}]))
21+
22+
(defn- terms
23+
[terms-accepted? set-terms-accepted?]
24+
[rn/view {:style style/terms-privacy-container}
25+
[rn/view
26+
{:accessibility-label :terms-privacy-checkbox-container}
27+
[quo/selectors
28+
{:type :checkbox
29+
:blur? true
30+
:checked? terms-accepted?
31+
:on-change #(set-terms-accepted? not)}]]
32+
[rn/view {:style style/text-container}
33+
[quo/text
34+
{:style style/plain-text
35+
:size :paragraph-2}
36+
(str (i18n/label :t/accept-status-tos-prefix) " ")]
37+
[quo/text
38+
{:on-press show-terms-of-use
39+
:style style/highlighted-text
40+
:size :paragraph-2
41+
:weight :medium}
42+
(i18n/label :t/terms-of-service)]
43+
[quo/text
44+
{:style style/plain-text
45+
:size :paragraph-2}
46+
" " (i18n/label :t/and) " "]
47+
[quo/text
48+
{:on-press show-privacy-policy
49+
:style style/highlighted-text
50+
:size :paragraph-2
51+
:weight :medium}
52+
(i18n/label :t/intro-privacy-policy)]]])
53+
54+
(defn- explore-new-status
55+
[]
56+
(rf/dispatch [:profile/explore-new-status]))
57+
58+
(defn- sync-or-recover-profile
59+
[]
60+
(when-let [blur-show-fn @overlay/blur-show-fn-atom]
61+
(blur-show-fn))
62+
(rf/dispatch [:open-modal :screen/onboarding.sync-or-recover-profile]))
63+
64+
(defn- create-profile
65+
[]
66+
(when-let [blur-show-fn @overlay/blur-show-fn-atom]
67+
(blur-show-fn))
68+
(rf/dispatch [:open-modal :screen/onboarding.new-to-status]))
69+
1470
(defn view
1571
[]
16-
(let [[terms-accepted? set-terms-accepted?] (rn/use-state false)]
72+
(let [[terms-accepted? set-terms-accepted?] (rn/use-state false)
73+
from-v1-without-terms-accepted? (rf/sub [:profile/from-status-v1-without-terms-accepted?])]
1774
[rn/view {:style style/page-container}
1875
[background/view false]
1976
[quo/bottom-actions
20-
{:container-style (style/bottom-actions-container (safe-area/get-bottom))
21-
:actions :two-vertical-actions
22-
:description :top
23-
:description-top-text [rn/view
24-
{:style style/terms-privacy-container}
25-
[rn/view
26-
{:accessibility-label :terms-privacy-checkbox-container}
27-
[quo/selectors
28-
{:type :checkbox
29-
:blur? true
30-
:checked? terms-accepted?
31-
:on-change #(set-terms-accepted? not)}]]
32-
[rn/view {:style style/text-container}
33-
[quo/text
34-
{:style style/plain-text
35-
:size :paragraph-2}
36-
(str (i18n/label :t/accept-status-tos-prefix) " ")]
37-
[quo/text
38-
{:on-press #(rf/dispatch [:show-bottom-sheet
39-
{:content terms/terms-of-use
40-
:shell? true}])
41-
:style style/highlighted-text
42-
:size :paragraph-2
43-
:weight :medium}
44-
(i18n/label :t/terms-of-service)]
45-
[quo/text
46-
{:style style/plain-text
47-
:size :paragraph-2}
48-
" " (i18n/label :t/and) " "]
49-
[quo/text
50-
{:on-press #(rf/dispatch [:show-bottom-sheet
51-
{:content privacy/privacy-statement
52-
:shell? true}])
53-
:style style/highlighted-text
54-
:size :paragraph-2
55-
:weight :medium}
56-
(i18n/label :t/intro-privacy-policy)]]]
57-
:button-one-label (i18n/label :t/sync-or-recover-profile)
58-
:button-one-props {:type :dark-grey
59-
:disabled? (not terms-accepted?)
60-
:accessibility-label :already-use-status-button
61-
:on-press (fn []
62-
(when-let [blur-show-fn @overlay/blur-show-fn-atom]
63-
(blur-show-fn))
64-
(rf/dispatch
65-
[:open-modal
66-
:screen/onboarding.sync-or-recover-profile]))}
67-
:button-two-label (i18n/label :t/create-profile)
68-
:button-two-props {:accessibility-label :new-to-status-button
69-
:disabled? (not terms-accepted?)
70-
:on-press
71-
(fn []
72-
(when-let [blur-show-fn @overlay/blur-show-fn-atom]
73-
(blur-show-fn))
74-
(rf/dispatch
75-
[:open-modal :screen/onboarding.new-to-status]))}}]
77+
(cond->
78+
{:container-style (style/bottom-actions-container (safe-area/get-bottom))
79+
:actions :two-vertical-actions
80+
:description :top
81+
:description-top-text [terms terms-accepted? set-terms-accepted?]}
82+
from-v1-without-terms-accepted?
83+
(assoc
84+
:actions :one-action
85+
:button-one-label (i18n/label :t/explore-the-new-status)
86+
:button-one-props {:disabled? (not terms-accepted?)
87+
:accessibility-label :explore-new-status
88+
:on-press explore-new-status})
89+
90+
(not from-v1-without-terms-accepted?)
91+
(assoc
92+
:actions :two-vertical-actions
93+
:button-one-label (i18n/label :t/sync-or-recover-profile)
94+
:button-one-props {:type :dark-grey
95+
:disabled? (not terms-accepted?)
96+
:accessibility-label :already-use-status-button
97+
:on-press sync-or-recover-profile}
98+
:button-two-label (i18n/label :t/create-profile)
99+
:button-two-props {:accessibility-label :new-to-status-button
100+
:disabled? (not terms-accepted?)
101+
:on-press create-profile}))]
76102
[overlay/view]]))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
(ns status-im.contexts.profile.data-store)
2+
3+
(defn accepted-terms?
4+
[accounts]
5+
(some #(:hasAcceptedTerms %) accounts))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
(ns status-im.contexts.profile.effects
2+
(:require
3+
[native-module.core :as native-module]
4+
[promesa.core :as promesa]
5+
[taoensso.timbre :as log]
6+
[utils.re-frame :as rf]))
7+
8+
(rf/reg-fx :effects.profile/accept-terms
9+
(fn [{:keys [on-success]}]
10+
(-> (native-module/accept-terms)
11+
(promesa/then (fn []
12+
(rf/call-continuation on-success)))
13+
(promesa/catch (fn [error]
14+
(log/error "Failed to accept terms" {:error error}))))))

src/status_im/contexts/profile/events.cljs

+28-20
Original file line numberDiff line numberDiff line change
@@ -4,10 +4,12 @@
44
[legacy.status-im.multiaccounts.update.core :as multiaccounts.update]
55
[native-module.core :as native-module]
66
[status-im.config :as config]
7+
[status-im.contexts.profile.data-store :as profile.data-store]
78
[status-im.contexts.profile.edit.accent-colour.events]
89
[status-im.contexts.profile.edit.bio.events]
910
[status-im.contexts.profile.edit.header.events]
1011
[status-im.contexts.profile.edit.name.events]
12+
status-im.contexts.profile.effects
1113
status-im.contexts.profile.login.events
1214
[status-im.contexts.profile.rpc :as profile.rpc]
1315
[utils.re-frame :as rf]))
@@ -42,26 +44,27 @@
4244
(rf/reg-event-fx
4345
:profile/get-profiles-overview-success
4446
(fn [{:keys [db]} [{:keys [accounts] {:keys [userConfirmed enabled]} :centralizedMetricsInfo}]]
45-
(let [db-with-settings (assoc db
46-
:centralized-metrics/user-confirmed? userConfirmed
47-
:centralized-metrics/enabled? enabled)]
48-
(if (seq accounts)
49-
(let [profiles (reduce-profiles accounts)
50-
{:keys [key-uid]} (first (sort-by :timestamp > (vals profiles)))]
51-
{:db (if key-uid
52-
(-> db-with-settings
53-
(assoc :profile/profiles-overview profiles)
54-
(update :profile/login #(select-profile % key-uid)))
55-
db-with-settings)
56-
:fx [[:dispatch [:update-theme-and-init-root :screen/profile.profiles]]
57-
(when (and key-uid userConfirmed)
58-
[:effects.biometric/check-if-available
59-
{:key-uid key-uid
60-
:on-success (fn [auth-method]
61-
(rf/dispatch [:profile.login/check-biometric-success key-uid
62-
auth-method]))}])]})
63-
{:db db-with-settings
64-
:fx [[:dispatch [:update-theme-and-init-root :screen/onboarding.intro]]]}))))
47+
(let [db-with-settings (assoc db
48+
:centralized-metrics/user-confirmed? userConfirmed
49+
:centralized-metrics/enabled? enabled)
50+
profiles (reduce-profiles accounts)
51+
{:keys [key-uid]} (first (sort-by :timestamp > (vals profiles)))
52+
new-db (cond-> db-with-settings
53+
(seq profiles)
54+
(assoc :profile/profiles-overview profiles)
55+
56+
key-uid
57+
(update :profile/login #(select-profile % key-uid)))]
58+
{:db new-db
59+
:fx (if (profile.data-store/accepted-terms? accounts)
60+
[[:dispatch [:update-theme-and-init-root :screen/profile.profiles]]
61+
(when (and key-uid userConfirmed)
62+
[:effects.biometric/check-if-available
63+
{:key-uid key-uid
64+
:on-success (fn [auth-method]
65+
(rf/dispatch [:profile.login/check-biometric-success key-uid
66+
auth-method]))}])]
67+
[[:dispatch [:update-theme-and-init-root :screen/onboarding.intro]]])})))
6568

6669
(rf/reg-event-fx
6770
:profile/update-setting-from-backup
@@ -82,3 +85,8 @@
8285
:messages-from-contacts-only
8386
(not (get-in db [:profile/profile :messages-from-contacts-only]))
8487
{})))
88+
89+
(rf/reg-event-fx :profile/explore-new-status
90+
(fn []
91+
{:fx [[:effects.profile/accept-terms
92+
{:on-success [:navigate-to :screen/profile.profiles]}]]}))

src/status_im/subs/profile.cljs

+13
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
[re-frame.core :as re-frame]
1010
[status-im.common.pixel-ratio :as pixel-ratio]
1111
[status-im.constants :as constants]
12+
[status-im.contexts.profile.data-store :as profile.data-store]
1213
[status-im.contexts.profile.utils :as profile.utils]
1314
[utils.security.core :as security]))
1415

@@ -18,6 +19,18 @@
1819
(fn [{:keys [customization-color]}]
1920
(or customization-color constants/profile-default-color)))
2021

22+
(re-frame/reg-sub :profile/accepted-terms?
23+
:<- [:profile/profile]
24+
(fn [{:keys [hasAcceptedTerms]}]
25+
hasAcceptedTerms))
26+
27+
;; A profile can only be created without accepting terms in Status v1.
28+
(re-frame/reg-sub :profile/from-status-v1-without-terms-accepted?
29+
:<- [:profile/profiles-overview]
30+
(fn [profiles-overview]
31+
(and (seq profiles-overview)
32+
(not (profile.data-store/accepted-terms? (vals profiles-overview))))))
33+
2134
(re-frame/reg-sub
2235
:profile/currency
2336
(fn []

src/tests/test_utils.cljs

+3
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,9 @@
7070
{:initializeApplication
7171
(fn [request callback]
7272
(callback (.initializeApplication native-status request)))
73+
:acceptTerms
74+
(fn [callback]
75+
(callback (.acceptTerms native-status)))
7376
:createAccountAndLogin
7477
(fn [request] (.createAccountAndLogin native-status request))
7578
:restoreAccountAndLogin

status-go-version.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,6 @@
44
"owner": "status-im",
55
"repo": "status-go",
66
"version": "release/0.182.x",
7-
"commit-sha1": "4a18c85c3c1d58ea6c8493c46bfd2ed5772b1386",
8-
"src-sha256": "04fgykwk44r7f16bfxlkpl9kgnl7yssfyycqnddwszinnnlnfmpl"
7+
"commit-sha1": "14c996158cf1d651d44808d51686fdbfb2eb3b39",
8+
"src-sha256": "02dnz3327kz8cnhqp6cgcmvqvhcdc941iic7jz2v35hck221vbkg"
99
}

translations/en.json

+1
Original file line numberDiff line numberDiff line change
@@ -986,6 +986,7 @@
986986
"expand-all": "Expand all",
987987
"experienced-web3": "Experienced in Web3?",
988988
"explore-the-decentralized-web": "Explore and interact with the decentralized web",
989+
"explore-the-new-status": "Explore the new Status",
989990
"export-account": "Export account",
990991
"export-key": "Export private key",
991992
"external-link": "External link",

0 commit comments

Comments
 (0)