使用视图模型和实时数据的联系提供者

时间:2019-01-07 17:04:34

标签: android-architecture-components android-livedata

我的android应用正在使用contacts providers向用户显示所有联系人。我正在按照https://developer.android.com/training/contacts-provider/retrieve-names

上的教程/文档使用Loaders来加载联系人

但是在链接https://developer.android.com/guide/components/loaders中,提到从Android P开始不推荐使用加载程序。

  

从Android P(API 28)开始不推荐使用加载程序。推荐的   处理活动时处理加载数据的选项,以及   片段生命周期是结合使用ViewModels和   LiveData。 ViewModels可以像Loaders一样保留配置更改,但是   用更少的样板。 LiveData提供了一种生命周期感知的方法   加载可以在多个ViewModel中重用的数据。你也可以   使用MediatorLiveData和任何可观察到的查询来组合LiveData,   例如来自Room数据库的数据,可以用来观察   数据。在某些情况下还可以使用ViewModels和LiveData   您无权访问LoaderManager的地方,例如   服务。串联使用两者提供了一种轻松的方式来访问   无需处理UI生命周期的数据。至   了解有关LiveData的更多信息,请参见LiveData指南并了解更多信息   有关ViewModel的信息,请参见ViewModel指南。

所以我的问题是:

1。我们如何使用android view模型和联系人提供者的实时数据来获取联系人?
2。我们可以使用会议室数据库作为联系提供者吗?

在下面,您可以找到我尝试使用Android View模型和实时数据从ContactProviders获取联系人的源代码链接。

https://github.com/deepak786/phonebook-contacts
3。上面的源代码中有哪些可以改进的地方,以便提取会更快?

感谢与问候
迪帕克

2 个答案:

答案 0 :(得分:2)

下面您可以找到一个非常简单的使用MVVM加载联系人的解决方案:

https://github.com/NaarGes/Android-Contact-List

这里有一些代码,以防链接不再起作用。

首先,让我们为联系人 UserObject.java

创建一个简单的POJO
public class UserObject {

    private String email, name, phone;

    public UserObject() {
        // EMPTY CONSTRUCTOR FOR FIREBASE REALTIME DATABASE
    }

    public UserObject(String email, String name, String phone) {
        this.email = email;
        this.name = name;
        this.phone = phone;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getPhone() {
        return phone;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }
}

现在,让我们创建存储库 ContactRepository.java

public class ContactRepository {

    private Context context;

    private static final String TAG = "debinf ContRepo";

    public ContactRepository(Context context) {
        this.context = context;
    }

    public List<UserObject> fetchContacts() {
        List<UserObject> contacts = new ArrayList<>();

        Cursor cursor = context.getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, null, null, null);
        Log.i(TAG, "fetchContacts: cursor.getCount() is "+cursor.getCount());
        if ((cursor != null ? cursor.getCount() : 0) > 0) {
            while (cursor.moveToNext()) {

                String name = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
                String phone = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));

                UserObject contact = new UserObject("",name, phone);

                Log.i(TAG, "fetchContacts: phone is "+phone);

                contacts.add(contact);
            }
        }
        if (cursor != null) {
            cursor.close();
        }
        return contacts;
    }
}

接下来,我们创建我们的 ContactViewModel.java

public class ContactViewModel extends ViewModel {

    private ContactRepository repository;
    private MutableLiveData<List<UserObject>> contacts;

    public ContactViewModel(Context context) {
        repository = new ContactRepository(context);
        contacts = new MutableLiveData<>();
    }

    public MutableLiveData<List<UserObject>> getContacts() {
        contacts.setValue(repository.fetchContacts());
        return contacts;
    }
}

接下来,我们为factory创建一个ViewModel ContactViewModelFactory.java

public class ContactViewModelFactory implements ViewModelProvider.Factory {

    private Context context;

    public ContactViewModelFactory(Context context) {
        this.context = context;
    }

    @NonNull
    @Override
    public <T extends ViewModel> T create(@NonNull Class<T> modelClass) {
        if (modelClass.isAssignableFrom(ContactViewModel.class)) {
            return (T) new ContactViewModel(context);
        }
        throw new IllegalArgumentException("Unknown ViewModel class");
    }
}

我们不要忘记在我们的AndroidManifest中添加权限

<uses-permission android:name="android.permission.READ_CONTACTS" />

最后,我们在 MainActivity.java

中请求许可
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        requestPermissions(new String[]{Manifest.permission.WRITE_CONTACTS,Manifest.permission.READ_CONTACTS,
                Manifest.permission.WRITE_EXTERNAL_STORAGE,Manifest.permission.READ_EXTERNAL_STORAGE}, PERMISSION_REQUEST);
}

并使我们的联系人浮出水面

ContactViewModelFactory factory = new ContactViewModelFactory(this);
viewModel = ViewModelProviders.of(this, factory).get(ContactViewModel.class);

viewModel.getContacts().observe(this, new Observer<List<UserObject>>() {
    @Override
    public void onChanged(@Nullable List<UserObject> userObjects) {
        Log.i(TAG, "ViewModel: userObjects size is "+userObjects.size());
        Log.i(TAG, "ViewModel: userObjects size is "+userObjects.get(1).getPhone());
    }
});

答案 1 :(得分:0)

class ContactsViewModel(private val contentResolver: ContentResolver) : ViewModel() 
{

    lateinit var contactsList: LiveData<PagedList<Contact>>

    fun loadContacts() {
        val config = PagedList.Config.Builder()
                .setPageSize(20)
                .setEnablePlaceholders(false)
                .build()
        contactsList = LivePagedListBuilder<Int, Contact>(
                ContactsDataSourceFactory(contentResolver), config).build()
    }
}

class ContactsDataSourceFactory(private val contentResolver: ContentResolver) :
        DataSource.Factory<Int, Contact>() {

    override fun create(): DataSource<Int, Contact> {
        return ContactsDataSource(contentResolver)
    }
}

class ContactsDataSource(private val contentResolver: ContentResolver) :
        PositionalDataSource<Contact>() {

    companion object {
        private val PROJECTION = arrayOf(
                ContactsContract.Contacts._ID,
                ContactsContract.Contacts.LOOKUP_KEY,
                ContactsContract.Contacts.DISPLAY_NAME_PRIMARY
        )
    }

    override fun loadInitial(params: LoadInitialParams, callback: LoadInitialCallback<Contact>) {
        callback.onResult(getContacts(params.requestedLoadSize, params.requestedStartPosition), 0)
    }

    override fun loadRange(params: LoadRangeParams, callback: LoadRangeCallback<Contact>) {
        callback.onResult(getContacts(params.loadSize, params.startPosition))
    }

    private fun getContacts(limit: Int, offset: Int): MutableList<Contact> {
        val cursor = contentResolver.query(ContactsContract.Contacts.CONTENT_URI,
                PROJECTION,
                null,
                null,
                ContactsContract.Contacts.DISPLAY_NAME_PRIMARY +
                        " ASC LIMIT " + limit + " OFFSET " + offset)

        cursor.moveToFirst()
        val contacts: MutableList<Contact> = mutableListOf()
        while (!cursor.isAfterLast) {
            val id = cursor.getLong(cursor.getColumnIndex(PROJECTION[0]))
            val lookupKey = cursor.getString(cursor.getColumnIndex(PROJECTION[0]))
            val name = cursor.getString(cursor.getColumnIndex(PROJECTION[2]))
            contacts.add(Contact(id, lookupKey, name))
            cursor.moveToNext()
        }
        cursor.close()

        return contacts
    }
}

请找到完整的源代码here