Kotlin addTextChangeListenerラムダ?


103

KotlinでEditText addTextChangeListenerのラムダ式を作成するにはどうすればよいですか?以下はエラーになります:

passwordEditText.addTextChangedListener { charSequence  ->
    try {
        password = charSequence.toString()
    } catch (error: Throwable) {
        raise(error)
    }
}

2
どのようなエラーが発生しますか?
voddan 2016年

回答:


244

addTextChangedListener()かかるTextWatcher3つの方法とのインターフェースです。あなたが書いたものは、TextWatcherメソッドが1つしかない場合にのみ機能します。他の2つのメソッドを実装していないラムダに関連するエラーを推測します。今後は2つのオプションがあります。

  1. ラムダを捨てて、匿名の内部クラスを使用する
    editText.addTextChangedListener(object : TextWatcher {
      override fun afterTextChanged(s: Editable?) {
      }
    
      override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
      }
    
      override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
      }
    })
  1. ラムダ式を使用できるように拡張メソッドを作成します。
    fun EditText.afterTextChanged(afterTextChanged: (String) -> Unit) {
        this.addTextChangedListener(object : TextWatcher {
          override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
          }
    
          override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {
          }
    
          override fun afterTextChanged(editable: Editable?) {
            afterTextChanged.invoke(editable.toString())
          }
        })
    }

そして、次のように拡張機能を使用します。

editText.afterTextChanged { doSomethingWithText(it) }

4
個人的な好みまたはより良いスタイルかどうかはわかりませんが、拡張関数を式の本文に変換できます(fun foo() = ...
F. George

6
@ mEQ5aNLrK3lqs3kfSa5HbvsTWe0nIu変換できるのは当然です。ただし、関数が1行より長い場合は、括弧で囲んで、関数の開始と停止の場所を明確に示します。私はそれが読みやすさを向上させると信じていますが、それは完全にスタイルの好みです。私はそれを両方の方法で議論することができると思います:)
Andrew Orobator

2
関心のアウト:なぜ呼び出すafterTextChanged.invoke(...)のではなくafterTextChanged(...)
Felix D.

これでうまくいきました。私は再利用性のために2番目のオプションを選択しました。
Onie Maniego

21

少し古いですが、Kotlin Android拡張機能を使用すると、次のようなことができます。

editTextRequest.textChangedListener {
            afterTextChanged {
                // Do something here...
            }
}

追加のコードは必要ありません。追加するだけです。

implementation 'androidx.core:core-ktx:1.0.0'

4
android Xにリファクタリングした後でも、それは私にとってはうまくいきません。何が間違っているのでしょうか?
–NícolasSchirmer、2018

3
私にとってもうまくいきません。KTXはこの拡張機能を提供していないようですが、 KAndroid ソリューションは完全に機能します。
Igor Wojda


16

このコアktx依存関係を追加する

implementation 'androidx.core:core-ktx:1.0.0'

あなたは単にしなければならない

passwordEditText.doAfterTextChanged{ }


12

このKotlinサンプルがそれを明確にするのに役立つことを願っています:

class MainFragment : Fragment() {

    private lateinit var viewModel: MainViewModel

override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
                          savedInstanceState: Bundle?): View {
    val view = inflater.inflate(R.layout.main_fragment, container, false)

    view.user.addTextChangedListener(object : TextWatcher {
        override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {

        }

        override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {

        }

        override fun afterTextChanged(s: Editable) {
                userLayout.error =
                        if (s.length > userLayout.counterMaxLength) {
                            "Max character length is: ${userLayout.counterMaxLength}"
                        } else null
        }
    })
    return view
}

override fun onActivityCreated(savedInstanceState: Bundle?) {
    super.onActivityCreated(savedInstanceState)
    viewModel = ViewModelProviders.of(this).get(MainViewModel::class.java)
    // TODO: Use the ViewModel
   }
}

このXMLレイアウトでは:

<android.support.design.widget.TextInputLayout
    android:id="@+id/userLayout"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    app:counterMaxLength="5"
    app:counterEnabled="true"
    android:hint="user_name">

    <android.support.design.widget.TextInputEditText
        android:id="@+id/user"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
</android.support.design.widget.TextInputLayout>

そしてこれGradle

android {
    compileSdkVersion 'android-P'
...
}
    api 'com.android.support:design:28.0.0-alpha1'

    implementation 'com.android.support:appcompat-v7:28.0.0-alpha1' // appcompat library

12

試して :

passwordEditText.addTextChangedListener(object:TextWatcher{override fun afterTextChanged(s: Editable?) {

    }

    override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) {
    }

    override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) {

    }

})

10

使用するimplementation 'androidx.core:core-ktx:1.1.0-alpha05'場合は使用できます

For android.widget.TextView
TextWatcher 
TextView.doBeforeTextChanged(crossinline action: (text: CharSequence?, start: Int, count: Int, after: Int) -> Unit)
Add an action which will be invoked before the text changed.

TextWatcher 
TextView.doOnTextChanged(crossinline action: (text: CharSequence?, start: Int, count: Int, after: Int) -> Unit)
Add an action which will be invoked when the text is changing.

TextWatcher 
TextView.doAfterTextChanged(crossinline action: (text: Editable?) -> Unit)

https://developer.android.com/reference/kotlin/androidx/core/widget/package-summary#extension-functions


4

遅刻してすみません!

implementation 'androidx.core:core-ktx:1.1.0'モジュールのbuild.gradleファイルに追加すると、次のように使用できます

etPlayer1.doOnTextChanged { text, start, count, after -> // Do stuff }

2

別の選択肢はKAndroidライブラリです-

implementation 'com.pawegio.kandroid:kandroid:0.8.7@aar'

次に、あなたはこのようなことをすることができます...

editText.textWatcher { afterTextChanged { doSomething() } }

明らかに、ライブラリ全体を使用して問題を解決することは過剰ですが、Android SDKのボイラープレートコードを排除する他の便利な拡張機能も付属しています。


2

kotlinの名前付きパラメーターを使用できます。

private val beforeTextChangedStub: (CharSequence, Int, Int, Int) -> Unit = { _, _, _, _ -> }
private val onTextChangedStub: (CharSequence, Int, Int, Int) -> Unit = { _, _, _, _ -> }
private val afterTextChangedStub: (Editable) -> Unit = {}

fun EditText.addChangedListener(
        beforeTextChanged: (CharSequence, Int, Int, Int) -> Unit = beforeTextChangedStub,
        onTextChanged: (CharSequence, Int, Int, Int) -> Unit = onTextChangedStub,
        afterTextChanged: (Editable) -> Unit = afterTextChangedStub
) = addTextChangedListener(object : TextWatcher {
    override fun beforeTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) {
        beforeTextChanged(charSequence, i, i1, i2)
    }

    override fun onTextChanged(charSequence: CharSequence, i: Int, i1: Int, i2: Int) {
        onTextChanged(charSequence, i, i1, i2)
    }

    override fun afterTextChanged(editable: Editable) {
        afterTextChanged(editable)
    }
})

0

コアktx依存関係を追加する

implementation 'androidx.core:core-ktx:1.3.0'

そして、あなたは単にこのように実装することができます

    edit_text.addTextChangedListener { it: Editable? ->
      // Do your stuff here
    }

-9

これはきれいに見えます:

passwordEditText.setOnEditorActionListener { 
    textView, keyCode, keyEvent ->
    val DONE = 6

    if (keyCode == DONE) {                       
         // your code here
    }
    false
}

1
私は本当にあなたたちを知らないが、私の答えは私のために働いたとの上と下の答えよりも短い..です
レミュエルADANE

1
これはコードが意図するとおりに機能し、DONEが押されるとアクションを実行します。条件の外にトーストを置くことでコードを変更しましたが、TAB / DONE /などを押したときにのみ発火するようです。しかし、他のキャラクターにはありません。
Onie Maniego
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.