merge develop

This commit is contained in:
Adrian Tiberius 2016-05-23 19:53:07 +03:00
commit 979ce25f2a
36 changed files with 582 additions and 560 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 644 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 504 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 931 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

View File

@ -13,10 +13,9 @@
[status-im.chat.screen :refer [chat]] [status-im.chat.screen :refer [chat]]
[status-im.chats-list.screen :refer [chats-list]] [status-im.chats-list.screen :refer [chats-list]]
[status-im.new-group.screen :refer [new-group]] [status-im.new-group.screen :refer [new-group]]
[status-im.participants.views.create :refer [new-participants]] [status-im.participants.views.add :refer [new-participants]]
[status-im.participants.views.remove :refer [remove-participants]] [status-im.participants.views.remove :refer [remove-participants]]
[status-im.group-settings.screen :refer [group-settings]] [status-im.group-settings.screen :refer [group-settings]]
[status-im.group-settings.views.chat-name-edit :refer [chat-name-edit]]
[status-im.profile.screen :refer [profile my-profile]] [status-im.profile.screen :refer [profile my-profile]]
[status-im.utils.utils :refer [toast]] [status-im.utils.utils :refer [toast]]
[status-im.utils.encryption])) [status-im.utils.encryption]))
@ -44,7 +43,6 @@
:chat-list [main-tabs] :chat-list [main-tabs]
:new-group [new-group] :new-group [new-group]
:group-settings [group-settings] :group-settings [group-settings]
:chat-name-edit [chat-name-edit]
:contact-list [main-tabs] :contact-list [main-tabs]
:chat [chat] :chat [chat]
:profile [profile] :profile [profile]

View File

