Skip to content

Commit e1abe5c

Browse files
authored
feat(onboarding): Present Terms to users upgrading from v1 or those who need to accept updated Terms (#21487)
Cherry-pick d45eb5e. Fixes #21113 Related status-go PR: status-im/status-go#5977
1 parent cac96b5 commit e1abe5c

File tree

12 files changed

+188
-80
lines changed

12 files changed

+188
-80
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

+84-57
Original file line numberDiff line numberDiff line change
@@ -11,66 +11,93 @@
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+
has-profiles-and-unaccepted-terms? (rf/sub [:profile/has-profiles-and-unaccepted-terms?])]
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+
83+
has-profiles-and-unaccepted-terms?
84+
(assoc
85+
:actions :one-action
86+
:button-one-label (i18n/label :t/explore-the-new-status)
87+
:button-one-props {:disabled? (not terms-accepted?)
88+
:accessibility-label :explore-new-status
89+
:on-press explore-new-status})
90+
91+
(not has-profiles-and-unaccepted-terms?)
92+
(assoc
93+
:actions :two-vertical-actions
94+
:button-one-label (i18n/label :t/sync-or-recover-profile)
95+
:button-one-props {:type :dark-grey
96+
:disabled? (not terms-accepted?)
97+
:accessibility-label :already-use-status-button
98+
:on-press sync-or-recover-profile}
99+
:button-two-label (i18n/label :t/create-profile)
100+
:button-two-props {:accessibility-label :new-to-status-button
101+
:disabled? (not terms-accepted?)
102+
:on-press create-profile}))]
76103
[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

+10
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
[re-frame.core :as re-frame]
99
[status-im.common.pixel-ratio :as pixel-ratio]
1010
[status-im.constants :as constants]
11+
[status-im.contexts.profile.data-store :as profile.data-store]
1112
[status-im.contexts.profile.utils :as profile.utils]
1213
[utils.security.core :as security]))
1314

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

21+
;; A profile can only be created without accepting terms in Status v1. In Status
22+
;; v2, the terms may be unaccepted because we intentionally created a migration
23+
;; resetting the flag in case the privacy policy changed.
24+
(re-frame/reg-sub :profile/has-profiles-and-unaccepted-terms?
25+
:<- [:profile/profiles-overview]
26+
(fn [profiles-overview]
27+
(and (seq profiles-overview)
28+
(not (profile.data-store/accepted-terms? (vals profiles-overview))))))
29+
2030
(re-frame/reg-sub
2131
:profile/currency
2232
:<- [:profile/profile]

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

+3-3
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
"_comment": "Instead use: scripts/update-status-go.sh <rev>",
44
"owner": "status-im",
55
"repo": "status-go",
6-
"version": "v3.3.0",
7-
"commit-sha1": "43659f0c7e1a64102e7f8e38bd532b72714c5819",
8-
"src-sha256": "01fpviw1g22f4ni3li2wnhyz0l517jfipvnyfysckds6yi94l0hf"
6+
"version": "v3.5.0",
7+
"commit-sha1": "39511298cdc8f50e7659ac8079d5bc8a4763ddc7",
8+
"src-sha256": "1nn8gq9d91ix5mzndr306rq3n68rzj4drym3pfaxzaa73k4v9r07"
99
}

translations/en.json

+1
Original file line numberDiff line numberDiff line change
@@ -1002,6 +1002,7 @@
10021002
"expand-all": "Expand all",
10031003
"experienced-web3": "Experienced in Web3?",
10041004
"explore-the-decentralized-web": "Explore and interact with the decentralized web",
1005+
"explore-the-new-status": "Explore the new Status",
10051006
"export-account": "Export account",
10061007
"export-key": "Export private key",
10071008
"external-link": "External link",

0 commit comments

Comments
 (0)