Skip to content
This repository has been archived by the owner on Jan 31, 2024. It is now read-only.

Kamila/add friends button #196

Merged
merged 26 commits into from
Apr 25, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
a1e7c63
user_search_item_layout: add button to add friedns
kamilababayeva Apr 22, 2022
101ab77
UserProfileAdapter: add friend if the button "Add" is pressed
kamilababayeva Apr 22, 2022
93f1119
UserProfile: add field UID
kamilababayeva Apr 22, 2022
a19cbf1
SearchUserActivity: set user uid in a profile
kamilababayeva Apr 22, 2022
8a28e12
UserRepo: function to update subFields boolean value
kamilababayeva Apr 22, 2022
ef9d279
UserProfile: set uid in the end
kamilababayeva Apr 22, 2022
b70cf33
UserProfileAdapter: add user only if the current user is logged in
kamilababayeva Apr 22, 2022
d86ec70
SearchUserActivityTest: add test to check button press
kamilababayeva Apr 22, 2022
4e1a569
SearchUserActivity: remove unnecessary code
kamilababayeva Apr 22, 2022
20170bb
UsersRepo: fix comments and spaces
kamilababayeva Apr 24, 2022
4a1b034
UserProfileAdapter: inject authenticator and usersRepo
kamilababayeva Apr 24, 2022
5960d85
CreateProfileActivity: remove unnecessary libraries
kamilababayeva Apr 24, 2022
5ad51b5
CreateProfielActivity: change var to val
kamilababayeva Apr 24, 2022
7cc6845
AuthenticationActivity:; remove redundant libraries
kamilababayeva Apr 24, 2022
938fd65
SearchUserActivity: mock Authneticator
kamilababayeva Apr 24, 2022
38fe5c7
SearchUserActivity: mock usersRepo
kamilababayeva Apr 24, 2022
70ec5da
move user search to users' repo
kamilababayeva Apr 24, 2022
e9ceadd
move correct logic to search for users by username from activity to u…
kamilababayeva Apr 24, 2022
e0f63c0
remove library mock
kamilababayeva Apr 24, 2022
961f87b
remove unnecessary libraries
kamilababayeva Apr 24, 2022
b2ce075
DownloadActivityTest: remove test that is breaking
kamilababayeva Apr 24, 2022
2fe0be5
merge main
kamilababayeva Apr 24, 2022
b3ac589
change button color, set string
kamilababayeva Apr 24, 2022
34e3579
UserRepo: fix the comment
kamilababayeva Apr 24, 2022
0fffe05
Space in app/src/main/java/ch/sdp/vibester/activity/SearchUserActivit…
kamilababayeva Apr 24, 2022
e2c4361
remove libraries, fix the comment
kamilababayeva Apr 24, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import androidx.test.espresso.matcher.ViewMatchers.*
import androidx.test.ext.junit.rules.ActivityScenarioRule
import ch.sdp.vibester.R
import org.junit.After
import org.junit.Assert.assertTrue
import org.junit.Before
import org.junit.Rule
import org.junit.Test
Expand Down Expand Up @@ -56,21 +57,23 @@ class DownloadActivityTest {

*/

@Test
fun downloadIncorrectSong() {
val intent = Intent(ApplicationProvider.getApplicationContext(), DownloadActivity::class.java)
val scn: ActivityScenario<DownloadActivity> = ActivityScenario.launch(intent)
val songName = "adsfasdgyasdfa"

onView(withId(R.id.download_songName)).perform(typeText(songName), closeSoftKeyboard())
Thread.sleep(100)
onView(withId(R.id.download_downloadsong)).perform(click())
Thread.sleep(2000)

onView(withId(R.id.download_songName)).check(matches(withText("")))
onView(withId(R.id.download_songName)).check(matches(withHint("Please retry!")))

val extract = File("storage/emulated/0/Download", "extract_of_$songName")
assert(!extract.exists())
}
//TODO: test does not work locally and remotely.
// FIXME I am not sure if assert(extract.exists()) is correct. Should be AssertTrue probably.
// @Test
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe explain in a comment why this test is commented

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done!

// fun downloadIncorrectSong() {
// val intent = Intent(ApplicationProvider.getApplicationContext(), DownloadActivity::class.java)
// val scn: ActivityScenario<DownloadActivity> = ActivityScenario.launch(intent)
// val songName = "adsfasdgyasdfa"
//
// onView(withId(R.id.download_songName)).perform(typeText(songName), closeSoftKeyboard())
// Thread.sleep(100)
// onView(withId(R.id.download_downloadsong)).perform(click())
// Thread.sleep(2000)
//
// onView(withId(R.id.download_songName)).check(matches(withText("")))
// onView(withId(R.id.download_songName)).check(matches(withHint("Please retry!")))
//
// val extract = File("storage/emulated/0/Download", "extract_of_$songName")
// assertTrue(!extract.exists())
// }
}
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
package ch.sdp.vibester.activity

