yeon's blog
[Kotlin] 팁 페이지 본문
1. 레이아웃 설정 및 아이콘 넣기
src/main/res/drawable 파일에 팁 페이지에 넣을 아이콘 png들을 넣어준 후 아래와 같이 코드를 작성한다.
fragment_tip.xml 추가코드
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginBottom="70dp"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="150dp">
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_all"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_cook"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_economy"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="150dp">
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_else"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_hobby"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_interior"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="150dp">
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_life"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:src="@drawable/main_icon_oneroom"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
<ImageView
android:layout_marginTop="8dp"
android:layout_weight="1"
android:layout_width="120dp"
android:layout_height="120dp" />
</LinearLayout>
</LinearLayout>
실행 화면
2. 컨텐츠 리스트 만들기 - RecyclerView
*RecyclerView 사용법은 따로 정리해 포스팅 할 예정이므로 자세한 설명은 -- 생략 --
activity_content_list.xml 전체코드
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".contentsList.ContentListActivity">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="20dp"
android:text="카테고리 영역 텍스트"
android:textSize="20sp"
android:textStyle="bold"
android:textColor="#000000"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/rv"
android:layout_marginTop="80dp"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
content_rv_item.xml 전체코드
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="200dp"
android:orientation="vertical">
<ImageView
android:id="@+id/imageArea"
android:src="@drawable/main_icon_oneroom"
android:layout_marginTop="8dp"
android:layout_marginBottom="8dp"
android:layout_width="match_parent"
android:layout_height="100dp" />
<TextView
android:id="@+id/textArea"
android:text="Text Area"
android:textStyle="bold"
android:textSize="15sp"
android:gravity="center"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<ImageView
android:id="@+id/bookmarkArea"
android:src="@drawable/bookmark_white"
android:layout_marginTop="10dp"
android:layout_gravity="center"
android:layout_width="30dp"
android:layout_height="30dp" />
</LinearLayout>
ContentModel 전체코드
package com.example.mysololife.contentsList
data class ContentModel (
var title: String = "",
var imageUrl: String = ""
)
ContentRVAdapter 전체코드
package com.example.mysololife.contentsList
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.ImageView
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import com.example.mysololife.R
class ContentRVAdapter(val items : ArrayList<ContentModel>) : RecyclerView.Adapter<ContentRVAdapter.Viewholder>() {
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ContentRVAdapter.Viewholder {
val v = LayoutInflater.from(parent.context).inflate(R.layout.content_rv_item, parent, false)
return Viewholder(v)
}
override fun onBindViewHolder(holder: ContentRVAdapter.Viewholder, position: Int) {
holder.bindItems(items[position])
}
override fun getItemCount(): Int {
return items.size
}
inner class Viewholder(itemView : View) : RecyclerView.ViewHolder(itemView) {
fun bindItems(item : ContentModel) {
// itemView == content_rv_item
val contentTitle = itemView.findViewById<TextView>(R.id.textArea)
contentTitle.text = item.title
}
}
}
ContentListActivity 전체코드
package com.example.mysololife.contentsList
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import androidx.recyclerview.widget.GridLayoutManager
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import com.example.mysololife.R
class ContentListActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_content_list)
val rv : RecyclerView = findViewById(R.id.rv)
val items = ArrayList<ContentModel>()
items.add(ContentModel("imageUrl1", "title1"))
items.add(ContentModel("imageUrl2", "title2"))
items.add(ContentModel("imageUrl3", "title3"))
val rvAdapter = ContentRVAdapter(items)
rv.adapter = rvAdapter
rv.layoutManager = GridLayoutManager(this, 2)
}
}
TipFragment 전체코드
package com.example.mysololife.fragments
import android.content.Intent
import android.os.Bundle
import androidx.fragment.app.Fragment
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.databinding.DataBindingUtil
import androidx.navigation.findNavController
import com.example.mysololife.R
import com.example.mysololife.contentsList.ContentListActivity
import com.example.mysololife.databinding.FragmentHomeBinding
import com.example.mysololife.databinding.FragmentTipBinding
class TipFragment : Fragment() {
private lateinit var binding: FragmentTipBinding
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
}
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
binding = DataBindingUtil.inflate(inflater, R.layout.fragment_tip, container, false)
binding.category1.setOnClickListener {
val intent = Intent(context, ContentListActivity::class.java)
startActivity(intent)
}
binding.homeTap.setOnClickListener {
it.findNavController().navigate(R.id.action_tipFragment_to_homeFragment)
}
binding.talkTap.setOnClickListener {
it.findNavController().navigate(R.id.action_tipFragment_to_talkFragment)
}
binding.bookmarkTap.setOnClickListener {
it.findNavController().navigate(R.id.action_tipFragment_to_bookmarkFragment)
}
binding.storeTap.setOnClickListener {
it.findNavController().navigate(R.id.action_tipFragment_to_storeFragment)
}
return binding.root
}
}
3. 컨텐츠 리스트 만들기 - Glide
Glide를 사용하기 위해 build.gradle과 AndroidManifest.xml에 미리 설정해야 하는 것이 있다.
build.gradle 추가코드
implementation("com.github.bumptech.glide:glide:4.12.0")
annotationProcessor("com.github.bumptech.glide:compiler:4.12.0")
AndroidManifest.xml 추가코드
<uses-permission android:name="android.permission.INTERNET" />
ContentListActivity 파일에 임의로 넣었던,
items.add(ContentModel("imageUrl1", "title1"))
items.add(ContentModel("imageUrl2", "title2"))
items.add(ContentModel("imageUrl3", "title3"))
val rvAdapter = ContentRVAdapter(items)
이 코드 부분을 실제 웹사이트의 url로 바꾸어 주었다.
실제 url이기 때문에 코드를 첨부하지는 않는다.
Glide를 추가함에 따라, Adapter 코드도 아래와 같이 수정하였다.
ContentRVAdapter 수정 후 전체코드
package com.example.mysololife.contentsList
import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.ImageView
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import com.bumptech.glide.Glide
import com.example.mysololife.R
class ContentRVAdapter(val context : Context, val items : ArrayList<ContentModel>) : RecyclerView.Adapter<ContentRVAdapter.Viewholder>() {
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ContentRVAdapter.Viewholder {
val v = LayoutInflater.from(parent.context).inflate(R.layout.content_rv_item, parent, false)
return Viewholder(v)
}
override fun onBindViewHolder(holder: ContentRVAdapter.Viewholder, position: Int) {
holder.bindItems(items[position])
}
override fun getItemCount(): Int {
return items.size
}
inner class Viewholder(itemView : View) : RecyclerView.ViewHolder(itemView) {
fun bindItems(item : ContentModel) {
// itemView == content_rv_item
val contentTitle = itemView.findViewById<TextView>(R.id.textArea)
val imageViewArea = itemView.findViewById<ImageView>(R.id.imageArea)
contentTitle.text = item.title
Glide.with(context)
.load(item.imageUrl)
.into(imageViewArea)
}
}
}
4. 컨텐츠 리스트 만들기 - RecyclerView Item Click
마지막으로 각 아이콘을 클릭했을 때, 실제 webUrl로 접속할 수 있도록 구현할 것이다.
우선 title과 imageUrl만 담도록 구현했던 ContentModel 코드에 webUrl을 추가해준다.
ContentModel 전체코드
data class ContentModel (
var title: String = "",
var imageUrl: String = "",
var webUrl: String = ""
)
ContentRVAdapter 추가 후 전체코드
package com.example.mysololife.contentsList
import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.ImageView
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView
import com.bumptech.glide.Glide
import com.example.mysololife.R
class ContentRVAdapter(val context : Context, val items : ArrayList<ContentModel>) : RecyclerView.Adapter<ContentRVAdapter.Viewholder>() {
// ItemClick 수동 생성 - 추가코드
interface ItemClick {
fun onCLick(view : View, position: Int)
}
var itemClick : ItemClick? = null
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ContentRVAdapter.Viewholder {
val v = LayoutInflater.from(parent.context).inflate(R.layout.content_rv_item, parent, false)
return Viewholder(v)
}
override fun onBindViewHolder(holder: ContentRVAdapter.Viewholder, position: Int) {
// 추가코드
if (itemClick != null) {
holder.itemView.setOnClickListener { v ->
itemClick?.onCLick(v, position)
}
}
holder.bindItems(items[position])
}
override fun getItemCount(): Int {
return items.size
}
inner class Viewholder(itemView : View) : RecyclerView.ViewHolder(itemView) {
fun bindItems(item : ContentModel) {
// itemView == content_rv_item
val contentTitle = itemView.findViewById<TextView>(R.id.textArea)
val imageViewArea = itemView.findViewById<ImageView>(R.id.imageArea)
contentTitle.text = item.title
Glide.with(context)
.load(item.imageUrl)
.into(imageViewArea)
}
}
}
위와 같이 ItemClick 인터페이스를 수동으로 구현해주었다.
그리고나서 ContentListActivity 파일에 클릭 이벤트 코드를 추가해 주었다. (마찬가지로 webUrl 추가는 첨부 x)
ContentListActivity 전체코드
rvAdapter.itemClick = object : ContentRVAdapter.ItemClick {
override fun onCLick(view: View, position: Int) {
val intent = Intent(this@ContentListActivity, ContentShowActivity::class.java)
intent.putExtra("url", items[position].webUrl)
startActivity(intent)
}
}
클릭 이벤트를 구현했으니 이제 web로 이동할 수 있는 webView를 구현한다.
클릭했을 때 이동하는 Activity는 ContentShowActivity파일에 작성했다.
activity_content_show.xml 전체코드
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".contentsList.ContentShowActivity">
<WebView
android:id="@+id/webView"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
ContentShowActivity 전체코드
package com.example.mysololife.contentsList
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.webkit.WebView
import com.example.mysololife.R
class ContentShowActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_content_show)
val getUrl = intent.getStringExtra("url")
val webView : WebView = findViewById(R.id.webView)
webView.loadUrl(getUrl.toString())
}
}
실행 화면
5. Firebase Realtime Database
*Firebase의 Realtime Database 이용 방법은 https://hyeyeon-ii.tistory.com/47 참고!
꿀팁 페이지에 있는 아이콘 별로 해당 아이콘에 해당하는 정보들을 저장하고, 불러올 수 있도록 realtime database를 이용해 구현하였다.
'Kotlin > 커뮤니티 앱' 카테고리의 다른 글
[Kotlin] 메인화면 레이아웃 (Navigation) (0) | 2024.01.06 |
---|---|
[Kotlin] Firebase 회원가입, 로그인, 로그아웃, 익명로그인 기능 구현 (2) | 2024.01.03 |
[Kotlin] 레이아웃 구현 (1) | 2024.01.02 |