Fragment와 함께 데이터 바인딩을 사용하는 방법


182

공식 Google 문서 https://developer.android.com/tools/data-binding/guide.html의 데이터 바인딩 예제를 따르려고합니다.

활동이 아닌 조각에 데이터 금지를 적용하려고한다는 것을 제외하고.

컴파일 할 때 현재 발생하는 오류는

Error:(37, 27) No resource type specified (at 'text' with value '@{marsdata.martianSols}.

onCreate 조각은 다음과 같습니다.

@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    MartianDataBinding binding = MartianDataBinding.inflate(getActivity().getLayoutInflater());
    binding.setMarsdata(this);
}

onCreateView 조각은 다음과 같습니다.

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    return inflater.inflate(R.layout.martian_data, container, false);
}

조각에 대한 레이아웃 파일의 일부는 다음과 같습니다.

<?xml version="1.0" encoding="utf-8"?>

<layout xmlns:android="http://schemas.android.com/apk/res/android">
    <data>
        <variable
            name="marsdata"
            type="uk.co.darkruby.app.myapp.MarsDataProvider" />
    </data>
...

        <TextView
            android:layout_height="wrap_content"
            android:layout_width="wrap_content"
            android:text="@{marsdata.martianSols}"
        />

    </RelativeLayout>
</layout>

내 의심은 MartianDataBinding그것이 어떤 레이아웃 파일을 바인딩해야하는지 모르기 때문에 오류입니다. 어떤 제안?

답변:


354

데이터 바인딩 구현이 있어야합니다 onCreateView당신에 그 존재를 바인딩 데이터 삭제, 조각의 방법 OnCreate방법을, 당신 onCreateView과 같아야합니다 :

public View onCreateView(LayoutInflater inflater, 
                         @Nullable ViewGroup container, 
                         @Nullable Bundle savedInstanceState) {
    MartianDataBinding binding = DataBindingUtil.inflate(
            inflater, R.layout.martian_data, container, false);
    View view = binding.getRoot();
    //here data must be an instance of the class MarsDataProvider
    binding.setMarsdata(data);
    return view;
}

Binding 클래스를 생성하려면 super에 대한 호출을 추가해야했습니다.
joey_g216

3
나는 몇 시간 동안이 문제로 어려움을 겪고 있습니다. 문제는 내가 잘못된 견해를 반환했다는 것입니다. +1
TharakaNirmana

1
View view = binding.getRoot(); 나는 오랫동안이 문제에 갇혀있어서 developer.android.com에서 그것에 관한 문서를 찾을 수 없을 정도로 합법적으로 꽤 화가났습니다 ... 문제를 해결했습니다. 감사합니다!
Victor Ude

1
LiveData 및 ViewModel을 사용하는 경우이 답변 을 읽으십시오 .
Big McLargeHuge

1
setMarsdata () 란 무엇입니까? 여기에 setViewModel ()을 사용한다고 생각합니까 ??
suv

59

실제로 inflateDataBindingUtil이 아니라 생성 된 Binding 의 메소드 를 사용하는 것이 좋습니다 .

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    MainFragmentBinding binding = MainFragmentBinding.inflate(inflater, container, false);
    //set variables in Binding
    return binding.getRoot();
}

DataBindingUtil.inflate ()에 대한 문서 :

layoutId를 미리 알 수없는 경우에만이 버전을 사용하십시오. 그렇지 않으면 생성 된 바인딩의 팽창 방법을 사용하여 형식이 안전한 팽창을 보장하십시오.


불행히도 이것은 cannot be resolved to a type빌드 오류로 인해 나를 죽이고 있습니다. 내 의견으로는 신뢰할 수 없습니다. 먼저 가서으로 DataBindingUtil.inflate(inflater, R.layout.fragment_camera, container, false);변경 FragmentCameraBinding.inflate(inflater, container, false);하면 작동하지만 다시 빌드하면 오류가 다시 발생합니다.
Alex Burdusel

잘 작동합니다. 실제로 생성 된 바인딩 파일에서 자동으로 선택되므로 레이아웃 res id (이전에 궁금해 했음)를 지정할 필요가 없습니다.
eC Droid