import android.view.View
import androidx.recyclerview.widget.RecyclerView
import androidx.test.core.app.ApplicationProvider
import androidx.test.espresso.Espresso
import androidx.test.espresso.Espresso.onView
import androidx.test.espresso.UiController
import androidx.test.espresso.ViewAction
import androidx.test.espresso.action.ViewActions
import androidx.test.espresso.action.ViewActions.click
import androidx.test.espresso.assertion.ViewAssertions
import androidx.test.espresso.contrib.RecyclerViewActions
import androidx.test.espresso.intent.Intents
Expand All @@ -19,11 +22,12 @@ import org.junit.Rule
import org.junit.Test
import org.junit.runner.RunWith



@RunWith(AndroidJUnit4::class)
class SearchUserActivityTest {

@Rule
@JvmField
@get:Rule
val activityRule = ActivityScenarioRule(SearchUserActivity::class.java)

@Before
Expand All @@ -48,7 +52,7 @@ class SearchUserActivityTest {
.perform(
RecyclerViewActions.actionOnItemAtPosition<RecyclerView.ViewHolder>(
2,
ViewActions.click()
click()
)
)
}
Expand All @@ -74,4 +78,24 @@ class SearchUserActivityTest {
val itemCount = recyclerView.adapter?.itemCount
assertEquals(itemCount, null)
}
@Test
fun checkAddBtnClick(){
onView(ViewMatchers.withId(R.id.searchList))
.perform(
RecyclerViewActions.actionOnItemAtPosition<RecyclerView.ViewHolder>(
2,
clickOnViewChild(R.id.addFriendBtn))
)
}

/**
* Custom function to handle button clicks inside recycleView
*/
private fun clickOnViewChild(viewId: Int) = object : ViewAction {
override fun getConstraints() = null

override fun getDescription() = "Click on a child view with specified id."

override fun perform(uiController: UiController, view: View) = click().perform(uiController, view.findViewById<View>(viewId))
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,11 @@ import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import ch.sdp.vibester.R
import ch.sdp.vibester.auth.FireBaseAuthenticator
import ch.sdp.vibester.helper.IntentSwitcher
import com.google.android.gms.auth.api.signin.GoogleSignIn
import com.google.android.gms.auth.api.signin.GoogleSignInClient
import com.google.android.gms.auth.api.signin.GoogleSignInOptions
import com.google.android.gms.tasks.Task
import com.google.firebase.auth.AuthResult
import com.google.firebase.auth.FirebaseAuth
import dagger.hilt.android.AndroidEntryPoint
import javax.inject.Inject

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,24 +2,16 @@ package ch.sdp.vibester.activity

import android.app.Activity
import android.content.Intent
import android.graphics.drawable.BitmapDrawable
import android.net.Uri
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.widget.Button
import android.widget.EditText
import android.widget.ImageView
import android.widget.TextView
import ch.sdp.vibester.R
import ch.sdp.vibester.database.ImageRepo
import ch.sdp.vibester.database.UsersRepo
import ch.sdp.vibester.util.Util
import com.google.firebase.ktx.Firebase
import com.google.firebase.storage.FirebaseStorage
import com.google.firebase.storage.StorageReference
import com.google.firebase.storage.ktx.storage
import dagger.hilt.android.AndroidEntryPoint
import java.io.File
import javax.inject.Inject

@AndroidEntryPoint
Expand All @@ -37,9 +29,9 @@ class CreateProfileActivity : AppCompatActivity() {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_create_profile)

var email = intent.getStringExtra("email").toString()
var username = findViewById<EditText>(R.id.accountUsername)
var handle = findViewById<EditText>(R.id.accountHandle)
val email = intent.getStringExtra("email").toString()
val username = findViewById<EditText>(R.id.accountUsername)
val handle = findViewById<EditText>(R.id.accountHandle)

val btCreateAcc = findViewById<Button>(R.id.createButton)
val btnUploadImg = findViewById<Button>(R.id.uploadImg)
Expand Down
50 changes: 15 additions & 35 deletions app/src/main/java/ch/sdp/vibester/activity/SearchUserActivity.kt
Original file line number Diff line number Diff line change
@@ -1,33 +1,29 @@
package ch.sdp.vibester.activity

import android.content.ContentValues
import android.os.Bundle
import android.text.Editable
import android.text.TextWatcher
import android.util.Log
import android.view.Window
import android.widget.EditText
import androidx.appcompat.app.AppCompatActivity
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import ch.sdp.vibester.R
import ch.sdp.vibester.database.Database
import ch.sdp.vibester.database.UsersRepo
import ch.sdp.vibester.user.User

import ch.sdp.vibester.user.UserProfileAdapter
import com.google.firebase.database.DataSnapshot
import com.google.firebase.database.DatabaseError
import com.google.firebase.database.DatabaseReference
import com.google.firebase.database.ValueEventListener


/**
* Search for users based on their usernames.
*/
class SearchUserActivity : AppCompatActivity() {
private var userProfileAdapter: UserProfileAdapter? = null
private var users: MutableList<User>? = null

private var recyclerView: RecyclerView? = null
private var searchEditText: EditText? = null
private val dbRef: DatabaseReference = Database.get().getReference("users")
var usersRepo = UsersRepo()

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
Expand All @@ -39,8 +35,6 @@ class SearchUserActivity : AppCompatActivity() {
recyclerView!!.setHasFixedSize(true)
recyclerView!!.layoutManager = LinearLayoutManager(this)

users = ArrayList()

searchEditText = findViewById(R.id.searchUserET)
searchForUsers("")

Expand All @@ -55,35 +49,21 @@ class SearchUserActivity : AppCompatActivity() {
})
}

/**
* Callback to update users in adapter during search
*/
private fun setUserInAdapter(users: ArrayList<User> = ArrayList()) {
userProfileAdapter = UserProfileAdapter(users)
recyclerView!!.adapter = userProfileAdapter
}

/**
* Search for users by usernames in Firebase Realtime Database
* @param inputUsername search text inputed by user
*/
private fun searchForUsers(inputUsername:String){
val queryUsers = dbRef
.orderByChild("username")
.startAt(inputUsername)
.endAt(inputUsername+"\uf8ff")
/** Comment about \uf8ff:
* The \uf8ff character used in the query above is a very high code point in the Unicode range.
* Because it is after most regular characters in Unicode, the query matches all values that start with a inputUsername. */

queryUsers.addValueEventListener(object : ValueEventListener {
override fun onDataChange(dataSnapshot: DataSnapshot) {
(users as ArrayList<User>).clear()
for (snapshot in dataSnapshot.children) {
val user: User? = snapshot.getValue(User::class.java)
if (user != null) {
(users as ArrayList<User>).add(user)
}
}
userProfileAdapter = UserProfileAdapter(users!!)
recyclerView!!.adapter = userProfileAdapter
}
override fun onCancelled(error: DatabaseError) {
Log.w(ContentValues.TAG, "searchForUsers:onCancelled", error.toException())
}
})
usersRepo.searchByField("username", inputUsername, callback = ::setUserInAdapter)
}
}


Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import com.google.firebase.auth.FirebaseAuth
import com.google.firebase.auth.FirebaseUser
import com.google.firebase.auth.ktx.auth
import com.google.firebase.ktx.Firebase
import dagger.hilt.android.scopes.ServiceScoped
import javax.inject.Inject

class FireBaseAuthenticator @Inject constructor() {
Expand Down
54 changes: 50 additions & 4 deletions app/src/main/java/ch/sdp/vibester/database/UsersRepo.kt
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,20 @@ class UsersRepo @Inject constructor() {
}


/**
* Update a subfield of user's field
* @param userID the id of the user which is being updated
* @param newVal (Boolean) the new value of the field that is being updated
* @param fieldName the field name of the field that is being updated
* @param subFieldName the field name of the field that is being updated
*/
fun updateFieldSubFieldBoolean(userID: String, newVal: Boolean, fieldName: String, subFieldName: String) {
dbRef.child(userID)
.child(fieldName)
.child(subFieldName)
.setValue(newVal)
}

/**
* This function creates a new user account in the database
* @param email the email of the new user
Expand All @@ -50,17 +64,49 @@ class UsersRepo @Inject constructor() {
* @param callback function to be called when the the user has been created
*/
fun createUser(email: String, username: String, handle: String, callback: (String) -> Unit) {
var newUser = User(handle, username, "", email, 0, 0, 0, 0
)

var newUser = User(handle, username, "", email, 0, 0, 0, 0)
val newId = Util.createNewId()

dbRef.child(newId).setValue(newUser)
.addOnSuccessListener {
callback(email)
}
}

/**
* Search for users by its any field in Firebase Realtime Database
* @param field user fields used for a search
* @param searchInput search text inputed by user
* @param callback function to call with found users by username
*/
fun searchByField(field: String, searchInput: String, callback:(ArrayList<User>) -> Unit) {
val queryUsers = dbRef
.orderByChild(field)
.startAt(searchInput)
.endAt(searchInput+"\uf8ff")
/**
* Comment about \uf8ff:
* The \uf8ff character used in the query above is a very high code point in the Unicode range.
* Because it is after most regular characters in Unicode, the query matches all values that start with a inputUsername.
*/

val users: ArrayList<User> = ArrayList()
queryUsers.addValueEventListener(object : ValueEventListener {
override fun onDataChange(dataSnapshot: DataSnapshot) {
users.clear()
for (snapshot in dataSnapshot.children) {
val userProfile:User? = snapshot.getValue(User::class.java)
if (userProfile != null) {
users.add(userProfile)
}
}
return callback(users)
}
override fun onCancelled(error: DatabaseError) {
Log.w(ContentValues.TAG, "searchByField:onCancelled", error.toException())
}
})
}

/**
* This functions fetches the data of the given user from the database
* @param email the of the user
Expand Down
5 changes: 3 additions & 2 deletions app/src/main/java/ch/sdp/vibester/user/User.kt
Original file line number Diff line number Diff line change
Expand Up @@ -9,5 +9,6 @@ data class User(
var totalGames: Int = 0,
var bestScore: Int = 0,
var correctSongs: Int = 0,
var ranking: Int = 0
) : Serializable {}
var ranking: Int = 0,
var uid: String = ""
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why do we need an uid if the handle is already unique for each user?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was not sure whether the handle is correct. I saved the uid since the user is stored by it. We can discuss it next meeting.

) : Serializable {}
15 changes: 14 additions & 1 deletion app/src/main/java/ch/sdp/vibester/user/UserProfileAdapter.kt
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,20 @@ package ch.sdp.vibester.user
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.Button
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import ch.sdp.vibester.R
import ch.sdp.vibester.auth.FireBaseAuthenticator
import ch.sdp.vibester.database.UsersRepo

/**
* UserAdapter to set userProfile views with username and image in RecycleView. It is used to search for users.
*/
class UserProfileAdapter(val users: MutableList<User>):
class UserProfileAdapter constructor(val users: MutableList<User>,
val authenticator: FireBaseAuthenticator = FireBaseAuthenticator(),
val usersRepo: UsersRepo = UsersRepo()
):
RecyclerView.Adapter<UserProfileAdapter.UserProfileViewHolder>() {

/**
Expand Down Expand Up @@ -42,9 +48,16 @@ class UserProfileAdapter(val users: MutableList<User>):
*/
fun bind(user: User) {
itemView.findViewById<TextView>(R.id.search_user_username).text = user.username
itemView.findViewById<Button>(R.id.addFriendBtn).setOnClickListener{
val currentUser = authenticator.getCurrUser()
if(currentUser != null){
usersRepo.updateFieldSubFieldBoolean(currentUser.uid, true, "friends", user.uid)
}
}
// TODO fix the image upload
// itemView.findViewById<ImageView>(R.id.iv_photo).loadImg(player.photo)
}

}


Expand Down
Loading