2016-05-19 18:31:56 +02:00
|
|
|
(ns status-im.chat.screen
|
|
|
|
(:require-macros [status-im.utils.views :refer [defview]])
|
2016-05-19 15:45:16 +03:00
|
|
|
(:require [re-frame.core :refer [subscribe dispatch]]
|
2016-05-20 16:18:12 +03:00
|
|
|
[clojure.string :as s]
|
2016-05-19 18:31:56 +02:00
|
|
|
[status-im.components.react :refer [view
|
2016-06-02 16:23:50 +03:00
|
|
|
animated-view
|
|
|
|
text
|
|
|
|
icon
|
|
|
|
touchable-highlight
|
|
|
|
list-view
|
|
|
|
list-item]]
|
2016-07-22 11:53:12 +03:00
|
|
|
[status-im.components.status-bar :refer [status-bar]]
|
2016-05-20 14:20:48 +03:00
|
|
|
[status-im.components.chat-icon.screen :refer [chat-icon-view-action
|
|
|
|
chat-icon-view-menu-item]]
|
2016-05-19 18:31:56 +02:00
|
|
|
[status-im.chat.styles.screen :as st]
|
2016-06-08 14:24:35 +03:00
|
|
|
[status-im.utils.listview :refer [to-datasource-inverted]]
|
2016-05-20 16:36:00 +03:00
|
|
|
[status-im.utils.utils :refer [truncate-str]]
|
2016-08-04 18:36:13 +03:00
|
|
|
[status-im.utils.datetime :as time]
|
2016-05-19 18:31:56 +02:00
|
|
|
[status-im.components.invertible-scroll-view :refer [invertible-scroll-view]]
|
|
|
|
[status-im.components.toolbar :refer [toolbar]]
|
|
|
|
[status-im.chat.views.message :refer [chat-message]]
|
2016-06-25 18:44:37 +03:00
|
|
|
[status-im.chat.views.suggestions :refer [suggestion-container]]
|
2016-06-03 14:09:34 +03:00
|
|
|
[status-im.chat.views.response :refer [response-view]]
|
2016-05-25 04:27:03 +03:00
|
|
|
[status-im.chat.views.new-message :refer [chat-message-new]]
|
2016-06-03 22:34:24 +03:00
|
|
|
[status-im.i18n :refer [label label-pluralize]]
|
|
|
|
[status-im.components.animation :as anim]
|
2016-08-04 18:36:13 +03:00
|
|
|
[reagent.core :as r]
|
|
|
|
[clojure.string :as str]))
|
2016-03-25 13:36:16 +03:00
|
|
|
|
|
|
|
|
2016-04-06 16:13:31 +03:00
|
|
|
(defn contacts-by-identity [contacts]
|
|
|
|
(->> contacts
|
|
|
|
(map (fn [{:keys [identity] :as contact}]
|
|
|
|
[identity contact]))
|
|
|
|
(into {})))
|
|
|
|
|
|
|
|
(defn add-msg-color [{:keys [from] :as msg} contact-by-identity]
|
2016-04-07 14:01:41 +03:00
|
|
|
(if (= "system" from)
|
2016-05-10 13:58:37 +03:00
|
|
|
(assoc msg :text-color :#4A5258
|
|
|
|
:background-color :#D3EEEF)
|
2016-04-07 14:01:41 +03:00
|
|
|
(let [{:keys [text-color background-color]} (get contact-by-identity from)]
|
|
|
|
(assoc msg :text-color text-color
|
|
|
|
:background-color background-color))))
|
2016-04-06 16:13:31 +03:00
|
|
|
|
2016-05-19 15:45:16 +03:00
|
|
|
(defview chat-icon []
|
2016-06-27 14:35:33 +03:00
|
|
|
[chat-id [:chat :chat-id]
|
2016-05-19 15:45:16 +03:00
|
|
|
group-chat [:chat :group-chat]
|
2016-06-27 14:35:33 +03:00
|
|
|
name [:chat :name]
|
|
|
|
color [:chat :color]]
|
2016-05-19 15:45:16 +03:00
|
|
|
;; TODO stub data ('online' property)
|
|
|
|
[chat-icon-view-action chat-id group-chat name color true])
|
2016-04-18 13:38:38 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn typing [member platform-specific]
|
2016-05-07 16:39:03 +03:00
|
|
|
[view st/typing-view
|
|
|
|
[view st/typing-background
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/typing-text
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :default}
|
2016-05-25 18:57:06 +03:00
|
|
|
(str member " " (label :t/is-typing))]]])
|
2016-04-19 15:38:16 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn typing-all [platform-specific]
|
2016-05-08 23:13:24 +03:00
|
|
|
[view st/typing-all
|
2016-05-20 14:38:20 +03:00
|
|
|
;; TODO stub data
|
2016-04-19 15:38:16 +03:00
|
|
|
(for [member ["Geoff" "Justas"]]
|
2016-07-22 11:53:12 +03:00
|
|
|
^{:key member} [typing member platform-specific])])
|
2016-04-19 15:38:16 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn message-row [{:keys [contact-by-identity platform-specific group-chat messages-count]}]
|
2016-05-08 23:06:38 +03:00
|
|
|
(fn [row _ idx]
|
2016-05-06 14:06:58 +03:00
|
|
|
(let [msg (-> row
|
|
|
|
(add-msg-color contact-by-identity)
|
2016-05-08 23:06:38 +03:00
|
|
|
(assoc :group-chat group-chat)
|
2016-06-08 14:24:35 +03:00
|
|
|
(assoc :last-msg (= (js/parseInt idx) (dec messages-count))))]
|
2016-07-22 11:53:12 +03:00
|
|
|
(list-item [chat-message msg platform-specific]))))
|
2016-05-06 14:06:58 +03:00
|
|
|
|
|
|
|
(defn on-action-selected [position]
|
|
|
|
(case position
|
2016-05-20 19:58:07 +03:00
|
|
|
0 (dispatch [:navigate-to :add-participants])
|
|
|
|
1 (dispatch [:navigate-to :remove-participants])
|
2016-05-12 12:56:47 +03:00
|
|
|
2 (dispatch [:leave-group-chat])))
|
2016-05-06 14:06:58 +03:00
|
|
|
|
|
|
|
(defn overlay [{:keys [on-click-outside]} items]
|
2016-05-07 16:39:03 +03:00
|
|
|
[view st/actions-overlay
|
|
|
|
[touchable-highlight {:on-press on-click-outside
|
|
|
|
:style st/overlay-highlight}
|
2016-05-06 14:06:58 +03:00
|
|
|
[view nil]]
|
|
|
|
items])
|
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn action-view [{{:keys [icon-style
|
|
|
|
custom-icon
|
|
|
|
handler
|
|
|
|
title
|
|
|
|
subtitle]
|
|
|
|
icon-name :icon} :action
|
|
|
|
platform-specific :platform-specific}]
|
2016-05-03 16:39:08 +03:00
|
|
|
[touchable-highlight {:on-press (fn []
|
|
|
|
(dispatch [:set-show-actions false])
|
2016-05-11 14:37:31 +03:00
|
|
|
(when handler
|
|
|
|
(handler)))}
|
2016-05-07 16:39:03 +03:00
|
|
|
[view st/action-icon-row
|
|
|
|
[view st/action-icon-view
|
2016-05-11 14:37:31 +03:00
|
|
|
(or custom-icon
|
|
|
|
[icon icon-name icon-style])]
|
2016-05-08 23:13:24 +03:00
|
|
|
[view st/action-view
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/action-title
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :medium} title]
|
2016-05-07 16:39:03 +03:00
|
|
|
(when-let [subtitle subtitle]
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/action-subtitle
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :default}
|
2016-05-02 16:56:42 +03:00
|
|
|
subtitle])]]])
|
|
|
|
|
2016-05-19 15:45:16 +03:00
|
|
|
(defview menu-item-icon-profile []
|
2016-06-27 14:35:33 +03:00
|
|
|
[chat-id [:chat :chat-id]
|
2016-05-19 15:45:16 +03:00
|
|
|
group-chat [:chat :group-chat]
|
2016-06-27 14:35:33 +03:00
|
|
|
name [:chat :name]
|
|
|
|
color [:chat :color]]
|
2016-05-19 15:45:16 +03:00
|
|
|
;; TODO stub data ('online' property)
|
|
|
|
[chat-icon-view-menu-item chat-id group-chat name color true])
|
2016-05-11 14:37:31 +03:00
|
|
|
|
2016-05-20 16:18:12 +03:00
|
|
|
(defn members-text [members]
|
2016-05-25 18:57:06 +03:00
|
|
|
(truncate-str (str (s/join ", " (map #(:name %) members)) " " (label :t/and-you)) 35))
|
2016-05-20 16:18:12 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn actions-list-view [{styles :styles :as platform-specific}]
|
|
|
|
(let [{:keys [group-chat chat-id]} (subscribe [:chat-properties [:group-chat :chat-id]])
|
|
|
|
members (subscribe [:current-chat-contacts])
|
|
|
|
status-bar-height (get-in styles [:components :status-bar :default :height])]
|
2016-05-09 17:33:41 +03:00
|
|
|
(when-let [actions (if @group-chat
|
2016-05-25 18:57:06 +03:00
|
|
|
[{:title (label :t/members-title)
|
2016-05-20 16:18:12 +03:00
|
|
|
:subtitle (members-text @members)
|
2016-05-07 16:39:03 +03:00
|
|
|
:icon :menu_group
|
2016-05-06 14:06:58 +03:00
|
|
|
:icon-style {:width 25
|
|
|
|
:height 19}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Members
|
|
|
|
:handler nil}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/search-chat)
|
|
|
|
:subtitle (label :t/not-implemented)
|
2016-05-07 16:39:03 +03:00
|
|
|
:icon :search_gray_copy
|
2016-05-06 14:06:58 +03:00
|
|
|
:icon-style {:width 17
|
|
|
|
:height 17}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Search chat
|
|
|
|
:handler nil}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/notifications-title)
|
|
|
|
:subtitle (label :t/not-implemented)
|
2016-05-20 15:37:24 +03:00
|
|
|
;;:subtitle "Chat muted"
|
2016-05-07 16:39:03 +03:00
|
|
|
:icon :muted
|
2016-05-06 14:06:58 +03:00
|
|
|
:icon-style {:width 18
|
|
|
|
:height 21}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Notifications
|
|
|
|
:handler nil}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/settings)
|
2016-05-07 16:39:03 +03:00
|
|
|
:icon :settings
|
2016-05-06 14:06:58 +03:00
|
|
|
:icon-style {:width 20
|
|
|
|
:height 13}
|
2016-05-12 12:13:09 +03:00
|
|
|
:handler #(dispatch [:show-group-settings])}]
|
2016-06-27 14:35:33 +03:00
|
|
|
[{:title (label :t/profile)
|
2016-05-11 14:37:31 +03:00
|
|
|
:custom-icon [menu-item-icon-profile]
|
2016-06-27 14:35:33 +03:00
|
|
|
:icon :menu_group
|
|
|
|
:icon-style {:width 25
|
|
|
|
:height 19}
|
|
|
|
:handler #(dispatch [:show-profile @chat-id])}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/search-chat)
|
|
|
|
:subtitle (label :t/not-implemented)
|
2016-05-11 14:37:31 +03:00
|
|
|
:icon :search_gray_copy
|
|
|
|
:icon-style {:width 17
|
|
|
|
:height 17}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Search chat
|
2016-05-12 12:13:09 +03:00
|
|
|
:handler nil}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/notifications-title)
|
|
|
|
:subtitle (label :t/not-implemented)
|
2016-05-20 15:37:24 +03:00
|
|
|
;;:subtitle "Notifications on"
|
2016-05-11 14:37:31 +03:00
|
|
|
:icon :muted
|
|
|
|
:icon-style {:width 18
|
|
|
|
:height 21}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Notifications
|
2016-05-12 12:13:09 +03:00
|
|
|
:handler nil}
|
2016-05-25 18:57:06 +03:00
|
|
|
{:title (label :t/settings)
|
|
|
|
:subtitle (label :t/not-implemented)
|
2016-05-11 14:37:31 +03:00
|
|
|
:icon :settings
|
|
|
|
:icon-style {:width 20
|
|
|
|
:height 13}
|
2016-05-20 15:37:24 +03:00
|
|
|
;; TODO not implemented: action Settings
|
|
|
|
:handler nil}])]
|
2016-07-22 11:53:12 +03:00
|
|
|
[view (-> (st/actions-wrapper status-bar-height)
|
|
|
|
(merge (get-in styles [:components :actions-list-view])))
|
2016-05-07 16:39:03 +03:00
|
|
|
[view st/actions-separator]
|
2016-05-08 23:13:24 +03:00
|
|
|
[view st/actions-view
|
2016-05-06 14:06:58 +03:00
|
|
|
(for [action actions]
|
2016-07-22 11:53:12 +03:00
|
|
|
^{:key action} [action-view {:platform-specific platform-specific
|
|
|
|
:action action}])]])))
|
2016-05-02 16:56:42 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn actions-view [platform-specific]
|
2016-05-06 14:06:58 +03:00
|
|
|
[overlay {:on-click-outside #(dispatch [:set-show-actions false])}
|
2016-07-22 11:53:12 +03:00
|
|
|
[actions-list-view platform-specific]])
|
2016-05-03 16:39:08 +03:00
|
|
|
|
2016-08-04 18:36:13 +03:00
|
|
|
(defn online-text [contact chat-id]
|
|
|
|
(if contact
|
|
|
|
(if (> (get contact :last-online) 0)
|
|
|
|
(time/time-ago (time/to-date (get contact :last-online)))
|
|
|
|
(label :t/active-unknown))
|
|
|
|
(if (= chat-id "console")
|
|
|
|
(label :t/active-online)
|
|
|
|
(label :t/active-unknown))))
|
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn toolbar-content [platform-specific]
|
2016-08-04 18:36:13 +03:00
|
|
|
(let [{:keys [group-chat chat-id name contacts]} (subscribe [:chat-properties [:group-chat :chat-id :name :contacts]])
|
|
|
|
contact (subscribe [:get-in [:contacts @chat-id]])
|
2016-05-06 14:06:58 +03:00
|
|
|
show-actions (subscribe [:show-actions])]
|
|
|
|
(fn []
|
2016-05-11 13:53:38 +03:00
|
|
|
[view (st/chat-name-view @show-actions)
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/chat-name-text
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :medium}
|
2016-08-04 18:36:13 +03:00
|
|
|
(if (str/blank? @name)
|
|
|
|
(label :t/user-anonymous)
|
|
|
|
(truncate-str (or @name (label :t/chat-name)) 30))]
|
2016-05-11 13:53:38 +03:00
|
|
|
(if @group-chat
|
|
|
|
[view {:flexDirection :row}
|
|
|
|
[icon :group st/group-icon]
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/members
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :medium}
|
2016-05-20 14:38:20 +03:00
|
|
|
(let [cnt (inc (count @contacts))]
|
2016-06-01 14:44:04 +03:00
|
|
|
(label-pluralize cnt :t/members))]]
|
2016-07-22 11:53:12 +03:00
|
|
|
[text {:style st/last-activity
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:font :default}
|
2016-08-04 18:36:13 +03:00
|
|
|
(online-text @contact @chat-id)])])))
|
2016-05-11 13:53:38 +03:00
|
|
|
|
|
|
|
(defn toolbar-action []
|
|
|
|
(let [show-actions (subscribe [:show-actions])]
|
|
|
|
(fn []
|
|
|
|
(if @show-actions
|
|
|
|
[touchable-highlight
|
|
|
|
{:on-press #(dispatch [:set-show-actions false])}
|
2016-05-19 15:45:16 +03:00
|
|
|
[view st/action
|
2016-05-11 13:53:38 +03:00
|
|
|
[icon :up st/up-icon]]]
|
|
|
|
[touchable-highlight
|
|
|
|
{:on-press #(dispatch [:set-show-actions true])}
|
2016-05-19 15:45:16 +03:00
|
|
|
[view st/action
|
|
|
|
[chat-icon]]]))))
|
2016-05-11 13:53:38 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defn chat-toolbar [platform-specific]
|
|
|
|
(let [{:keys [group-chat name contacts]} (subscribe [:chat-properties [:group-chat :name :contacts]])
|
|
|
|
show-actions (subscribe [:show-actions])]
|
|
|
|
[view
|
|
|
|
[status-bar {:platform-specific platform-specific}]
|
|
|
|
[toolbar {:hide-nav? @show-actions
|
|
|
|
:custom-content [toolbar-content platform-specific]
|
|
|
|
:custom-action [toolbar-action]
|
|
|
|
:style (get-in platform-specific [:styles :components :toolbar])}]]))
|
2016-05-03 16:39:08 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defview messages-view [platform-specific group-chat]
|
2016-06-27 14:35:33 +03:00
|
|
|
[messages [:chat :messages]
|
|
|
|
contacts [:chat :contacts]]
|
|
|
|
(let [contacts' (contacts-by-identity contacts)]
|
2016-07-22 11:53:12 +03:00
|
|
|
[list-view {:renderRow (message-row {:contact-by-identity contacts'
|
|
|
|
:platform-specific platform-specific
|
|
|
|
:group-chat group-chat
|
|
|
|
:messages-count (count messages)})
|
2016-06-27 14:35:33 +03:00
|
|
|
:renderScrollComponent #(invertible-scroll-view (js->clj %))
|
|
|
|
:onEndReached #(dispatch [:load-more-messages])
|
|
|
|
:enableEmptySections true
|
|
|
|
:keyboardShouldPersistTaps true
|
|
|
|
:dataSource (to-datasource-inverted messages)}]))
|
2016-06-03 22:34:24 +03:00
|
|
|
|
2016-06-25 18:44:37 +03:00
|
|
|
(defn messages-container-animation-logic
|
2016-06-27 14:35:33 +03:00
|
|
|
[{:keys [offset val]}]
|
2016-06-03 22:34:24 +03:00
|
|
|
(fn [_]
|
2016-06-27 14:35:33 +03:00
|
|
|
(anim/start (anim/spring val {:toValue @offset}))))
|
2016-06-03 22:34:24 +03:00
|
|
|
|
|
|
|
(defn messages-container [messages]
|
2016-06-27 14:35:33 +03:00
|
|
|
(let [offset (subscribe [:messages-offset])
|
2016-06-25 18:44:37 +03:00
|
|
|
messages-offset (anim/create-value 0)
|
2016-06-27 14:35:33 +03:00
|
|
|
context {:offset offset
|
|
|
|
:val messages-offset}
|
2016-06-03 22:34:24 +03:00
|
|
|
on-update (messages-container-animation-logic context)]
|
|
|
|
(r/create-class
|
|
|
|
{:component-did-mount
|
|
|
|
on-update
|
|
|
|
:component-did-update
|
|
|
|
on-update
|
|
|
|
:reagent-render
|
|
|
|
(fn [messages]
|
2016-06-27 14:35:33 +03:00
|
|
|
@offset
|
2016-06-03 22:34:24 +03:00
|
|
|
[animated-view {:style (st/messages-container messages-offset)}
|
|
|
|
messages])})))
|
2016-04-19 15:38:16 +03:00
|
|
|
|
2016-07-22 11:53:12 +03:00
|
|
|
(defview chat [{platform-specific :platform-specific}]
|
2016-05-17 13:26:20 +03:00
|
|
|
[group-chat [:chat :group-chat]
|
2016-06-02 15:53:13 +03:00
|
|
|
show-actions-atom [:show-actions]
|
|
|
|
command [:get-chat-command]
|
2016-06-21 12:46:02 +03:00
|
|
|
command? [:command?]
|
2016-06-25 18:44:37 +03:00
|
|
|
suggestions [:get-suggestions]
|
|
|
|
to-msg-id [:get-chat-command-to-msg-id]
|
|
|
|
layout-height [:get :layout-height]]
|
|
|
|
[view {:style st/chat-view
|
2016-06-06 15:05:35 +03:00
|
|
|
:onLayout (fn [event]
|
|
|
|
(let [height (.. event -nativeEvent -layout -height)]
|
2016-06-25 18:44:37 +03:00
|
|
|
(when (not= height layout-height)
|
2016-06-30 15:54:44 +03:00
|
|
|
(dispatch [:set-layout-height height]))))}
|
2016-07-22 11:53:12 +03:00
|
|
|
[chat-toolbar platform-specific]
|
2016-06-03 22:34:24 +03:00
|
|
|
[messages-container
|
2016-07-22 11:53:12 +03:00
|
|
|
[messages-view platform-specific group-chat]]
|
|
|
|
(when group-chat [typing-all platform-specific])
|
2016-06-21 10:27:07 +03:00
|
|
|
[response-view]
|
2016-06-25 18:44:37 +03:00
|
|
|
(when-not command? [suggestion-container])
|
2016-07-22 11:53:12 +03:00
|
|
|
[chat-message-new platform-specific]
|
|
|
|
(when show-actions-atom [actions-view platform-specific])])
|