[#11046] Add local contact names

Signed-off-by: andrey <motor4ik@gmail.com>
This commit is contained in:
andrey 2020-08-20 16:26:40 +02:00
parent fea916590a
commit 4304a5dd97
No known key found for this signature in database
GPG Key ID: 89B67245FD2F0272
23 changed files with 375 additions and 205 deletions

View File

@ -12,7 +12,8 @@
[status-im.navigation :as navigation] [status-im.navigation :as navigation]
[status-im.utils.fx :as fx] [status-im.utils.fx :as fx]
[status-im.waku.core :as waku] [status-im.waku.core :as waku]
[taoensso.timbre :as log])) [taoensso.timbre :as log]
[clojure.string :as string]))
(fx/defn load-contacts (fx/defn load-contacts
{:events [::contacts-loaded]} {:events [::contacts-loaded]}
@ -59,7 +60,9 @@
[{:keys [db]} contacts] [{:keys [db]} contacts]
{:db (update db :contacts/contacts {:db (update db :contacts/contacts
#(reduce (fn [acc {:keys [public-key] :as contact}] #(reduce (fn [acc {:keys [public-key] :as contact}]
(update acc public-key merge contact)) (-> acc
(update public-key merge contact)
(assoc-in [public-key :nickname] (:nickname contact))))
% %
contacts))}) contacts))})
@ -81,12 +84,15 @@
(fx/defn add-contact (fx/defn add-contact
"Add a contact and set pending to false" "Add a contact and set pending to false"
[{:keys [db] :as cofx} public-key] [{:keys [db] :as cofx} public-key nickname]
(when (not= (get-in db [:multiaccount :public-key]) public-key) (when (not= (get-in db [:multiaccount :public-key]) public-key)
(let [contact (-> (get-in db [:contacts/contacts public-key] (let [contact (cond-> (get-in db [:contacts/contacts public-key]
(build-contact cofx public-key)) (build-contact cofx public-key))
(update :system-tags nickname
(fnil #(conj % :contact/added) #{})))] (assoc :nickname nickname)
:else
(update :system-tags
(fnil #(conj % :contact/added) #{})))]
(fx/merge cofx (fx/merge cofx
{:db (dissoc db :contacts/new-identity)} {:db (dissoc db :contacts/new-identity)}
(upsert-contact contact) (upsert-contact contact)
@ -179,3 +185,13 @@
:ens-verified true})} :ens-verified true})}
(upsert-contact {:public-key public-key}))) (upsert-contact {:public-key public-key})))
(fx/defn update-nickname
{:events [:contacts/update-nickname]}
[{:keys [db] :as cofx} public-key nickname]
(fx/merge cofx
{:db (if (string/blank? nickname)
(update-in db [:contacts/contacts public-key] dissoc :nickname)
(assoc-in db [:contacts/contacts public-key :nickname] nickname))}
(upsert-contact {:public-key public-key})
(navigation/navigate-back)))

View File

@ -3,7 +3,8 @@
[clojure.string :as clojure.string] [clojure.string :as clojure.string]
[status-im.ethereum.core :as ethereum] [status-im.ethereum.core :as ethereum]
[status-im.utils.gfycat.core :as gfycat] [status-im.utils.gfycat.core :as gfycat]
[status-im.utils.identicon :as identicon])) [status-im.utils.identicon :as identicon]
[status-im.multiaccounts.core :as multiaccounts]))
(defn public-key->new-contact [public-key] (defn public-key->new-contact [public-key]
(let [alias (gfycat/generate-gfy public-key)] (let [alias (gfycat/generate-gfy public-key)]
@ -135,7 +136,8 @@
(assoc :pending? (pending? contact) (assoc :pending? (pending? contact)
:blocked? (blocked? contact) :blocked? (blocked? contact)
:active? (active? contact) :active? (active? contact)
:added? (contains? system-tags :contact/added)))) :added? (contains? system-tags :contact/added))
(multiaccounts/contact-with-names)))
(defn enrich-contacts (defn enrich-contacts
[contacts] [contacts]

View File

@ -32,18 +32,18 @@
admins admins
contacts contacts
current-multiaccount) current-multiaccount)
[{:name "generated" [{:name "generated"
:identicon "generated" :identicon "generated"
:alias "generated" :alias "generated"
:admin? true :admin? true
:public-key "0x04fcf40c526b09ff9fb22f4a5dbd08490ef9b64af700870f8a0ba2133f4251d5607ed83cd9047b8c2796576bc83fa0de23a13a4dced07654b8ff137fe744047917" :public-key "0x04fcf40c526b09ff9fb22f4a5dbd08490ef9b64af700870f8a0ba2133f4251d5607ed83cd9047b8c2796576bc83fa0de23a13a4dced07654b8ff137fe744047917"
:system-tags #{}} :system-tags #{}}
{:alias "User A" {:alias "User A"
:photo-path "photo2" :photo-path "photo2"
:public-key "0x048a2f8b80c60f89a91b4c1316e56f75b087f446e7b8701ceca06a40142d8efe1f5aa36bd0fee9e248060a8d5207b43ae98bef4617c18c71e66f920f324869c09f"} :public-key "0x048a2f8b80c60f89a91b4c1316e56f75b087f446e7b8701ceca06a40142d8efe1f5aa36bd0fee9e248060a8d5207b43ae98bef4617c18c71e66f920f324869c09f"}
{:last-updated 0 {:last-updated 0
:name "User B" :name "User B"
:photo-path "photo1" :photo-path "photo1"
:last-online 0 :last-online 0
:public-key "0x04985040682b77a32bb4bb58268a0719bd24ca4d07c255153fe1eb2ccd5883669627bd1a092d7cc76e8e4b9104327667b19dcda3ac469f572efabe588c38c1985f" :public-key "0x04985040682b77a32bb4bb58268a0719bd24ca4d07c255153fe1eb2ccd5883669627bd1a092d7cc76e8e4b9104327667b19dcda3ac469f572efabe588c38c1985f"
:system-tags #{}}])))))) :system-tags #{}}]))))))

View File

@ -27,7 +27,8 @@
:ensVerificationRetries :ens-verification-retries :ensVerificationRetries :ens-verification-retries
:lastENSClockValue :last-ens-clock-value :lastENSClockValue :last-ens-clock-value
:systemTags :system-tags :systemTags :system-tags
:lastUpdated :last-updated}))) :lastUpdated :last-updated
:localNickname :nickname})))
(defn ->rpc [contact] (defn ->rpc [contact]
(-> contact (-> contact
@ -41,7 +42,8 @@
:photo-path :photoPath :photo-path :photoPath
:tribute-to-talk :tributeToTalk :tribute-to-talk :tributeToTalk
:system-tags :systemTags :system-tags :systemTags
:last-updated :lastUpdated}))) :last-updated :lastUpdated
:nickname :localNickname})))
(fx/defn fetch-contacts-rpc (fx/defn fetch-contacts-rpc
[cofx on-success] [cofx on-success]

View File

@ -816,7 +816,7 @@
:contact.ui/add-to-contact-pressed :contact.ui/add-to-contact-pressed
[(re-frame/inject-cofx :random-id-generator)] [(re-frame/inject-cofx :random-id-generator)]
(fn [cofx [_ public-key]] (fn [cofx [_ public-key]]
(contact/add-contact cofx public-key))) (contact/add-contact cofx public-key nil)))
(handlers/register-handler-fx (handlers/register-handler-fx
:contact.ui/block-contact-confirmed :contact.ui/block-contact-confirmed
@ -842,11 +842,11 @@
(handlers/register-handler-fx (handlers/register-handler-fx
:contact.ui/contact-code-submitted :contact.ui/contact-code-submitted
[(re-frame/inject-cofx :random-id-generator)] [(re-frame/inject-cofx :random-id-generator)]
(fn [{{:contacts/keys [new-identity]} :db :as cofx} [_ new-contact?]] (fn [{{:contacts/keys [new-identity]} :db :as cofx} [_ new-contact? nickname]]
(let [{:keys [public-key ens-name]} new-identity] (let [{:keys [public-key ens-name]} new-identity]
(fx/merge cofx (fx/merge cofx
#(if new-contact? #(if new-contact?
(contact/add-contact % public-key) (contact/add-contact % public-key nickname)
(chat/start-chat % public-key)) (chat/start-chat % public-key))
#(when new-contact? #(when new-contact?
(navigation/navigate-back %)) (navigation/navigate-back %))

View File

@ -8,7 +8,40 @@
[status-im.utils.gfycat.core :as gfycat] [status-im.utils.gfycat.core :as gfycat]
[status-im.utils.identicon :as identicon] [status-im.utils.identicon :as identicon]
[status-im.utils.theme :as utils.theme] [status-im.utils.theme :as utils.theme]
[status-im.theme.core :as theme])) [status-im.theme.core :as theme]
[status-im.utils.utils :as utils]
[clojure.string :as string]))
(defn contact-names
"Returns map of all existing names for contact"
[{:keys [name preferred-name alias public-key ens-verified nickname]}]
(let [ens-name (or preferred-name
name)]
(cond-> {:nickname nickname
:three-words-name (or alias (gfycat/generate-gfy public-key))}
;; Preferred name is our own otherwise we make sure it's verified
(or preferred-name (and ens-verified name))
(assoc :ens-name (str "@" (or (stateofus/username ens-name) ens-name))))))
(defn contact-two-names
"Returns vector of two names in next order nickname, ens name, three word name, public key"
[{:keys [names public-key] :as contact} public-key?]
(let [{:keys [nickname ens-name three-words-name]} (or names (contact-names contact))
short-public-key (when public-key? (utils/get-shortened-address public-key))]
(cond (not (string/blank? nickname))
[nickname (or ens-name three-words-name short-public-key)]
(not (string/blank? ens-name))
[ens-name (or three-words-name short-public-key)]
(not (string/blank? three-words-name))
[three-words-name short-public-key]
:else
(when public-key?
[short-public-key short-public-key]))))
(defn contact-with-names
"Returns contact with :names map "
[contact]
(assoc contact :names (contact-names contact)))
(defn displayed-name (defn displayed-name
"Use preferred name, name or alias in that order" "Use preferred name, name or alias in that order"

View File

@ -2,7 +2,8 @@
(:require [reagent.core :as reagent] (:require [reagent.core :as reagent]
[status-im.ui.components.react :as react] [status-im.ui.components.react :as react]
[status-im.react-native.resources :as resources] [status-im.react-native.resources :as resources]
[status-im.ui.components.colors :as colors])) [status-im.ui.components.colors :as colors]
[re-frame.core :as re-frame]))
(def cnt (reagent/atom 0)) (def cnt (reagent/atom 0))
(defonce cnt-prev (reagent/atom 0)) (defonce cnt-prev (reagent/atom 0))
@ -14,6 +15,7 @@
(defn reload [] (defn reload []
(reset! warning? false) (reset! warning? false)
(reset! label "reloading UI") (reset! label "reloading UI")
(re-frame/clear-subscription-cache!)
(swap! cnt inc)) (swap! cnt inc))
(defn build-competed [] (defn build-competed []

View File

@ -1543,16 +1543,17 @@
:<- [:contacts/contacts] :<- [:contacts/contacts]
:<- [:contacts/current-contact-identity] :<- [:contacts/current-contact-identity]
(fn [[contacts identity]] (fn [[contacts identity]]
(or (contacts identity) (or (get contacts identity)
(-> identity (-> identity
contact.db/public-key->new-contact contact.db/public-key->new-contact
contact.db/enrich-contact)))) contact.db/enrich-contact))))
(re-frame/reg-sub (re-frame/reg-sub
:contacts/contact-by-identity :contacts/contact-by-identity
:<- [::contacts] :<- [:contacts/contacts]
(fn [contacts [_ identity]] (fn [contacts [_ identity]]
(get contacts identity))) (or (get contacts identity)
(multiaccounts/contact-with-names {:public-key identity}))))
(re-frame/reg-sub (re-frame/reg-sub
:contacts/contact-added? :contacts/contact-added?
@ -1562,27 +1563,23 @@
(contact.db/added? contact))) (contact.db/added? contact)))
(re-frame/reg-sub (re-frame/reg-sub
:contacts/raw-contact-name-by-identity :contacts/contact-two-names-by-identity
(fn [[_ identity] _] (fn [[_ identity] _]
[(re-frame/subscribe [:contacts/contact-by-identity identity])]) [(re-frame/subscribe [:contacts/contact-by-identity identity])
(fn [[db-contact] _] (re-frame/subscribe [:multiaccount])])
(if (and (:ens-verified db-contact) (seq (:name db-contact))) (fn [[contact current-multiaccount] [_ identity]]
(str "@" (:name db-contact)) (let [me? (= (:public-key current-multiaccount) identity)]
(:alias db-contact)))) (if me?
[(or (:preferred-name current-multiaccount)
(gfycat/generate-gfy identity))]
(multiaccounts/contact-two-names contact false)))))
(re-frame/reg-sub (re-frame/reg-sub
:contacts/contact-name-by-identity :contacts/contact-name-by-identity
(fn [[_ identity] _] (fn [[_ identity] _]
[(re-frame/subscribe [:contacts/raw-contact-name-by-identity identity]) [(re-frame/subscribe [:contacts/contact-two-names-by-identity identity])])
(re-frame/subscribe [:multiaccount])]) (fn [[names] _]
(fn [[contact-name current-multiaccount] [_ identity]] (first names)))
(let [me? (= (:public-key current-multiaccount) identity)]
(if me?
(or (:preferred-name current-multiaccount)
(gfycat/generate-gfy identity))
(or (stateofus/username contact-name)
contact-name
(gfycat/generate-gfy identity))))))
(re-frame/reg-sub (re-frame/reg-sub
:messages/quote-info :messages/quote-info
@ -1612,8 +1609,7 @@
:<- [::query-current-chat-contacts remove] :<- [::query-current-chat-contacts remove]
(fn [contacts] (fn [contacts]
(->> contacts (->> contacts
(filter contact.db/added?) (filter contact.db/added?))))
(sort-by (comp clojure.string/lower-case multiaccounts/displayed-name)))))
(re-frame/reg-sub (re-frame/reg-sub
:contacts/current-chat-contacts :contacts/current-chat-contacts
@ -1764,16 +1760,19 @@
(string/lower-case (or alias (string/lower-case (or alias
(get-in contacts [chat-id :alias]) (get-in contacts [chat-id :alias])
(gfycat/generate-gfy chat-id))) (gfycat/generate-gfy chat-id)))
"")] "")
nickname (get-in contacts [chat-id :nickname])]
(or (or
(string/includes? (string/lower-case (str name)) search-filter) (string/includes? (string/lower-case (str name)) search-filter)
(string/includes? (string/lower-case alias) search-filter) (string/includes? (string/lower-case alias) search-filter)
(when nickname
(string/includes? (string/lower-case nickname) search-filter))
(and (and
(get-in contacts [chat-id :ens-verified]) (get-in contacts [chat-id :ens-verified])
(string/includes? (string/lower-case (string/includes? (string/lower-case
(str (get-in contacts [chat-id :name]))) (str (get-in contacts [chat-id :name])))
search-filter))))) search-filter)))))
(re-frame/reg-sub (re-frame/reg-sub
:search/filtered-chats :search/filtered-chats
:<- [:chats/active-chats] :<- [:chats/active-chats]

View File

@ -1,8 +0,0 @@
(ns status-im.ui.components.contact.contact
(:require [status-im.ethereum.stateofus :as stateofus]
[status-im.utils.gfycat.core :as gfycat]))
(defn format-name [{:keys [ens-verified name public-key]}]
(if ens-verified
(str "@" (or (stateofus/username name) name))
(gfycat/generate-gfy public-key)))

View File

@ -93,7 +93,7 @@
(if-not validation-result (if-not validation-result
(if new-contact? (if new-contact?
(fx/merge cofx (fx/merge cofx
(contact/add-contact chat-key) (contact/add-contact chat-key nil)
(navigation/navigate-to-cofx :contacts-list {})) (navigation/navigate-to-cofx :contacts-list {}))
(chat/start-chat cofx chat-key)) (chat/start-chat cofx chat-key))
{:utils/show-popup {:title (i18n/label :t/unable-to-read-this-code) {:utils/show-popup {:title (i18n/label :t/unable-to-read-this-code)

View File

@ -13,17 +13,20 @@
[status-im.ui.components.topbar :as topbar] [status-im.ui.components.topbar :as topbar]
[status-im.ui.screens.add-new.new-chat.styles :as styles] [status-im.ui.screens.add-new.new-chat.styles :as styles]
[status-im.utils.debounce :as debounce] [status-im.utils.debounce :as debounce]
[status-im.utils.utils :as utils]) [status-im.utils.utils :as utils]
[reagent.core :as reagent])
(:require-macros [status-im.utils.views :as views])) (:require-macros [status-im.utils.views :as views]))
(defn- render-row [row _ _] (defn- render-row [row _ _]
[quo/list-item (let [[first-name second-name] (multiaccounts/contact-two-names row false)]
{:title (multiaccounts/displayed-name row) [quo/list-item
:icon [chat-icon/contact-icon-contacts-tab {:title first-name
(multiaccounts/displayed-photo row)] :subtitle second-name
:chevron true :icon [chat-icon/contact-icon-contacts-tab
:on-press #(re-frame/dispatch [:chat.ui/start-chat (multiaccounts/displayed-photo row)]
(:public-key row)])}]) :chevron true
:on-press #(re-frame/dispatch [:chat.ui/start-chat
(:public-key row)])}]))
(defn- icon-wrapper [color icon] (defn- icon-wrapper [color icon]
[react/view [react/view
@ -36,7 +39,7 @@
icon]) icon])
(defn- input-icon (defn- input-icon
[state new-contact?] [state new-contact? entered-nickname]
(let [icon (if new-contact? :main-icons/add :main-icons/arrow-right)] (let [icon (if new-contact? :main-icons/add :main-icons/arrow-right)]
(case state (case state
:searching :searching
@ -45,7 +48,7 @@
:valid :valid
[react/touchable-highlight [react/touchable-highlight
{:on-press #(debounce/dispatch-and-chill [:contact.ui/contact-code-submitted new-contact?] 3000)} {:on-press #(debounce/dispatch-and-chill [:contact.ui/contact-code-submitted new-contact? entered-nickname] 3000)}
[icon-wrapper colors/blue [icon-wrapper colors/blue
[vector-icons/icon icon {:color colors/white-persist}]]] [vector-icons/icon icon {:color colors/white-persist}]]]
@ -83,7 +86,7 @@
(debounce/debounce-and-dispatch [:new-chat/set-new-identity %] 600)) (debounce/debounce-and-dispatch [:new-chat/set-new-identity %] 600))
:on-submit-editing :on-submit-editing
#(when (= state :valid) #(when (= state :valid)
(debounce/dispatch-and-chill [:contact.ui/contact-code-submitted false] 3000)) (debounce/dispatch-and-chill [:contact.ui/contact-code-submitted false nil] 3000))
:placeholder (i18n/label :t/enter-contact-code) :placeholder (i18n/label :t/enter-contact-code)
:show-cancel false :show-cancel false
:accessibility-label :enter-contact-code-input :accessibility-label :enter-contact-code-input
@ -91,7 +94,7 @@
:return-key-type :go}]] :return-key-type :go}]]
[react/view {:justify-content :center [react/view {:justify-content :center
:align-items :center} :align-items :center}
[input-icon state false]]] [input-icon state false nil]]]
[react/view {:min-height 30 :justify-content :flex-end} [react/view {:min-height 30 :justify-content :flex-end}
[quo/text {:style styles/message [quo/text {:style styles/message
:size :small :size :small
@ -118,8 +121,20 @@
:enableEmptySections true :enableEmptySections true
:keyboardShouldPersistTaps :always}]])) :keyboardShouldPersistTaps :always}]]))
(defn- nickname-input [entered-nickname]
[quo/text-input
{:on-change-text #(reset! entered-nickname %)
:auto-capitalize :none
:max-length 32
:auto-focus false
:accessibility-label :nickname-input
:placeholder (i18n/label :t/add-nickname)
:return-key-type :done
:auto-correct false}])
(views/defview new-contact [] (views/defview new-contact []
(views/letsubs [{:keys [state ens-name public-key error]} [:contacts/new-identity]] (views/letsubs [{:keys [state ens-name public-key error]} [:contacts/new-identity]
entered-nickname (reagent/atom "")]
[react/view {:style {:flex 1}} [react/view {:style {:flex 1}}
[topbar/topbar [topbar/topbar
{:title (i18n/label :t/new-contact) {:title (i18n/label :t/new-contact)
@ -142,7 +157,7 @@
(debounce/debounce-and-dispatch [:new-chat/set-new-identity %] 600)) (debounce/debounce-and-dispatch [:new-chat/set-new-identity %] 600))
:on-submit-editing :on-submit-editing
#(when (= state :valid) #(when (= state :valid)
(debounce/dispatch-and-chill [:contact.ui/contact-code-submitted true] 3000)) (debounce/dispatch-and-chill [:contact.ui/contact-code-submitted true @entered-nickname] 3000))
:placeholder (i18n/label :t/enter-contact-code) :placeholder (i18n/label :t/enter-contact-code)
:show-cancel false :show-cancel false
:accessibility-label :enter-contact-code-input :accessibility-label :enter-contact-code-input
@ -150,12 +165,23 @@
:return-key-type :go}]] :return-key-type :go}]]
[react/view {:justify-content :center [react/view {:justify-content :center
:align-items :center} :align-items :center}
[input-icon state true]]] [input-icon state true @entered-nickname]]]
[react/view {:min-height 30 :justify-content :flex-end} [react/view {:min-height 30 :justify-content :flex-end :margin-bottom 16}
[react/text {:style styles/message} [quo/text {:style styles/message
:size :small
:align :center
:color :secondary}
(cond (= state :error) (cond (= state :error)
(get-validation-label error) (get-validation-label error)
(= state :valid) (= state :valid)
(str (when ens-name (str ens-name " • ")) (str (when ens-name (str ens-name " • "))
(utils/get-shortened-address public-key)) (utils/get-shortened-address public-key))
:else "")]]])) :else "")]]
[react/text {:style {:margin-horizontal 16 :color colors/gray}}
(i18n/label :t/nickname-description)]
[react/view {:padding 16}
[nickname-input entered-nickname]
[react/text {:style {:align-self :flex-end :margin-top 16
:color colors/gray}}
(str (count @entered-nickname) " / 32")]]]))

