/
PostListFragment.kt
162 lines (133 loc) · 6.08 KB
/
PostListFragment.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
package com.google.firebase.quickstart.database.kotlin.fragment
import android.content.Intent
import android.os.Bundle
import android.util.Log
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.Fragment
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import com.firebase.ui.database.FirebaseRecyclerAdapter
import com.firebase.ui.database.FirebaseRecyclerOptions
import com.google.firebase.auth.FirebaseAuth
import com.google.firebase.database.DataSnapshot
import com.google.firebase.database.DatabaseError
import com.google.firebase.database.DatabaseReference
import com.google.firebase.database.FirebaseDatabase
import com.google.firebase.database.MutableData
import com.google.firebase.database.Query
import com.google.firebase.database.Transaction
import com.google.firebase.quickstart.database.R
import com.google.firebase.quickstart.database.kotlin.PostDetailActivity
import com.google.firebase.quickstart.database.kotlin.models.Post
import com.google.firebase.quickstart.database.kotlin.viewholder.PostViewHolder
import kotlinx.android.synthetic.main.fragment_all_posts.view.messagesList
abstract class PostListFragment : Fragment() {
// [START define_database_reference]
private lateinit var database: DatabaseReference
// [END define_database_reference]
private lateinit var recycler: RecyclerView
private lateinit var manager: LinearLayoutManager
private var adapter: FirebaseRecyclerAdapter<Post, PostViewHolder>? = null
val uid: String
get() = FirebaseAuth.getInstance().currentUser!!.uid
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
super.onCreateView(inflater, container, savedInstanceState)
val rootView = inflater.inflate(R.layout.fragment_all_posts, container, false)
// [START create_database_reference]
database = FirebaseDatabase.getInstance().reference
// [END create_database_reference]
recycler = rootView.findViewById(R.id.messagesList)
rootView.messagesList.setHasFixedSize(true)
return rootView
}
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
// Set up Layout Manager, reverse layout
manager = LinearLayoutManager(activity)
manager.reverseLayout = true
manager.stackFromEnd = true
recycler.layoutManager = manager
// Set up FirebaseRecyclerAdapter with the Query
val postsQuery = getQuery(database)
val options = FirebaseRecyclerOptions.Builder<Post>()
.setQuery(postsQuery, Post::class.java)
.build()
adapter = object : FirebaseRecyclerAdapter<Post, PostViewHolder>(options) {
override fun onCreateViewHolder(viewGroup: ViewGroup, i: Int): PostViewHolder {
val inflater = LayoutInflater.from(viewGroup.context)
return PostViewHolder(inflater.inflate(R.layout.item_post, viewGroup, false))
}
override fun onBindViewHolder(viewHolder: PostViewHolder, position: Int, model: Post) {
val postRef = getRef(position)
// Set click listener for the whole post view
val postKey = postRef.key
viewHolder.itemView.setOnClickListener {
// Launch PostDetailActivity
val intent = Intent(activity, PostDetailActivity::class.java)
intent.putExtra(PostDetailActivity.EXTRA_POST_KEY, postKey)
startActivity(intent)
}
// Determine if the current user has liked this post and set UI accordingly
viewHolder.setLikedState(model.stars.containsKey(uid))
// Bind Post to ViewHolder, setting OnClickListener for the star button
viewHolder.bindToPost(model, View.OnClickListener {
// Need to write to both places the post is stored
val globalPostRef = database.child("posts").child(postRef.key!!)
val userPostRef = database.child("user-posts").child(model.uid!!).child(postRef.key!!)
// Run two transactions
onStarClicked(globalPostRef)
onStarClicked(userPostRef)
})
}
}
recycler.adapter = adapter
}
// [START post_stars_transaction]
private fun onStarClicked(postRef: DatabaseReference) {
postRef.runTransaction(object : Transaction.Handler {
override fun doTransaction(mutableData: MutableData): Transaction.Result {
val p = mutableData.getValue(Post::class.java)
?: return Transaction.success(mutableData)
if (p.stars.containsKey(uid)) {
// Unstar the post and remove self from stars
p.starCount = p.starCount - 1
p.stars.remove(uid)
} else {
// Star the post and add self to stars
p.starCount = p.starCount + 1
p.stars[uid] = true
}
// Set value and report transaction success
mutableData.value = p
return Transaction.success(mutableData)
}
override fun onComplete(
databaseError: DatabaseError?,
b: Boolean,
dataSnapshot: DataSnapshot?
) {
// Transaction completed
Log.d(TAG, "postTransaction:onComplete:" + databaseError!!)
}
})
}
// [END post_stars_transaction]
override fun onStart() {
super.onStart()
adapter?.startListening()
}
override fun onStop() {
super.onStop()
adapter?.stopListening()
}
abstract fun getQuery(databaseReference: DatabaseReference): Query
companion object {
private const val TAG = "PostListFragment"
}
}