From 8ae483b5608fdf6c2bc64bc816b2613e04768933 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Mon, 15 Jul 2024 14:01:05 +0900 Subject: [PATCH 01/19] =?UTF-8?q?Rename[]:=20=EC=9D=B4=EB=8F=99=203?= =?UTF-8?q?=EC=A3=BC=EC=B0=A8=20=EA=B3=BC=EC=A0=9C=20=EB=82=B4=EC=9A=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/build.gradle.kts | 18 ++- app/src/main/AndroidManifest.xml | 8 +- .../tech/kakao/map/Adapter/DocumentAdapter.kt | 56 +++++++++ .../tech/kakao/map/Adapter/WordAdapter.kt | 53 +++++++++ .../kakao/map/DBHelper/SearchWordDbHelper.kt | 110 ++++++++++++++++++ .../campus/tech/kakao/map/DTO/Document.kt | 22 ++++ .../java/campus/tech/kakao/map/DTO/Meta.kt | 14 +++ .../tech/kakao/map/DTO/PlaceResponse.kt | 6 + .../campus/tech/kakao/map/DTO/SameName.kt | 7 ++ .../campus/tech/kakao/map/DTO/SearchWord.kt | 3 + .../tech/kakao/map/DTO/SearchWordContract.kt | 11 ++ .../campus/tech/kakao/map/DTO/UrlContract.kt | 8 ++ .../campus/tech/kakao/map/MainViewModel.kt | 42 +++++++ .../java/campus/tech/kakao/map/MapActivity.kt | 54 +++++++++ .../campus/tech/kakao/map/MyApplication.kt | 11 ++ .../campus/tech/kakao/map/RetrofitData.kt | 54 +++++++++ .../campus/tech/kakao/map/RetrofitService.kt | 15 +++ .../campus/tech/kakao/map/SearchActivity.kt | 86 ++++++++++++++ app/src/main/res/drawable/magnifier.png | Bin 0 -> 9712 bytes app/src/main/res/drawable/marker.png | Bin 0 -> 16096 bytes app/src/main/res/drawable/x.png | Bin 0 -> 3272 bytes app/src/main/res/layout/activity_map.xml | 36 ++++++ app/src/main/res/layout/activity_search.xml | 61 ++++++++++ app/src/main/res/layout/place_item.xml | 47 ++++++++ app/src/main/res/layout/word_item.xml | 23 ++++ app/src/main/res/values/strings.xml | 2 + .../main/res/xml/network_security_config.xml | 6 + settings.gradle.kts | 1 + 28 files changed, 752 insertions(+), 2 deletions(-) create mode 100644 app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/Document.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/MainViewModel.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/MapActivity.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/MyApplication.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/RetrofitData.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/RetrofitService.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/SearchActivity.kt create mode 100644 app/src/main/res/drawable/magnifier.png create mode 100644 app/src/main/res/drawable/marker.png create mode 100644 app/src/main/res/drawable/x.png create mode 100644 app/src/main/res/layout/activity_map.xml create mode 100644 app/src/main/res/layout/activity_search.xml create mode 100644 app/src/main/res/layout/place_item.xml create mode 100644 app/src/main/res/layout/word_item.xml create mode 100644 app/src/main/res/xml/network_security_config.xml diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 33c4ca53..9b4327c9 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -1,3 +1,5 @@ +import com.android.build.gradle.internal.cxx.configure.gradleLocalProperties + plugins { id("com.android.application") id("org.jetbrains.kotlin.android") @@ -15,6 +17,14 @@ android { versionName = "1.0" testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + buildConfigField("String", "KAKAO_API_KEY", getApiKey("KAKAO_API_KEY")) + buildConfigField("String", "KAKAO_REST_API_KEY", getApiKey("KAKAO_REST_API_KEY")) + ndk { + abiFilters.add("arm64-v8a") + abiFilters.add("armeabi-v7a") + abiFilters.add("x86") + abiFilters.add("x86_64") + } } buildTypes { @@ -36,19 +46,24 @@ android { buildFeatures { viewBinding = true + buildConfig = true } } dependencies { - + val lifecycle_version = "2.8.3" + implementation("androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version") + implementation("androidx.lifecycle:lifecycle-livedata-ktx:$lifecycle_version") implementation("androidx.core:core-ktx:1.12.0") implementation("androidx.appcompat:appcompat:1.6.1") implementation("com.google.android.material:material:1.11.0") implementation("androidx.constraintlayout:constraintlayout:2.1.4") implementation("androidx.recyclerview:recyclerview:1.3.2") + implementation("androidx.datastore:datastore-preferences:1.0.0") implementation("com.squareup.retrofit2:retrofit:2.11.0") implementation("com.squareup.retrofit2:converter-gson:2.11.0") implementation("com.kakao.maps.open:android:2.9.5") + implementation("com.kakao.sdk:v2-all:2.20.3") implementation("androidx.activity:activity:1.8.0") implementation("androidx.test:core-ktx:1.5.0") testImplementation("junit:junit:4.13.2") @@ -61,3 +76,4 @@ dependencies { androidTestImplementation("androidx.test:rules:1.5.0") androidTestImplementation("androidx.test.espresso:espresso-intents:3.5.1") } +fun getApiKey(key: String): String = gradleLocalProperties(rootDir, providers).getProperty(key) \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 6bca2f54..ec1a5ec1 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -2,18 +2,22 @@ + + @@ -21,6 +25,8 @@ + diff --git a/app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt b/app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt new file mode 100644 index 00000000..74492c58 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt @@ -0,0 +1,56 @@ +package campus.tech.kakao.map.Adapter + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.TextView +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import androidx.recyclerview.widget.RecyclerView +import campus.tech.kakao.map.DTO.Document +import campus.tech.kakao.map.R + +class DocumentAdapter( + val addWord: (Document) -> Unit +): ListAdapter( + object : DiffUtil.ItemCallback(){ + override fun areItemsTheSame(oldItem: Document, newItem: Document): Boolean { + return oldItem === newItem + } + + override fun areContentsTheSame(oldItem: Document, newItem: Document): Boolean { + return oldItem == newItem + } + + } +) { + private var placeClicked = { position:Int -> + val document: Document = getItem(position) + addWord(document) + } + inner class ViewHolder( + itemView: View + ): RecyclerView.ViewHolder(itemView) { + val name:TextView = itemView.findViewById(R.id.name) + val address:TextView = itemView.findViewById(R.id.address) + val type:TextView = itemView.findViewById(R.id.type) + init { + itemView.setOnClickListener { + placeClicked(bindingAdapterPosition) + } + } + } + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { + return ViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.place_item, parent, false)) + } + + override fun onBindViewHolder(holder: ViewHolder, position: Int) { + val document: Document = getItem(position) + holder.name.text = document.placeName + holder.address.text = document.addressName + holder.type.text = document.categoryGroupName + } +} + + diff --git a/app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt b/app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt new file mode 100644 index 00000000..9b051e3d --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt @@ -0,0 +1,53 @@ +package campus.tech.kakao.map.Adapter + +import android.view.LayoutInflater +import android.view.View +import android.view.ViewGroup +import android.widget.ImageView +import android.widget.TextView +import androidx.recyclerview.widget.DiffUtil +import androidx.recyclerview.widget.ListAdapter +import androidx.recyclerview.widget.RecyclerView +import campus.tech.kakao.map.R +import campus.tech.kakao.map.DTO.SearchWord + +class WordAdapter( + val deleteWord: (SearchWord) -> Unit +): ListAdapter( + object : DiffUtil.ItemCallback(){ + override fun areItemsTheSame(oldItem: SearchWord, newItem: SearchWord): Boolean { + return oldItem === newItem + } + + override fun areContentsTheSame(oldItem: SearchWord, newItem: SearchWord): Boolean { + return oldItem == newItem + } + + } +) { + inner class ViewHolder( + itemView: View + ): RecyclerView.ViewHolder(itemView) { + val searchWord: TextView = itemView.findViewById(R.id.search_word) + val delete: ImageView = itemView.findViewById(R.id.x) + init { + delete.setOnClickListener { + deletedWords(bindingAdapterPosition) + } + } + } + + override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder { + return ViewHolder(LayoutInflater.from(parent.context).inflate(R.layout.word_item, parent, false)) + } + + override fun onBindViewHolder(holder: ViewHolder, position: Int) { + val word = getItem(position) + holder.searchWord.text = word.name + } + + private val deletedWords = { position:Int -> + val word = getItem(position) + deleteWord(word) + } +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt b/app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt new file mode 100644 index 00000000..a736d865 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt @@ -0,0 +1,110 @@ +package campus.tech.kakao.map.DBHelper + +import android.content.ContentValues +import android.content.Context +import android.database.sqlite.SQLiteDatabase +import android.database.sqlite.SQLiteOpenHelper +import android.util.Log +import androidx.lifecycle.LiveData +import androidx.lifecycle.MutableLiveData +import campus.tech.kakao.map.DTO.SearchWord +import campus.tech.kakao.map.DTO.SearchWordContract +import campus.tech.kakao.map.DTO.SearchWordContract.DB_VERSION + +class SearchWordDbHelper(context: Context): SQLiteOpenHelper( + context, SearchWordContract.DB_NAME, null, DB_VERSION) { + private val _searchWords = MutableLiveData>() + val searchSameSelection = "${SearchWordContract.COLUMN_NAME_NAME} = ? AND " + + "${SearchWordContract.COLUMN_NAME_ADDRESS} = ? AND " + + "${SearchWordContract.COLUMN_NAME_TYPE} = ?" + override fun onCreate(db: SQLiteDatabase?) { + createTable(db) + } + + private fun createTable(db: SQLiteDatabase?) { + db?.execSQL( + "CREATE TABLE ${SearchWordContract.TABLE_NAME} " + + "(${SearchWordContract.COLUMN_NAME_NAME} TEXT, " + + "${SearchWordContract.COLUMN_NAME_ADDRESS} TEXT, " + + "${SearchWordContract.COLUMN_NAME_TYPE} TEXT)" + ) + } + + override fun onUpgrade(db: SQLiteDatabase?, oldVersion: Int, newVersion: Int) { + db?.execSQL("DROP TABLE IF EXISTS ${SearchWordContract.TABLE_NAME}") + createTable(db) + } + + fun getSearchWords(): LiveData> { + return _searchWords + } + + fun addWord(word: SearchWord) { + val db = writableDatabase + if (existWord(word, db)){ + deleteWord(word) + } + val values = ContentValues() + values.put(SearchWordContract.COLUMN_NAME_NAME, word.name) + values.put(SearchWordContract.COLUMN_NAME_ADDRESS, word.address) + values.put(SearchWordContract.COLUMN_NAME_TYPE, word.type) + db.insert(SearchWordContract.TABLE_NAME, null, values) + db.close() + updateSearchWords() + } + + fun existWord(word: SearchWord, db: SQLiteDatabase): Boolean{ + val selection = searchSameSelection + val cursor = db.query( + SearchWordContract.TABLE_NAME, + arrayOf(SearchWordContract.COLUMN_NAME_NAME), + selection, + arrayOf(word.name, word.address, word.type), + null, + null, + "${SearchWordContract.COLUMN_NAME_NAME} DESC" + ).use { + it.moveToFirst() + } + return cursor + } + + fun deleteWord(word: SearchWord){ + val db = writableDatabase + val selection = searchSameSelection + val selectionArgs = arrayOf(word.name, word.address, word.type) + db.delete(SearchWordContract.TABLE_NAME, selection, selectionArgs) + updateSearchWords() + } + + fun updateSearchWords(){ + val db = readableDatabase + val resultList = mutableListOf() + val cursor = db.query( + SearchWordContract.TABLE_NAME, + arrayOf( + SearchWordContract.COLUMN_NAME_NAME, + SearchWordContract.COLUMN_NAME_ADDRESS, + SearchWordContract.COLUMN_NAME_TYPE + ), + null, + null, + null, + null, + null + ).use {cursor -> + while (cursor.moveToNext()) { + val name = cursor.getString( + cursor.getColumnIndexOrThrow(SearchWordContract.COLUMN_NAME_NAME) + ) + val address = cursor.getString( + cursor.getColumnIndexOrThrow(SearchWordContract.COLUMN_NAME_ADDRESS) + ) + val type = cursor.getString( + cursor.getColumnIndexOrThrow(SearchWordContract.COLUMN_NAME_TYPE)) + resultList.add(SearchWord(name, address, type)) + } + } + _searchWords.value = resultList + } +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/Document.kt b/app/src/main/java/campus/tech/kakao/map/DTO/Document.kt new file mode 100644 index 00000000..1b33c093 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/Document.kt @@ -0,0 +1,22 @@ +package campus.tech.kakao.map.DTO + +import com.google.gson.annotations.SerializedName + +//필요 없는 데이터 주석화 +data class Document( + //val id: String, + @SerializedName("place_name") + val placeName: String, + //val category_name: String, + //val category_group_code: String, + @SerializedName("category_group_name") + val categoryGroupName: String, + //val phone: String, + @SerializedName("address_name") + val addressName: String, + //val road_address_name: String, + //val x: String, + //val y: String, + //val place_url: String, + //val distance: String, +) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt b/app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt new file mode 100644 index 00000000..e34d3048 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt @@ -0,0 +1,14 @@ +package campus.tech.kakao.map.DTO + +import com.google.gson.annotations.SerializedName + +data class Meta( + @SerializedName("total_count") + val totalCount: Int, + @SerializedName("pageable_count") + val pageableCount: Int, + @SerializedName("is_end") + val isEnd: Boolean, + @SerializedName("same_name") + val sameName: SameName +) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt b/app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt new file mode 100644 index 00000000..9ab49a5d --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt @@ -0,0 +1,6 @@ +package campus.tech.kakao.map.DTO + +data class PlaceResponse( + val meta: Meta, + val documents: List +) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt b/app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt new file mode 100644 index 00000000..b3b9f3bc --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt @@ -0,0 +1,7 @@ +package campus.tech.kakao.map.DTO + +data class SameName( + val region: List, + val keyword: String, + val selected_region: String +) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt b/app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt new file mode 100644 index 00000000..2823ba00 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt @@ -0,0 +1,3 @@ +package campus.tech.kakao.map.DTO + +data class SearchWord(val name: String, val address: String, val type: String) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt b/app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt new file mode 100644 index 00000000..5c97b8cd --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt @@ -0,0 +1,11 @@ +package campus.tech.kakao.map.DTO + +object SearchWordContract { + const val DB_NAME = "search_word.db" + const val DB_VERSION = 1 + const val TABLE_NAME = "SearchWord" + const val COLUMN_NAME_NAME = "name" + const val COLUMN_NAME_ADDRESS = "address" + const val COLUMN_NAME_TYPE = "type" + +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt b/app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt new file mode 100644 index 00000000..88d057c7 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt @@ -0,0 +1,8 @@ +package campus.tech.kakao.map.DTO + +import campus.tech.kakao.map.BuildConfig + +object UrlContract { + const val BASE_URL = "https://dapi.kakao.com/v2/local/search/" + const val AUTHORIZATION = "KakaoAK ${BuildConfig.KAKAO_REST_API_KEY}" +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt new file mode 100644 index 00000000..d5344630 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt @@ -0,0 +1,42 @@ +package campus.tech.kakao.map + +import android.app.Application +import androidx.lifecycle.AndroidViewModel +import androidx.lifecycle.LiveData +import campus.tech.kakao.map.DBHelper.SearchWordDbHelper +import campus.tech.kakao.map.DTO.Document +import campus.tech.kakao.map.DTO.SearchWord +import campus.tech.kakao.map.RetrofitData.Companion.getInstance + +class MainViewModel(application: Application): AndroidViewModel(application) { + + private val wordDbHelper = SearchWordDbHelper(application) + val wordList: LiveData> get() = wordDbHelper.getSearchWords() + + private val retrofitData = getInstance() + val documentList: LiveData> get() = retrofitData.getDocuments() + + fun addWord(document: Document){ + wordDbHelper.addWord(wordfromDocument(document)) + } + + private fun wordfromDocument(document: Document): SearchWord { + return SearchWord(document.placeName, document.categoryGroupName, document.addressName) + } + fun deleteWord(word: SearchWord){ + wordDbHelper.deleteWord(word) + } + + fun loadWord(){ + wordDbHelper.updateSearchWords() + } + + fun searchLocalAPI(query: String){ + retrofitData.searchPlace(query) + } + + override fun onCleared() { + super.onCleared() + wordDbHelper.close() + } +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/MapActivity.kt b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt new file mode 100644 index 00000000..f71c0cb5 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt @@ -0,0 +1,54 @@ +package campus.tech.kakao.map + +import android.content.Intent +import android.os.Bundle +import android.util.Log +import android.widget.LinearLayout +import androidx.appcompat.app.AppCompatActivity +import com.kakao.vectormap.KakaoMap +import com.kakao.vectormap.KakaoMapReadyCallback +import com.kakao.vectormap.MapLifeCycleCallback +import com.kakao.vectormap.MapView +import java.lang.Exception + +class MapActivity : AppCompatActivity() { + private lateinit var mapView: MapView + private var map: KakaoMap? = null + private lateinit var searchBar: LinearLayout + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_map) + mapView = findViewById(R.id.map_view) + mapView.start(object : MapLifeCycleCallback() { + override fun onMapDestroy() { + + } + + override fun onMapError(p0: Exception?) { + Log.e("MapActivity", "onMapError: ${p0?.message}", p0) + } + + }, object: KakaoMapReadyCallback() { + override fun onMapReady(kakaoMap: KakaoMap) { + map = kakaoMap + } + + }) + searchBar = findViewById(R.id.search_bar) + searchBar.setOnClickListener { + val intent = Intent(this@MapActivity, SearchActivity::class.java) + startActivity(intent) + } + } + + override fun onResume() { + super.onResume() + mapView.resume() + } + + override fun onPause() { + super.onPause() + mapView.pause() + } + +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/MyApplication.kt b/app/src/main/java/campus/tech/kakao/map/MyApplication.kt new file mode 100644 index 00000000..2d2aa825 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/MyApplication.kt @@ -0,0 +1,11 @@ +package campus.tech.kakao.map + +import android.app.Application +import com.kakao.vectormap.KakaoMapSdk + +class MyApplication : Application() { + override fun onCreate() { + super.onCreate() + KakaoMapSdk.init(this, BuildConfig.KAKAO_API_KEY) + } +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt b/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt new file mode 100644 index 00000000..de3f13a0 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt @@ -0,0 +1,54 @@ +package campus.tech.kakao.map + +import androidx.lifecycle.MutableLiveData +import campus.tech.kakao.map.DTO.Document +import campus.tech.kakao.map.DTO.PlaceResponse +import campus.tech.kakao.map.DTO.UrlContract +import campus.tech.kakao.map.DTO.UrlContract.AUTHORIZATION +import retrofit2.Call +import retrofit2.Callback +import retrofit2.Response +import retrofit2.Retrofit +import retrofit2.converter.gson.GsonConverterFactory + +class RetrofitData private constructor() { + private val _documents = MutableLiveData>() + val retrofitService = Retrofit.Builder() + .baseUrl(UrlContract.BASE_URL) + .addConverterFactory(GsonConverterFactory.create()) + .build() + .create(RetrofitService::class.java) + + fun searchPlace(query : String){ + retrofitService.requestPlaces(AUTHORIZATION,query).enqueue(object : Callback { + override fun onResponse( + call: Call, + response: Response + ) { + if (response.isSuccessful) { + val documentList = mutableListOf() + val body = response.body() + body?.documents?.forEach { + documentList.add(Document(it.placeName, it.categoryGroupName, it.addressName)) + } + _documents.value = documentList + } + } + + override fun onFailure(call: Call, t: Throwable) { + } + }) + } + fun getDocuments() = _documents + + companion object { + private var instance: RetrofitData? = null + + fun getInstance(): RetrofitData { + if (instance == null) { + instance = RetrofitData() + } + return instance!! + } + } +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt b/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt new file mode 100644 index 00000000..7f4e7be9 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt @@ -0,0 +1,15 @@ +package campus.tech.kakao.map + +import campus.tech.kakao.map.DTO.PlaceResponse +import retrofit2.Call +import retrofit2.http.GET +import retrofit2.http.Header +import retrofit2.http.Query + +interface RetrofitService { + @GET("keyword.json") + fun requestPlaces( + @Header("Authorization") authorization: String, + @Query("query") query: String + ): Call +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt new file mode 100644 index 00000000..3a10cb0b --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt @@ -0,0 +1,86 @@ +package campus.tech.kakao.map + +import android.os.Bundle +import android.view.View +import android.widget.EditText +import android.widget.TextView +import androidx.appcompat.app.AppCompatActivity +import androidx.core.widget.doOnTextChanged +import androidx.lifecycle.Observer +import androidx.lifecycle.ViewModelProvider +import androidx.recyclerview.widget.LinearLayoutManager +import androidx.recyclerview.widget.RecyclerView +import campus.tech.kakao.map.Adapter.DocumentAdapter +import campus.tech.kakao.map.Adapter.WordAdapter + +class SearchActivity : AppCompatActivity() { + + private lateinit var model: MainViewModel + private lateinit var search:EditText + private lateinit var clear: TextView + private lateinit var noResult: TextView + private lateinit var searchResult: RecyclerView + private lateinit var searchWordResult: RecyclerView + private lateinit var documentAdapter: DocumentAdapter + private lateinit var wordAdapter: WordAdapter + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + setContentView(R.layout.activity_search) + setupUI() + searchResult.layoutManager = LinearLayoutManager(this) + searchWordResult.layoutManager = LinearLayoutManager(this, LinearLayoutManager.HORIZONTAL, false) + documentAdapter = DocumentAdapter(){ Document -> + model.addWord(Document) + } + wordAdapter = WordAdapter() { SearchWord -> + model.deleteWord(SearchWord) + } + search.doOnTextChanged { text, start, before, count -> + val query = text.toString() + if (query.isEmpty()){ + noResult.visibility = View.VISIBLE + searchResult.visibility = View.GONE + }else{ + noResult.visibility = View.GONE + searchResult.visibility = View.VISIBLE + model.searchLocalAPI(query) + } + } + model = ViewModelProvider(this)[MainViewModel::class.java] + model.documentList.observe(this, Observer {documents -> + if (documents.isNullOrEmpty()){ + noResult.visibility = View.VISIBLE + searchResult.visibility = View.GONE + }else{ + noResult.visibility = View.GONE + searchResult.visibility = View.VISIBLE + documentAdapter.submitList(documents) + searchResult.adapter = documentAdapter + } + }) + model.loadWord() + model.wordList.observe(this, Observer {searchWords -> + if (searchWords.isNullOrEmpty()){ + searchWordResult.visibility = View.GONE + } + else{ + searchWordResult.visibility = View.VISIBLE + wordAdapter.submitList(searchWords) + searchWordResult.adapter = wordAdapter + } + }) + } + + fun setupUI(){ + search = findViewById(R.id.search) + clear = findViewById(R.id.search_clear) + noResult = findViewById(R.id.no_search_result) + searchResult = findViewById(R.id.search_result_recycler_view) + searchWordResult = findViewById(R.id.search_word_recycler_view) + clear.setOnClickListener { + search.setText("") + } + } + + +} diff --git a/app/src/main/res/drawable/magnifier.png b/app/src/main/res/drawable/magnifier.png new file mode 100644 index 0000000000000000000000000000000000000000..3dad26ecb6fbb28a250e63e9bec157d755148306 GIT binary patch literal 9712 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 W(6f^&lH?Fy(B59Yo<$$ + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/activity_search.xml b/app/src/main/res/layout/activity_search.xml new file mode 100644 index 00000000..97d7e02e --- /dev/null +++ b/app/src/main/res/layout/activity_search.xml @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + diff --git a/app/src/main/res/layout/place_item.xml b/app/src/main/res/layout/place_item.xml new file mode 100644 index 00000000..fa5da065 --- /dev/null +++ b/app/src/main/res/layout/place_item.xml @@ -0,0 +1,47 @@ + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/word_item.xml b/app/src/main/res/layout/word_item.xml new file mode 100644 index 00000000..a14c1c28 --- /dev/null +++ b/app/src/main/res/layout/word_item.xml @@ -0,0 +1,23 @@ + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index e5ba5b9c..290224da 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -1,3 +1,5 @@ Map + 검색어를 입력해 주세요. + 검색 결과가 없습니다. \ No newline at end of file diff --git a/app/src/main/res/xml/network_security_config.xml b/app/src/main/res/xml/network_security_config.xml new file mode 100644 index 00000000..96ed3bbe --- /dev/null +++ b/app/src/main/res/xml/network_security_config.xml @@ -0,0 +1,6 @@ + + + + 54.180.95.212 + + diff --git a/settings.gradle.kts b/settings.gradle.kts index ff9d5255..e556d5a4 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -10,6 +10,7 @@ dependencyResolutionManagement { repositories { google() mavenCentral() + maven (url = "https://devrepo.kakao.com/nexus/content/groups/public/") } } From 7c76c973b90e7faa9382a6d2396c0d6308d93d56 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Mon, 15 Jul 2024 14:25:51 +0900 Subject: [PATCH 02/19] =?UTF-8?q?Refactor[]:=20=EB=B3=80=EA=B2=BD=20warnin?= =?UTF-8?q?g=20=EB=B6=80=EB=B6=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/src/main/res/layout/word_item.xml | 2 +- app/src/main/res/values/strings.xml | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/app/src/main/res/layout/word_item.xml b/app/src/main/res/layout/word_item.xml index a14c1c28..7b543d86 100644 --- a/app/src/main/res/layout/word_item.xml +++ b/app/src/main/res/layout/word_item.xml @@ -14,7 +14,7 @@ android:id="@+id/search_word" android:layout_width="wrap_content" android:layout_height="wrap_content" - android:text="word" + android:text="@string/word_item_default_text" android:textSize="25sp" android:gravity="center" android:layout_margin="8dp"/> diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 290224da..bfa0bb3e 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -2,4 +2,5 @@ Map 검색어를 입력해 주세요. 검색 결과가 없습니다. + word \ No newline at end of file From ba17eaf234aabc582513b32d6f567bdb22b5399f Mon Sep 17 00:00:00 2001 From: jmc98 Date: Mon, 15 Jul 2024 14:39:21 +0900 Subject: [PATCH 03/19] =?UTF-8?q?Docs[README.md]:=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?=EC=9A=94=EA=B5=AC=20=EC=82=AC=ED=95=AD=20=EC=A0=95=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/README.md b/README.md index 91dbd079..dc23030f 100644 --- a/README.md +++ b/README.md @@ -1 +1,13 @@ # android-map-location + +## 1단계 - 카카오맵 API 심화 +### 기능 요구 사항 +- 저장된 검색어를 선택하면 해당 검색어의 검색 결과가 표시된다. +- 검색 결과 목록 중 하나의 항목을 선택하면 해당 항목의 위치를 지도에 표시한다. +- 앱 종료 시 마지막 위치를 저장하여 다시 앱 실행 시 해당 위치로 포커스 한다. +- 카카오지도 onMapError() 호출 시 에러 화면을 보여준다. +### 프로그래밍 요구 사항 +- BottomSheet를 사용한다. +- 카카오 API 사용을 위한 앱 키를 외부에 노출하지 않는다. +- 가능한 MVVM 아키텍처 패턴을 적용하도록 한다. +- 코드 컨벤션을 준수하며 프로그래밍한다. \ No newline at end of file From 4c2c1117176391c9b3fa11674c34279854746414 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Tue, 16 Jul 2024 15:19:47 +0900 Subject: [PATCH 04/19] =?UTF-8?q?Feat[]:=20=EC=B6=94=EA=B0=80=20=EC=84=A0?= =?UTF-8?q?=ED=83=9D=20=ED=95=AD=EB=AA=A9=20=EC=9C=84=EC=B9=98,=20?= =?UTF-8?q?=EC=A0=95=EB=B3=B4=EB=A5=BC=20=EC=A7=80=EB=8F=84=EC=97=90=20?= =?UTF-8?q?=ED=91=9C=EC=8B=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 라벨과 바텀시트 사용 --- .../campus/tech/kakao/map/MainActivity.kt | 11 --- .../campus/tech/kakao/map/MainViewModel.kt | 15 ++- .../java/campus/tech/kakao/map/MapActivity.kt | 96 +++++++++++++++++++ .../campus/tech/kakao/map/MyApplication.kt | 6 ++ .../campus/tech/kakao/map/RetrofitData.kt | 17 ++-- .../campus/tech/kakao/map/RetrofitService.kt | 2 +- .../campus/tech/kakao/map/SearchActivity.kt | 11 ++- .../{Adapter => adapter}/DocumentAdapter.kt | 10 +- .../map/{Adapter => adapter}/WordAdapter.kt | 4 +- .../SearchWordDbHelper.kt | 9 +- .../tech/kakao/map/{DTO => dto}/Document.kt | 8 +- .../tech/kakao/map/dto/MapPositionContract.kt | 9 ++ .../kakao/map/dto/MapPositionPreferences.kt | 17 ++++ .../tech/kakao/map/{DTO => dto}/Meta.kt | 2 +- .../kakao/map/{DTO => dto}/PlaceResponse.kt | 2 +- .../tech/kakao/map/{DTO => dto}/SameName.kt | 2 +- .../tech/kakao/map/{DTO => dto}/SearchWord.kt | 2 +- .../map/{DTO => dto}/SearchWordContract.kt | 2 +- .../kakao/map/{DTO => dto}/UrlContract.kt | 2 +- app/src/main/res/layout/activity_main.xml | 19 ---- app/src/main/res/layout/activity_map.xml | 68 ++++++------- app/src/main/res/layout/bottom_sheet.xml | 30 ++++++ app/src/main/res/values/strings.xml | 2 + 23 files changed, 251 insertions(+), 95 deletions(-) delete mode 100644 app/src/main/java/campus/tech/kakao/map/MainActivity.kt rename app/src/main/java/campus/tech/kakao/map/{Adapter => adapter}/DocumentAdapter.kt (84%) rename app/src/main/java/campus/tech/kakao/map/{Adapter => adapter}/WordAdapter.kt (94%) rename app/src/main/java/campus/tech/kakao/map/{DBHelper => dbHelper}/SearchWordDbHelper.kt (93%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/Document.kt (79%) create mode 100644 app/src/main/java/campus/tech/kakao/map/dto/MapPositionContract.kt create mode 100644 app/src/main/java/campus/tech/kakao/map/dto/MapPositionPreferences.kt rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/Meta.kt (89%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/PlaceResponse.kt (69%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/SameName.kt (75%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/SearchWord.kt (70%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/SearchWordContract.kt (87%) rename app/src/main/java/campus/tech/kakao/map/{DTO => dto}/UrlContract.kt (85%) delete mode 100644 app/src/main/res/layout/activity_main.xml create mode 100644 app/src/main/res/layout/bottom_sheet.xml diff --git a/app/src/main/java/campus/tech/kakao/map/MainActivity.kt b/app/src/main/java/campus/tech/kakao/map/MainActivity.kt deleted file mode 100644 index 95b43803..00000000 --- a/app/src/main/java/campus/tech/kakao/map/MainActivity.kt +++ /dev/null @@ -1,11 +0,0 @@ -package campus.tech.kakao.map - -import android.os.Bundle -import androidx.appcompat.app.AppCompatActivity - -class MainActivity : AppCompatActivity() { - override fun onCreate(savedInstanceState: Bundle?) { - super.onCreate(savedInstanceState) - setContentView(R.layout.activity_main) - } -} diff --git a/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt index d5344630..a875368b 100644 --- a/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt +++ b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt @@ -3,9 +3,11 @@ package campus.tech.kakao.map import android.app.Application import androidx.lifecycle.AndroidViewModel import androidx.lifecycle.LiveData -import campus.tech.kakao.map.DBHelper.SearchWordDbHelper -import campus.tech.kakao.map.DTO.Document -import campus.tech.kakao.map.DTO.SearchWord +import campus.tech.kakao.map.dbHelper.SearchWordDbHelper +import campus.tech.kakao.map.dto.Document +import campus.tech.kakao.map.dto.MapPositionContract +import campus.tech.kakao.map.dto.SearchWord +import campus.tech.kakao.map.MyApplication.Companion.mapPosition import campus.tech.kakao.map.RetrofitData.Companion.getInstance class MainViewModel(application: Application): AndroidViewModel(application) { @@ -39,4 +41,11 @@ class MainViewModel(application: Application): AndroidViewModel(application) { super.onCleared() wordDbHelper.close() } + + fun getMapInfo(document: Document){ + mapPosition.setPreferences(MapPositionContract.PREFERENCE_KEY_LATITUDE, document.latitude) + mapPosition.setPreferences(MapPositionContract.PREFERENCE_KEY_LONGITUDE, document.longitude) + mapPosition.setPreferences(MapPositionContract.PREFERENCE_KEY_PLACENAME, document.placeName) + mapPosition.setPreferences(MapPositionContract.PREFERENCE_KEY_ADDRESSNAME, document.addressName) + } } \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/MapActivity.kt b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt index f71c0cb5..7cdb2025 100644 --- a/app/src/main/java/campus/tech/kakao/map/MapActivity.kt +++ b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt @@ -1,24 +1,58 @@ package campus.tech.kakao.map import android.content.Intent +import android.graphics.Bitmap +import android.graphics.BitmapFactory +import android.graphics.Color import android.os.Bundle import android.util.Log +import android.view.View import android.widget.LinearLayout +import android.widget.TextView import androidx.appcompat.app.AppCompatActivity +import campus.tech.kakao.map.MyApplication.Companion.mapPosition +import com.google.android.material.bottomsheet.BottomSheetBehavior import com.kakao.vectormap.KakaoMap import com.kakao.vectormap.KakaoMapReadyCallback +import com.kakao.vectormap.LatLng import com.kakao.vectormap.MapLifeCycleCallback import com.kakao.vectormap.MapView +import com.kakao.vectormap.camera.CameraUpdate +import com.kakao.vectormap.camera.CameraUpdateFactory +import com.kakao.vectormap.label.Label +import com.kakao.vectormap.label.LabelLayer +import com.kakao.vectormap.label.LabelOptions +import com.kakao.vectormap.label.LabelStyle +import com.kakao.vectormap.label.LabelStyles import java.lang.Exception class MapActivity : AppCompatActivity() { private lateinit var mapView: MapView private var map: KakaoMap? = null private lateinit var searchBar: LinearLayout + private var latitude = 37.402005 + private var longitude = 127.108621 + private lateinit var placeName:String + private lateinit var addressName:String + private var styles: LabelStyles? = null + private lateinit var options:LabelOptions + private var layer: LabelLayer? = null + private var label: Label? = null + private lateinit var bitmapImage: Bitmap + private lateinit var markerImage: Bitmap + private lateinit var bottomSheetBehavior: BottomSheetBehavior + private val bottomSheet by lazy { findViewById(R.id.bottom_sheet) } + private val bottomSheetName by lazy { findViewById(R.id.name) } + private val bottomSheetAddress by lazy { findViewById(R.id.address) } + companion object{ + var documentClicked = false + } + override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_map) mapView = findViewById(R.id.map_view) + setMapInfo() mapView.start(object : MapLifeCycleCallback() { override fun onMapDestroy() { @@ -33,17 +67,37 @@ class MapActivity : AppCompatActivity() { map = kakaoMap } + override fun getPosition(): LatLng { + return LatLng.from(latitude, longitude) + } + + override fun getZoomLevel(): Int { + return 17 + } }) searchBar = findViewById(R.id.search_bar) searchBar.setOnClickListener { val intent = Intent(this@MapActivity, SearchActivity::class.java) startActivity(intent) } + initBottomSheet() } override fun onResume() { super.onResume() + setMapInfo() mapView.resume() + if(documentClicked){ + makeMarker() + setBottomSheet() + documentClicked = false + } + else{ + layer?.remove(label) + bottomSheetBehavior.state = BottomSheetBehavior.STATE_HIDDEN + } + val cameraUpdate: CameraUpdate = CameraUpdateFactory.newCenterPosition(LatLng.from(latitude, longitude)) + map?.moveCamera(cameraUpdate) } override fun onPause() { @@ -51,4 +105,46 @@ class MapActivity : AppCompatActivity() { mapView.pause() } + private fun setMapInfo(){ + latitude = mapPosition.getPreferences("latitude","37.406960").toDouble() + longitude = mapPosition.getPreferences("longitude","127.110030").toDouble() + placeName = mapPosition.getPreferences("placeName","") + addressName = mapPosition.getPreferences("addressName","") + } + + private fun makeMarker(){ + bitmapImage = BitmapFactory.decodeResource(resources, R.drawable.marker) + markerImage = Bitmap.createScaledBitmap(bitmapImage, 100, 100, true) + styles = map?.labelManager?.addLabelStyles(LabelStyles.from(LabelStyle.from(markerImage).setTextStyles(40, Color.BLACK))) + if(styles != null){ + options = LabelOptions.from(LatLng.from(latitude, longitude)).setStyles(styles).setTexts(placeName) + layer = map?.labelManager?.layer + if(label != null){ + layer?.remove(label) + } + label = layer?.addLabel(options) + } + else{ + Log.e("MapActivity", "makeMarker: styles is null") + } + } + + private fun initBottomSheet(){ + bottomSheetBehavior = BottomSheetBehavior.from(bottomSheet) + bottomSheetBehavior.addBottomSheetCallback(object : BottomSheetBehavior.BottomSheetCallback(){ + override fun onStateChanged(bottomSheet: View, newState: Int) { + } + + override fun onSlide(bottomSheet: View, slideOffset: Float) { + } + + }) + bottomSheetBehavior.state = BottomSheetBehavior.STATE_HIDDEN + } + + private fun setBottomSheet(){ + bottomSheetBehavior.state = BottomSheetBehavior.STATE_COLLAPSED + bottomSheetName.text = placeName + bottomSheetAddress.text = addressName + } } \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/MyApplication.kt b/app/src/main/java/campus/tech/kakao/map/MyApplication.kt index 2d2aa825..343ca5f8 100644 --- a/app/src/main/java/campus/tech/kakao/map/MyApplication.kt +++ b/app/src/main/java/campus/tech/kakao/map/MyApplication.kt @@ -1,11 +1,17 @@ package campus.tech.kakao.map import android.app.Application +import campus.tech.kakao.map.dto.MapPositionPreferences import com.kakao.vectormap.KakaoMapSdk class MyApplication : Application() { override fun onCreate() { super.onCreate() + mapPosition = MapPositionPreferences(this) KakaoMapSdk.init(this, BuildConfig.KAKAO_API_KEY) } + + companion object{ + lateinit var mapPosition : MapPositionPreferences + } } \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt b/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt index de3f13a0..c9a232ee 100644 --- a/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt +++ b/app/src/main/java/campus/tech/kakao/map/RetrofitData.kt @@ -1,10 +1,10 @@ package campus.tech.kakao.map import androidx.lifecycle.MutableLiveData -import campus.tech.kakao.map.DTO.Document -import campus.tech.kakao.map.DTO.PlaceResponse -import campus.tech.kakao.map.DTO.UrlContract -import campus.tech.kakao.map.DTO.UrlContract.AUTHORIZATION +import campus.tech.kakao.map.dto.Document +import campus.tech.kakao.map.dto.PlaceResponse +import campus.tech.kakao.map.dto.UrlContract +import campus.tech.kakao.map.dto.UrlContract.AUTHORIZATION import retrofit2.Call import retrofit2.Callback import retrofit2.Response @@ -28,8 +28,13 @@ class RetrofitData private constructor() { if (response.isSuccessful) { val documentList = mutableListOf() val body = response.body() - body?.documents?.forEach { - documentList.add(Document(it.placeName, it.categoryGroupName, it.addressName)) + body?.documents?.forEach {document -> + documentList.add(Document( + document.placeName, + document.categoryGroupName, + document.addressName, + document.longitude, + document.latitude)) } _documents.value = documentList } diff --git a/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt b/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt index 7f4e7be9..d2101f00 100644 --- a/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt +++ b/app/src/main/java/campus/tech/kakao/map/RetrofitService.kt @@ -1,6 +1,6 @@ package campus.tech.kakao.map -import campus.tech.kakao.map.DTO.PlaceResponse +import campus.tech.kakao.map.dto.PlaceResponse import retrofit2.Call import retrofit2.http.GET import retrofit2.http.Header diff --git a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt index 3a10cb0b..0f8f34a6 100644 --- a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt +++ b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt @@ -10,8 +10,8 @@ import androidx.lifecycle.Observer import androidx.lifecycle.ViewModelProvider import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.RecyclerView -import campus.tech.kakao.map.Adapter.DocumentAdapter -import campus.tech.kakao.map.Adapter.WordAdapter +import campus.tech.kakao.map.adapter.DocumentAdapter +import campus.tech.kakao.map.adapter.WordAdapter class SearchActivity : AppCompatActivity() { @@ -29,9 +29,12 @@ class SearchActivity : AppCompatActivity() { setupUI() searchResult.layoutManager = LinearLayoutManager(this) searchWordResult.layoutManager = LinearLayoutManager(this, LinearLayoutManager.HORIZONTAL, false) - documentAdapter = DocumentAdapter(){ Document -> + documentAdapter = DocumentAdapter({ Document -> model.addWord(Document) - } + },{Document -> + model.getMapInfo(Document) + finish() + }) wordAdapter = WordAdapter() { SearchWord -> model.deleteWord(SearchWord) } diff --git a/app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt b/app/src/main/java/campus/tech/kakao/map/adapter/DocumentAdapter.kt similarity index 84% rename from app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt rename to app/src/main/java/campus/tech/kakao/map/adapter/DocumentAdapter.kt index 74492c58..b141f6b9 100644 --- a/app/src/main/java/campus/tech/kakao/map/Adapter/DocumentAdapter.kt +++ b/app/src/main/java/campus/tech/kakao/map/adapter/DocumentAdapter.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.Adapter +package campus.tech.kakao.map.adapter import android.view.LayoutInflater import android.view.View @@ -7,11 +7,13 @@ import android.widget.TextView import androidx.recyclerview.widget.DiffUtil import androidx.recyclerview.widget.ListAdapter import androidx.recyclerview.widget.RecyclerView -import campus.tech.kakao.map.DTO.Document +import campus.tech.kakao.map.dto.Document +import campus.tech.kakao.map.MapActivity.Companion.documentClicked import campus.tech.kakao.map.R class DocumentAdapter( - val addWord: (Document) -> Unit + val addWord: (Document) -> Unit, + val sendDocumentInfo: (Document) -> Unit ): ListAdapter( object : DiffUtil.ItemCallback(){ override fun areItemsTheSame(oldItem: Document, newItem: Document): Boolean { @@ -27,6 +29,8 @@ class DocumentAdapter( private var placeClicked = { position:Int -> val document: Document = getItem(position) addWord(document) + sendDocumentInfo(document) + documentClicked = true } inner class ViewHolder( itemView: View diff --git a/app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt b/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt similarity index 94% rename from app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt rename to app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt index 9b051e3d..7af25014 100644 --- a/app/src/main/java/campus/tech/kakao/map/Adapter/WordAdapter.kt +++ b/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.Adapter +package campus.tech.kakao.map.adapter import android.view.LayoutInflater import android.view.View @@ -9,7 +9,7 @@ import androidx.recyclerview.widget.DiffUtil import androidx.recyclerview.widget.ListAdapter import androidx.recyclerview.widget.RecyclerView import campus.tech.kakao.map.R -import campus.tech.kakao.map.DTO.SearchWord +import campus.tech.kakao.map.dto.SearchWord class WordAdapter( val deleteWord: (SearchWord) -> Unit diff --git a/app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt b/app/src/main/java/campus/tech/kakao/map/dbHelper/SearchWordDbHelper.kt similarity index 93% rename from app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt rename to app/src/main/java/campus/tech/kakao/map/dbHelper/SearchWordDbHelper.kt index a736d865..bf22f9bf 100644 --- a/app/src/main/java/campus/tech/kakao/map/DBHelper/SearchWordDbHelper.kt +++ b/app/src/main/java/campus/tech/kakao/map/dbHelper/SearchWordDbHelper.kt @@ -1,15 +1,14 @@ -package campus.tech.kakao.map.DBHelper +package campus.tech.kakao.map.dbHelper import android.content.ContentValues import android.content.Context import android.database.sqlite.SQLiteDatabase import android.database.sqlite.SQLiteOpenHelper -import android.util.Log import androidx.lifecycle.LiveData import androidx.lifecycle.MutableLiveData -import campus.tech.kakao.map.DTO.SearchWord -import campus.tech.kakao.map.DTO.SearchWordContract -import campus.tech.kakao.map.DTO.SearchWordContract.DB_VERSION +import campus.tech.kakao.map.dto.SearchWord +import campus.tech.kakao.map.dto.SearchWordContract +import campus.tech.kakao.map.dto.SearchWordContract.DB_VERSION class SearchWordDbHelper(context: Context): SQLiteOpenHelper( context, SearchWordContract.DB_NAME, null, DB_VERSION) { diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/Document.kt b/app/src/main/java/campus/tech/kakao/map/dto/Document.kt similarity index 79% rename from app/src/main/java/campus/tech/kakao/map/DTO/Document.kt rename to app/src/main/java/campus/tech/kakao/map/dto/Document.kt index 1b33c093..637f7059 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/Document.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/Document.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto import com.google.gson.annotations.SerializedName @@ -15,8 +15,10 @@ data class Document( @SerializedName("address_name") val addressName: String, //val road_address_name: String, - //val x: String, - //val y: String, + @SerializedName("x") + val longitude : String, + @SerializedName("y") + val latitude: String, //val place_url: String, //val distance: String, ) diff --git a/app/src/main/java/campus/tech/kakao/map/dto/MapPositionContract.kt b/app/src/main/java/campus/tech/kakao/map/dto/MapPositionContract.kt new file mode 100644 index 00000000..b0af0619 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/dto/MapPositionContract.kt @@ -0,0 +1,9 @@ +package campus.tech.kakao.map.dto + +object MapPositionContract { + const val PREFERENCE_NAME = "MapPosition" + const val PREFERENCE_KEY_LATITUDE = "latitude" + const val PREFERENCE_KEY_LONGITUDE = "longitude" + const val PREFERENCE_KEY_PLACENAME = "placeName" + const val PREFERENCE_KEY_ADDRESSNAME = "addressName" +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/dto/MapPositionPreferences.kt b/app/src/main/java/campus/tech/kakao/map/dto/MapPositionPreferences.kt new file mode 100644 index 00000000..bb52e031 --- /dev/null +++ b/app/src/main/java/campus/tech/kakao/map/dto/MapPositionPreferences.kt @@ -0,0 +1,17 @@ +package campus.tech.kakao.map.dto + +import android.content.Context + +class MapPositionPreferences(context: Context) { + + private val mapPosition = context.getSharedPreferences(MapPositionContract.PREFERENCE_NAME, Context.MODE_PRIVATE) + + fun setPreferences(key: String, value: String) { + mapPosition.edit().putString(key, value).apply() + } + + fun getPreferences(key: String, defaultValue: String): String { + return mapPosition.getString(key, defaultValue).toString() + } + +} \ No newline at end of file diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt b/app/src/main/java/campus/tech/kakao/map/dto/Meta.kt similarity index 89% rename from app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt rename to app/src/main/java/campus/tech/kakao/map/dto/Meta.kt index e34d3048..60b41d75 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/Meta.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/Meta.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto import com.google.gson.annotations.SerializedName diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt b/app/src/main/java/campus/tech/kakao/map/dto/PlaceResponse.kt similarity index 69% rename from app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt rename to app/src/main/java/campus/tech/kakao/map/dto/PlaceResponse.kt index 9ab49a5d..3abf2475 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/PlaceResponse.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/PlaceResponse.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto data class PlaceResponse( val meta: Meta, diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt b/app/src/main/java/campus/tech/kakao/map/dto/SameName.kt similarity index 75% rename from app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt rename to app/src/main/java/campus/tech/kakao/map/dto/SameName.kt index b3b9f3bc..833cddfe 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/SameName.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/SameName.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto data class SameName( val region: List, diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt b/app/src/main/java/campus/tech/kakao/map/dto/SearchWord.kt similarity index 70% rename from app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt rename to app/src/main/java/campus/tech/kakao/map/dto/SearchWord.kt index 2823ba00..8ee333ed 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWord.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/SearchWord.kt @@ -1,3 +1,3 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto data class SearchWord(val name: String, val address: String, val type: String) diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt b/app/src/main/java/campus/tech/kakao/map/dto/SearchWordContract.kt similarity index 87% rename from app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt rename to app/src/main/java/campus/tech/kakao/map/dto/SearchWordContract.kt index 5c97b8cd..2acdb952 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/SearchWordContract.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/SearchWordContract.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto object SearchWordContract { const val DB_NAME = "search_word.db" diff --git a/app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt b/app/src/main/java/campus/tech/kakao/map/dto/UrlContract.kt similarity index 85% rename from app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt rename to app/src/main/java/campus/tech/kakao/map/dto/UrlContract.kt index 88d057c7..2d46eb85 100644 --- a/app/src/main/java/campus/tech/kakao/map/DTO/UrlContract.kt +++ b/app/src/main/java/campus/tech/kakao/map/dto/UrlContract.kt @@ -1,4 +1,4 @@ -package campus.tech.kakao.map.DTO +package campus.tech.kakao.map.dto import campus.tech.kakao.map.BuildConfig diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml deleted file mode 100644 index 24d17df2..00000000 --- a/app/src/main/res/layout/activity_main.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - - - diff --git a/app/src/main/res/layout/activity_map.xml b/app/src/main/res/layout/activity_map.xml index 4c302a6f..8b7646e4 100644 --- a/app/src/main/res/layout/activity_map.xml +++ b/app/src/main/res/layout/activity_map.xml @@ -1,36 +1,40 @@ - - + - - - - - - \ No newline at end of file + android:layout_height="match_parent" + tools:context=".MapActivity"> + + + + + + + + diff --git a/app/src/main/res/layout/bottom_sheet.xml b/app/src/main/res/layout/bottom_sheet.xml new file mode 100644 index 00000000..4c14d032 --- /dev/null +++ b/app/src/main/res/layout/bottom_sheet.xml @@ -0,0 +1,30 @@ + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index bfa0bb3e..4d7654b1 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -3,4 +3,6 @@ 검색어를 입력해 주세요. 검색 결과가 없습니다. word + 카카오 + 경기 성남시 분당구 백현동 532 \ No newline at end of file From c69b1815a3059714915e086c6b1521d97dff0b37 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Tue, 16 Jul 2024 15:22:45 +0900 Subject: [PATCH 05/19] =?UTF-8?q?Refactor[]:=20=EB=B3=80=EA=B2=BD=20warnin?= =?UTF-8?q?g=20=EC=9A=94=EC=86=8C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/SearchActivity.kt | 16 ++++++++-------- app/src/main/res/layout/bottom_sheet.xml | 1 + 2 files changed, 9 insertions(+), 8 deletions(-) diff --git a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt index 0f8f34a6..0837c69d 100644 --- a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt +++ b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt @@ -29,16 +29,16 @@ class SearchActivity : AppCompatActivity() { setupUI() searchResult.layoutManager = LinearLayoutManager(this) searchWordResult.layoutManager = LinearLayoutManager(this, LinearLayoutManager.HORIZONTAL, false) - documentAdapter = DocumentAdapter({ Document -> - model.addWord(Document) - },{Document -> - model.getMapInfo(Document) + documentAdapter = DocumentAdapter({ document -> + model.addWord(document) + },{document -> + model.getMapInfo(document) finish() }) - wordAdapter = WordAdapter() { SearchWord -> - model.deleteWord(SearchWord) + wordAdapter = WordAdapter { searchWord -> + model.deleteWord(searchWord) } - search.doOnTextChanged { text, start, before, count -> + search.doOnTextChanged { text, _, _, _ -> val query = text.toString() if (query.isEmpty()){ noResult.visibility = View.VISIBLE @@ -74,7 +74,7 @@ class SearchActivity : AppCompatActivity() { }) } - fun setupUI(){ + private fun setupUI(){ search = findViewById(R.id.search) clear = findViewById(R.id.search_clear) noResult = findViewById(R.id.no_search_result) diff --git a/app/src/main/res/layout/bottom_sheet.xml b/app/src/main/res/layout/bottom_sheet.xml index 4c14d032..dc44d733 100644 --- a/app/src/main/res/layout/bottom_sheet.xml +++ b/app/src/main/res/layout/bottom_sheet.xml @@ -5,6 +5,7 @@ android:layout_width="match_parent" android:layout_height="wrap_content" android:paddingStart="20dp" + android:paddingEnd="20dp" app:layout_behavior="com.google.android.material.bottomsheet.BottomSheetBehavior" app:behavior_hideable = "true" app:behavior_peekHeight="100dp" From 2613a7d4cee30dc3c4623b88808a0b910823daf0 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Tue, 16 Jul 2024 15:32:36 +0900 Subject: [PATCH 06/19] =?UTF-8?q?Design[bottom=5Fsheet]:=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD=20=ED=8C=A8=EB=94=A9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/src/main/res/layout/bottom_sheet.xml | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/app/src/main/res/layout/bottom_sheet.xml b/app/src/main/res/layout/bottom_sheet.xml index dc44d733..a020f694 100644 --- a/app/src/main/res/layout/bottom_sheet.xml +++ b/app/src/main/res/layout/bottom_sheet.xml @@ -4,11 +4,10 @@ android:orientation="vertical" android:layout_width="match_parent" android:layout_height="wrap_content" - android:paddingStart="20dp" - android:paddingEnd="20dp" + android:padding="10dp" app:layout_behavior="com.google.android.material.bottomsheet.BottomSheetBehavior" app:behavior_hideable = "true" - app:behavior_peekHeight="100dp" + app:behavior_peekHeight="120dp" android:focusable="true" android:id="@+id/bottom_sheet" android:background="@color/white"> From daefd3549d5fd23a91cffe0ad0f75dcfee025e31 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Tue, 16 Jul 2024 15:51:39 +0900 Subject: [PATCH 07/19] =?UTF-8?q?Feat[]:=20=EC=B6=94=EA=B0=80=20=EC=A0=80?= =?UTF-8?q?=EC=9E=A5=EB=90=9C=20=EA=B2=80=EC=83=89=EC=96=B4=20=EC=84=A0?= =?UTF-8?q?=ED=83=9D=20=EC=8B=9C=20=ED=95=B4=EB=8B=B9=20=EA=B2=80=EC=83=89?= =?UTF-8?q?=EC=96=B4=20=EA=B2=80=EC=83=89=20=EA=B2=B0=EA=B3=BC=20=ED=91=9C?= =?UTF-8?q?=EC=8B=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/SearchActivity.kt | 10 +++++++--- .../campus/tech/kakao/map/adapter/WordAdapter.kt | 12 +++++++++++- 2 files changed, 18 insertions(+), 4 deletions(-) diff --git a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt index 0837c69d..d3098e30 100644 --- a/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt +++ b/app/src/main/java/campus/tech/kakao/map/SearchActivity.kt @@ -35,9 +35,13 @@ class SearchActivity : AppCompatActivity() { model.getMapInfo(document) finish() }) - wordAdapter = WordAdapter { searchWord -> - model.deleteWord(searchWord) - } + wordAdapter = WordAdapter( + { searchWord -> + model.deleteWord(searchWord) + },{ searchWord -> + model.searchLocalAPI(searchWord.name) + } + ) search.doOnTextChanged { text, _, _, _ -> val query = text.toString() if (query.isEmpty()){ diff --git a/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt b/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt index 7af25014..7f9a30da 100644 --- a/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt +++ b/app/src/main/java/campus/tech/kakao/map/adapter/WordAdapter.kt @@ -12,7 +12,8 @@ import campus.tech.kakao.map.R import campus.tech.kakao.map.dto.SearchWord class WordAdapter( - val deleteWord: (SearchWord) -> Unit + val deleteWord: (SearchWord) -> Unit, + val clickWord: (SearchWord) -> Unit ): ListAdapter( object : DiffUtil.ItemCallback(){ override fun areItemsTheSame(oldItem: SearchWord, newItem: SearchWord): Boolean { @@ -34,6 +35,9 @@ class WordAdapter( delete.setOnClickListener { deletedWords(bindingAdapterPosition) } + searchWord.setOnClickListener { + clickedWord(bindingAdapterPosition) + } } } @@ -50,4 +54,10 @@ class WordAdapter( val word = getItem(position) deleteWord(word) } + + private val clickedWord = { position:Int -> + val word = getItem(position) + clickWord(word) + } + } \ No newline at end of file From 4bc4784eaaf9d628655d0a9ebfdf90c1b851c4b9 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Wed, 17 Jul 2024 14:31:29 +0900 Subject: [PATCH 08/19] =?UTF-8?q?Feat[]:=20=EC=B6=94=EA=B0=80=20=EC=B9=B4?= =?UTF-8?q?=EC=B9=B4=EC=98=A4=EC=A7=80=EB=8F=84=20onMapError()=20=ED=98=B8?= =?UTF-8?q?=EC=B6=9C=20=EC=8B=9C=20=EC=97=90=EB=9F=AC=20=ED=99=94=EB=A9=B4?= =?UTF-8?q?=20=ED=91=9C=EC=8B=9C?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/MapActivity.kt | 4 ++- app/src/main/res/layout/map_error.xml | 32 +++++++++++++++++++ app/src/main/res/values/strings.xml | 1 + 3 files changed, 36 insertions(+), 1 deletion(-) create mode 100644 app/src/main/res/layout/map_error.xml diff --git a/app/src/main/java/campus/tech/kakao/map/MapActivity.kt b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt index 7cdb2025..d979b1dc 100644 --- a/app/src/main/java/campus/tech/kakao/map/MapActivity.kt +++ b/app/src/main/java/campus/tech/kakao/map/MapActivity.kt @@ -59,7 +59,9 @@ class MapActivity : AppCompatActivity() { } override fun onMapError(p0: Exception?) { - Log.e("MapActivity", "onMapError: ${p0?.message}", p0) + setContentView(R.layout.map_error) + val errorText = findViewById(R.id.map_error_text) + errorText.text = p0?.message } }, object: KakaoMapReadyCallback() { diff --git a/app/src/main/res/layout/map_error.xml b/app/src/main/res/layout/map_error.xml new file mode 100644 index 00000000..4d0a94b9 --- /dev/null +++ b/app/src/main/res/layout/map_error.xml @@ -0,0 +1,32 @@ + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 4d7654b1..9175c622 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -5,4 +5,5 @@ word 카카오 경기 성남시 분당구 백현동 532 + 오류가 발생했습니다.\n다시 시도해 주세요. \ No newline at end of file From 47c2fca80a37b74a4b65333b0ef3c4d669d015ec Mon Sep 17 00:00:00 2001 From: jmc98 Date: Thu, 18 Jul 2024 17:18:34 +0900 Subject: [PATCH 09/19] =?UTF-8?q?Docs[README.md]:=202=EB=8B=A8=EA=B3=84=20?= =?UTF-8?q?=EC=9A=94=EA=B5=AC=20=EC=82=AC=ED=95=AD=20=EC=9E=91=EC=84=B1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/README.md b/README.md index dc23030f..2bd5276d 100644 --- a/README.md +++ b/README.md @@ -10,4 +10,12 @@ - BottomSheet를 사용한다. - 카카오 API 사용을 위한 앱 키를 외부에 노출하지 않는다. - 가능한 MVVM 아키텍처 패턴을 적용하도록 한다. +- 코드 컨벤션을 준수하며 프로그래밍한다. + +## 2단계 - 테스트 +### 기능 요구 사항 +- 테스트 코드를 작성한다. +### 프로그래밍 요구 사항 +- 카카오 API 사용을 위한 앱 키를 외부에 노출하지 않는다. +- 가능한 MVVM 아키텍처 패턴을 적용하도록 한다. - 코드 컨벤션을 준수하며 프로그래밍한다. \ No newline at end of file From 3c24b64d4121e52ef56f912f5ddda9756e2b5176 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Thu, 18 Jul 2024 22:17:59 +0900 Subject: [PATCH 10/19] =?UTF-8?q?Test[FunTest]:=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?=EA=B2=80=EC=83=89=EC=96=B4=EB=A5=BC=5F=EC=9E=85=EB=A0=A5?= =?UTF-8?q?=ED=95=98=EB=A9=B4=5F=EA=B2=80=EC=83=89=5F=EA=B2=B0=EA=B3=BC=5F?= =?UTF-8?q?=ED=91=9C=EC=8B=9C=20=ED=95=A8=EC=88=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/FunTest.kt | 49 +++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 app/src/test/java/campus/tech/kakao/map/FunTest.kt diff --git a/app/src/test/java/campus/tech/kakao/map/FunTest.kt b/app/src/test/java/campus/tech/kakao/map/FunTest.kt new file mode 100644 index 00000000..8e50483d --- /dev/null +++ b/app/src/test/java/campus/tech/kakao/map/FunTest.kt @@ -0,0 +1,49 @@ +package campus.tech.kakao.map + +import android.app.Application +import android.content.Context +import android.os.Looper +import androidx.lifecycle.LifecycleOwner +import androidx.lifecycle.ViewModelProvider +import androidx.lifecycle.ViewModelStore +import androidx.lifecycle.map +import androidx.test.core.app.ApplicationProvider +import org.junit.Assert.assertEquals +import org.junit.Assert.assertTrue +import org.junit.Before +import org.junit.Test +import org.junit.runner.RunWith +import org.robolectric.RobolectricTestRunner +import org.robolectric.RuntimeEnvironment +import org.robolectric.Shadows +import org.robolectric.annotation.Config + +@RunWith(RobolectricTestRunner::class) +@Config(manifest = Config.NONE) +class FunTest { + private lateinit var model: MainViewModel + private lateinit var viewModelStore: ViewModelStore + private lateinit var context: Context + @Before + fun setUp() { + context = RuntimeEnvironment.getApplication() + viewModelStore = ViewModelStore() + model = MainViewModel(context as Application) + } + @Test + fun 검색어를_입력하면_검색_결과_표시(){ + val query = "이안아파트" + val expectedQueryResultName = arrayOf("이안금곡아파트", "대우이안아파트", "이안금곡아파트 관리사무소", + "대우이안아파트 정문", "이안동래센트럴시티아파트", "이안금곡아파트 입주자대표회의 전기차충전소", + "대우이안아파트 상가동", "CU 화명대우이안점", "대우이안아파트 지하주차장", + "이안공인중개사사무소", "부산시 북구 대우이안아파트 전기차충전소", "대우이안공인중개사사무소", + "삼계이안아파트", "이안센트럴포레장유1단지아파트", "성일이안시티아파트") + val expectedQueryResultCount = 15 + model.searchLocalAPI(query) + val actualQueryResult = model.documentList.value + actualQueryResult?.forEach { document -> + assertTrue(expectedQueryResultName.contains(document.placeName)) + } + + } +} \ No newline at end of file From 2c5abdfbd2ebcdcec062ea46a682989a20dbb56c Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 15:30:36 +0900 Subject: [PATCH 11/19] =?UTF-8?q?Test[SearchActivityUITest]:=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EC=9E=85=EB=A0=A5=ED=95=9C=5F=EA=B2=80=EC=83=89?= =?UTF-8?q?=EC=96=B4=5FX=5F=EB=88=8C=EB=9F=AC=5F=EC=82=AD=EC=A0=9C=20?= =?UTF-8?q?=ED=95=A8=EC=88=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../tech/kakao/map/SearchActivityUITest.kt | 33 +++++++++++++++++++ app/src/main/res/layout/activity_map.xml | 3 +- app/src/main/res/layout/activity_search.xml | 2 +- .../java/campus/tech/kakao/map/FunTest.kt | 12 ++----- 4 files changed, 38 insertions(+), 12 deletions(-) create mode 100644 app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt diff --git a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt new file mode 100644 index 00000000..2092bfe9 --- /dev/null +++ b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt @@ -0,0 +1,33 @@ +package campus.tech.kakao.map + +import androidx.test.espresso.Espresso.onView +import androidx.test.espresso.action.ViewActions.click +import androidx.test.espresso.action.ViewActions.replaceText +import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.matcher.ViewMatchers.withId +import androidx.test.espresso.matcher.ViewMatchers.withParent +import androidx.test.espresso.matcher.ViewMatchers.withText +import androidx.test.ext.junit.rules.ActivityScenarioRule +import androidx.test.ext.junit.runners.AndroidJUnit4 +import org.hamcrest.core.AllOf.allOf +import org.junit.Rule +import org.junit.Test +import org.junit.runner.RunWith +@RunWith(AndroidJUnit4::class) +class SearchActivityUITest { + @get:Rule + val searchActivityRule = ActivityScenarioRule(SearchActivity::class.java) + + + @Test + fun 입력한_검색어_X_눌러_삭제() { + val query = "박물관" + val search = onView(allOf(withId(R.id.search), withParent(withId(R.id.search_main)))) + val clear = onView(withId(R.id.search_clear)) + search.perform(replaceText(query)) + clear.perform(click()) + search.check(matches(withText(""))) + + } + +} \ No newline at end of file diff --git a/app/src/main/res/layout/activity_map.xml b/app/src/main/res/layout/activity_map.xml index 8b7646e4..ea1351fd 100644 --- a/app/src/main/res/layout/activity_map.xml +++ b/app/src/main/res/layout/activity_map.xml @@ -4,7 +4,6 @@ android:layout_height="match_parent" android:layout_width="match_parent"> @@ -22,7 +21,7 @@ android:elevation="5dp" android:padding="10dp"> diff --git a/app/src/test/java/campus/tech/kakao/map/FunTest.kt b/app/src/test/java/campus/tech/kakao/map/FunTest.kt index 8e50483d..4ecf2b11 100644 --- a/app/src/test/java/campus/tech/kakao/map/FunTest.kt +++ b/app/src/test/java/campus/tech/kakao/map/FunTest.kt @@ -2,20 +2,14 @@ package campus.tech.kakao.map import android.app.Application import android.content.Context -import android.os.Looper -import androidx.lifecycle.LifecycleOwner -import androidx.lifecycle.ViewModelProvider +import android.util.Log import androidx.lifecycle.ViewModelStore -import androidx.lifecycle.map -import androidx.test.core.app.ApplicationProvider -import org.junit.Assert.assertEquals import org.junit.Assert.assertTrue import org.junit.Before import org.junit.Test import org.junit.runner.RunWith import org.robolectric.RobolectricTestRunner import org.robolectric.RuntimeEnvironment -import org.robolectric.Shadows import org.robolectric.annotation.Config @RunWith(RobolectricTestRunner::class) @@ -38,12 +32,12 @@ class FunTest { "대우이안아파트 상가동", "CU 화명대우이안점", "대우이안아파트 지하주차장", "이안공인중개사사무소", "부산시 북구 대우이안아파트 전기차충전소", "대우이안공인중개사사무소", "삼계이안아파트", "이안센트럴포레장유1단지아파트", "성일이안시티아파트") - val expectedQueryResultCount = 15 model.searchLocalAPI(query) val actualQueryResult = model.documentList.value actualQueryResult?.forEach { document -> - assertTrue(expectedQueryResultName.contains(document.placeName)) + assert(expectedQueryResultName.contains(document.placeName)) } + } } \ No newline at end of file From 504a272b5c0663f1496a2329e308e89b73f041e2 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 17:15:28 +0900 Subject: [PATCH 12/19] =?UTF-8?q?Test[SearchActivityUITest]:=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EA=B2=80=EC=83=89=5F=EA=B2=B0=EA=B3=BC=5F=ED=81=B4?= =?UTF-8?q?=EB=A6=AD=EC=8B=9C=20=ED=95=A8=EC=88=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/build.gradle.kts | 4 ++++ .../tech/kakao/map/SearchActivityUITest.kt | 16 ++++++++++++++++ 2 files changed, 20 insertions(+) diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 9b4327c9..760dea52 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -48,6 +48,9 @@ android { viewBinding = true buildConfig = true } + testOptions { + animationsDisabled = true + } } dependencies { @@ -75,5 +78,6 @@ dependencies { androidTestImplementation("androidx.test.espresso:espresso-core:3.5.1") androidTestImplementation("androidx.test:rules:1.5.0") androidTestImplementation("androidx.test.espresso:espresso-intents:3.5.1") + androidTestImplementation("androidx.test.espresso:espresso-contrib:3.5.1") } fun getApiKey(key: String): String = gradleLocalProperties(rootDir, providers).getProperty(key) \ No newline at end of file diff --git a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt index 2092bfe9..a540f793 100644 --- a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt +++ b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt @@ -1,18 +1,26 @@ package campus.tech.kakao.map +import androidx.test.InstrumentationRegistry import androidx.test.espresso.Espresso.onView import androidx.test.espresso.action.ViewActions.click import androidx.test.espresso.action.ViewActions.replaceText import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.contrib.RecyclerViewActions +import androidx.test.espresso.matcher.ViewMatchers.isDisplayed import androidx.test.espresso.matcher.ViewMatchers.withId import androidx.test.espresso.matcher.ViewMatchers.withParent import androidx.test.espresso.matcher.ViewMatchers.withText import androidx.test.ext.junit.rules.ActivityScenarioRule import androidx.test.ext.junit.runners.AndroidJUnit4 +import campus.tech.kakao.map.adapter.DocumentAdapter +import okhttp3.internal.wait import org.hamcrest.core.AllOf.allOf +import org.junit.Before import org.junit.Rule import org.junit.Test import org.junit.runner.RunWith +import java.lang.Thread.sleep + @RunWith(AndroidJUnit4::class) class SearchActivityUITest { @get:Rule @@ -27,7 +35,15 @@ class SearchActivityUITest { search.perform(replaceText(query)) clear.perform(click()) search.check(matches(withText(""))) + } + @Test + fun 검색_결과_클릭시(){ + val query = "박물관" + val search = onView(allOf(withId(R.id.search), withParent(withId(R.id.search_main)))) + search.perform(replaceText(query)) + val searchResult = onView(withId(R.id.search_result_recycler_view)) + searchResult.check(matches(isDisplayed())) } } \ No newline at end of file From e3da3c0ab2a897b74f7e510d22432cb9fd345ca1 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 17:20:50 +0900 Subject: [PATCH 13/19] =?UTF-8?q?Test[SearchActivityUITest]:=20=EB=B3=80?= =?UTF-8?q?=EA=B2=BD=20=EA=B2=80=EC=83=89=5F=EA=B2=B0=EA=B3=BC=5F=ED=81=B4?= =?UTF-8?q?=EB=A6=AD=EC=8B=9C=20=ED=95=A8=EC=88=98=20=EC=9D=B4=EB=A6=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/SearchActivityUITest.kt | 8 +------- 1 file changed, 1 insertion(+), 7 deletions(-) diff --git a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt index a540f793..0bc09135 100644 --- a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt +++ b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt @@ -1,25 +1,19 @@ package campus.tech.kakao.map -import androidx.test.InstrumentationRegistry import androidx.test.espresso.Espresso.onView import androidx.test.espresso.action.ViewActions.click import androidx.test.espresso.action.ViewActions.replaceText import androidx.test.espresso.assertion.ViewAssertions.matches -import androidx.test.espresso.contrib.RecyclerViewActions import androidx.test.espresso.matcher.ViewMatchers.isDisplayed import androidx.test.espresso.matcher.ViewMatchers.withId import androidx.test.espresso.matcher.ViewMatchers.withParent import androidx.test.espresso.matcher.ViewMatchers.withText import androidx.test.ext.junit.rules.ActivityScenarioRule import androidx.test.ext.junit.runners.AndroidJUnit4 -import campus.tech.kakao.map.adapter.DocumentAdapter -import okhttp3.internal.wait import org.hamcrest.core.AllOf.allOf -import org.junit.Before import org.junit.Rule import org.junit.Test import org.junit.runner.RunWith -import java.lang.Thread.sleep @RunWith(AndroidJUnit4::class) class SearchActivityUITest { @@ -38,7 +32,7 @@ class SearchActivityUITest { } @Test - fun 검색_결과_클릭시(){ + fun 검색_결과_클릭시_검색_결과_나오는지_확인(){ val query = "박물관" val search = onView(allOf(withId(R.id.search), withParent(withId(R.id.search_main)))) search.perform(replaceText(query)) From a6c8a27a58ca3c824d7593cf355c02a8e21347d8 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 17:22:15 +0900 Subject: [PATCH 14/19] =?UTF-8?q?Test[SearchActivityUITest]:=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EA=B2=80=EC=83=89=EC=B0=BD=5F=ED=91=9C=EC=8B=9C=5F?= =?UTF-8?q?=ED=99=95=EC=9D=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/SearchActivityUITest.kt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt index 0bc09135..472d404e 100644 --- a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt +++ b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt @@ -20,7 +20,11 @@ class SearchActivityUITest { @get:Rule val searchActivityRule = ActivityScenarioRule(SearchActivity::class.java) - + @Test + fun 검색창_표시_확인(){ + val search = onView(withId(R.id.search)) + search.check(matches(isDisplayed())) + } @Test fun 입력한_검색어_X_눌러_삭제() { val query = "박물관" From 5bb33efc5abb23f5d3f1b5b49073f796d08d7584 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 17:23:21 +0900 Subject: [PATCH 15/19] =?UTF-8?q?Test[SearchActivityUITest]:=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20X=5F=EB=B2=84=ED=8A=BC=5F=ED=99=95=EC=9D=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/SearchActivityUITest.kt | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt index 472d404e..5fd92d8c 100644 --- a/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt +++ b/app/src/androidTest/java/campus/tech/kakao/map/SearchActivityUITest.kt @@ -25,6 +25,12 @@ class SearchActivityUITest { val search = onView(withId(R.id.search)) search.check(matches(isDisplayed())) } + + @Test + fun X_버튼_확인(){ + val clear = onView(withId(R.id.search_clear)) + clear.check(matches(isDisplayed())) + } @Test fun 입력한_검색어_X_눌러_삭제() { val query = "박물관" From bacca26dee3d0bff528ea7fe514809e1a6fdf117 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 17:34:50 +0900 Subject: [PATCH 16/19] =?UTF-8?q?Test[MapActivityUITest]:=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80=20=EA=B2=80=EC=83=89=EC=B0=BD=5F=EB=B3=B4=EC=9D=B4?= =?UTF-8?q?=EB=8A=94=EC=A7=80=5F=ED=99=95=EC=9D=B8=20=ED=95=A8=EC=88=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../tech/kakao/map/MapActivityUITest.kt | 22 +++++++++++++++++++ 1 file changed, 22 insertions(+) create mode 100644 app/src/androidTest/java/campus/tech/kakao/map/MapActivityUITest.kt diff --git a/app/src/androidTest/java/campus/tech/kakao/map/MapActivityUITest.kt b/app/src/androidTest/java/campus/tech/kakao/map/MapActivityUITest.kt new file mode 100644 index 00000000..ca9159a8 --- /dev/null +++ b/app/src/androidTest/java/campus/tech/kakao/map/MapActivityUITest.kt @@ -0,0 +1,22 @@ +package campus.tech.kakao.map + +import androidx.test.espresso.Espresso.onView +import androidx.test.espresso.assertion.ViewAssertions.matches +import androidx.test.espresso.matcher.ViewMatchers.isDisplayed +import androidx.test.espresso.matcher.ViewMatchers.withId +import androidx.test.ext.junit.rules.ActivityScenarioRule +import org.junit.Rule +import org.junit.Test + + +class MapActivityUITest { + @get:Rule + val mapActivityRule = ActivityScenarioRule(MapActivity::class.java) + + @Test + fun 검색창_보이는지_확인(){ + val mapSearch = onView(withId(R.id.map_search)) + mapSearch.check(matches(isDisplayed())) + } + +} \ No newline at end of file From 236eb18c66991a9bac16784a7538b23362d01957 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 20:50:01 +0900 Subject: [PATCH 17/19] =?UTF-8?q?Fix[MainViewModel]:=20=EC=88=98=EC=A0=95?= =?UTF-8?q?=20wordfromDocument?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit SearchWord 매개 변수 저장 순서 변경 --- app/src/main/java/campus/tech/kakao/map/MainViewModel.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt index a875368b..4a699f38 100644 --- a/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt +++ b/app/src/main/java/campus/tech/kakao/map/MainViewModel.kt @@ -23,7 +23,7 @@ class MainViewModel(application: Application): AndroidViewModel(application) { } private fun wordfromDocument(document: Document): SearchWord { - return SearchWord(document.placeName, document.categoryGroupName, document.addressName) + return SearchWord(document.placeName, document.addressName, document.categoryGroupName) } fun deleteWord(word: SearchWord){ wordDbHelper.deleteWord(word) From 42b46d9903bb6383d8b874de632f4eb5c2d5a7b8 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 20:51:05 +0900 Subject: [PATCH 18/19] =?UTF-8?q?Feat[FunTest]:=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?=EA=B2=80=EC=83=89=EC=96=B4=5F=EC=A0=80=EC=9E=A5=5F=EB=90=98?= =?UTF-8?q?=EB=8A=94=EC=A7=80=5F=ED=99=95=EC=9D=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../java/campus/tech/kakao/map/FunTest.kt | 21 ++++++++++++++++--- 1 file changed, 18 insertions(+), 3 deletions(-) diff --git a/app/src/test/java/campus/tech/kakao/map/FunTest.kt b/app/src/test/java/campus/tech/kakao/map/FunTest.kt index 4ecf2b11..54ba9347 100644 --- a/app/src/test/java/campus/tech/kakao/map/FunTest.kt +++ b/app/src/test/java/campus/tech/kakao/map/FunTest.kt @@ -2,9 +2,13 @@ package campus.tech.kakao.map import android.app.Application import android.content.Context -import android.util.Log +import androidx.lifecycle.LiveData import androidx.lifecycle.ViewModelStore -import org.junit.Assert.assertTrue +import campus.tech.kakao.map.dbHelper.SearchWordDbHelper +import campus.tech.kakao.map.dto.Document +import campus.tech.kakao.map.dto.SearchWord +import io.mockk.every +import io.mockk.mockk import org.junit.Before import org.junit.Test import org.junit.runner.RunWith @@ -37,7 +41,18 @@ class FunTest { actualQueryResult?.forEach { document -> assert(expectedQueryResultName.contains(document.placeName)) } + } - + @Test + fun 검색어_저장_되는지_확인(){ + val query = Document( + "이안아파트", "아파트", + "남양주", "10", + "10") + val expectedResult = SearchWord( + "이안아파트", "남양주", "아파트") + model.addWord(query) + val result = model.wordList.value?.contains(expectedResult) + if (result != null) assert(result) else assert(false) } } \ No newline at end of file From a88c1a72a43ed52122f7a32a7aab65d66d1aa157 Mon Sep 17 00:00:00 2001 From: jmc98 Date: Fri, 19 Jul 2024 21:35:17 +0900 Subject: [PATCH 19/19] =?UTF-8?q?Feat[FunTest]:=20=EC=B6=94=EA=B0=80=20?= =?UTF-8?q?=EA=B2=80=EC=83=89=EC=96=B4=5F=EC=A0=80=EC=9E=A5=5F=EB=90=98?= =?UTF-8?q?=EB=8A=94=EC=A7=80=5F=ED=99=95=EC=9D=B8=20=ED=95=A8=EC=88=98?= =?UTF-8?q?=EC=97=90=20=EC=82=AD=EC=A0=9C=20=ED=99=95=EC=9D=B8=20=EA=B8=B0?= =?UTF-8?q?=EB=8A=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/src/test/java/campus/tech/kakao/map/FunTest.kt | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/app/src/test/java/campus/tech/kakao/map/FunTest.kt b/app/src/test/java/campus/tech/kakao/map/FunTest.kt index 54ba9347..95127e48 100644 --- a/app/src/test/java/campus/tech/kakao/map/FunTest.kt +++ b/app/src/test/java/campus/tech/kakao/map/FunTest.kt @@ -9,6 +9,7 @@ import campus.tech.kakao.map.dto.Document import campus.tech.kakao.map.dto.SearchWord import io.mockk.every import io.mockk.mockk +import org.junit.Assert.assertFalse import org.junit.Before import org.junit.Test import org.junit.runner.RunWith @@ -44,7 +45,7 @@ class FunTest { } @Test - fun 검색어_저장_되는지_확인(){ + fun 검색어_저장_되고_삭제도_되는지_확인(){ val query = Document( "이안아파트", "아파트", "남양주", "10", @@ -54,5 +55,11 @@ class FunTest { model.addWord(query) val result = model.wordList.value?.contains(expectedResult) if (result != null) assert(result) else assert(false) + + val query2 = SearchWord( + "이안아파트", "남양주", "아파트") + model.deleteWord(query2) + val result2 = model.wordList.value?.contains(query2) + if (result2 != null) assertFalse(result2) else assert(false) } } \ No newline at end of file