View File

@ -175,13 +175,13 @@
(= outgoing-status :not-sent)) (= outgoing-status :not-sent))
[message-not-sent-text chat-id message-id])) [message-not-sent-text chat-id message-id]))
(defview message-author-name [from alias modal] (defview message-author-name [from modal]
(letsubs [contact-name [:contacts/raw-contact-name-by-identity from]] (letsubs [contact-with-names [:contacts/contact-by-identity from]]
(chat.utils/format-author (or contact-name alias) modal))) (chat.utils/format-author contact-with-names modal)))
(defn message-content-wrapper (defn message-content-wrapper
"Author, userpic and delivery wrapper" "Author, userpic and delivery wrapper"
[{:keys [alias first-in-group? display-photo? identicon display-username? [{:keys [first-in-group? display-photo? identicon display-username?
from outgoing] from outgoing]
:as message} content {:keys [modal close-modal]}] :as message} content {:keys [modal close-modal]}]
[react/view {:style (style/message-wrapper message) [react/view {:style (style/message-wrapper message)
@ -200,7 +200,7 @@
[react/touchable-opacity {:style style/message-author-touchable [react/touchable-opacity {:style style/message-author-touchable
:on-press #(do (when modal (close-modal)) :on-press #(do (when modal (close-modal))
(re-frame/dispatch [:chat.ui/show-profile from]))} (re-frame/dispatch [:chat.ui/show-profile from]))}
[message-author-name from alias modal]]) [message-author-name from modal]])
;;MESSAGE CONTENT ;;MESSAGE CONTENT
[react/view [react/view
content]]] content]]]