2
이 예제에서 프래그먼트 레이아웃 ID (예 : R.layout.fragment_)를 어디에 설정합니까?
레닌 Raj Rajasekaran 1

이것이 정답이어야합니다. 레이아웃 생성 바인딩 대신 다음을 사용하는 것이 좋습니다.DataBindingUtil.inflate
mochadwi

@LeninRajRajasekaran 레이아웃 ID는 MainFragmentBinding클래스 사용을 통해 암시됩니다 . 해당 클래스는 레이아웃 파일에서 생성되므로 원하는 레이아웃이 자동으로 적용됩니다.
Emil S.

19

다른 답변조차 잘 작동 할 수 있지만 최선의 방법을 말하고 싶습니다.

Android 설명서Binding class's inflate 에서 권장하는대로 사용하십시오 .

하나의 옵션은 팽창시키는 DataBindingUtil 것이지만 바인딩 클래스를 생성했는지 모른다는 것 입니다.

-자동 생성 binding class을 사용하는 대신 해당 클래스를 사용하십시오 DataBindingUtil.

자바에서

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    HomeFragmentBinding binding = HomeFragmentBinding.inflate(inflater, container, false);
    //set binding variables here
    return binding.getRoot();
}

코 틀린에서

lateinit var binding: HomeFragmentBinding 
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    binding = HomeFragmentBinding.inflate(inflater, container, false)
    return binding.root
}

에서 DataBindingUtil의 클래스 문서 를 알 수 있습니다.

부풀게 하다

T inflate (LayoutInflater inflater, 
                int layoutId, 
                ViewGroup parent, 
                boolean attachToParent)

layoutId를 미리 알 수없는 경우에만이 버전을 사용하십시오. 그렇지 않으면 생성 된 바인딩의 팽창 방법을 사용하여 형식이 안전한 팽창을 보장하십시오.

레이아웃 바이너리 클래스가 생성되지 않은 경우 @ 이 답변을 참조하십시오 .


왜 유일한 인수로 사용하는 inflate방법을 사용하지 LayoutInflater않습니까?
Florian Walther

@FlorianWalther 없이도 작동 ViewGroup container합니까?
Khemraj

글쎄, 나는이 의견을 언제 썼는지 몰랐다. 그러나 여기에 좋은 대답이 있습니다. stackoverflow.com/questions/61571381/…
Florian Walther

1
@FlorianWalther 좋아, 내가 대답, 읽은 container때 필요한 attachToRoot입니다 true.
Khemraj

16

당신이 사용하는 경우 의 ViewModel을 하고 LiveData 이 충분한 구문입니다

코 틀린 구문 :

override fun onCreateView(
    inflater: LayoutInflater,
    container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    return MartianDataBinding.inflate(
        inflater,
        container,
        false
    ).apply {
        lifecycleOwner = viewLifecycleOwner
        vm = viewModel    // Attach your view model here
    }.root
}

10

Android DataBinding에서 이것을 시도하십시오

FragmentMainBinding binding;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        binding = DataBindingUtil.inflate(inflater, R.layout.fragment_main, container, false);
        View rootView = binding.getRoot();
        initInstances(savedInstanceState);
        return rootView;
}

7

아래에서 언급 한 것처럼 단순히 뷰 객체를 검색 할 수 있습니다

public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

View view = DataBindingUtil.inflate(inflater, R.layout.layout_file, container, false).getRoot();

return view;

}

7

코 틀린 구문 :

lateinit var binding: MartianDataBinding
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    binding = DataBindingUtil.inflate(inflater, R.layout.martian_data, container, false)
    return binding.root
}

7

다만 대부분이 말했다,하지만 설정하는 것을 잊지 해달라고로 LifeCycleOwner의
자바 샘플을 즉,

public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {
    super.onCreateView(inflater, container, savedInstanceState);
    BindingClass binding = DataBindingUtil.inflate(inflater, R.layout.fragment_layout, container, false);
    ModelClass model = ViewModelProviders.of(getActivity()).get(ViewModelClass.class);
    binding.setLifecycleOwner(getActivity());
    binding.setViewmodelclass(model);

    //Your codes here

    return binding.getRoot();
}

