├── app ├── .gitignore ├── src │ ├── main │ │ ├── res │ │ │ ├── mipmap-hdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-mdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xxhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xxxhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── drawable-hdpi │ │ │ │ └── ic_fire_emoji.png │ │ │ ├── drawable-mdpi │ │ │ │ └── ic_fire_emoji.png │ │ │ ├── drawable-xhdpi │ │ │ │ └── ic_fire_emoji.png │ │ │ ├── drawable-xxhdpi │ │ │ │ └── ic_fire_emoji.png │ │ │ ├── drawable-xxxhdpi │ │ │ │ └── ic_fire_emoji.png │ │ │ ├── values │ │ │ │ ├── colors.xml │ │ │ │ ├── dimens.xml │ │ │ │ ├── strings.xml │ │ │ │ └── styles.xml │ │ │ ├── mipmap-anydpi-v26 │ │ │ │ ├── ic_launcher.xml │ │ │ │ └── ic_launcher_round.xml │ │ │ ├── drawable │ │ │ │ ├── splash_screen_background.xml │ │ │ │ ├── ic_send_black_24dp.xml │ │ │ │ ├── rect_oval_white.xml │ │ │ │ ├── rect_round_white.xml │ │ │ │ ├── rect_round_primary_color.xml │ │ │ │ ├── ic_image_black_24dp.xml │ │ │ │ ├── ic_account_circle_black_24dp.xml │ │ │ │ ├── ic_people_black_24dp.xml │ │ │ │ └── ic_launcher_background.xml │ │ │ ├── menu │ │ │ │ └── navigation.xml │ │ │ ├── layout │ │ │ │ ├── fragment_people.xml │ │ │ │ ├── activity_main.xml │ │ │ │ ├── item_image_message.xml │ │ │ │ ├── item_text_message.xml │ │ │ │ ├── activity_sign_in.xml │ │ │ │ ├── item_person.xml │ │ │ │ ├── activity_chat.xml │ │ │ │ └── fragment_my_account.xml │ │ │ └── drawable-v24 │ │ │ │ └── ic_launcher_foreground.xml │ │ ├── java │ │ │ └── com │ │ │ │ └── resocoder │ │ │ │ └── firemessage │ │ │ │ ├── AppConstants.kt │ │ │ │ ├── model │ │ │ │ ├── ChatChannel.kt │ │ │ │ ├── User.kt │ │ │ │ ├── Message.kt │ │ │ │ ├── TextMessage.kt │ │ │ │ └── ImageMessage.kt │ │ │ │ ├── service │ │ │ │ ├── MyFirebaseMessagingService.kt │ │ │ │ └── MyFirebaseInstanceIDService.kt │ │ │ │ ├── SplashActivity.kt │ │ │ │ ├── glide │ │ │ │ └── FireMessageGlideModule.kt │ │ │ │ ├── recyclerview │ │ │ │ └── item │ │ │ │ │ ├── PersonItem.kt │ │ │ │ │ ├── TextMessageItem.kt │ │ │ │ │ ├── ImageMessageItem.kt │ │ │ │ │ └── MessageItem.kt │ │ │ │ ├── MainActivity.kt │ │ │ │ ├── util │ │ │ │ ├── StorageUtil.kt │ │ │ │ └── FirestoreUtil.kt │ │ │ │ ├── fragment │ │ │ │ ├── PeopleFragment.kt │ │ │ │ └── MyAccountFragment.kt │ │ │ │ ├── SignInActivity.kt │ │ │ │ └── ChatActivity.kt │ │ └── AndroidManifest.xml │ ├── test │ │ └── java │ │ │ └── com │ │ │ └── resocoder │ │ │ └── firemessage │ │ │ └── ExampleUnitTest.kt │ └── androidTest │ │ └── java │ │ └── com │ │ └── resocoder │ │ └── firemessage │ │ └── ExampleInstrumentedTest.kt ├── proguard-rules.pro ├── google-services.json └── build.gradle ├── settings.gradle ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── .gitignore ├── .idea ├── modules.xml ├── runConfigurations.xml ├── gradle.xml ├── misc.xml └── codeStyles │ └── Project.xml ├── gradle.properties ├── gradlew.bat └── gradlew /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app' 2 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/drawable-hdpi/ic_fire_emoji.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/drawable-hdpi/ic_fire_emoji.png -------------------------------------------------------------------------------- /app/src/main/res/drawable-mdpi/ic_fire_emoji.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/drawable-mdpi/ic_fire_emoji.png -------------------------------------------------------------------------------- /app/src/main/res/drawable-xhdpi/ic_fire_emoji.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/drawable-xhdpi/ic_fire_emoji.png -------------------------------------------------------------------------------- /app/src/main/res/drawable-xxhdpi/ic_fire_emoji.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/drawable-xxhdpi/ic_fire_emoji.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.iml 2 | .gradle 3 | /local.properties 4 | /.idea/workspace.xml 5 | /.idea/libraries 6 | .DS_Store 7 | /build 8 | /captures 9 | .externalNativeBuild 10 | -------------------------------------------------------------------------------- /app/src/main/res/drawable-xxxhdpi/ic_fire_emoji.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/drawable-xxxhdpi/ic_fire_emoji.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ResoCoder/firebase-firestore-chat-app/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/AppConstants.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage 2 | 3 | 4 | object AppConstants { 5 | const val USER_NAME = "USER_NAME" 6 | const val USER_ID = "USER_ID" 7 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/model/ChatChannel.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.model 2 | 3 | 4 | data class ChatChannel(val userIds: MutableList) { 5 | constructor() : this(mutableListOf()) 6 | } -------------------------------------------------------------------------------- /app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #F4511E 4 | #D84315 5 | #FFD600 6 | 7 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Thu Mar 29 21:19:27 CEST 2018 2 | distributionBase=GRADLE_USER_HOME 3 | distributionPath=wrapper/dists 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | distributionUrl=https\://services.gradle.org/distributions/gradle-4.4-all.zip 7 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/model/User.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.model 2 | 3 | 4 | data class User(val name: String, 5 | val bio: String, 6 | val profilePicturePath: String?, 7 | val registrationTokens: MutableList) { 8 | constructor(): this("", "", null, mutableListOf()) 9 | } -------------------------------------------------------------------------------- /app/src/main/res/drawable/splash_screen_background.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 9 | 10 | 11 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_send_black_24dp.xml: -------------------------------------------------------------------------------- 1 | 6 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/rect_oval_white.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/rect_round_white.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/model/Message.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.model 2 | 3 | import java.util.* 4 | 5 | 6 | object MessageType { 7 | const val TEXT = "TEXT" 8 | const val IMAGE = "IMAGE" 9 | } 10 | 11 | interface Message { 12 | val time: Date 13 | val senderId: String 14 | val recipientId: String 15 | val senderName: String 16 | val type: String 17 | } -------------------------------------------------------------------------------- /app/src/main/res/drawable/rect_round_primary_color.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | -------------------------------------------------------------------------------- /app/src/test/java/com/resocoder/firemessage/ExampleUnitTest.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage 2 | 3 | import org.junit.Test 4 | 5 | import org.junit.Assert.* 6 | 7 | /** 8 | * Example local unit test, which will execute on the development machine (host). 9 | * 10 | * See [testing documentation](http://d.android.com/tools/testing). 11 | */ 12 | class ExampleUnitTest { 13 | @Test 14 | fun addition_isCorrect() { 15 | assertEquals(4, 2 + 2) 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_image_black_24dp.xml: -------------------------------------------------------------------------------- 1 | 6 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/res/menu/navigation.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 8 | 9 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /app/src/main/res/layout/fragment_people.xml: -------------------------------------------------------------------------------- 1 | 6 | 7 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/model/TextMessage.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.model 2 | 3 | import java.util.* 4 | 5 | 6 | data class TextMessage(val text: String, 7 | override val time: Date, 8 | override val senderId: String, 9 | override val recipientId: String, 10 | override val senderName: String, 11 | override val type: String = MessageType.TEXT) 12 | : Message { 13 | constructor() : this("", Date(0), "", "", "") 14 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/model/ImageMessage.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.model 2 | 3 | import java.util.* 4 | 5 | 6 | data class ImageMessage(val imagePath: String, 7 | override val time: Date, 8 | override val senderId: String, 9 | override val recipientId: String, 10 | override val senderName: String, 11 | override val type: String = MessageType.IMAGE) 12 | : Message { 13 | constructor() : this("", Date(0), "", "", "") 14 | } -------------------------------------------------------------------------------- /app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | FireMessage 3 | Home 4 | Dashboard 5 | Notifications 6 | Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. 7 | 8 | 9 | Hello blank fragment 10 | 11 | -------------------------------------------------------------------------------- /.idea/runConfigurations.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 11 | 12 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_account_circle_black_24dp.xml: -------------------------------------------------------------------------------- 1 | 6 | 9 | 10 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/service/MyFirebaseMessagingService.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.service 2 | 3 | import android.util.Log 4 | import com.google.firebase.messaging.FirebaseMessagingService 5 | import com.google.firebase.messaging.RemoteMessage 6 | 7 | 8 | class MyFirebaseMessagingService : FirebaseMessagingService() { 9 | 10 | override fun onMessageReceived(remoteMessage: RemoteMessage) { 11 | if (remoteMessage.notification != null) { 12 | //TODO: Show notification if we're not online 13 | Log.d("FCM", remoteMessage.data.toString()) 14 | } 15 | } 16 | } -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 10 | 11 | 14 | 15 | 16 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/SplashActivity.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage 2 | 3 | import android.support.v7.app.AppCompatActivity 4 | import android.os.Bundle 5 | import com.google.firebase.auth.FirebaseAuth 6 | import org.jetbrains.anko.startActivity 7 | 8 | class SplashActivity : AppCompatActivity() { 9 | 10 | override fun onCreate(savedInstanceState: Bundle?) { 11 | super.onCreate(savedInstanceState) 12 | 13 | if (FirebaseAuth.getInstance().currentUser == null) 14 | startActivity() 15 | else 16 | startActivity() 17 | finish() 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /app/src/main/res/drawable/ic_people_black_24dp.xml: -------------------------------------------------------------------------------- 1 | 6 | 9 | 10 | -------------------------------------------------------------------------------- /.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 17 | 18 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/glide/FireMessageGlideModule.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.glide 2 | 3 | import android.content.Context 4 | import com.bumptech.glide.Glide 5 | import com.bumptech.glide.Registry 6 | import com.bumptech.glide.annotation.GlideModule 7 | import com.bumptech.glide.module.AppGlideModule 8 | import com.firebase.ui.storage.images.FirebaseImageLoader 9 | import com.google.firebase.storage.StorageReference 10 | import java.io.InputStream 11 | 12 | 13 | @GlideModule 14 | class FireMessageGlideModule: AppGlideModule() { 15 | override fun registerComponents(context: Context, glide: Glide, registry: Registry) { 16 | registry.append(StorageReference::class.java, InputStream::class.java, 17 | FirebaseImageLoader.Factory()) 18 | } 19 | } -------------------------------------------------------------------------------- /gradle.properties: -------------------------------------------------------------------------------- 1 | # Project-wide Gradle settings. 2 | 3 | # IDE (e.g. Android Studio) users: 4 | # Gradle settings configured through the IDE *will override* 5 | # any settings specified in this file. 6 | 7 | # For more details on how to configure your build environment visit 8 | # http://www.gradle.org/docs/current/userguide/build_environment.html 9 | 10 | # Specifies the JVM arguments used for the daemon process. 11 | # The setting is particularly useful for tweaking memory settings. 12 | org.gradle.jvmargs=-Xmx1536m 13 | 14 | # When configured, Gradle will run in incubating parallel mode. 15 | # This option should only be used with decoupled projects. More details, visit 16 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 17 | # org.gradle.parallel=true 18 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/resocoder/firemessage/ExampleInstrumentedTest.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage 2 | 3 | import android.support.test.InstrumentationRegistry 4 | import android.support.test.runner.AndroidJUnit4 5 | 6 | import org.junit.Test 7 | import org.junit.runner.RunWith 8 | 9 | import org.junit.Assert.* 10 | 11 | /** 12 | * Instrumented test, which will execute on an Android device. 13 | * 14 | * See [testing documentation](http://d.android.com/tools/testing). 15 | */ 16 | @RunWith(AndroidJUnit4::class) 17 | class ExampleInstrumentedTest { 18 | @Test 19 | fun useAppContext() { 20 | // Context of the app under test. 21 | val appContext = InstrumentationRegistry.getTargetContext() 22 | assertEquals("com.resocoder.firemessage", appContext.packageName) 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # You can control the set of applied configuration files using the 3 | # proguardFiles setting in build.gradle. 4 | # 5 | # For more details, see 6 | # http://developer.android.com/guide/developing/tools/proguard.html 7 | 8 | # If your project uses WebView with JS, uncomment the following 9 | # and specify the fully qualified class name to the JavaScript interface 10 | # class: 11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 | # public *; 13 | #} 14 | 15 | # Uncomment this to preserve the line number information for 16 | # debugging stack traces. 17 | #-keepattributes SourceFile,LineNumberTable 18 | 19 | # If you keep the line number information, uncomment this to 20 | # hide the original source file name. 21 | #-renamesourcefileattribute SourceFile 22 | -------------------------------------------------------------------------------- /app/google-services.json: -------------------------------------------------------------------------------- 1 | { 2 | "project_info": { 3 | "project_number": "277485433091", 4 | "firebase_url": "https://firemessage-db177.firebaseio.com", 5 | "project_id": "firemessage-db177", 6 | "storage_bucket": "firemessage-db177.appspot.com" 7 | }, 8 | "client": [ 9 | { 10 | "client_info": { 11 | "mobilesdk_app_id": "1:277485433091:android:ba89bfa487b92988", 12 | "android_client_info": { 13 | "package_name": "com.resocoder.firemessage" 14 | } 15 | }, 16 | "oauth_client": [ 17 | { 18 | "client_id": "277485433091-ijnbcps9nv94spj9p1sk153774rt4432.apps.googleusercontent.com", 19 | "client_type": 3 20 | } 21 | ], 22 | "api_key": [ 23 | { 24 | "current_key": "AIzaSyBx9HueAK6SYyaxXDKULgpecQJuMFTNivM" 25 | } 26 | ], 27 | "services": { 28 | "analytics_service": { 29 | "status": 1 30 | }, 31 | "appinvite_service": { 32 | "status": 1, 33 | "other_platform_oauth_client": [] 34 | }, 35 | "ads_service": { 36 | "status": 2 37 | } 38 | } 39 | } 40 | ], 41 | "configuration_version": "1" 42 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/service/MyFirebaseInstanceIDService.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.service 2 | 3 | import com.google.firebase.auth.FirebaseAuth 4 | import com.google.firebase.iid.FirebaseInstanceId 5 | import com.google.firebase.iid.FirebaseInstanceIdService 6 | import com.resocoder.firemessage.util.FirestoreUtil 7 | 8 | 9 | class MyFirebaseInstanceIDService : FirebaseInstanceIdService() { 10 | 11 | override fun onTokenRefresh() { 12 | val newRegistrationToken = FirebaseInstanceId.getInstance().token 13 | 14 | if (FirebaseAuth.getInstance().currentUser != null) 15 | addTokenToFirestore(newRegistrationToken) 16 | } 17 | 18 | companion object { 19 | fun addTokenToFirestore(newRegistrationToken: String?) { 20 | if (newRegistrationToken == null) throw NullPointerException("FCM token is null.") 21 | 22 | FirestoreUtil.getFCMRegistrationTokens { tokens -> 23 | if (tokens.contains(newRegistrationToken)) 24 | return@getFCMRegistrationTokens 25 | 26 | tokens.add(newRegistrationToken) 27 | FirestoreUtil.setFCMRegistrationTokens(tokens) 28 | } 29 | } 30 | } 31 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/recyclerview/item/PersonItem.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.recyclerview.item 2 | 3 | import android.content.Context 4 | import com.resocoder.firemessage.R 5 | import com.resocoder.firemessage.glide.GlideApp 6 | import com.resocoder.firemessage.model.User 7 | import com.resocoder.firemessage.util.StorageUtil 8 | import com.xwray.groupie.kotlinandroidextensions.Item 9 | import com.xwray.groupie.kotlinandroidextensions.ViewHolder 10 | import kotlinx.android.synthetic.main.item_person.* 11 | 12 | 13 | class PersonItem(val person: User, 14 | val userId: String, 15 | private val context: Context) 16 | : Item() { 17 | 18 | override fun bind(viewHolder: ViewHolder, position: Int) { 19 | viewHolder.textView_name.text = person.name 20 | viewHolder.textView_bio.text = person.bio 21 | if (person.profilePicturePath != null) 22 | GlideApp.with(context) 23 | .load(StorageUtil.pathToReference(person.profilePicturePath)) 24 | .placeholder(R.drawable.ic_account_circle_black_24dp) 25 | .into(viewHolder.imageView_profile_picture) 26 | } 27 | 28 | override fun getLayout() = R.layout.item_person 29 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/recyclerview/item/TextMessageItem.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.recyclerview.item 2 | 3 | import android.content.Context 4 | import com.resocoder.firemessage.R 5 | import com.resocoder.firemessage.model.TextMessage 6 | import com.xwray.groupie.kotlinandroidextensions.ViewHolder 7 | import kotlinx.android.synthetic.main.item_text_message.* 8 | 9 | 10 | class TextMessageItem(val message: TextMessage, 11 | val context: Context) 12 | : MessageItem(message) { 13 | override fun bind(viewHolder: ViewHolder, position: Int) { 14 | viewHolder.textView_message_text.text = message.text 15 | super.bind(viewHolder, position) 16 | } 17 | 18 | override fun getLayout() = R.layout.item_text_message 19 | 20 | override fun isSameAs(other: com.xwray.groupie.Item<*>?): Boolean { 21 | if (other !is TextMessageItem) 22 | return false 23 | if (this.message != other.message) 24 | return false 25 | return true 26 | } 27 | 28 | override fun equals(other: Any?): Boolean { 29 | return isSameAs(other as? TextMessageItem) 30 | } 31 | 32 | override fun hashCode(): Int { 33 | var result = message.hashCode() 34 | result = 31 * result + context.hashCode() 35 | return result 36 | } 37 | 38 | } -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/MainActivity.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage 2 | 3 | import android.annotation.SuppressLint 4 | import android.os.Bundle 5 | import android.support.design.widget.BottomNavigationView 6 | import android.support.v4.app.Fragment 7 | import android.support.v7.app.AppCompatActivity 8 | import com.resocoder.firemessage.fragment.MyAccountFragment 9 | import com.resocoder.firemessage.fragment.PeopleFragment 10 | import kotlinx.android.synthetic.main.activity_main.* 11 | 12 | class MainActivity : AppCompatActivity() { 13 | 14 | override fun onCreate(savedInstanceState: Bundle?) { 15 | super.onCreate(savedInstanceState) 16 | setContentView(R.layout.activity_main) 17 | 18 | replaceFragment(PeopleFragment()) 19 | 20 | navigation.setOnNavigationItemSelectedListener { 21 | when (it.itemId) { 22 | R.id.navigation_people -> { 23 | replaceFragment(PeopleFragment()) 24 | true 25 | } 26 | R.id.navigation_my_account -> { 27 | replaceFragment(MyAccountFragment()) 28 | true 29 | } 30 | else -> false 31 | } 32 | } 33 | } 34 | 35 | private fun replaceFragment(fragment: Fragment) { 36 | supportFragmentManager.beginTransaction() 37 | .replace(R.id.fragment_layout, fragment) 38 | .commit() 39 | } 40 | } 41 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/util/StorageUtil.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.util 2 | 3 | import com.google.firebase.auth.FirebaseAuth 4 | import com.google.firebase.storage.FirebaseStorage 5 | import com.google.firebase.storage.StorageReference 6 | import java.util.* 7 | 8 | 9 | object StorageUtil { 10 | private val storageInstance: FirebaseStorage by lazy { FirebaseStorage.getInstance() } 11 | 12 | private val currentUserRef: StorageReference 13 | get() = storageInstance.reference 14 | .child(FirebaseAuth.getInstance().currentUser?.uid 15 | ?: throw NullPointerException("UID is null.")) 16 | 17 | fun uploadProfilePhoto(imageBytes: ByteArray, 18 | onSuccess: (imagePath: String) -> Unit) { 19 | val ref = currentUserRef.child("profilePictures/${UUID.nameUUIDFromBytes(imageBytes)}") 20 | ref.putBytes(imageBytes) 21 | .addOnSuccessListener { 22 | onSuccess(ref.path) 23 | } 24 | } 25 | 26 | fun uploadMessageImage(imageBytes: ByteArray, 27 | onSuccess: (imagePath: String) -> Unit) { 28 | val ref = currentUserRef.child("messages/${UUID.nameUUIDFromBytes(imageBytes)}") 29 | ref.putBytes(imageBytes) 30 | .addOnSuccessListener { 31 | onSuccess(ref.path) 32 | } 33 | } 34 | 35 | fun pathToReference(path: String) = storageInstance.getReference(path) 36 | } -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 20 | 21 | 32 | 33 | 34 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/recyclerview/item/ImageMessageItem.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.recyclerview.item 2 | 3 | import android.content.Context 4 | import com.resocoder.firemessage.R 5 | import com.resocoder.firemessage.glide.GlideApp 6 | import com.resocoder.firemessage.model.ImageMessage 7 | import com.resocoder.firemessage.util.StorageUtil 8 | import com.xwray.groupie.kotlinandroidextensions.ViewHolder 9 | import kotlinx.android.synthetic.main.item_image_message.* 10 | 11 | 12 | class ImageMessageItem(val message: ImageMessage, 13 | val context: Context) 14 | : MessageItem(message) { 15 | 16 | override fun bind(viewHolder: ViewHolder, position: Int) { 17 | super.bind(viewHolder, position) 18 | GlideApp.with(context) 19 | .load(StorageUtil.pathToReference(message.imagePath)) 20 | .placeholder(R.drawable.ic_image_black_24dp) 21 | .into(viewHolder.imageView_message_image) 22 | } 23 | 24 | override fun getLayout() = R.layout.item_image_message 25 | 26 | override fun isSameAs(other: com.xwray.groupie.Item<*>?): Boolean { 27 | if (other !is ImageMessageItem) 28 | return false 29 | if (this.message != other.message) 30 | return false 31 | return true 32 | } 33 | 34 | override fun equals(other: Any?): Boolean { 35 | return isSameAs(other as? ImageMessageItem) 36 | } 37 | 38 | override fun hashCode(): Int { 39 | var result = message.hashCode() 40 | result = 31 * result + context.hashCode() 41 | return result 42 | } 43 | } -------------------------------------------------------------------------------- /app/src/main/res/layout/item_image_message.xml: -------------------------------------------------------------------------------- 1 | 2 | 11 | 17 | 27 | 36 | 37 | -------------------------------------------------------------------------------- /.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 16 | 26 | 27 | 28 | 29 | 30 | 31 | 33 | -------------------------------------------------------------------------------- /app/src/main/res/layout/item_text_message.xml: -------------------------------------------------------------------------------- 1 | 2 | 11 | 17 | 28 | 37 | 38 | -------------------------------------------------------------------------------- /.idea/codeStyles/Project.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 15 | 16 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | -------------------------------------------------------------------------------- /app/src/main/java/com/resocoder/firemessage/recyclerview/item/MessageItem.kt: -------------------------------------------------------------------------------- 1 | package com.resocoder.firemessage.recyclerview.item 2 | 3 | import android.view.Gravity 4 | import android.widget.FrameLayout 5 | import com.google.firebase.auth.FirebaseAuth 6 | import com.resocoder.firemessage.R 7 | import com.resocoder.firemessage.model.Message 8 | import com.xwray.groupie.kotlinandroidextensions.Item 9 | import com.xwray.groupie.kotlinandroidextensions.ViewHolder 10 | import kotlinx.android.synthetic.main.item_text_message.* 11 | import org.jetbrains.anko.backgroundResource 12 | import org.jetbrains.anko.wrapContent 13 | import java.text.SimpleDateFormat 14 | 15 | 16 | abstract class MessageItem(private val message: Message) 17 | : Item() { 18 | 19 | override fun bind(viewHolder: ViewHolder, position: Int) { 20 | setTimeText(viewHolder) 21 | setMessageRootGravity(viewHolder) 22 | } 23 | 24 | private fun setTimeText(viewHolder: ViewHolder) { 25 | val dateFormat = SimpleDateFormat 26 | .getDateTimeInstance(SimpleDateFormat.SHORT, SimpleDateFormat.SHORT) 27 | viewHolder.textView_message_time.text = dateFormat.format(message.time) 28 | } 29 | 30 | private fun setMessageRootGravity(viewHolder: ViewHolder) { 31 | if (message.senderId == FirebaseAuth.getInstance().currentUser?.uid) { 32 | viewHolder.message_root.apply { 33 | backgroundResource = R.drawable.rect_round_white 34 | val lParams = FrameLayout.LayoutParams(wrapContent, wrapContent, Gravity.END) 35 | this.layoutParams = lParams 36 | } 37 | } 38 | else { 39 | viewHolder.message_root.apply { 40 | backgroundResource = R.drawable.rect_round_primary_color 41 | val lParams = FrameLayout.LayoutParams(wrapContent, wrapContent, Gravity.START) 42 | this.layoutParams = lParams 43 | } 44 | } 45 | } 46 | } -------------------------------------------------------------------------------- /app/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 7 | 12 | 13 | 19 | 22 | 25 | 26 | 27 | 28 | 34 | 35 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_sign_in.xml: -------------------------------------------------------------------------------- 1 | 2 | 9 | 10 | 24 | 25 |