1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License
15  */
16 
17 package com.android.internal.widget;
18 
19 import android.app.ActivityManager;
20 import android.app.Notification;
21 import android.view.View;
22 
23 import java.util.Objects;
24 
25 /**
26  * A message of a {@link MessagingLayout}.
27  */
28 public interface MessagingMessage extends MessagingLinearLayout.MessagingChild {
29 
30     /**
31      * Prefix for supported image MIME types
32      **/
33     String IMAGE_MIME_TYPE_PREFIX = "image/";
34 
createMessage(MessagingLayout layout, Notification.MessagingStyle.Message m)35     static MessagingMessage createMessage(MessagingLayout layout,
36             Notification.MessagingStyle.Message m) {
37         if (hasImage(m) && !ActivityManager.isLowRamDeviceStatic()) {
38             return MessagingImageMessage.createMessage(layout, m);
39         } else {
40             return MessagingTextMessage.createMessage(layout, m);
41         }
42     }
43 
dropCache()44     static void dropCache() {
45         MessagingTextMessage.dropCache();
46         MessagingImageMessage.dropCache();
47     }
48 
hasImage(Notification.MessagingStyle.Message m)49     static boolean hasImage(Notification.MessagingStyle.Message m) {
50         return m.getDataUri() != null
51                 && m.getDataMimeType() != null
52                 && m.getDataMimeType().startsWith(IMAGE_MIME_TYPE_PREFIX);
53     }
54 
55     /**
56      * Set a message for this view.
57      * @return true if setting the message worked
58      */
setMessage(Notification.MessagingStyle.Message message)59     default boolean setMessage(Notification.MessagingStyle.Message message) {
60         getState().setMessage(message);
61         return true;
62     }
63 
getMessage()64     default Notification.MessagingStyle.Message getMessage() {
65         return getState().getMessage();
66     }
67 
sameAs(Notification.MessagingStyle.Message message)68     default boolean sameAs(Notification.MessagingStyle.Message message) {
69         Notification.MessagingStyle.Message ownMessage = getMessage();
70         if (!Objects.equals(message.getText(), ownMessage.getText())) {
71             return false;
72         }
73         if (!Objects.equals(message.getSender(), ownMessage.getSender())) {
74             return false;
75         }
76         boolean hasRemoteInputHistoryChanged = message.isRemoteInputHistory()
77                 != ownMessage.isRemoteInputHistory();
78         // When the remote input history has changed, we want to regard messages equal even when
79         // the timestamp changes. The main reason is that the message that the system inserts
80         // will have a different time set than the one that the app will update us with and we
81         // still want to reuse that message.
82         if (!hasRemoteInputHistoryChanged
83                 && !Objects.equals(message.getTimestamp(), ownMessage.getTimestamp())) {
84             return false;
85         }
86         if (!Objects.equals(message.getDataMimeType(), ownMessage.getDataMimeType())) {
87             return false;
88         }
89         if (!Objects.equals(message.getDataUri(), ownMessage.getDataUri())) {
90             return false;
91         }
92         return true;
93     }
94 
sameAs(MessagingMessage message)95     default boolean sameAs(MessagingMessage message) {
96         return sameAs(message.getMessage());
97     }
98 
removeMessage()99     default void removeMessage() {
100         getGroup().removeMessage(this);
101     }
102 
setMessagingGroup(MessagingGroup group)103     default void setMessagingGroup(MessagingGroup group) {
104         getState().setGroup(group);
105     }
106 
setIsHistoric(boolean isHistoric)107     default void setIsHistoric(boolean isHistoric) {
108         getState().setIsHistoric(isHistoric);
109     }
110 
getGroup()111     default MessagingGroup getGroup() {
112         return getState().getGroup();
113     }
114 
setIsHidingAnimated(boolean isHiding)115     default void setIsHidingAnimated(boolean isHiding) {
116         getState().setIsHidingAnimated(isHiding);
117     }
118 
119     @Override
isHidingAnimated()120     default boolean isHidingAnimated() {
121         return getState().isHidingAnimated();
122     }
123 
124     @Override
hideAnimated()125     default void hideAnimated() {
126         setIsHidingAnimated(true);
127         getGroup().performRemoveAnimation(getView(), () -> setIsHidingAnimated(false));
128     }
129 
hasOverlappingRendering()130     default boolean hasOverlappingRendering() {
131         return false;
132     }
133 
recycle()134     default void recycle() {
135         getState().recycle();
136     }
137 
getView()138     default View getView() {
139         return (View) this;
140     }
141 
setColor(int textColor)142     default void setColor(int textColor) {}
143 
getState()144     MessagingMessageState getState();
145 
setVisibility(int visibility)146     void setVisibility(int visibility);
147 
getVisibility()148     int getVisibility();
149 }
150