Collections

|

코틀린은 자바에서와는 달리 리스트(List)나 맵(Map) 등의 자료 구조에 읽기 전용 객체와 수정이 가능한 객체로 나누어 놓았습니다.

코틀린에서는 리스트를 생성할 때 클래스를 선언하지 않아도 만들 수 있는 함수를 제공하고 있습니다. 리스트를 생성하는 함수는 listOf이며, 맵을 생성하는 함수는 mapOf입니다. 이 함수를 이용하면 읽기 전용의 리스트가 만들어집니다.

fun init() {
    val numList = listOf(1, 3, 7, 5, 10)
    val strList = listOf("snowdeer", "ran", "yang")

    val map = mapOf(1 to "one", 2 to "two", 3 to "three")
}

읽기 전용 리스트를 만들면 get, first, last 등 데이터를 읽을 수 있는 함수들만 제공됩니다.

만약, 수정이 가능한 리스트들을 만들고 싶으면 mutableListOf, mutableMapOf 함수를 사용하면 됩니다.

fun init() {
    val numList = mutableListOf(1, 3, 7, 5, 10)
    val strList = mutableListOf("snowdeer", "ran", "yang")
    val map = mutableMapOf(1 to "one", 2 to "two", 3 to "three")
}

Kotlin Fragment 템플릿

|

팩토리 메소드(newInstance)로 Fragment를 생성하는 예제 코드입니다.

import android.os.Bundle
import android.support.v4.app.Fragment
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup

private const val ARG_TITLE = "TITLE"

class TodoListFragment : Fragment() {

    private var title: String? = ""

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        arguments?.let {
            title = it.getString(ARG_TITLE)
        }
    }

    companion object {
        fun newInstance(title: String): TodoListFragment {
            return TodoListFragment().apply {
                arguments = Bundle().apply {
                    putString(ARG_TITLE, title)
                }
            }
        }
    }

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
                              savedInstanceState: Bundle?): View? {
        return inflater.inflate(R.layout.fragment_todolist, container, false)

    }
}

consider adding at least one Activity with an ACTION-VIEW intent filter

|

AndroidManifest.xml 에서 다음과 같은 오류가 발생할 경우 해결하는 방법입니다.

App is not indexable by Google Search; consider adding at least one Activity with an ACTION-VIEW intent filter.

적어도 하나 이상의 액티비티의 인텐트 필터에 다음 라인을 추가하면 됩니다.

<action android:name="android.intent.action.VIEW"/>

for 문과 when 문

|

for 키워드

1 부터 10까지의 반복문은 다음과 같습니다. (1 <= i < 10)

for(i in 1..9) {
    println(i)
}

또는 다음과 같이 사용할 수 있습니다.

for(i in 1 until 10) {
    println(i)
}

만약 9 부터 1까지 거꾸로 내려가고 싶을 때는 다음과 같이 downTo 키워드를 사용합니다.

for(i in 9 downTo 1) {
    println(i)
}

만약 증가폭을 특정 숫자만큼 하고 싶을 때는 step 키워드를 사용합니다.

for(i in 1 until 10 step 2) {
    println(i)
}

ArrayList 등의 리스트를 사용할 경우 다음과 같이 for 문을 작성할 수 있습니다.

fun test() {
    val list = ArrayList<String>()

    list.add("Hello")
    list.add("Nice to meet you")
    list.add("Good bye")

    for (i in 0 until list.size) {
        println(list[i])
    }
}

만약 iterator를 사용할 경우 다음과 같이 코드를 작성할 수 있습니다.

fun test() {
    val list = ArrayList<String>()

    list.add("Hello")
    list.add("Nice to meet you")
    list.add("Good bye")

    for (text: String in list) {
        println(text)
    }
}


when 키워드

when은 Java에서의 switch와 사용법이 유사합니다.

fun test(menu: Int) {
    when (menu) {
        R.id.menu_start -> {

        }
        R.id.menu_stop -> {
            
        }
    }
}

break 구문 없이 중괄호({}) 만으로 분기를 나눌 수 있습니다. 중괄호 없이 사용하더라도 다음 -> 구문이 나올 때까지 실행합니다.

when 문 안에서 간단한 연산도 가능합니다.

fun main(args: Array<String>) {
    test(2)
    test(10)
    test(4)
    test(1)
}

fun test(menu: Int) {
    when {
        menu <= 3 -> {
            println("$menu <= 3")
        }
        menu <= 7 -> {
            println("3 < $menu <= 7")
        }
        else -> {
            println("$menu > 7")
        }
    }
}

Kotlin ViewPager 예제

|

build.gradle

dependencies {
    implementation 'com.android.support:design:28.0.0'
}


activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  tools:context=".MainActivity">

  <android.support.design.widget.TabLayout
    android:id="@+id/tab"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    app:tabIndicatorColor="@color/colorAccent"
    app:tabSelectedTextColor="@color/colorAccent"
    app:tabTextColor="@color/colorPrimary"/>

  <android.support.v4.view.ViewPager
    android:id="@+id/viewpager"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"/>

</LinearLayout>


fragment_page.xml

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent">

  <TextView
    android:id="@+id/textview"
    android:textStyle="bold"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:text="Hello"
    android:textColor="@android:color/holo_blue_dark"
    android:textSize="32sp"/>

</FrameLayout>


BaseFragment.kt

import android.support.v4.app.Fragment

abstract class BaseFragment : Fragment() {
    abstract fun title(): String
}


FirstPageFragment.kt

import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.TextView
import com.snowdeer.hellokotlin.R

class FirstPageFragment : BaseFragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        val view = inflater.inflate(R.layout.fragment_page, container, false)
        view.findViewById<TextView>(R.id.textview).text = "First Page"

        return view
    }

    override fun title(): String {
        return "First"
    }
}


SecondPageFragment.kt

import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.TextView
import com.snowdeer.hellokotlin.R

class SecondPageFragment : BaseFragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        val view = inflater.inflate(R.layout.fragment_page, container, false)
        view.findViewById<TextView>(R.id.textview).text = "Second Page"

        return view
    }

    override fun title(): String {
        return "Second"
    }
}


ThirdPageFragment.kt

import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.TextView
import com.snowdeer.hellokotlin.R

class ThirdPageFragment : BaseFragment() {

    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        val view = inflater.inflate(R.layout.fragment_page, container, false)
        view.findViewById<TextView>(R.id.textview).text = "Third Page"

        return view
    }

    override fun title(): String {
        return "Third"
    }
}


SamplePagerAdapter.kt

class SamplePagerAdapter : FragmentPagerAdapter {

    private val list: ArrayList<BaseFragment> = ArrayList();

    constructor(fragmentManager: FragmentManager) : super(fragmentManager) {
        list.add(FirstPageFragment())
        list.add(SecondPageFragment())
        list.add(ThirdPageFragment())
    }

    override fun getPageTitle(position: Int): CharSequence? {
        return list[position].title()
    }

    override fun getItem(position: Int): Fragment {
        return list.get(position)
    }

    override fun getCount(): Int {
        return list.size
    }
}


MainActivity.kt

import android.support.v7.app.AppCompatActivity
import android.os.Bundle
import com.snowdeer.hellokotlin.viewpager.SamplePagerAdapter
import kotlinx.android.synthetic.main.activity_main.*

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        val adapter = SamplePagerAdapter(supportFragmentManager)
        viewpager.adapter = adapter

        tab.setupWithViewPager(viewpager)
    }
}