View File

@ -17,7 +17,11 @@
(i18n/label-pluralize cnt :t/members-active))))]]) (i18n/label-pluralize cnt :t/members-active))))]])
(defn one-to-one-name [from] (defn one-to-one-name [from]
@(re-frame.core/subscribe [:contacts/contact-name-by-identity from])) (let [[first-name _] @(re-frame.core/subscribe [:contacts/contact-two-names-by-identity from])]
[react/text {:style st/chat-name-text
:number-of-lines 1
:accessibility-label :chat-name-text}
first-name]))
(defn contact-indicator [contact-id] (defn contact-indicator [contact-id]
(let [added? @(re-frame/subscribe [:contacts/contact-added? contact-id])] (let [added? @(re-frame/subscribe [:contacts/contact-added? contact-id])]
@ -39,12 +43,12 @@
[react/view {:margin-right 10} [react/view {:margin-right 10}
[chat-icon.screen/chat-icon-view-toolbar chat-id group-chat chat-name color]] [chat-icon.screen/chat-icon-view-toolbar chat-id group-chat chat-name color]]
[react/view {:style st/chat-name-view} [react/view {:style st/chat-name-view}
[react/text {:style st/chat-name-text (if group-chat
:number-of-lines 1 [react/text {:style st/chat-name-text
:accessibility-label :chat-name-text} :number-of-lines 1
(if group-chat :accessibility-label :chat-name-text}
chat-name chat-name]
[one-to-one-name chat-id])] [one-to-one-name chat-id])
(when-not group-chat (when-not group-chat
[contact-indicator chat-id]) [contact-indicator chat-id])
(when group-chat (when group-chat

View File

@ -2,26 +2,31 @@
(:require [status-im.ethereum.stateofus :as stateofus] (:require [status-im.ethereum.stateofus :as stateofus]
[status-im.i18n :as i18n] [status-im.i18n :as i18n]
[status-im.ui.components.react :as react] [status-im.ui.components.react :as react]
[status-im.ui.components.colors :as colors])) [status-im.ui.components.colors :as colors]
[status-im.multiaccounts.core :as multiaccounts]))
(def ^:private reply-symbol "↪ ") (def ^:private reply-symbol "↪ ")
(defn format-author (defn format-author
([contact-name] (format-author contact-name false)) ([contact] (format-author contact false))
([contact-name modal] ([{:keys [names] :as contact} modal]
(if (= (aget contact-name 0) "@") (let [{:keys [nickname ens-name]} names
(let [trimmed-name (subs contact-name 0 81)] [first-name second-name] (multiaccounts/contact-two-names contact false)]
[react/text {:number-of-lines 2 (if (or nickname ens-name)
:style {:color (if modal colors/white-persist colors/blue) [react/nested-text {:number-of-lines 2
:font-size 13 :style {:color (if modal colors/white-persist colors/blue)
:line-height 18 :font-size 13
:font-weight "500"}} :line-height 18
(or (stateofus/username trimmed-name) trimmed-name)]) :font-weight "500"}}
[react/text {:style {:color (if modal colors/white-persist colors/gray) (subs first-name 0 81)
:font-size 12 (when nickname
:line-height 18 [{:style {:color colors/gray :font-weight "400"}}
:font-weight "400"}} (str " " (subs second-name 0 81))])]
contact-name]))) [react/text {:style {:color (if modal colors/white-persist colors/gray)
:font-size 12
:line-height 18
:font-weight "400"}}
first-name]))))
(defn format-reply-author [from username current-public-key style] (defn format-reply-author [from username current-public-key style]
(let [contact-name (str reply-symbol username)] (let [contact-name (str reply-symbol username)]

View File

@ -73,7 +73,8 @@
:default-chat-icon-text style/intro-header-icon-text :default-chat-icon-text style/intro-header-icon-text
:size 120}]] :size 120}]]
;; Chat title section ;; Chat title section
[react/text {:style (style/intro-header-chat-name)} (if group-chat chat-name contact-name)] [react/text {:style (style/intro-header-chat-name)}
(if group-chat chat-name contact-name)]
;; Description section ;; Description section
(if group-chat (if group-chat
[chat.group/group-chat-description-container {:chat-id chat-id [chat.group/group-chat-description-container {:chat-id chat-id
@ -89,9 +90,9 @@
contact-name)])]) contact-name)])])
(defn chat-intro-one-to-one [{:keys [chat-id] :as opts}] (defn chat-intro-one-to-one [{:keys [chat-id] :as opts}]
(let [contact-name @(re-frame/subscribe (let [contact-names @(re-frame/subscribe
[:contacts/contact-name-by-identity chat-id])] [:contacts/contact-two-names-by-identity chat-id])]
(chat-intro (assoc opts :contact-name contact-name)))) (chat-intro (assoc opts :contact-name (first contact-names)))))
(defn chat-intro-header-container (defn chat-intro-header-container
[{:keys [group-chat [{:keys [group-chat

View File

@ -12,12 +12,14 @@
(:require-macros [status-im.utils.views :refer [defview letsubs]])) (:require-macros [status-im.utils.views :refer [defview letsubs]]))
(defn contacts-list-item [{:keys [public-key] :as contact}] (defn contacts-list-item [{:keys [public-key] :as contact}]
[quo/list-item (let [[first-name second-name] (multiaccounts/contact-two-names contact true)]
{:title (multiaccounts/displayed-name contact) [quo/list-item
:icon [chat-icon.screen/contact-icon-contacts-tab {:title first-name
(multiaccounts/displayed-photo contact)] :subtitle second-name
:chevron true :icon [chat-icon.screen/contact-icon-contacts-tab
:on-press #(re-frame/dispatch [:chat.ui/show-profile public-key])}]) (multiaccounts/displayed-photo contact)]
:chevron true
:on-press #(re-frame/dispatch [:chat.ui/show-profile public-key])}]))
(defn add-new-contact [] (defn add-new-contact []
[quo/list-item [quo/list-item

View File

@ -628,7 +628,10 @@
(views/defview my-name [] (views/defview my-name []
(views/letsubs [contact-name [:multiaccount/preferred-name]] (views/letsubs [contact-name [:multiaccount/preferred-name]]
(when-not (string/blank? contact-name) (when-not (string/blank? contact-name)
(chat.utils/format-author (str "@" contact-name))))) (chat.utils/format-author {:names {:ens-name
(str "@"
(or (stateofus/username contact-name)
contact-name))}}))))
(views/defview registered [names {:keys [preferred-name] :as account} _ registrations] (views/defview registered [names {:keys [preferred-name] :as account} _ registrations]
[react/view {:style {:flex 1}} [react/view {:style {:flex 1}}

View File

@ -6,7 +6,6 @@
[status-im.constants :as constants] [status-im.constants :as constants]
[status-im.i18n :as i18n] [status-im.i18n :as i18n]
[status-im.ui.components.chat-icon.screen :as chat-icon] [status-im.ui.components.chat-icon.screen :as chat-icon]
[status-im.ui.components.contact.contact :as contact]
[status-im.multiaccounts.core :as multiaccounts] [status-im.multiaccounts.core :as multiaccounts]
[status-im.ui.components.keyboard-avoid-presentation [status-im.ui.components.keyboard-avoid-presentation
:as :as
@ -23,10 +22,12 @@
(:require-macros [status-im.utils.views :as views])) (:require-macros [status-im.utils.views :as views]))
(defn- render-contact [row] (defn- render-contact [row]
[quo/list-item (let [[first-name second-name] (multiaccounts/contact-two-names row false)]
{:title (contact/format-name row) [quo/list-item
:icon [chat-icon/contact-icon-contacts-tab {:title first-name
(multiaccounts/displayed-photo row)]}]) :subtitle second-name
:icon [chat-icon/contact-icon-contacts-tab
(multiaccounts/displayed-photo row)]}]))
(defn- on-toggle [allow-new-users? checked? public-key] (defn- on-toggle [allow-new-users? checked? public-key]
(cond (cond
@ -52,9 +53,11 @@
(defn- toggle-item [] (defn- toggle-item []
(fn [allow-new-users? subs-name {:keys [public-key] :as contact} on-toggle] (fn [allow-new-users? subs-name {:keys [public-key] :as contact} on-toggle]
(let [contact-selected? @(re-frame/subscribe [subs-name public-key])] (let [contact-selected? @(re-frame/subscribe [subs-name public-key])
[first-name second-name] (multiaccounts/contact-two-names contact true)]
[quo/list-item [quo/list-item
{:title (contact/format-name contact) {:title first-name
:subtitle second-name
:icon [chat-icon/contact-icon-contacts-tab :icon [chat-icon/contact-icon-contacts-tab
(multiaccounts/displayed-photo contact)] (multiaccounts/displayed-photo contact)]
:on-press #(on-toggle allow-new-users? contact-selected? public-key) :on-press #(on-toggle allow-new-users? contact-selected? public-key)
@ -83,10 +86,12 @@
(defn filter-contacts [filter-text contacts] (defn filter-contacts [filter-text contacts]
(let [lower-filter-text (string/lower-case (str filter-text)) (let [lower-filter-text (string/lower-case (str filter-text))
filter-fn (fn [{:keys [name alias]}] filter-fn (fn [{:keys [name alias nickname]}]
(or (or
(string/includes? (string/lower-case (str name)) lower-filter-text) (string/includes? (string/lower-case (str name)) lower-filter-text)
(string/includes? (string/lower-case (str alias)) lower-filter-text)))] (string/includes? (string/lower-case (str alias)) lower-filter-text)
(when nickname
(string/includes? (string/lower-case (str nickname)) lower-filter-text))))]
(if filter-text (if filter-text
(filter filter-fn contacts) (filter filter-fn contacts)
contacts))) contacts)))

View File

@ -163,7 +163,7 @@
;; This looks a bit odd, but I would like only to subscribe ;; This looks a bit odd, but I would like only to subscribe
;; if it's a one-to-one. If wrapped in a component styling ;; if it's a one-to-one. If wrapped in a component styling
;; won't be applied correctly. ;; won't be applied correctly.
@(re-frame/subscribe [:contacts/contact-name-by-identity chat-id]))]] (first @(re-frame/subscribe [:contacts/contact-two-names-by-identity chat-id])))]]
[message-timestamp (if (pos? (:whisper-timestamp last-message)) [message-timestamp (if (pos? (:whisper-timestamp last-message))
(:whisper-timestamp last-message) (:whisper-timestamp last-message)
timestamp)]] timestamp)]]

View File

@ -9,8 +9,13 @@
[status-im.ui.components.react :as react] [status-im.ui.components.react :as react]
[status-im.ui.screens.profile.components.sheets :as sheets] [status-im.ui.screens.profile.components.sheets :as sheets]
[status-im.ui.screens.profile.contact.styles :as styles] [status-im.ui.screens.profile.contact.styles :as styles]
[status-im.utils.gfycat.core :as gfy] [status-im.utils.utils :as utils]
[status-im.utils.utils :as utils]) [status-im.ui.components.topbar :as topbar]
[status-im.ui.components.colors :as colors]
[status-im.ui.components.toolbar :as toolbar]
[status-im.ui.components.keyboard-avoid-presentation :as kb-presentation]
[reagent.core :as reagent]
[clojure.string :as string])
(:require-macros [status-im.utils.views :as views])) (:require-macros [status-im.utils.views :as views]))
(defn actions (defn actions
@ -47,26 +52,47 @@
; :content-height 150} ; :content-height 150}
; contact]) ; contact])
(defn render-detail [{:keys [public-key names name] :as detail}]
[quo/list-item
{:title (:three-words-name names)
:subtitle [quo/text {:monospace true
:color :secondary}
(utils/get-shortened-address public-key)]
:icon [chat-icon/contact-icon-contacts-tab
(multiaccounts/displayed-photo detail)]
:accessibility-label :profile-public-key
:on-press #(re-frame/dispatch [:show-popover (merge {:view :share-chat-key
:address public-key}
(when (and (:ens-name names) name)
{:ens-name name}))])
:accessory [icons/icon :main-icons/share styles/contact-profile-detail-share-icon]}])
(defn profile-details [{:keys [alias public-key ens-name] :as contact}] (defn profile-details [contact]
(when contact (when contact
[react/view [react/view
[quo/list-header [quo/list-header
[quo/text {:accessibility-label :profile-details [quo/text {:accessibility-label :profile-details
:color :inherit} :color :inherit}
(i18n/label :t/profile-details)]] (i18n/label :t/profile-details)]]
[quo/list-item [render-detail contact]]))
{:title (or alias ens-name)
:subtitle [quo/text {:monospace true (defn render-chat-settings [{:keys [names]}]
:color :secondary} [quo/list-item
(utils/get-shortened-address public-key)] {:title (i18n/label :t/nickname)
:icon [chat-icon/contact-icon-contacts-tab :size :small
(multiaccounts/displayed-photo contact)] :accessibility-label :profile-nickname-item
:accessibility-label :profile-public-key :accessory :text
:on-press #(re-frame/dispatch [:show-popover {:view :share-chat-key :accessory-text (or (:nickname names) (i18n/label :t/none))
:address public-key :on-press #(re-frame/dispatch [:navigate-to :nickname])
:ens-name ens-name}]) :chevron true}])
:accessory [icons/icon :main-icons/share styles/contact-profile-detail-share-icon]}]]))
(defn chat-settings [contact]
[react/view
[quo/list-header
[quo/text {:accessibility-label :chat-settings
:color :inherit}
(i18n/label :t/chat-settings)]]
[render-chat-settings contact]])
;; TODO: List item ;; TODO: List item
(defn block-contact-action [{:keys [blocked? public-key]}] (defn block-contact-action [{:keys [blocked? public-key]}]
@ -84,10 +110,56 @@
(i18n/label :t/unblock-contact) (i18n/label :t/unblock-contact)
(i18n/label :t/block-contact))]]) (i18n/label :t/block-contact))]])
(defn save-nickname [public-key nickname]
(re-frame/dispatch [:contacts/update-nickname public-key nickname]))
(defn valid-nickname? [nickname]
(not (string/blank? nickname)))
(defn- nickname-input [nickname entered-nickname public-key]
[quo/text-input
{:on-change-text #(reset! entered-nickname %)
:on-submit-editing #(when (valid-nickname? @entered-nickname)
(save-nickname public-key @entered-nickname))
:auto-capitalize :none
:auto-focus false
:max-length 32
:accessibility-label :nickname-input
:default-value nickname
:placeholder (i18n/label :t/nickname)
:return-key-type :done
:auto-correct false}])
(defn nickname-view [public-key {:keys [nickname ens-name three-words-name]}]
(let [entered-nickname (reagent/atom nickname)]
(fn []
[kb-presentation/keyboard-avoiding-view {:style {:flex 1}}
[topbar/topbar {:title (i18n/label :t/nickname)
:subtitle (or ens-name three-words-name)
:modal? true}]
[react/view {:flex 1 :padding 16}
[react/text {:style {:color colors/gray :margin-bottom 16}}
(i18n/label :t/nickname-description)]
[nickname-input nickname entered-nickname public-key]
[react/text {:style {:align-self :flex-end :margin-top 16
:color colors/gray}}
(str (count @entered-nickname) " / 32")]]
[toolbar/toolbar {:show-border? true
:center
[quo/button
{:type :secondary
:on-press #(save-nickname public-key @entered-nickname)}
(i18n/label :t/done)]}]])))
(views/defview nickname []
(views/letsubs [{:keys [public-key names]} [:contacts/current-contact]]
[nickname-view public-key names]))
(views/defview profile [] (views/defview profile []
(views/letsubs [{:keys [ens-verified name public-key] (views/letsubs [{:keys [public-key name ens-verified]
:as contact} [:contacts/current-contact]] :as contact} [:contacts/current-contact]]
(let [on-share #(re-frame/dispatch [:show-popover (merge (let [[first-name second-name] (multiaccounts/contact-two-names contact true)
on-share #(re-frame/dispatch [:show-popover (merge
{:view :share-chat-key {:view :share-chat-key
:address public-key} :address public-key}
(when (and ens-verified name) (when (and ens-verified name)
@ -104,13 +176,11 @@
:accessibility-label :back-button :accessibility-label :back-button
:on-press #(re-frame/dispatch [:navigate-back])}] :on-press #(re-frame/dispatch [:navigate-back])}]
:extended-header (profile-header/extended-header :extended-header (profile-header/extended-header
{:on-press on-share {:on-press on-share
:title (multiaccounts/displayed-name contact) :title first-name
:photo (multiaccounts/displayed-photo contact) :photo (multiaccounts/displayed-photo contact)
:monospace (not ens-verified) :monospace (not ens-verified)
:subtitle (if (and ens-verified public-key) :subtitle second-name})}
(gfy/generate-gfy public-key)
public-key)})}
[react/view {:padding-top 12} [react/view {:padding-top 12}
(for [{:keys [label subtext accessibility-label icon action disabled?]} (actions contact)] (for [{:keys [label subtext accessibility-label icon action disabled?]} (actions contact)]
@ -124,7 +194,6 @@
:disabled disabled? :disabled disabled?
:on-press action}]))] :on-press action}]))]
[react/view styles/contact-profile-details-container [react/view styles/contact-profile-details-container
[profile-details (cond-> contact [profile-details contact]
(and ens-verified name) [chat-settings contact]]
(assoc :ens-name name))]]
[block-contact-action contact]]])))) [block-contact-action contact]]]))))