5

내 코드에서 일하고 있습니다.

private FragmentSampleBinding dataBiding;
private SampleListAdapter mAdapter;

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    super.onCreateView(inflater, container, savedInstanceState);
    dataBiding = DataBindingUtil.inflate(inflater, R.layout.fragment_sample, null, false);
    return mView = dataBiding.getRoot();
}

5

데이터 바인딩 프래그먼트의 완전한 예제

FragmentMyProgramsBinding은 res / layout / fragment_my_programs에 대해 생성 된 바인딩 클래스입니다.

public class MyPrograms extends Fragment {
    FragmentMyProgramsBinding fragmentMyProgramsBinding;

    public MyPrograms() {
        // Required empty public constructor
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // Inflate the layout for this fragment
    FragmentMyProgramsBinding    fragmentMyProgramsBinding = DataBindingUtil.inflate(inflater, R
                .layout.fragment_my_programs, container, false);
        return fragmentMyProgramsBinding.getRoot();
    }

    @Override
    public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

    }
}

2

데이터 바인딩에 대한 매우 유용한 블로그 : https://link.medium.com/HQY2VizKO1

class FragmentBinding<out T : ViewDataBinding>(
    @LayoutRes private val resId: Int
) : ReadOnlyProperty<Fragment, T> {

    private var binding: T? = null

    override operator fun getValue(
        thisRef: Fragment,
        property: KProperty<*>
    ): T = binding ?: createBinding(thisRef).also { binding = it }

    private fun createBinding(
        activity: Fragment
    ): T = DataBindingUtil.inflate(LayoutInflater.from(activity.context),resId,null,true)
}

Fragment에서 다음과 같이 바인딩 값을 선언하십시오.

private val binding by FragmentBinding<FragmentLoginBinding>(R.layout.fragment_login)

이것을 조각으로 쓰는 것을 잊지 마십시오

override fun onCreateView(
    inflater: LayoutInflater,
    container: ViewGroup?,
    savedInstanceState: Bundle?
): View? {
    return binding.root
}

1

코 틀린의 또 다른 예 :

override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? {
    val binding = DataBindingUtil
            .inflate< MartianDataBinding >(
                    inflater,
                    R.layout.bla,
                    container,
                    false
            )

    binding.modelName = // ..

    return binding.root
}

"MartianDataBinding"이라는 이름은 레이아웃 파일의 이름에 따라 다릅니다. 파일 이름이 "martian_data"이면 올바른 이름은 MartianDataBinding입니다.


0

모두에 대해 말하지만 inflate(), 우리가 그것을 사용하려면 어떻게해야 onViewCreated()합니까?

bind(view)구체적인 바인딩 클래스의 메소드를 사용 하여의 ViewDataBinding인스턴스 를 가져올 수 있습니다 view.


일반적으로 BaseFragment를 다음과 같이 작성합니다 (간체).

// BaseFragment.kt
abstract fun layoutId(): Int

override fun onCreateView(inflater, container, savedInstanceState) = 
    inflater.inflate(layoutId(), container, false)

그리고 자식 조각에 사용하십시오.

// ConcreteFragment.kt
override fun layoutId() = R.layout.fragment_concrete

override fun onViewCreated(view, savedInstanceState) {
    val binding = FragmentConcreteBinding.bind(view)
    // or
    val binding = DataBindingUtil.bind<FragmentConcreteBinding>(view)
}


모든 Fragments가 데이터 바인딩을 사용하는 경우 type 매개 변수를 사용하여 더 간단하게 만들 수도 있습니다.

abstract class BaseFragment<B: ViewDataBinding> : Fragment() {
    abstract fun onViewCreated(binding: B, savedInstanceState: Bundle?)

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        onViewCreated(DataBindingUtil.bind<B>(view)!!, savedInstanceState)
    }
}

나는 널이 아닌 것을 주장하는 것이 괜찮다는 것을 모른다. 그러나 당신은 아이디어를 얻는다. 널 입력 가능하게하려면 할 수 있습니다.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.