Синтетический и пользовательский макет Kotlin в DialogFragment
Скажем, у меня есть этот макет:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<ImageButton
android:id="@+id/add_dep_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentEnd="true"
android:layout_alignParentRight="true"
android:layout_marginEnd="5dp"
android:layout_marginRight="5dp"
android:src="@android:drawable/ic_input_add" />
<EditText
android:id="@+id/add_dep_text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignBottom="@id/add_dep_btn"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true"
android:layout_alignTop="@id/add_dep_btn"
android:layout_marginLeft="5dp"
android:layout_marginStart="5dp"
android:layout_toLeftOf="@id/add_dep_btn"
android:layout_toStartOf="@id/add_dep_btn" />
<android.support.v7.widget.RecyclerView
android:id="@+id/dep_list"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/add_dep_btn" />
<TextView
android:id="@+id/empty_text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_below="@id/add_dep_text"
android:layout_margin="20dp"
android:gravity="center"
android:text="@string/no_dep"
android:textSize="22sp" />
</RelativeLayout>
И я использую его в DialogFragment:
class DepartmentChoiceDialog : DialogFragment() {
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
val builder = AlertDialog.Builder(activity)
builder.setTitle(R.string.choose_or_create_dep)
.setView(R.layout.department_chooser_dialog)
.setNegativeButton(android.R.string.cancel, { d, i ->
d.cancel()
})
return builder.create()
}
}
если я ссылаюсь на виджет, используя синтез:
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
dep_list.layoutManager = LinearLayoutManager(activity)
dep_list.itemAnimator = DefaultItemAnimator()
dep_list.setHasFixedSize(true)
}
Я получил эту ошибку во время выполнения:
java.lang.NullPointerException: попытка вызвать виртуальный метод 'android.view.View android.view.View.findViewById(int)' в ссылке на нулевой объект в MyDialog._ $_ findCachedViewById (DepartmentChoiceDialog.kt: 0)
Я не понимаю, как использовать синтез в случае DialogFragment. Он отлично работает во Фрагменте и Деятельности.
Ответы
Ответ 1
Я нашел способ, который работает для пользовательских диалогов.
class ServerPickerDialogFragment: AppCompatDialogFragment()
{
// Save your custom view at the class level
lateinit var customView: View;
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View?
{
// Simply return the already inflated custom view
return customView
}
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
// Inflate your view here
customView = context!!.layoutInflater.inflate(R.layout.dialog_server_picker, null)
// Create Alert Dialog with your custom view
return AlertDialog.Builder(context!!)
.setTitle(R.string.server_picker_dialog_title)
.setView(customView)
.setNegativeButton(android.R.string.cancel, null)
.create()
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?)
{
super.onViewCreated(view, savedInstanceState)
// Perform remaining operations here. No null issues.
rbgSelectType.setOnCheckedChangeListener({ _, checkedId ->
if(checkedId == R.id.rbSelectFromList) {
// XYZ
} else {
// ABC
}
})
}
}
Ответ 2
Похоже, что это пока не поддерживается по умолчанию, но я нашел самый простой способ сделать это так. В базовом диалоговом классе:
protected abstract val containerView: View
override fun getView() = containerView
В подклассе:
override val containerView by unsafeLazy {
View.inflate(context, R.layout.dialog_team_details, null) as ViewGroup
}
Затем вы можете использовать синтетические представления, как обычно, и использовать containerView
как представление для своего диалога.
Ответ 3
Изменить на onCreateView.
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return inflater.inflate(R.layout.department_chooser_dialog, container, false)
}
и используйте пользовательский заголовок (TextView) и отмените (Button) в разделе department_chooser_dialog
onActivityCreated будет запускаться после onCreateView и будет в порядке.
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
dep_list.layoutManager = LinearLayoutManager(activity)
dep_list.itemAnimator = DefaultItemAnimator()
dep_list.setHasFixedSize(true)
}
Ответ 4
Поэтому я не уверен, что это было решено... Я просто натолкнулся на это. Если у вас есть пользовательское диалоговое окно, создайте класс, который расширяет DialogFragment и использует объект "dialog" для импорта представлений в макете. Я использую Android Studio 3.1.3
и Kotlin version 1.2.41
на момент написания.
import kotlinx.android.synthetic.main.your_custom_layout.*
class SelectCountryBottomSheet : BottomSheetDialogFragment() {
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
val dialog = super.onCreateDialog(savedInstanceState) as BottomSheetDialog
dialog.setContentView(R.layout.your_custom_layout)
dialog.some_custom_close_button.setOnClickListener { dismiss() }
return dialog
}
}
Ответ 5
Предыдущий ответ не будет работать, потому что onViewCreated не вызывается, когда вы используете onCreateDialog. Сначала нужно импортировать kotlinx... department_chooser_dialog. просмотреть.dep_list, затем использовать его следующим образом:
import kotlinx.android.synthetic.main.department_chooser_dialog.view.dep_list
...
class DepartmentChoiceDialog : DialogFragment() {
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
val builder = AlertDialog.Builder(activity)
val dialog = inflater.inflate(R.layout.department_chooser_dialog, null)
dialog.dep_list.layoutManager = LinearLayoutManager(activity)
dialog.dep_list.itemAnimator = DefaultItemAnimator()
dialog.dep_list.setHasFixedSize(true)
builder.setTitle(R.string.choose_or_create_dep)
.setView(dialog)
...
Ответ 6
Переместите код с onActivityCreated
на onViewCreated
.
Вот так:
import kotlinx.android.synthetic.main.department_chooser_dialog.dep_list
override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
dep_list.apply {
layoutManager = LinearLayoutManager(activity)
itemAnimator = DefaultItemAnimator()
setHasFixedSize(true)
}
}
На самом деле я не смотрел глубже в сгенерированный код, и, возможно, есть ошибка.
Ответ 7
Поскольку значение представления по умолчанию из фрагмента (kotlin generate method _ $ _ findCachedViewById), но если мы создаем диалог "Просмотр из диалога", представление о фрагменте имеет нулевое значение, поэтому мы не можем напрямую использовать по умолчанию xxx, но мы можем использовать dialog.xxx заменить значение по умолчанию ххх
Ответ 8
SetContentView находится внутри вызовов OnActivityCreated. Так что по синтетическому набору элементов управления для мониторинга событий необходимо позвонить сюда:
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return inflater.inflate(R.layout.layout_email_ga_code, container)
}
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
}
override fun onActivityCreated(savedInstanceState: Bundle?) {
dialog?.window?.requestFeature(Window.FEATURE_NO_TITLE)
super.onActivityCreated(savedInstanceState)
dialog?.window?.setLayout(WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.MATCH_PARENT)
btn_back?.setOnClickListener {
mOnClickListener?.onClickCancel()
dismiss()
}
}
И это сработало.
Ответ 9
Код kotlin в фрагменте выглядит так:
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return inflater.inflate(R.layout.your_layout, container, false).apply {
mContentView = this
button1.setOnClickListener {
//do something
}
}
}
После декомпиляции байт-кода вы можете увидеть реализацию синтетических свойств: ((Button)this._$_findCachedViewById(id.button1))
и метод _$_findCachedViewById
:
public View _$_findCachedViewById(int var1) {
if (this._$_findViewCache == null) {
this._$_findViewCache = new HashMap();
}
View var2 = (View)this._$_findViewCache.get(var1);
if (var2 == null) {
View var10000 = this.getView();
if (var10000 == null) {
return null;
}
var2 = var10000.findViewById(var1);
this._$_findViewCache.put(var1, var2);
}
return var2;
}
поэтому магией является только this.getView()
. Свойство Fragment.mView
назначается после Fragment.onCreateView(inflater, container, savedInstanceState)
, если вы используете Kotlin Synthetic Properties в методе onCreateView(), будет NPE. Код из FragmentManager.moveToState()
:
case Fragment.CREATED:
...
f.mView = f.performCreateView(f.performGetLayoutInflater(
f.mSavedFragmentState), container,
f.mSavedFragmentState);
...
Чтобы исправить NPE, убедитесь, что метод getView
возвращает ненулевой вид.
private var mContentView: View? = null
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
return inflater.inflate(R.layout.your_layout, container, false).apply {
mContentView = this
}
}
override fun getView(): View? {
return mContentView
}
и в onDestroyView()
цикла onDestroyView()
установите для mContentView
значение null.
override fun onDestroyView() {
super.onDestroyView()
mView = null
}
Ответ 10
Представления доступны через представление, которое вы надуваете в onCreateDialog
. Итак, если вы сохраняете представление в переменной (rootView
), вы можете получить доступ к представлениям из любого метода внутри YourDialogFragment
.
// ...
import kotlinx.android.synthetic.main.your_layout.view.*
class YourDialogFragment : DialogFragment() {
private lateinit var rootView: View
override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
rootView = activity.layoutInflater.inflate(R.layout.your_layout, null as ViewGroup?)
rootView.someTextView.text = "Hello" // works
}
}