yeon's blog

[Kotlin] 팁 페이지 본문

Kotlin/커뮤니티 앱

[Kotlin] 팁 페이지

yeonii 2024. 1. 12. 21:22

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를 이용해 구현하였다.