View File

@ -5,7 +5,6 @@
[status-im.i18n :as i18n] [status-im.i18n :as i18n]
[status-im.multiaccounts.core :as multiaccounts] [status-im.multiaccounts.core :as multiaccounts]
[status-im.ui.components.chat-icon.screen :as chat-icon] [status-im.ui.components.chat-icon.screen :as chat-icon]
[status-im.ui.components.contact.contact :as contact]
[status-im.ui.components.list.views :as list] [status-im.ui.components.list.views :as list]
[status-im.ui.components.profile-header.view :as profile-header] [status-im.ui.components.profile-header.view :as profile-header]
[status-im.ui.components.react :as react] [status-im.ui.components.react :as react]
@ -16,56 +15,59 @@
(:require-macros [status-im.utils.views :refer [defview letsubs]])) (:require-macros [status-im.utils.views :refer [defview letsubs]]))
(defn member-sheet [chat-id member us-admin?] (defn member-sheet [chat-id member us-admin?]
[react/view (let [[first-name _] (multiaccounts/contact-two-names member false)]
[quo/list-item [react/view
{:theme :accent
:icon [chat-icon/contact-icon-contacts-tab
(multiaccounts/displayed-photo member)]
:title (contact/format-name member)
:subtitle (i18n/label :t/view-profile)
:accessibility-label :view-chat-details-button
:chevron true
:on-press #(chat.sheets/hide-sheet-and-dispatch
[:chat.ui/show-profile
(:public-key member)])}]
(when (and us-admin?
(not (:admin? member)))
[quo/list-item [quo/list-item
{:theme :accent {:theme :accent
:title (i18n/label :t/make-admin) :icon [chat-icon/contact-icon-contacts-tab
:accessibility-label :make-admin (multiaccounts/displayed-photo member)]
:icon :main-icons/make-admin :title first-name
:on-press #(chat.sheets/hide-sheet-and-dispatch [:group-chats.ui/make-admin-pressed chat-id (:public-key member)])}]) :subtitle (i18n/label :t/view-profile)
(when-not (:admin? member) :accessibility-label :view-chat-details-button
[quo/list-item :chevron true
{:theme :accent :on-press #(chat.sheets/hide-sheet-and-dispatch
:title (i18n/label :t/remove-from-chat) [:chat.ui/show-profile
:accessibility-label :remove-from-chat (:public-key member)])}]
:icon :main-icons/remove-contact (when (and us-admin?
:on-press #(chat.sheets/hide-sheet-and-dispatch [:group-chats.ui/remove-member-pressed chat-id (:public-key member)])}])]) (not (:admin? member)))
[quo/list-item
{:theme :accent
:title (i18n/label :t/make-admin)
:accessibility-label :make-admin
:icon :main-icons/make-admin
:on-press #(chat.sheets/hide-sheet-and-dispatch [:group-chats.ui/make-admin-pressed chat-id (:public-key member)])}])
(when-not (:admin? member)
[quo/list-item
{:theme :accent
:title (i18n/label :t/remove-from-chat)
:accessibility-label :remove-from-chat
:icon :main-icons/remove-contact
:on-press #(chat.sheets/hide-sheet-and-dispatch [:group-chats.ui/remove-member-pressed chat-id (:public-key member)])}])]))
(defn render-member [chat-id {:keys [public-key] :as member} admin? current-user-identity] (defn render-member [chat-id {:keys [public-key] :as member} admin? current-user-identity]
[quo/list-item (let [[first-name second-name] (multiaccounts/contact-two-names member false)]
(merge [quo/list-item
{:title (contact/format-name member) (merge
:accessibility-label :member-item {:title first-name
:icon [chat-icon/contact-icon-contacts-tab :subtitle second-name
(multiaccounts/displayed-photo member)] :accessibility-label :member-item
:on-press (when (not= public-key current-user-identity) :icon [chat-icon/contact-icon-contacts-tab
#(re-frame/dispatch [:chat.ui/show-profile public-key]))} (multiaccounts/displayed-photo member)]
(when (:admin? member) :on-press (when (not= public-key current-user-identity)
{:accessory :text #(re-frame/dispatch [:chat.ui/show-profile public-key]))}
:accessory-text (i18n/label :t/group-chat-admin)}) (when (:admin? member)
(when (and admin? {:accessory :text
(not (:admin? member)) :accessory-text (i18n/label :t/group-chat-admin)})
(not= public-key current-user-identity)) (when (and admin?
{:accessory [quo/button {:on-press #(re-frame/dispatch [:bottom-sheet/show-sheet (not (:admin? member))
{:content (fn [] (not= public-key current-user-identity))
[member-sheet chat-id member admin?])}]) {:accessory [quo/button {:on-press #(re-frame/dispatch [:bottom-sheet/show-sheet
:type :icon {:content (fn []
:theme :icon [member-sheet chat-id member admin?])}])
:accessibility-label :menu-option} :type :icon
:main-icons/more]}))]) :theme :icon
:accessibility-label :menu-option}
:main-icons/more]}))]))
(defview chat-group-members-view [chat-id admin? current-user-identity] (defview chat-group-members-view [chat-id admin? current-user-identity]
(letsubs [members [:contacts/current-chat-contacts]] (letsubs [members [:contacts/current-chat-contacts]]

View File

@ -23,7 +23,8 @@
[quo.previews.main :as quo.preview] [quo.previews.main :as quo.preview]
[status-im.utils.config :as config] [status-im.utils.config :as config]
[status-im.ui.screens.chat.image.preview.views :as image-preview] [status-im.ui.screens.chat.image.preview.views :as image-preview]
[status-im.ui.screens.notifications-settings.views :as notifications-settings])) [status-im.ui.screens.notifications-settings.views :as notifications-settings]
[status-im.ui.screens.profile.contact.views :as contact]))
(defonce main-stack (navigation/create-stack)) (defonce main-stack (navigation/create-stack))
(defonce bottom-tabs (navigation/create-bottom-tabs)) (defonce bottom-tabs (navigation/create-bottom-tabs))
@ -82,6 +83,10 @@
:transition :presentation-ios :transition :presentation-ios
:insets {:bottom true} :insets {:bottom true}
:component new-public-chat/new-public-chat} :component new-public-chat/new-public-chat}
{:name :nickname
:transition :presentation-ios
:insets {:bottom true}
:component contact/nickname}
{:name :edit-group-chat-name {:name :edit-group-chat-name
:transition :presentation-ios :transition :presentation-ios
:insets {:bottom true} :insets {:bottom true}

View File

@ -770,7 +770,6 @@
"new-contract": "New Contract", "new-contract": "New Contract",
"new-group": "New group", "new-group": "New group",
"new-group-chat": "New group chat", "new-group-chat": "New group chat",
"add-members": "Add members",
"new-network": "New network", "new-network": "New network",
"new-pin-description": "Enter new 6-digit passcode", "new-pin-description": "Enter new 6-digit passcode",
"new-public-group-chat": "Join public chat", "new-public-group-chat": "Join public chat",
@ -1228,5 +1227,8 @@
"audio": "Audio", "audio": "Audio",
"update-to-see-image": "Update to latest version to see a nice image here!", "update-to-see-image": "Update to latest version to see a nice image here!",
"update-to-listen-audio": "Update to latest version to listen to an audio message here!", "update-to-listen-audio": "Update to latest version to listen to an audio message here!",
"update-to-see-sticker": "Update to latest version to see a nice sticker here!" "update-to-see-sticker": "Update to latest version to see a nice sticker here!",
"nickname": "Nickname",
"add-nickname": "Add a nickname (optional)",
"nickname-description": "Nicknames help you identify others in Status.\nOnly you can see the nicknames youve added"
} }