This library provides the GroupsQuery
API that allows you to get groups associated with an
Account
or no account.
An instance of the GroupsQuery
API is obtained by,
val query = Contacts(context).groups().query()
To get all of the groups for all accounts (including no/null account),
val groupsFromAllAccounts = Contacts(context)
.groups()
.query()
.find()
ℹ️ It is recommended to get sets of groups for a single account at a time to avoid confusion.
To limit the search to only those Groups associated with one of the given accounts,
.accounts(accounts)
For example, to limit the search to groups belonging to only one account,
.accounts(Account("[email protected]", "com.google"))
ℹ️ For more info, read Query for Accounts.
If no accounts are specified (this function is not called or called with no Accounts), then all Groups from all accounts (including the null account) are included in the search.
To order resulting Groups using one or more fields,
.orderBy(fieldOrder)
For example, to order groups by account name,
.orderBy(GroupsFields.AccountName.asc())
String comparisons ignores case by default. Each orderBys provides ignoreCase
as an optional
parameter.
Use GroupsFields
to construct the orderBys.
To limit the amount of groups returned and/or offset (skip) a specified number of groups, use
the limit
and offset
functions;
.limit(limit)
.offset(offset)
For more info, read Using limit and offset in queries.
To execute the query,
.find()
To cancel a query amid execution,
.find { returnTrueIfQueryShouldBeCancelled() }
The find
function optionally takes in a function that, if it returns true, will cancel query
processing as soon as possible. The function is called numerous times during query processing to
check if processing should stop or continue. This gives you the option to cancel the query.
This is useful when used in multi-threaded environments. One scenario where this would be frequently used is when performing queries as the user types a search text. You are able to cancel the current query when the user enters new text.
For example, to automatically cancel the query inside a Kotlin coroutine when the coroutine is cancelled,
launch {
withContext(coroutineContext) {
val groups = query.find { !isActive }
}
}
Queries are executed when the find
function is invoked. The work is done in the same thread as
the call-site. This may result in a choppy UI.
To perform the work in a different thread, use the Kotlin coroutine extensions provided in the async
module.
For more info, read Execute work outside of the UI thread using coroutines.
You may, of course, use other multi-threading libraries or just do it yourself =)
ℹ️ Extensions for Kotlin Flow and RxJava are also in the project roadmap.
Queries require the android.permission.READ_CONTACTS
permission. If not granted, the query will
do nothing and return an empty list.
To perform the query with permission, use the extensions provided in the permissions
module.
For more info, read Permissions handling using coroutines.
You may, of course, use other permission handling libraries or just do it yourself =)
Use the contacts.core.GroupsFields
combined with the extensions from contacts.core.Where
to form
WHERE clauses.
ℹ️ This docs page will not provide a tutorial on database where clauses. It assumes that you know the basics. If you don't know the basics, then search for sqlite where clause.
For example, to find groups with a specific title,
.where { Title equalToIgnoreCase "friends" }
To get a list of groups by IDs,
.where { Id `in` groupIds }
Each account will have its own set of system and user-created groups. This means that there may be multiple groups with the same title belonging to different accounts. This is not a bug. This is why it is recommended to only get sets of groups per account, especially if there is more than one account in the system.
When you perform a query that returns groups from more than one account, you will get everything
in the same GroupsList
. This list is just like any other List
except it also provides an extra
function that allows you to get a sublist with groups belonging only to a particular account.
val groupsFromAccount = groupsList.from(account)
This is equivalent to,
val groupsFromAccount = groupsList.filter { it.account == account }
It serves more as documentation and hint that you should really not be mixing groups from different accounts in the same list as it could cause confusion. However, if you know what you are doing and you are not confused, then do what you like :D
This is also nice for Java users to not have to perform the filtering themselves.