mirror of
https://github.com/status-im/status-react.git
synced 2025-01-11 11:34:45 +00:00
feat: Keypair name handling (#19423)
feat: Keypair name handling (#19423)
This commit is contained in:
parent
a03cc3c3ab
commit
e2b9e12038
@ -23,7 +23,8 @@
|
|||||||
:text-color colors/white ;; text color override
|
:text-color colors/white ;; text color override
|
||||||
:icon-color colors/white ;; icon color override
|
:icon-color colors/white ;; icon color override
|
||||||
:no-icon-color? false ;; disable tint color for icon"
|
:no-icon-color? false ;; disable tint color for icon"
|
||||||
[{:keys [type size theme icon text-color icon-color no-icon-color? style accessibility-label]} message]
|
[{:keys [type size theme icon text-color icon-color no-icon-color? style accessibility-label
|
||||||
|
container-style]} message]
|
||||||
(let [weight (if (= size :default) :regular :medium)
|
(let [weight (if (= size :default) :regular :medium)
|
||||||
icon-size (if (= size :default) 16 12)
|
icon-size (if (= size :default) 16 12)
|
||||||
size (if (= size :default) :paragraph-2 :label)
|
size (if (= size :default) :paragraph-2 :label)
|
||||||
@ -32,7 +33,8 @@
|
|||||||
[rn/view
|
[rn/view
|
||||||
{:style (merge {:flex-direction :row
|
{:style (merge {:flex-direction :row
|
||||||
:align-items :center}
|
:align-items :center}
|
||||||
style)}
|
style
|
||||||
|
container-style)}
|
||||||
[quo.icons/icon icon
|
[quo.icons/icon icon
|
||||||
{:color icon-color
|
{:color icon-color
|
||||||
:no-color no-icon-color?
|
:no-color no-icon-color?
|
||||||
|
@ -134,9 +134,9 @@
|
|||||||
:align-items :flex-end})
|
:align-items :flex-end})
|
||||||
|
|
||||||
(defn counter-color
|
(defn counter-color
|
||||||
[current-chars char-limit variant-colors]
|
[{:keys [current-chars char-limit variant-colors theme]}]
|
||||||
{:color (if (> current-chars char-limit)
|
{:color (if (> current-chars char-limit)
|
||||||
colors/danger-60
|
(colors/resolve-color :danger theme)
|
||||||
(:label variant-colors))})
|
(:label variant-colors))})
|
||||||
|
|
||||||
(defn button
|
(defn button
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
[react-native.platform :as platform]))
|
[react-native.platform :as platform]))
|
||||||
|
|
||||||
(defn- label-&-counter
|
(defn- label-&-counter
|
||||||
[{:keys [label current-chars char-limit variant-colors]}]
|
[{:keys [label current-chars char-limit variant-colors theme]}]
|
||||||
[rn/view
|
[rn/view
|
||||||
{:accessibility-label :input-labels
|
{:accessibility-label :input-labels
|
||||||
:style style/texts-container}
|
:style style/texts-container}
|
||||||
@ -22,7 +22,10 @@
|
|||||||
(str current-chars "/"))]
|
(str current-chars "/"))]
|
||||||
[rn/view {:style style/counter-container}
|
[rn/view {:style style/counter-container}
|
||||||
[text/text
|
[text/text
|
||||||
{:style (style/counter-color current-chars char-limit variant-colors)
|
{:style (style/counter-color {:current-chars current-chars
|
||||||
|
:char-limit char-limit
|
||||||
|
:variant-colors variant-colors
|
||||||
|
:theme theme})
|
||||||
:weight :regular
|
:weight :regular
|
||||||
:size :paragraph-2}
|
:size :paragraph-2}
|
||||||
count-text]])])
|
count-text]])])
|
||||||
@ -114,7 +117,8 @@
|
|||||||
{:variant-colors variant-colors
|
{:variant-colors variant-colors
|
||||||
:label label
|
:label label
|
||||||
:current-chars char-count
|
:current-chars char-count
|
||||||
:char-limit char-limit}])
|
:char-limit char-limit
|
||||||
|
:theme theme}])
|
||||||
[rn/view {:style (style/input-container colors-by-status small? disabled?)}
|
[rn/view {:style (style/input-container colors-by-status small? disabled?)}
|
||||||
(when-let [{:keys [icon-name]} left-icon]
|
(when-let [{:keys [icon-name]} left-icon]
|
||||||
[left-accessory
|
[left-accessory
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
[legacy.status-im.ethereum.ens :as ens]
|
[legacy.status-im.ethereum.ens :as ens]
|
||||||
[native-module.core :as native-module]
|
[native-module.core :as native-module]
|
||||||
[re-frame.core :as re-frame]
|
[re-frame.core :as re-frame]
|
||||||
[status-im.common.validators :as validators]
|
[status-im.common.validation.general :as validators]
|
||||||
[status-im.constants :as constants]
|
[status-im.constants :as constants]
|
||||||
[status-im.contexts.chat.events :as chat.events]
|
[status-im.contexts.chat.events :as chat.events]
|
||||||
[taoensso.timbre :as log]
|
[taoensso.timbre :as log]
|
||||||
|
25
src/status_im/common/validation/general.cljs
Normal file
25
src/status_im/common/validation/general.cljs
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
(ns status-im.common.validation.general
|
||||||
|
(:require
|
||||||
|
[status-im.constants :as constants]
|
||||||
|
[utils.emojilib :as emoji]))
|
||||||
|
|
||||||
|
(defn valid-public-key?
|
||||||
|
[s]
|
||||||
|
(and (string? s)
|
||||||
|
(not-empty s)
|
||||||
|
(boolean (re-matches constants/regx-public-key s))))
|
||||||
|
|
||||||
|
(defn valid-compressed-key?
|
||||||
|
[s]
|
||||||
|
(and (string? s)
|
||||||
|
(not-empty s)
|
||||||
|
(boolean (re-matches constants/regx-compressed-key s))))
|
||||||
|
|
||||||
|
(defn has-emojis? [s] (boolean (re-find emoji/emoji-regex s)))
|
||||||
|
|
||||||
|
(def no-special-chars-regex #"^[a-zA-Z0-9\-_ ]+$")
|
||||||
|
(defn has-special-characters?
|
||||||
|
[s]
|
||||||
|
(and (not (= s ""))
|
||||||
|
(not (re-find no-special-chars-regex s))))
|
||||||
|
|
@ -1,7 +1,7 @@
|
|||||||
(ns status-im.common.validators-test
|
(ns status-im.common.validation.general-test
|
||||||
(:require
|
(:require
|
||||||
[cljs.test :refer-macros [deftest testing is]]
|
[cljs.test :refer-macros [deftest testing is]]
|
||||||
[status-im.common.validators :refer [valid-compressed-key?]]))
|
[status-im.common.validation.general :refer [valid-compressed-key?]]))
|
||||||
|
|
||||||
(deftest test-valid-compressed-key
|
(deftest test-valid-compressed-key
|
||||||
(testing "valid"
|
(testing "valid"
|
||||||
@ -32,3 +32,4 @@
|
|||||||
(testing "contains l"
|
(testing "contains l"
|
||||||
(is (not (valid-compressed-key?
|
(is (not (valid-compressed-key?
|
||||||
"zQ3shWj4WaBdf2zYKCkXe6PHxDxNTzZyid1i75879Ue9cX9gl")))))
|
"zQ3shWj4WaBdf2zYKCkXe6PHxDxNTzZyid1i75879Ue9cX9gl")))))
|
||||||
|
|
@ -1,5 +1,6 @@
|
|||||||
(ns status-im.common.validation.profile
|
(ns status-im.common.validation.profile
|
||||||
(:require [clojure.string :as string]
|
(:require [clojure.string :as string]
|
||||||
|
[status-im.common.validation.general :as validators]
|
||||||
[status-im.constants :as constants]
|
[status-im.constants :as constants]
|
||||||
utils.emojilib
|
utils.emojilib
|
||||||
[utils.i18n :as i18n]))
|
[utils.i18n :as i18n]))
|
||||||
@ -8,16 +9,10 @@
|
|||||||
;; https://github.com/status-im/status-desktop/blob/2ba96803168461088346bf5030df750cb226df4c/ui/imports/utils/Constants.qml#L468
|
;; https://github.com/status-im/status-desktop/blob/2ba96803168461088346bf5030df750cb226df4c/ui/imports/utils/Constants.qml#L468
|
||||||
(def min-length 5)
|
(def min-length 5)
|
||||||
|
|
||||||
(def status-regex #"^[a-zA-Z0-9\-_ ]+$")
|
|
||||||
|
|
||||||
(def common-names ["Ethereum" "Bitcoin"])
|
(def common-names ["Ethereum" "Bitcoin"])
|
||||||
|
|
||||||
(defn has-emojis? [s] (boolean (re-find utils.emojilib/emoji-regex s)))
|
|
||||||
|
|
||||||
(defn has-common-names? [s] (pos? (count (filter #(string/includes? s %) common-names))))
|
(defn has-common-names? [s] (pos? (count (filter #(string/includes? s %) common-names))))
|
||||||
|
|
||||||
(defn has-special-characters? [s] (not (re-find status-regex s)))
|
|
||||||
|
|
||||||
(defn name-too-short? [s] (< (count (string/trim (str s))) min-length))
|
(defn name-too-short? [s] (< (count (string/trim (str s))) min-length))
|
||||||
|
|
||||||
(defn name-too-long? [s] (> (count (string/trim (str s))) constants/profile-name-max-length))
|
(defn name-too-long? [s] (> (count (string/trim (str s))) constants/profile-name-max-length))
|
||||||
@ -27,33 +22,37 @@
|
|||||||
(defn validation-name
|
(defn validation-name
|
||||||
[s]
|
[s]
|
||||||
(cond
|
(cond
|
||||||
(string/blank? s) nil
|
(string/blank? s) nil
|
||||||
(string/ends-with? s "-eth") (i18n/label :t/ending-not-allowed {:ending "-eth"})
|
(string/ends-with? s "-eth") (i18n/label :t/ending-not-allowed {:ending "-eth"})
|
||||||
(string/ends-with? s "_eth") (i18n/label :t/ending-not-allowed {:ending "_eth"})
|
(string/ends-with? s "_eth") (i18n/label :t/ending-not-allowed {:ending "_eth"})
|
||||||
(string/ends-with? s ".eth") (i18n/label :t/ending-not-allowed {:ending ".eth"})
|
(string/ends-with? s ".eth") (i18n/label :t/ending-not-allowed {:ending ".eth"})
|
||||||
(string/starts-with? s " ") (i18n/label :t/start-with-space)
|
(string/starts-with? s " ") (i18n/label :t/start-with-space)
|
||||||
(string/ends-with? s " ") (i18n/label :t/ends-with-space)
|
(string/ends-with? s " ") (i18n/label :t/ends-with-space)
|
||||||
(has-common-names? s) (i18n/label :t/are-not-allowed {:check (i18n/label :t/common-names)})
|
(has-common-names? s) (i18n/label :t/are-not-allowed
|
||||||
(has-emojis? s) (i18n/label :t/are-not-allowed {:check (i18n/label :t/emojis)})
|
{:check (i18n/label :t/common-names)})
|
||||||
(has-special-characters? s) (i18n/label :t/are-not-allowed
|
(validators/has-emojis? s) (i18n/label :t/are-not-allowed
|
||||||
{:check (i18n/label :t/special-characters)})
|
{:check (i18n/label :t/emojis)})
|
||||||
(name-too-short? s) (i18n/label :t/minimum-characters {:min-chars min-length})
|
(validators/has-special-characters? s) (i18n/label :t/are-not-allowed
|
||||||
(name-too-long? s) (i18n/label :t/profile-name-is-too-long)))
|
{:check (i18n/label :t/special-characters)})
|
||||||
|
(name-too-short? s) (i18n/label :t/minimum-characters {:min-chars min-length})
|
||||||
|
(name-too-long? s) (i18n/label :t/profile-name-is-too-long)))
|
||||||
|
|
||||||
(defn validation-bio
|
(defn validation-bio
|
||||||
[s]
|
[s]
|
||||||
(cond
|
(cond
|
||||||
(string/blank? s) nil
|
(string/blank? s) nil
|
||||||
(has-emojis? s) (i18n/label :t/are-not-allowed {:check (i18n/label :t/emojis)})
|
(validators/has-emojis? s) (i18n/label :t/are-not-allowed
|
||||||
(has-special-characters? s) (i18n/label :t/are-not-allowed
|
{:check (i18n/label :t/emojis)})
|
||||||
{:check (i18n/label :t/special-characters)})
|
(validators/has-special-characters? s) (i18n/label :t/are-not-allowed
|
||||||
(bio-too-long? s) (i18n/label :t/bio-is-too-long)))
|
{:check (i18n/label :t/special-characters)})
|
||||||
|
(bio-too-long? s) (i18n/label :t/bio-is-too-long)))
|
||||||
|
|
||||||
(defn validation-nickname
|
(defn validation-nickname
|
||||||
[s]
|
[s]
|
||||||
(cond
|
(cond
|
||||||
(string/blank? s) nil
|
(string/blank? s) nil
|
||||||
(has-emojis? s) (i18n/label :t/are-not-allowed {:check (i18n/label :t/emojis)})
|
(validators/has-emojis? s) (i18n/label :t/are-not-allowed
|
||||||
(has-special-characters? s) (i18n/label :t/are-not-allowed
|
{:check (i18n/label :t/emojis)})
|
||||||
{:check (i18n/label :t/special-characters)})
|
(validators/has-special-characters? s) (i18n/label :t/are-not-allowed
|
||||||
(name-too-long? s) (i18n/label :t/nickname-is-too-long)))
|
{:check (i18n/label :t/special-characters)})
|
||||||
|
(name-too-long? s) (i18n/label :t/nickname-is-too-long)))
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
(ns status-im.common.validation.profile-test
|
(ns status-im.common.validation.profile-test
|
||||||
(:require
|
(:require
|
||||||
[cljs.test :refer-macros [deftest are]]
|
[cljs.test :refer-macros [deftest are]]
|
||||||
|
[status-im.common.validation.general :as validator]
|
||||||
[status-im.common.validation.profile :as profile-validator]
|
[status-im.common.validation.profile :as profile-validator]
|
||||||
[utils.i18n :as i18n]))
|
[utils.i18n :as i18n]))
|
||||||
|
|
||||||
(deftest has-emojis-test
|
(deftest has-emojis-test
|
||||||
(are [arg expected]
|
(are [arg expected]
|
||||||
(expected (profile-validator/has-emojis? arg))
|
(expected (validator/has-emojis? arg))
|
||||||
"Hello 😊" true?
|
"Hello 😊" true?
|
||||||
"Hello" false?))
|
"Hello" false?))
|
||||||
|
|
||||||
@ -18,7 +19,7 @@
|
|||||||
|
|
||||||
(deftest has-special-characters-test
|
(deftest has-special-characters-test
|
||||||
(are [arg expected]
|
(are [arg expected]
|
||||||
(expected (profile-validator/has-special-characters? arg))
|
(expected (validator/has-special-characters? arg))
|
||||||
"@name" true?
|
"@name" true?
|
||||||
"name" false?))
|
"name" false?))
|
||||||
|
|
||||||
|
@ -1,15 +0,0 @@
|
|||||||
(ns status-im.common.validators
|
|
||||||
(:require
|
|
||||||
[status-im.constants :as constants]))
|
|
||||||
|
|
||||||
(defn valid-public-key?
|
|
||||||
[s]
|
|
||||||
(and (string? s)
|
|
||||||
(not-empty s)
|
|
||||||
(boolean (re-matches constants/regx-public-key s))))
|
|
||||||
|
|
||||||
(defn valid-compressed-key?
|
|
||||||
[s]
|
|
||||||
(and (string? s)
|
|
||||||
(not-empty s)
|
|
||||||
(boolean (re-matches constants/regx-compressed-key s))))
|
|
@ -2,7 +2,7 @@
|
|||||||
(:require
|
(:require
|
||||||
[clojure.string :as string]
|
[clojure.string :as string]
|
||||||
[re-frame.core :as re-frame]
|
[re-frame.core :as re-frame]
|
||||||
[status-im.common.validators :as validators]
|
[status-im.common.validation.general :as validators]
|
||||||
[status-im.contexts.chat.contacts.events :as data-store.contacts]
|
[status-im.contexts.chat.contacts.events :as data-store.contacts]
|
||||||
status-im.contexts.chat.home.add-new-contact.effects
|
status-im.contexts.chat.home.add-new-contact.effects
|
||||||
[utils.ens.stateofus :as stateofus]
|
[utils.ens.stateofus :as stateofus]
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
[react-native.hooks :as hooks]
|
[react-native.hooks :as hooks]
|
||||||
[status-im.common.router :as router]
|
[status-im.common.router :as router]
|
||||||
[status-im.common.scan-qr-code.view :as scan-qr-code]
|
[status-im.common.scan-qr-code.view :as scan-qr-code]
|
||||||
[status-im.common.validators :as validators]
|
[status-im.common.validation.general :as validators]
|
||||||
[status-im.contexts.communities.events]
|
[status-im.contexts.communities.events]
|
||||||
[status-im.contexts.wallet.common.validation :as wallet-validation]
|
[status-im.contexts.wallet.common.validation :as wallet-validation]
|
||||||
[utils.debounce :as debounce]
|
[utils.debounce :as debounce]
|
||||||
|
@ -143,4 +143,3 @@
|
|||||||
:delete-key? true}])]))))
|
:delete-key? true}])]))))
|
||||||
|
|
||||||
(def view (quo.theme/with-theme view-internal))
|
(def view (quo.theme/with-theme view-internal))
|
||||||
|
|
||||||
|
@ -7,3 +7,7 @@
|
|||||||
|
|
||||||
(def bottom-action
|
(def bottom-action
|
||||||
{:margin-horizontal -20})
|
{:margin-horizontal -20})
|
||||||
|
|
||||||
|
(def error-container
|
||||||
|
{:margin-left 20
|
||||||
|
:margin-vertical 8})
|
||||||
|
@ -2,44 +2,72 @@
|
|||||||
(:require
|
(:require
|
||||||
[quo.core :as quo]
|
[quo.core :as quo]
|
||||||
[react-native.core :as rn]
|
[react-native.core :as rn]
|
||||||
[reagent.core :as reagent]
|
|
||||||
[status-im.common.floating-button-page.view :as floating-button-page]
|
[status-im.common.floating-button-page.view :as floating-button-page]
|
||||||
|
[status-im.common.validation.general :as validators]
|
||||||
[status-im.contexts.wallet.create-account.new-keypair.keypair-name.style :as style]
|
[status-im.contexts.wallet.create-account.new-keypair.keypair-name.style :as style]
|
||||||
[utils.i18n :as i18n]
|
[utils.i18n :as i18n]
|
||||||
[utils.re-frame :as rf]))
|
[utils.re-frame :as rf]))
|
||||||
|
|
||||||
(def keypair-name-max-length 15)
|
(def keypair-name-max-length 15)
|
||||||
|
(def keypair-name-min-length 5)
|
||||||
|
|
||||||
|
(def error-messages
|
||||||
|
{:length (i18n/label :t/key-name-error-length)
|
||||||
|
:emoji (i18n/label :t/key-name-error-emoji)
|
||||||
|
:special-char (i18n/label :t/key-name-error-special-char)})
|
||||||
|
|
||||||
|
(defn navigate-back [] (rf/dispatch [:navigate-back]))
|
||||||
|
|
||||||
(defn view
|
(defn view
|
||||||
[]
|
[]
|
||||||
(let [keypair-name (reagent/atom "")]
|
(let [[keypair-name set-keypair-name] (rn/use-state "")
|
||||||
(fn []
|
customization-color (rf/sub [:profile/customization-color])
|
||||||
(let [customization-color (rf/sub [:profile/customization-color])]
|
[error set-error] (rn/use-state false)
|
||||||
[rn/view {:style {:flex 1}}
|
on-change-text (rn/use-callback (fn [value]
|
||||||
[floating-button-page/view
|
(set-keypair-name value)
|
||||||
{:header [quo/page-nav
|
(cond
|
||||||
{:icon-name :i/arrow-left
|
(> (count value) keypair-name-max-length)
|
||||||
:on-press #(rf/dispatch [:navigate-back])
|
(set-error :length)
|
||||||
:accessibility-label :top-bar}]
|
(validators/has-emojis? value) (set-error
|
||||||
:footer [quo/bottom-actions
|
:emoji)
|
||||||
{:actions :one-action
|
(validators/has-special-characters? value)
|
||||||
:button-one-label (i18n/label :t/continue)
|
(set-error :special-char)
|
||||||
:button-one-props {:disabled? (or (zero? (count @keypair-name))
|
:else (set-error nil))))
|
||||||
(> (count @keypair-name)
|
on-continue (rn/use-callback #(rf/dispatch [:wallet/new-keypair-continue
|
||||||
keypair-name-max-length))
|
{:keypair-name
|
||||||
:customization-color customization-color
|
keypair-name}])
|
||||||
:on-press #(rf/dispatch [:wallet/new-keypair-continue
|
[keypair-name])]
|
||||||
{:keypair-name
|
[rn/view {:style {:flex 1}}
|
||||||
@keypair-name}])}
|
[floating-button-page/view
|
||||||
:container-style style/bottom-action}]}
|
{:header [quo/page-nav
|
||||||
[quo/text-combinations
|
{:icon-name :i/arrow-left
|
||||||
{:container-style style/header-container
|
:on-press navigate-back
|
||||||
:title (i18n/label :t/keypair-name)
|
:accessibility-label :top-bar}]
|
||||||
:description (i18n/label :t/keypair-name-description)}]
|
:footer [quo/bottom-actions
|
||||||
[quo/input
|
{:actions :one-action
|
||||||
{:container-style {:margin-horizontal 20}
|
:button-one-label (i18n/label :t/continue)
|
||||||
:placeholder (i18n/label :t/keypair-name-input-placeholder)
|
:button-one-props {:disabled? (or (pos? error)
|
||||||
:label (i18n/label :t/keypair-name)
|
(<= (count keypair-name)
|
||||||
:char-limit keypair-name-max-length
|
keypair-name-min-length))
|
||||||
:auto-focus true
|
:customization-color customization-color
|
||||||
:on-change-text #(reset! keypair-name %)}]]]))))
|
:on-press on-continue}
|
||||||
|
:container-style style/bottom-action}]}
|
||||||
|
[quo/text-combinations
|
||||||
|
{:container-style style/header-container
|
||||||
|
:title (i18n/label :t/keypair-name)
|
||||||
|
:description (i18n/label :t/keypair-name-description)}]
|
||||||
|
[quo/input
|
||||||
|
{:container-style {:margin-horizontal 20}
|
||||||
|
:placeholder (i18n/label :t/keypair-name-input-placeholder)
|
||||||
|
:label (i18n/label :t/keypair-name)
|
||||||
|
:char-limit keypair-name-max-length
|
||||||
|
:auto-focus true
|
||||||
|
:on-change-text on-change-text
|
||||||
|
:error error}]
|
||||||
|
(when error
|
||||||
|
[quo/info-message
|
||||||
|
{:type :error
|
||||||
|
:size :default
|
||||||
|
:icon :i/info
|
||||||
|
:container-style style/error-container}
|
||||||
|
(get error-messages error)])]]))
|
||||||
|
@ -2562,5 +2562,8 @@
|
|||||||
"one": "1 address",
|
"one": "1 address",
|
||||||
"other": "{{count}} addresses"
|
"other": "{{count}} addresses"
|
||||||
},
|
},
|
||||||
"max": "Max: {{number}}"
|
"max": "Max: {{number}}",
|
||||||
|
"key-name-error-length": "Key name too long",
|
||||||
|
"key-name-error-emoji": "Emojis are not allowed",
|
||||||
|
"key-name-error-special-char": "Special characters are not allowed"
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user