@ -6,12 +6,13 @@
[status-im.protocol.api :as api] [status-im.protocol.api :as api]
[status-im.models.messages :as messages] [status-im.models.messages :as messages]
[status-im.constants :refer [text-content-type [status-im.constants :refer [text-content-type
content-type-command]] content-type-command]]
[status-im.utils.random :as random] [status-im.utils.random :as random]
[status-im.chat.sign-up :as sign-up-service] [status-im.chat.sign-up :as sign-up-service]
[status-im.models.chats :as chats] [status-im.models.chats :as chats]
[status-im.navigation.handlers :as nav] [status-im.navigation.handlers :as nav]
[status-im.utils.handlers :as u])) [status-im.utils.handlers :as u]
[status-im.persistence.realm :as r]))
(register-handler :set-show-actions (register-handler :set-show-actions
(fn [db [_ show-actions]] (fn [db [_ show-actions]]
@ -187,7 +188,8 @@
((enrich add-commands)) ((enrich add-commands))
((enrich clear-input)) ((enrich clear-input))
((enrich clear-staged-commands)) ((enrich clear-staged-commands))
((after send-message!)) ;; todo uncomment once
;((after send-message!))
((after save-message-to-realm!)) ((after save-message-to-realm!))
((after save-commands-to-realm!)) ((after save-commands-to-realm!))
((after handle-commands)))) ((after handle-commands))))
@ -283,10 +285,14 @@
(defmethod nav/preload-data! :chat (defmethod nav/preload-data! :chat
[{:keys [current-chat-id] :as db} [_ _ id]] [{:keys [current-chat-id] :as db} [_ _ id]]
(-> db (let [chat-id (or id current-chat-id)
(assoc :current-chat-id (or id current-chat-id)) messages (get-in db [:chats chat-id :messages])
load-messages! db' (assoc db :current-chat-id chat-id)]
init-chat)) (if (seq messages)
db'
(-> db'
load-messages!
init-chat))))
(defn prepare-chat (defn prepare-chat
[{:keys [contacts] :as db} [_ contcat-id]] [{:keys [contacts] :as db} [_ contcat-id]]
@ -321,3 +327,45 @@
(register-handler :switch-command-suggestions (register-handler :switch-command-suggestions
(fn [db [_]] (fn [db [_]]
(suggestions/switch-command-suggestions db))) (suggestions/switch-command-suggestions db)))
(defn remove-chat
[{:keys [current-chat-id] :as db} _]
(update db :chats dissoc current-chat-id))
(defn notify-about-leaving!
[{:keys [current-chat-id]} _]
(api/leave-group-chat current-chat-id))
; todo do we really need this message?
(defn leaving-message!
[{:keys [current-chat-id]} _]
(messages/save-message
current-chat-id
{:from "system"
:msg-id (random/id)
:content "You left this chat"
:content-type text-content-type}))
(defn delete-messages!
[{:keys [current-chat-id]} _]
(r/write
(fn []
(r/delete (r/get-by-field :msgs :chat-id current-chat-id)))))
(defn delete-chat!
[{:keys [current-chat-id]} _]
(r/write
(fn []
(-> (r/get-by-field :chats :chat-id current-chat-id)
(r/single)
(r/delete)))))
(register-handler :leave-group-chat
;; todo oreder of operations tbd
(after (fn [_ _] (dispatch [:navigation-replace :chat-list])))
(-> remove-chat
;; todo uncomment
;((after notify-about-leaving!))
;((after leaving-message!))
((after delete-messages!))
((after delete-chat!))))

View File

@ -67,8 +67,8 @@
(defn on-action-selected [position] (defn on-action-selected [position]
(case position (case position
0 (dispatch [:show-add-participants]) 0 (dispatch [:navigate-to :add-participants])
1 (dispatch [:show-remove-participants]) 1 (dispatch [:navigate-to :remove-participants])
2 (dispatch [:leave-group-chat]))) 2 (dispatch [:leave-group-chat])))
(defn overlay [{:keys [on-click-outside]} items] (defn overlay [{:keys [on-click-outside]} items]
@ -119,13 +119,13 @@
:icon :menu_group :icon :menu_group
:icon-style {:width 25 :icon-style {:width 25
:height 19} :height 19}
:handler #(dispatch [:show-add-participants])} :handler #(dispatch [:navigate-to :add-participants])}
{:title "Remove Contact from chat" {:title "Remove Contact from chat"
:subtitle "Alex, John" :subtitle "Alex, John"
:icon :search_gray_copy :icon :search_gray_copy
:icon-style {:width 17 :icon-style {:width 17
:height 17} :height 17}
:handler #(dispatch [:show-remove-participants])} :handler #(dispatch [:navigate-to :remove-participants])}
{:title "Leave Chat" {:title "Leave Chat"
:icon :muted :icon :muted
:icon-style {:width 18 :icon-style {:width 18

View File

@ -10,7 +10,7 @@
(defn chat-list-item [{:keys [chat-id] :as chat}] (defn chat-list-item [{:keys [chat-id] :as chat}]
[touchable-highlight [touchable-highlight
{:on-press #(dispatch [:show-chat chat-id :push])} {:on-press #(dispatch [:navigate-to :chat chat-id])}
;; TODO add [photo-path delivery-status new-messages-count online] values to chat-obj ;; TODO add [photo-path delivery-status new-messages-count online] values to chat-obj
[view [chat-list-item-inner-view (merge chat [view [chat-list-item-inner-view (merge chat
{:photo-path nil {:photo-path nil

View File

@ -35,9 +35,11 @@
(def picker-item (r/adapt-react-class (.-Item (.-Picker js/React)))) (def picker-item (r/adapt-react-class (.-Item (.-Picker js/React))))
(defn icon [n style] (defn icon
[image {:source {:uri (keyword (str "icon_" (name n)))} ([n] (icon n {}))
:style style}]) ([n style]
[image {:source {:uri (keyword (str "icon_" (name n)))}
:style style}]))
;(def react-linear-gradient (.-default (js/require "react-native-linear-gradient"))) ;(def react-linear-gradient (.-default (js/require "react-native-linear-gradient")))
;(def linear-gradient (r/adapt-react-class react-linear-gradient)) ;(def linear-gradient (r/adapt-react-class react-linear-gradient))

View File

@ -7,27 +7,28 @@
(def default-view :chat-list) (def default-view :chat-list)
;; initial state of app-db ;; initial state of app-db
(def app-db {:identity-password "replace-me-with-user-entered-password" (def app-db {:identity-password "replace-me-with-user-entered-password"
:identity "me" :identity "me"
:contacts [] :contacts []
:contacts-ids #{} :contacts-ids #{}
:selected-contacts #{} :selected-contacts #{}
:current-chat-id "console" :current-chat-id "console"
:chat {:command nil :chat {:command nil
:last-message nil} :last-message nil}
:chats {} :chats {}
:chats-updated-signal 0 :chats-updated-signal 0
:show-actions false :show-actions false
:new-participants #{} :selected-participants #{}
:signed-up true :signed-up true
:view-id default-view :view-id default-view
:navigation-stack (list default-view) :navigation-stack (list default-view)
;; TODO fix hardcoded values ;; TODO fix hardcoded values
:username "My Name" :username "My Name"
:phone-number "3147984309" :phone-number "3147984309"
:email "myemail@gmail.com" :email "myemail@gmail.com"
:status "Hi, this is my status" :status "Hi, this is my status"
:current-tag nil}) :current-tag nil
:disable-group-creation false})
(def protocol-initialized-path [:protocol-initialized]) (def protocol-initialized-path [:protocol-initialized])
(defn chat-input-text-path [chat-id] (defn chat-input-text-path [chat-id]

View File

@ -1,72 +1,144 @@
(ns status-im.group-settings.handlers (ns status-im.group-settings.handlers
(:require [re-frame.core :refer [register-handler debug dispatch]] (:require [re-frame.core :refer [register-handler debug dispatch after
enrich]]
[status-im.persistence.realm :as r] [status-im.persistence.realm :as r]
[status-im.models.messages :refer [clear-history]])) [status-im.chat.handlers :refer [delete-messages!]]
[status-im.protocol.api :as api]
[status-im.utils.random :as random]
[status-im.models.contacts :as contacts]
[status-im.models.messages :as messages]
[status-im.models.chats :as chats]
[status-im.constants :refer [text-content-type]]
[status-im.utils.handlers :as u]
[status-im.navigation.handlers :as nav]))
(defn set-chat-name [db] (defmethod nav/preload-data! :group-settings
(let [chat-id (:current-chat-id db) [db _]
name (:new-chat-name db)] (assoc db :selected-participants #{}))
(r/write (fn []
(-> (r/get-by-field :chats :chat-id chat-id)
(r/single)
(aset "name" name))))
(assoc-in db [:chats chat-id :name] name)))
(defn set-chat-color [db] (defn save-chat-property!
(let [chat-id (:current-chat-id db) [db-name property-name]
color (:new-chat-color db)] (fn [{:keys [current-chat-id] :as db} _]
(r/write (fn [] (let [property (db-name db)]
(-> (r/get-by-field :chats :chat-id chat-id) (r/write (fn []
(r/single) (-> (r/get-by-field :chats :chat-id current-chat-id)
(aset "color" color)))) (r/single)
(assoc-in db [:chats chat-id :color] color))) (aset (name property-name) property)))))))
(defn delete-chat [chat-id] (defn update-chat-property
(r/write [db-name property-name]
(fn [] (fn [{:keys [current-chat-id] :as db} _]
(-> (r/get-by-field :chats :chat-id chat-id) (let [property (db-name db)]
(r/single) (assoc-in db [:chats current-chat-id property-name] property))))
(r/delete))))
;; TODO temp. Update chat in db atom (defn prepare-chat-settings
(dispatch [:initialize-chats])) [{:keys [current-chat-id] :as db} _]
(let [{:keys [name color]} (-> db
(get-in [:chats current-chat-id])
(select-keys [:name :color]))]
(-> db
(assoc :new-chat-name name
:new-chat-color color
:group-settings {}))))
(register-handler :show-group-settings (register-handler :show-group-settings
(fn [db _] (after (fn [_ _] (dispatch [:navigate-to :group-settings])))
(let [chat-id (:current-chat-id db) prepare-chat-settings)
chat-name (get-in db [:chats chat-id :name])
chat-color (get-in db [:chats chat-id :color])
db (assoc db
:new-chat-name chat-name
:new-chat-color chat-color
:group-settings-show-color-picker false
:group-settings-selected-member nil)]
(dispatch [:navigate-to :group-settings])
db)))
(register-handler :set-chat-name (register-handler :set-chat-name
(fn [db [action]] (after (save-chat-property! :new-chat-name :name))
(set-chat-name db))) (update-chat-property :new-chat-name :name))
(register-handler :set-chat-color (register-handler :set-chat-color
(fn [db [action]] (after (save-chat-property! :new-chat-color :color))
(set-chat-color db))) (update-chat-property :new-chat-color :color))
(register-handler :set-new-chat-name (defn clear-messages
(fn [db [action name]] [{:keys [current-chat-id] :as db} _]
(assoc db :new-chat-name name))) (assoc-in db [:chats current-chat-id :messages] '()))
(register-handler :set-new-chat-color
(fn [db [action color]]
(assoc db :new-chat-color color)))
(register-handler :select-group-chat-member
(fn [db [action identity]]
(assoc db :group-settings-selected-member identity)))
(register-handler :set-group-settings-show-color-picker
(fn [db [action show?]]
(assoc db :group-settings-show-color-picker show?)))
(register-handler :clear-history (register-handler :clear-history
(fn [db [action]] (after delete-messages!)
(clear-history (:current-chat-id db)))) clear-messages)
(register-handler :group-settings
(fn [db [_ k v]]
(assoc-in db [:group-settings k] v)))
(defn remove-identities [collection identities]
(remove #(identities (:identity %)) collection))
(defn remove-members
[{:keys [current-chat-id selected-participants] :as db} _]
(update-in db [:chats current-chat-id :contacts]
remove-identities selected-participants))
(defn remove-members-from-realm!
[{:keys [current-chat-id selected-participants] :as db} _]
(let [chat (get-in db [:chats current-chat-id])]
(r/write
(fn []
(r/create
:chats
(update chat :contacts remove-identities selected-participants)
true)))))
(defn notify-about-removing!
[{:keys [current-chat-id selected-participants]} _]
(doseq [participant selected-participants]
(api/group-remove-participant current-chat-id participant)))
(defn system-message [msg-id content]
{:from "system"
:msg-id msg-id
:content content
:content-type text-content-type})
(defn removed-participant-msg [chat-id identity]
(let [contact-name (:name (contacts/contact-by-identity identity))]
(->> (str "You've removed " (or contact-name identity))
(system-message (random/id))
(messages/save-message chat-id))))
(defn create-removing-messages!
[{:keys [current-chat-id selected-participants]} _]
(doseq [participant selected-participants]
(removed-participant-msg current-chat-id participant)))
(defn deselect-members [db _]
(assoc db :selected-participants #{}))
(register-handler :remove-participants
;; todo check if user have rights to add/remove participants
;; todo order of operations tbd
(-> remove-members
;; todo shouldn't this be done only after receiving of the "ack message"
;; about the api call that removes participants from the group?
((after remove-members-from-realm!))
;; todo uncomment
;((after notify-about-removing!))
((after create-removing-messages!))
((enrich deselect-members))
debug))
(defn add-memebers
[{:keys [current-chat-id selected-participants] :as db} _]
(let [new-identities (map #(hash-map :identity %) selected-participants)]
(update db [:chats current-chat-id :contacts] concat new-identities)))
(defn add-members-to-realm!
[{:keys [current-chat-id selected-participants]} _]
(chats/chat-add-participants current-chat-id selected-participants))
(defn notify-about-new-members!
[{:keys [current-chat-id selected-participants]} _]
(doseq [identity selected-participants]
(api/group-add-participant current-chat-id identity)))
(register-handler :add-new-participants
;; todo order of operations tbd
(-> add-memebers
((after add-members-to-realm!))
;; todo uncomment
;((after notify-about-new-members!))
((enrich deselect-members))))

View File

@ -15,25 +15,25 @@
[status-im.group-settings.styles.group-settings :as st] [status-im.group-settings.styles.group-settings :as st]
[status-im.group-settings.views.member :refer [member-view]])) [status-im.group-settings.views.member :refer [member-view]]))
(defn remove-member [{:keys [whisper-identity]}] (defn remove-member []
(dispatch [:chat-remove-member whisper-identity])) (dispatch [:remove-participants]))
(defn close-member-menu [] (defn close-member-menu []
(dispatch [:select-group-chat-member nil])) (dispatch [:set :selected-participants #{}]))
(defview member-menu [] (defview member-menu []
[member [:group-settings-selected-member]] [{:keys [name] :as participant} [:selected-participant]]
[modal {:animated false (when participant
:transparent false [modal {:animated false
:onRequestClose close-member-menu} :transparent false
[touchable-highlight {:style st/modal-container :onRequestClose close-member-menu}
:on-press close-member-menu} [touchable-highlight {:style st/modal-container
[view st/modal-inner-container :on-press close-member-menu}
[text {:style st/modal-member-name} [view st/modal-inner-container
(:name member)] [text {:style st/modal-member-name} name]
[touchable-highlight {:on-press #(remove-member member)} [touchable-highlight {:on-press remove-member}
[text {:style st/modal-remove-text} [text {:style st/modal-remove-text}
"Remove"]]]]]) "Remove"]]]]]))
(defview chat-members [] (defview chat-members []
[members [:current-chat-contacts]] [members [:current-chat-contacts]]
@ -41,9 +41,6 @@
(for [member members] (for [member members]
^{:key member} [member-view member])]) ^{:key member} [member-view member])])
(defn show-chat-name-edit []
(dispatch [:navigate-to :chat-name-edit]))
(defn setting-view [{:keys [icon-style custom-icon handler title subtitle] (defn setting-view [{:keys [icon-style custom-icon handler title subtitle]
icon-name :icon}] icon-name :icon}]
[touchable-highlight {:on-press handler} [touchable-highlight {:on-press handler}
@ -58,15 +55,15 @@
subtitle])]]]) subtitle])]]])
(defn close-chat-color-picker [] (defn close-chat-color-picker []
(dispatch [:set-group-settings-show-color-picker false])) (dispatch [:group-settings :show-color-picker false]))
(defn set-chat-color [] (defn set-chat-color []
(close-chat-color-picker) (close-chat-color-picker)
(dispatch [:set-chat-color])) (dispatch [:set-chat-color]))
(defview chat-color-picker [] (defview chat-color-picker []
[show-color-picker [:get :group-settings-show-color-picker] [show-color-picker [:group-settings :show-color-picker]
new-color [:get :new-chat-color]] new-color [:get :new-chat-color]]
[modal {:animated false [modal {:animated false
:transparent false :transparent false
:onRequestClose close-chat-color-picker} :onRequestClose close-chat-color-picker}
@ -74,11 +71,11 @@
:on-press close-chat-color-picker} :on-press close-chat-color-picker}
[view st/modal-color-picker-inner-container [view st/modal-color-picker-inner-container
[picker {:selectedValue new-color [picker {:selectedValue new-color
:onValueChange #(dispatch [:set-new-chat-color %])} :onValueChange #(dispatch [:set :new-chat-color %])}
[picker-item {:label "Blue" :value "#7099e6"}] [picker-item {:label "Blue" :value "#7099e6"}]
[picker-item {:label "Purple" :value "#a187d5"}] [picker-item {:label "Purple" :value "#a187d5"}]
[picker-item {:label "Green" :value "green"}] [picker-item {:label "Green" :value "green"}]
[picker-item {:label "Red" :value "red"}]] [picker-item {:label "Red" :value "red"}]]
[touchable-highlight {:on-press set-chat-color} [touchable-highlight {:on-press set-chat-color}
[text {:style st/modal-color-picker-save-btn-text} [text {:style st/modal-color-picker-save-btn-text}
"Save"]]]]]) "Save"]]]]])
@ -88,7 +85,7 @@
[view {:style (st/chat-color-icon chat-color)}]) [view {:style (st/chat-color-icon chat-color)}])
(defn show-chat-color-picker [] (defn show-chat-color-picker []
(dispatch [:set-group-settings-show-color-picker true])) (dispatch [:group-settings :show-color-picker true]))
(defn settings-view [] (defn settings-view []
;; TODO implement settings handlers ;; TODO implement settings handlers
@ -105,22 +102,22 @@
{:icon :muted {:icon :muted
:icon-style {:width 18 :icon-style {:width 18
:height 21}})) :height 21}}))
{:icon :close-gray {:icon :close-gray
:icon-style {:width 12 :icon-style {:width 12
:height 12} :height 12}
:title "Clear history" :title "Clear history"
:handler #(dispatch [:clear-history])} :handler #(dispatch [:clear-history])}
{:icon :bin {:icon :bin
:icon-style {:width 12 :icon-style {:width 12
:height 18} :height 18}
:title "Delete and leave" :title "Delete and leave"
:handler #(dispatch [:leave-group-chat])}]] :handler #(dispatch [:leave-group-chat])}]]
[view st/settings-container [view st/settings-container
(for [setting settings] (for [setting settings]
^{:key setting} [setting-view setting])])) ^{:key setting} [setting-view setting])]))
(defview chat-icon [] (defview chat-icon []
[name [:chat :name] [name [:chat :name]
color [:chat :color]] color [:chat :color]]
[view (st/chat-icon color) [view (st/chat-icon color)
[text {:style st/chat-icon-text} (first name)]]) [text {:style st/chat-icon-text} (first name)]])
@ -129,25 +126,44 @@
[toolbar {:title "Chat settings" [toolbar {:title "Chat settings"
:custom-action [chat-icon]}]) :custom-action [chat-icon]}])
(defn focus []
(dispatch [:set ::name-input-focused true]))
(defn blur []
(dispatch [:set ::name-input-focused false]))
(defn save []
(dispatch [:set-chat-name]))
(defview chat-name []
[name [:chat :name]
new-name [:get :new-chat-name]
focused? [:get ::name-input-focused]]
[view
[text {:style st/chat-name-text} "Chat name"]
[view (st/chat-name-value-container focused?)
[text-input {:style st/chat-name-value
:ref #(when (and % focused?) (.focus %))
:on-change-text #(dispatch [:set :new-chat-name %])
:on-focus focus
:on-blur blur}
name]
(if (or focused? (not= name new-name))
[touchable-highlight {:style st/chat-name-btn-edit-container
:on-press save}
[view [icon :ok-purple st/add-members-icon]]]
[touchable-highlight {:style st/chat-name-btn-edit-container
:on-press focus}
[text {:style st/chat-name-btn-edit-text} "Edit"]])]])
(defview group-settings [] (defview group-settings []
[chat-name [:chat :name] [show-color-picker [:group-settings :show-color-picker]]
selected-member [:group-settings-selected-member]
show-color-picker [:get :group-settings-show-color-picker]]
[view st/group-settings [view st/group-settings
[new-group-toolbar] [new-group-toolbar]
[scroll-view st/body [scroll-view st/body
[text {:style st/chat-name-text} [chat-name]
"Chat name"] [text {:style st/members-text} "Members"]
[view st/chat-name-value-container [touchable-highlight {:on-press #(dispatch [:navigate-to :add-participants])}
[text {:style st/chat-name-value}
chat-name]
[touchable-highlight {:style st/chat-name-btn-edit-container
:on-press show-chat-name-edit}
[text {:style st/chat-name-btn-edit-text}
"Edit"]]]
[text {:style st/members-text}
"Members"]
[touchable-highlight {:on-press #(dispatch [:show-add-participants])}
[view st/add-members-container [view st/add-members-container
[icon :add-gray st/add-members-icon] [icon :add-gray st/add-members-icon]
[text {:style st/add-members-text} [text {:style st/add-members-text}
@ -158,5 +174,4 @@
[settings-view]] [settings-view]]
(when show-color-picker (when show-color-picker
[chat-color-picker]) [chat-color-picker])
(when selected-member [member-menu]])
[member-menu])])

View File

@ -1,19 +0,0 @@
(ns status-im.group-settings.styles.chat-name-edit
(:require [status-im.components.styles :refer [font
color-white
text1-color]]))
(def save-action-icon
{:width 18
:height 14})
(def chat-name-container
{:flex 1
:flexDirection :column
:backgroundColor color-white})
(def chat-name-input
{:marginLeft 12
:fontSize 14
:fontFamily font
:color text1-color})

View File

@ -24,17 +24,17 @@
:backgroundColor color-white}) :backgroundColor color-white})
(def modal-member-name (def modal-member-name
{:color text2-color {:color text2-color
:fontFamily font :fontFamily font
:fontSize 14 :fontSize 14
:lineHeight 20}) :lineHeight 20})
(def modal-remove-text (def modal-remove-text
{:margin 10 {:margin 10
:color text1-color :color text1-color
:fontFamily font :fontFamily font
:fontSize 14 :fontSize 14
:lineHeight 20}) :lineHeight 20})
(def modal-color-picker-inner-container (def modal-color-picker-inner-container
{:borderRadius 10 {:borderRadius 10
@ -42,12 +42,12 @@
:backgroundColor color-white}) :backgroundColor color-white})
(def modal-color-picker-save-btn-text (def modal-color-picker-save-btn-text
{:margin 10 {:margin 10
:alignSelf :center :alignSelf :center
:color text1-color :color text1-color
:fontFamily font :fontFamily font
:fontSize 14 :fontSize 14
:lineHeight 20}) :lineHeight 20})
(def chat-members-container (def chat-members-container
{:marginBottom 10}) {:marginBottom 10})
@ -85,14 +85,14 @@
:fontSize 14 :fontSize 14
:lineHeight 20}) :lineHeight 20})
(def chat-name-value-container (defn chat-name-value-container [focused?]
{:flexDirection :row {:flexDirection :row
:marginLeft 16 :marginLeft 16
:height 56 :height 56
:alignItems :center :alignItems :center
:justifyContent :center :justifyContent :center
:borderBottomWidth 1 :borderBottomWidth 2
:borderBottomColor separator-color}) :borderBottomColor (if focused? color-purple separator-color)})
(def chat-name-value (def chat-name-value
{:flex 1 {:flex 1
@ -105,11 +105,11 @@
:justifyContent :center}) :justifyContent :center})
(def chat-name-btn-edit-text (def chat-name-btn-edit-text
{:marginTop -1 {:marginTop -1
:color text2-color :color text2-color
:fontFamily font :fontFamily font
:fontSize 16 :fontSize 16
:lineHeight 20}) :lineHeight 20})
(def members-text (def members-text
{:marginTop 24 {:marginTop 24

View File

@ -1,14 +1,13 @@
(ns status-im.group-settings.subs (ns status-im.group-settings.subs
(:require-macros [reagent.ratom :refer [reaction]]) (:require-macros [reagent.ratom :refer [reaction]])
(:require [re-frame.core :refer [register-sub]] (:require [re-frame.core :refer [register-sub]]))
[status-im.models.contacts :refer [contact-by-identity]]))
(register-sub :group-settings-selected-member (register-sub :selected-participant
(fn [db [_]] (fn [db _]
(reaction (reaction
(let [identity (get @db :group-settings-selected-member)] (let [identity (first (:selected-participants @db))]
(contact-by-identity identity))))) (get-in @db [:contacts identity])))))
(register-sub :group-settings-show-color-picker (register-sub :group-settings
(fn [db [_]] (fn [db [_ k]]
(reaction (get @db :group-settings-show-color-picker)))) (reaction (get-in @db [:group-settings k]))))

View File

@ -1,31 +0,0 @@
(ns status-im.group-settings.views.chat-name-edit
(:require-macros [status-im.utils.views :refer [defview]])
(:require [reagent.core :as r]
[re-frame.core :refer [subscribe dispatch dispatch-sync]]
[status-im.components.react :refer [view text-input]]
[status-im.components.toolbar :refer [toolbar]]
[status-im.group-settings.styles.chat-name-edit :as st]
[status-im.components.styles :refer [toolbar-background2
text2-color]]))
(defn save-group-chat-name []
(dispatch [:set-chat-name])
(dispatch [:navigate-back]))
(defn chat-name-edit-toolbar [chat-name]
[toolbar {:background-color toolbar-background2
:title "Edit chat name"
;; TODO change to dark 'ok' icon
:action {:image {:source {:uri :icon_ok}
:style st/save-action-icon}
:handler save-group-chat-name}}])
(defview chat-name-edit []
[new-chat-name [:get :new-chat-name]]
[view st/chat-name-container
[chat-name-edit-toolbar]
[text-input {:style st/chat-name-input
:autoFocus true
:placeholderTextColor text2-color
:onChangeText #(dispatch [:set-new-chat-name %])}
new-chat-name]])

View File

@ -38,6 +38,6 @@
[text {:style st/role-text} [text {:style st/role-text}
role])] role])]
[touchable-highlight [touchable-highlight
{:on-press #(dispatch [:select-group-chat-member whisper-identity])} {:on-press #(dispatch [:set :selected-participants #{whisper-identity}])}
[view st/more-btn [view st/more-btn
[icon :more-vertical st/more-btn-icon]]]]) [icon :more-vertical st/more-btn-icon]]]])

View File

@ -1,46 +1,23 @@
(ns status-im.handlers (ns status-im.handlers
(:require (:require
[re-frame.core :refer [register-handler after dispatch debug enrich]] [re-frame.core :refer [register-handler after dispatch debug]]
[schema.core :as s :include-macros true] [schema.core :as s :include-macros true]
[status-im.persistence.realm :as r]
[status-im.db :refer [app-db schema]] [status-im.db :refer [app-db schema]]
[status-im.persistence.simple-kv-store :as kv] [status-im.persistence.simple-kv-store :as kv]
[status-im.protocol.state.storage :as storage] [status-im.protocol.state.storage :as storage]
[status-im.db :as db :refer [app-db schema]]
[status-im.protocol.api :refer [init-protocol]]
[status-im.protocol.protocol-handler :refer [make-handler]]
[status-im.models.protocol :refer [update-identity
set-initialized]]
[status-im.models.contacts :as contacts]
[status-im.models.messages :refer [save-message
update-message!
clear-history]]
[status-im.models.commands :refer [set-commands]] [status-im.models.commands :refer [set-commands]]
[status-im.handlers.server :as server]
[status-im.chat.suggestions :refer [load-commands]] [status-im.chat.suggestions :refer [load-commands]]
[status-im.models.chats :refer [chat-exists?
create-chat
chat-add-participants
chat-remove-participants
set-chat-active
re-join-group-chat
chat-by-id2]]
[status-im.utils.logging :as log] [status-im.utils.logging :as log]
[status-im.protocol.api :as api]
[status-im.constants :refer [text-content-type
content-type-command]]
[status-im.navigation :refer [nav-push
nav-replace
nav-pop]]
[status-im.utils.crypt :refer [gen-random-bytes]] [status-im.utils.crypt :refer [gen-random-bytes]]
[status-im.utils.random :as random]
[status-im.utils.handlers :as u] [status-im.utils.handlers :as u]
status-im.chat.handlers status-im.chat.handlers
[status-im.group-settings.handlers :refer [delete-chat]] status-im.group-settings.handlers
status-im.navigation.handlers status-im.navigation.handlers
status-im.discovery.handlers
status-im.contacts.handlers status-im.contacts.handlers
status-im.new-group.handlers)) status-im.discovery.handlers
status-im.new-group.handlers
status-im.participants.handlers
status-im.protocol.handlers))
;; -- Middleware ------------------------------------------------------------ ;; -- Middleware ------------------------------------------------------------
;; ;;
@ -63,6 +40,11 @@
(fn [db [_ k v]] (fn [db [_ k v]]
(assoc db k v)))) (assoc db k v))))
(register-handler :set-in
(debug
(fn [db [_ path v]]
(assoc-in db path v))))
(register-handler :initialize-db (register-handler :initialize-db
(fn [_ _] (fn [_ _]
(assoc app-db (assoc app-db
@ -100,192 +82,8 @@
(log/debug action commands) (log/debug action commands)
(set-commands db commands))) (set-commands db commands)))
;; -- Protocol --------------------------------------------------------------
(register-handler :initialize-protocol
(u/side-effect!
(fn [db [_]]
(init-protocol (make-handler db)))))
(register-handler :protocol-initialized
(fn [db [_ identity]]
(-> db
(update-identity identity)
(set-initialized true))))
(defn system-message [msg-id content]
{:from "system"
:msg-id msg-id
:content content
:content-type text-content-type})
(defn joined-chat-msg [chat-id from msg-id]
(let [contact-name (:name (contacts/contact-by-identity from))]
(save-message chat-id {:from "system"
:msg-id (str msg-id "_" from)
:content (str (or contact-name from) " received chat invitation")
:content-type text-content-type})))
(defn participant-invited-to-group-msg [chat-id identity from msg-id]
(let [inviter-name (:name (contacts/contact-by-identity from))
invitee-name (if (= identity (api/my-identity))
"You"
(:name (contacts/contact-by-identity identity)))]
(save-message chat-id {:from "system"
:msg-id msg-id
:content (str (or inviter-name from) " invited " (or invitee-name identity))
:content-type text-content-type})))
(defn participant-removed-from-group-msg [chat-id identity from msg-id]
(let [remover-name (:name (contacts/contact-by-identity from))
removed-name (:name (contacts/contact-by-identity identity))]
(->> (str (or remover-name from) " removed " (or removed-name identity))
(system-message msg-id)
(save-message chat-id))))
(defn you-removed-from-group-msg [chat-id from msg-id]
(let [remover-name (:name (contacts/contact-by-identity from))]
(->> (str (or remover-name from) " removed you from group chat")
(system-message msg-id)
(save-message chat-id))))
(defn participant-left-group-msg [chat-id from msg-id]
(let [left-name (:name (contacts/contact-by-identity from))]
(->> (str (or left-name from) " left")
(system-message msg-id)
(save-message chat-id))))
(defn removed-participant-msg [chat-id identity]
(let [contact-name (:name (contacts/contact-by-identity identity))]
(->> (str "You've removed " (or contact-name identity))
(system-message (random/id))
(save-message chat-id))))
(defn left-chat-msg [chat-id]
(save-message chat-id {:from "system"
:msg-id (random/id)
:content "You left this chat"
:content-type text-content-type}))
(register-handler :group-chat-invite-acked
(u/side-effect!
(fn [_ [action from group-id ack-msg-id]]
(log/debug action from group-id ack-msg-id)
(joined-chat-msg group-id from ack-msg-id))))
(register-handler :participant-removed-from-group
(u/side-effect!
(fn [_ [action from group-id identity msg-id]]
(log/debug action msg-id from group-id identity)
(chat-remove-participants group-id [identity])
(participant-removed-from-group-msg group-id identity from msg-id))))
(register-handler :you-removed-from-group
(u/side-effect!
(fn [_ [action from group-id msg-id]]
(log/debug action msg-id from group-id)
(you-removed-from-group-msg group-id from msg-id)
(set-chat-active group-id false))))
(register-handler :participant-left-group
(u/side-effect!
(fn [_ [action from group-id msg-id]]
(log/debug action msg-id from group-id)
(when-not (= (api/my-identity) from)
(participant-left-group-msg group-id from msg-id)))))
(register-handler :participant-invited-to-group
(u/side-effect!
(fn [_ [action from group-id identity msg-id]]
(log/debug action msg-id from group-id identity)
(participant-invited-to-group-msg group-id identity from msg-id))))
(register-handler :acked-msg
(u/side-effect!
(fn [_ [action from msg-id]]
(log/debug action from msg-id)
(update-message! {:msg-id msg-id
:delivery-status :delivered}))))
(register-handler :msg-delivery-failed
(u/side-effect!
(fn [_ [action msg-id]]
(log/debug action msg-id)
(update-message! {:msg-id msg-id
:delivery-status :failed}))))
(register-handler :leave-group-chat
(u/side-effect!
(fn [db [action]]
(log/debug action)
(let [chat-id (:current-chat-id db)]
(api/leave-group-chat chat-id)
(set-chat-active chat-id false)
(left-chat-msg chat-id)
(delete-chat chat-id)
(dispatch [:navigate-back])))))
;; -- User data -------------------------------------------------------------- ;; -- User data --------------------------------------------------------------
(register-handler :load-user-phone-number (register-handler :load-user-phone-number
(fn [db [_]] (fn [db [_]]
;; todo fetch phone number from db ;; todo fetch phone number from db
(assoc db :user-phone-number "123"))) (assoc db :user-phone-number "123")))
;; -- Chats --------------------------------------------------------------
(defn update-new-participants-selection [db identity add?]
(update db :new-participants (fn [new-participants]
(if add?
(conj new-participants identity)
(disj new-participants identity)))))
(register-handler :select-new-participant
(fn [db [action identity add?]]
(log/debug action identity add?)
(update-new-participants-selection db identity add?)))
(register-handler :remove-selected-participants
(fn [db [action]]
(log/debug action)
(let [identities (vec (:new-participants db))
chat-id (:current-chat-id db)]
(chat-remove-participants chat-id identities)
(dispatch [:navigate-back])
(doseq [ident identities]
(api/group-remove-participant chat-id ident)
(removed-participant-msg chat-id ident)))))
(register-handler :add-new-participants
(fn [db [action navigator]]
(log/debug action)
(let [identities (vec (:new-participants db))
chat-id (:current-chat-id db)]
(chat-add-participants chat-id identities)
(dispatch [:navigate-back])
(doseq [ident identities]
(api/group-add-participant chat-id ident))
db)))
(defn chat-remove-member [db]
(let [chat (get-in db [:chats (:current-chat-id db)])
identity (:group-settings-selected-member db)]
(r/write
(fn []
(r/create :chats
(update chat :contacts
(fn [members]
(filter #(not= (:identity %) identity) members)))
true)))
;; TODO temp. Update chat in db atom
(dispatch [:initialize-chats])
db))
(register-handler :chat-remove-member
(fn [db [action]]
(let [chat-id (:current-chat-id db)
identity (:group-settings-selected-member db)
db (chat-remove-member db)]
(dispatch [:select-group-chat-member nil])
;; TODO fix and uncomment
(api/group-remove-participant chat-id identity)
(removed-participant-msg chat-id identity)
db)))

View File

@ -62,10 +62,3 @@
(fn [] (fn []
(when (r/exists? :msgs :msg-id msg-id) (when (r/exists? :msgs :msg-id msg-id)
(r/create :msgs msg true))))) (r/create :msgs msg true)))))
(defn clear-history [chat-id]
(r/write
(fn []
(r/delete (r/get-by-field :msgs :chat-id chat-id))))
;; TODO temp. Update chat in db atom
(dispatch [:initialize-chats]))

View File

@ -1,17 +0,0 @@
(ns status-im.navigation)
(def ^{:dynamic true :private true} *nav-render*
"Flag to suppress navigator re-renders from outside om when pushing/popping."
true)
(defn nav-pop [nav]
(binding [*nav-render* true]
(.pop nav)))
(defn nav-push [nav route]
(binding [*nav-render* true]
(.push nav (clj->js route))))
(defn nav-replace [nav route]
(binding [*nav-render* true]
(.replace nav (clj->js route))))

View File

@ -18,7 +18,9 @@
(update :navigation-stack replace-top-element view-id) (update :navigation-stack replace-top-element view-id)
(assoc :view-id view-id))) (assoc :view-id view-id)))
(defmulti preload-data! (fn [_ [_ view-id]] view-id)) (defmulti preload-data!
(fn [db [_ view-id]] (or view-id (:view-id db))))
(defmethod preload-data! :default [db _] db) (defmethod preload-data! :default [db _] db)
(register-handler :navigate-to (register-handler :navigate-to
@ -27,10 +29,12 @@
(push-view db view-id))) (push-view db view-id)))
(register-handler :navigation-replace (register-handler :navigation-replace
(enrich preload-data!)
(fn [db [_ view-id]] (fn [db [_ view-id]]
(replace-view db view-id))) (replace-view db view-id)))
(register-handler :navigate-back (register-handler :navigate-back
(enrich preload-data!)
(fn [{:keys [navigation-stack] :as db} _] (fn [{:keys [navigation-stack] :as db} _]
(if (>= 1 (count navigation-stack)) (if (>= 1 (count navigation-stack))
db db
@ -46,32 +50,10 @@
(push-view :new-group) (push-view :new-group)
(assoc :new-group #{}))))) (assoc :new-group #{})))))
(register-handler :show-chat
(fn [db [_ chat-id nav-type]]
(let [update-view-id-fn (if (= :replace nav-type) replace-view push-view)]
(-> db
(update-view-id-fn :chat)
(assoc :current-chat-id chat-id)))))
(register-handler :show-contacts (register-handler :show-contacts
(fn [db _] (fn [db _]
(push-view db :contact-list))) (push-view db :contact-list)))
(defn clear-new-participants [db]
(assoc db :new-participants #{}))
(register-handler :show-remove-participants
(fn [db _]
(-> db
(push-view :remove-participants)
clear-new-participants)))
(register-handler :show-add-participants
(fn [db _]
(-> db
(push-view :add-participants)
clear-new-participants)))
(defn show-profile (defn show-profile
[db [_ identity]] [db [_ identity]]
(-> db (-> db

View File

@ -56,14 +56,31 @@
(defn show-chat! (defn show-chat!
[{:keys [new-group-id]} _] [{:keys [new-group-id]} _]
(dispatch [:show-chat new-group-id :replace])) (dispatch [:navigation-replace :chat new-group-id]))
(defn enable-creat-buttion
[db _]
(assoc db :disable-group-creation false))
(register-handler :create-new-group (register-handler :create-new-group
(-> start-group-chat! (-> start-group-chat!
((enrich prepare-chat)) ((enrich prepare-chat))
((enrich add-chat)) ((enrich add-chat))
((after create-chat!)) ((after create-chat!))
((after show-chat!)))) ((after show-chat!))
((enrich enable-creat-buttion))))
(defn disable-creat-button
[db _]
(assoc db :disable-group-creation true))
(defn dispatch-create-group
[_ [_ group-name]]
(dispatch [:create-new-group group-name]))
(register-handler :init-group-creation
(after dispatch-create-group)
disable-creat-button)
; todo rewrite ; todo rewrite
(register-handler :group-chat-invite-received (register-handler :group-chat-invite-received

View File

@ -18,12 +18,14 @@
(defview new-group-toolbar [] (defview new-group-toolbar []
[group-name [:get ::group-name]] [group-name [:get ::group-name]
creation-disabled? [:get :disable-group-creation]]
[toolbar [toolbar
{:title "New group chat" {:title "New group chat"
:action {:image {:source res/v ;; {:uri "icon_search"} :action {:image {:source res/v ;; {:uri "icon_search"}
:style st/toolbar-icon} :style st/toolbar-icon}
:handler #(dispatch [:create-new-group group-name])}}]) :handler (when-not creation-disabled?
#(dispatch [:init-group-creation group-name]))}}])
(defview group-name-input [] (defview group-name-input []
[group-name [:get ::group-name]] [group-name [:get ::group-name]]

View File

@ -1,9 +1,7 @@
(ns status-im.new-group.subs (ns status-im.new-group.subs
(:require-macros [reagent.ratom :refer [reaction]]) (:require-macros [reagent.ratom :refer [reaction]])
(:require [re-frame.core :refer [register-sub]])) (:require [re-frame.core :refer [register-sub]]
[status-im.utils.subs :as u]))
(register-sub :is-contact-selected? (register-sub :is-contact-selected?
(fn [db [_ id]] (u/contains-sub :selected-contacts))
(-> (:selected-contacts @db)
(contains? id)
(reaction))))

View File

@ -8,7 +8,6 @@
(defn on-toggle [whisper-identity] (defn on-toggle [whisper-identity]
(fn [checked?] (fn [checked?]
(println checked?)
(let [action (if checked? :select-contact :deselect-contact)] (let [action (if checked? :select-contact :deselect-contact)]
(dispatch [action whisper-identity])))) (dispatch [action whisper-identity]))))

View File

@ -0,0 +1,23 @@
(ns status-im.participants.handlers
(:require [status-im.navigation.handlers :as nav]
[re-frame.core :refer [register-handler debug]]))
(defmethod nav/preload-data! :add-participants
[db _]
(assoc db :selected-participants #{}))
(defmethod nav/preload-data! :remove-participants
[db _]
(assoc db :selected-participants #{}))
(defn deselect-participant
[db [_ id]]
(update db :selected-participants disj id))
(register-handler :deselect-participant deselect-participant)
(defn select-participant
[db [_ id]]
(update db :selected-participants conj id))
(register-handler :select-participant (debug select-participant))

View File

@ -0,0 +1,7 @@
(ns status-im.participants.subs
(:require-macros [reagent.ratom :refer [reaction]])
(:require [re-frame.core :refer [register-sub]]
[status-im.utils.subs :as u]))
(register-sub :is-participant-selected?
(u/contains-sub :selected-participants))

View File

@ -0,0 +1,30 @@
(ns status-im.participants.views.add
(:require-macros [status-im.utils.views :refer [defview]])
(:require [re-frame.core :refer [subscribe dispatch]]
[status-im.resources :as res]
[status-im.components.react :refer [view list-view list-item]]
[status-im.components.toolbar :refer [toolbar]]
[status-im.utils.listview :refer [to-datasource]]
[status-im.participants.views.contact :refer [participant-contact]]
[reagent.core :as r]
[status-im.participants.styles :as st]))
(defn new-participants-toolbar []
[toolbar
{:title "Add Participants"
:action {:image {:source res/v ;; {:uri "icon_search"}
:style st/new-participant-image}
:handler #(do (dispatch [:add-new-participants])
(dispatch [:navigate-back]))}}])
(defn new-participants-row
[row _ _]
(list-item [participant-contact row]))
(defview new-participants []
[contacts [:all-new-contacts]]
[view st/participants-container
[new-participants-toolbar]
[list-view {:dataSource (to-datasource contacts)
:renderRow new-participants-row
:style st/participants-list}]])

View File

@ -1,4 +1,5 @@
(ns status-im.participants.views.contact (ns status-im.participants.views.contact
(:require-macros [status-im.utils.views :refer [defview]])
(:require [re-frame.core :refer [subscribe dispatch dispatch-sync]] (:require [re-frame.core :refer [subscribe dispatch dispatch-sync]]
[status-im.components.react :refer [view]] [status-im.components.react :refer [view]]
[status-im.contacts.views.contact-inner :refer [contact-inner-view]] [status-im.contacts.views.contact-inner :refer [contact-inner-view]]
@ -6,14 +7,17 @@
[reagent.core :as r] [reagent.core :as r]
[status-im.participants.styles :as st])) [status-im.participants.styles :as st]))
(defn participant-contact [{:keys [whisper-identity] :as contact}] ;; todo duplication
;; todo must be moved to handlers (defn on-toggle [whisper-identity]
(let [checked (r/atom false)] (fn [checked?]
(fn [{:keys [whisper-identity] :as contact}] (let [action (if checked? :select-participant :deselect-participant)]
[view st/participant-container (dispatch [action whisper-identity]))))
[item-checkbox {:onToggle (fn [checked?]
(reset! checked checked?) (defview participant-contact
(dispatch [:select-new-participant whisper-identity checked?])) [{:keys [whisper-identity] :as contact}]
:checked @checked [checked [:is-participant-selected? whisper-identity]]
:size 30}] [view st/participant-container
[contact-inner-view contact]]))) [item-checkbox {:onToggle (on-toggle whisper-identity)
:checked checked
:size 30}]
[contact-inner-view contact]])

View File

@ -1,30 +0,0 @@
(ns status-im.participants.views.create
(:require [re-frame.core :refer [subscribe dispatch]]
[status-im.resources :as res]
[status-im.components.react :refer [view list-view list-item]]
[status-im.components.toolbar :refer [toolbar]]
[status-im.utils.listview :refer [to-datasource]]
[status-im.participants.views.contact :refer [participant-contact]]
[reagent.core :as r]
[status-im.participants.styles :as st]))
(defn new-participants-toolbar []
[toolbar
{:title "Add Participants"
:action {:image {:source res/v ;; {:uri "icon_search"}
:style st/new-participant-image}
:handler #(dispatch [:add-new-participants])}}])
(defn new-participants-row
[row _ _]
(list-item [participant-contact row]))
(defn new-participants []
(let [contacts (subscribe [:all-new-contacts])]
(fn []
(let [contacts-ds (to-datasource @contacts)]
[view st/participants-container
[new-participants-toolbar]
[list-view {:dataSource contacts-ds
:renderRow new-participants-row
:style st/participants-list}]]))))

View File

@ -1,4 +1,5 @@
(ns status-im.participants.views.remove (ns status-im.participants.views.remove
(:require-macros [status-im.utils.views :refer [defview]])
(:require [re-frame.core :refer [subscribe dispatch]] (:require [re-frame.core :refer [subscribe dispatch]]
[status-im.resources :as res] [status-im.resources :as res]
[status-im.components.react :refer [view text-input text image [status-im.components.react :refer [view text-input text image
@ -13,21 +14,20 @@
(defn remove-participants-toolbar [] (defn remove-participants-toolbar []
[toolbar [toolbar
{:title "Remove Participants" {:title "Remove Participants"
:action {:handler #(dispatch [:remove-selected-participants]) :action {:handler #(do (dispatch [:remove-participants])
:image {:source res/trash-icon ;; {:uri "icon_search"} (dispatch [:navigate-back]))
:style st/remove-participants-image}}}]) :image {:source res/trash-icon ;; {:uri "icon_search"}
:style st/remove-participants-image}}}])
(defn remove-participants-row (defn remove-participants-row
[row _ _] [row _ _]
(r/as-element [participant-contact row])) (r/as-element [participant-contact row]))
(defn remove-participants [] (defview remove-participants []
(let [contacts (subscribe [:current-chat-contacts])] [contacts [:current-chat-contacts]]
(fn [] [view st/participants-container
(let [contacts-ds (to-datasource @contacts)] [remove-participants-toolbar]
[view st/participants-container [list-view {:dataSource (to-datasource contacts)
[remove-participants-toolbar] :renderRow remove-participants-row
[list-view {:dataSource contacts-ds :style st/participants-list}]])
:renderRow remove-participants-row
:style st/participants-list}]]))))

View File

@ -32,7 +32,7 @@
(defn message-user [identity] (defn message-user [identity]
(when identity (when identity
(dispatch [:show-chat identity :push]))) (dispatch [:navigate-to :chat identity])))
(defview profile [] (defview profile []
[{:keys [name whisper-identity phone-number]} [:contact]] [{:keys [name whisper-identity phone-number]} [:contact]]

View File

@ -0,0 +1,115 @@
; todo everything inside this namespace must be revievew in common with future
; changes in protocol lib
(ns status-im.protocol.handlers
(:require [status-im.utils.handlers :as u]
[status-im.utils.logging :as log]
[status-im.protocol.api :as api]
[re-frame.core :refer [register-handler dispatch debug]]
[status-im.models.contacts :as contacts]
[status-im.protocol.api :refer [init-protocol]]
[status-im.protocol.protocol-handler :refer [make-handler]]
[status-im.models.protocol :refer [update-identity
set-initialized]]
[status-im.constants :refer [text-content-type]]
[status-im.models.messages :as messages]
[status-im.models.chats :as chats]))
(register-handler :initialize-protocol
(u/side-effect!
(fn [db [_]]
(init-protocol (make-handler db)))))
(register-handler :protocol-initialized
(fn [db [_ identity]]
(-> db
(update-identity identity)
(set-initialized true))))
(defn system-message [msg-id content]
{:from "system"
:msg-id msg-id
:content content
:content-type text-content-type})
(defn joined-chat-msg [chat-id from msg-id]
(let [contact-name (:name (contacts/contact-by-identity from))]
(messages/save-message chat-id {:from "system"
:msg-id (str msg-id "_" from)
:content (str (or contact-name from) " received chat invitation")
:content-type text-content-type})))
(defn participant-invited-to-group-msg [chat-id identity from msg-id]
(let [inviter-name (:name (contacts/contact-by-identity from))
invitee-name (if (= identity (api/my-identity))
"You"
(:name (contacts/contact-by-identity identity)))]
(messages/save-message chat-id {:from "system"
:msg-id msg-id
:content (str (or inviter-name from) " invited " (or invitee-name identity))
:content-type text-content-type})))
(defn participant-removed-from-group-msg [chat-id identity from msg-id]
(let [remover-name (:name (contacts/contact-by-identity from))
removed-name (:name (contacts/contact-by-identity identity))]
(->> (str (or remover-name from) " removed " (or removed-name identity))
(system-message msg-id)
(messages/save-message chat-id))))
(defn you-removed-from-group-msg [chat-id from msg-id]
(let [remover-name (:name (contacts/contact-by-identity from))]
(->> (str (or remover-name from) " removed you from group chat")
(system-message msg-id)
(messages/save-message chat-id))))
(defn participant-left-group-msg [chat-id from msg-id]
(let [left-name (:name (contacts/contact-by-identity from))]
(->> (str (or left-name from) " left")
(system-message msg-id)
(messages/save-message chat-id))))
(register-handler :group-chat-invite-acked
(u/side-effect!
(fn [_ [action from group-id ack-msg-id]]
(log/debug action from group-id ack-msg-id)
(joined-chat-msg group-id from ack-msg-id))))
(register-handler :participant-removed-from-group
(u/side-effect!
(fn [_ [action from group-id identity msg-id]]
(log/debug action msg-id from group-id identity)
(chats/chat-remove-participants group-id [identity])
(participant-removed-from-group-msg group-id identity from msg-id))))
(register-handler :you-removed-from-group
(u/side-effect!
(fn [_ [action from group-id msg-id]]
(log/debug action msg-id from group-id)
(you-removed-from-group-msg group-id from msg-id)
(chats/set-chat-active group-id false))))
(register-handler :participant-left-group
(u/side-effect!
(fn [_ [action from group-id msg-id]]
(log/debug action msg-id from group-id)
(when-not (= (api/my-identity) from)
(participant-left-group-msg group-id from msg-id)))))
(register-handler :participant-invited-to-group
(u/side-effect!
(fn [_ [action from group-id identity msg-id]]
(log/debug action msg-id from group-id identity)
(participant-invited-to-group-msg group-id identity from msg-id))))
(register-handler :acked-msg
(u/side-effect!
(fn [_ [action from msg-id]]
(log/debug action from msg-id)
(messages/update-message! {:msg-id msg-id
:delivery-status :delivered}))))
(register-handler :msg-delivery-failed
(u/side-effect!
(fn [_ [action msg-id]]
(log/debug action msg-id)
(messages/update-message! {:msg-id msg-id
:delivery-status :failed}))))

View File

@ -5,8 +5,13 @@
status-im.group-settings.subs status-im.group-settings.subs
status-im.discovery.subs status-im.discovery.subs
status-im.contacts.subs status-im.contacts.subs
status-im.new-group.subs)) status-im.new-group.subs
status-im.participants.subs))
(register-sub :get (register-sub :get
(fn [db [_ k]] (fn [db [_ k]]
(reaction (k @db)))) (reaction (k @db))))
(register-sub :get-in
(fn [db [_ path]]
(reaction (get-in @db path))))

View File

@ -0,0 +1,11 @@
(ns status-im.utils.subs
(:require-macros [reagent.ratom :refer [reaction]]))
(defn contains-sub
"Creates subscrition that cheks if collection (map or set) contains element"
[collection]
(fn [db [_ element]]
(println "WWWWWWWWWW" (type db))
(-> (collection @db)
(contains? element)
(reaction))))