12

背景

我已经搜索了一种与 Gmail 收件人字段具有相似外观和感觉的方法,它允许以非常酷的方式自动填充项目:

在此处输入图像描述

内置在 Android 框架中并负责此操作的类称为“ MultiAutoCompleteTextView ”。

问题

MultiAutoCompleteTextView 非常基本,但它没有足够的示例、教程和库来了解如何像在 Gmail 等上那样自定义它。

我想知道如何自定义它以处理任何类型的数据,并且我将完全控制它(例如添加、删除和获取它已自动完成的项目)。

我试过的

我找到了实现它的下一个可能的方法:

  1. 使用第三个库,例如splitwise-TokenAutoComplete。缺点:它非常有问题,并且在某些设备上效果不佳。
  2. 创建我自己的方式(如此处所示)。缺点:需要很长时间,我可能需要处理与图书馆相同的问题。
  3. 使用Google 的代码(在此处找到)。缺点:它真的不是可定制的。

我决定使用#3(谷歌的芯片库)。

目前用于获取 Google 库中使用的联系人列表的代码:

public List<RecipientEntry> doQuery() {
    final Cursor cursor = mContentResolver.query(mQuery.getContentUri(), mQuery.getProjection(), null, null, null);
    final LinkedHashMap<Long, List<RecipientEntry>> entryMap = new LinkedHashMap<Long, List<RecipientEntry>>();
    final List<RecipientEntry> nonAggregatedEntries = new ArrayList<RecipientEntry>();
    final Set<String> existingDestinations = new HashSet<String>();
    while (cursor.moveToNext())
        putOneEntry(new TemporaryEntry(cursor, false /* isGalContact */), true, entryMap, nonAggregatedEntries,
                existingDestinations);
    cursor.close();
    final List<RecipientEntry> entries = new ArrayList<RecipientEntry>();
    {
        for (final Map.Entry<Long, List<RecipientEntry>> mapEntry : entryMap.entrySet()) {
            final List<RecipientEntry> entryList = mapEntry.getValue();
            for (final RecipientEntry recipientEntry : entryList)
                entries.add(recipientEntry);
        }
        for (final RecipientEntry entry : nonAggregatedEntries)
            entries.add(entry);
    }
    return entries;
}

它工作正常,但我在添加和删除项目时遇到了困难。

我认为通过调用“getContactIds”来获取项目,但是关于修改芯片中的项目,这很难找到。

例如,我尝试向 "submitItemAtPosition" 添加类似的函数,这似乎添加了从适配器中找到的新实体。它确实添加了,但联系人的显示名称并未显示在芯片本身上。

问题

想了很多之后,我决定使用谷歌的代码。

遗憾的是,正如我所写的,视图及其类对它的使用非常严格。

  1. 如何解耦视图并使其更具可定制性?我怎样才能让它使用任何类型的数据,而不仅仅是谷歌所做的?

  2. 如何获取输入了哪些项目(变成了“筹码”),并且还能够从外部移除或添加项目?

4

2 回答 2

5

我已经成功添加了添加收件人的功能。唯一要记住的是只有在视图获得其大小后才调用它(如何在此处进行操作的示例):

/** adds a recipient to the view. note that it should be called when the view has determined its size */
public void addRecipient(final RecipientEntry entry) {
    if (entry == null)
        return;
    clearComposingText();

    final int end = getSelectionEnd();
    final int start = mTokenizer.findTokenStart(getText(), end);

    final Editable editable = getText();
    QwertyKeyListener.markAsReplaced(editable, start, end, "");
    final CharSequence chip = createChip(entry, false);
    if (chip != null && start >= 0 && end >= 0) {
        editable.replace(start, end, chip);
    }
    sanitizeBetween();
}

private void submitItemAtPosition(final int position) {
    final RecipientEntry entry = createValidatedEntry(getAdapter().getItem(position));
    if (entry == null)
        return;
    addRecipient(entry);
}

并且,对于删除:

/** removes a chip of a recipient from the view */
public void removeRecipient(final RecipientEntry entry) {
    final DrawableRecipientChip[] chips = getSpannable().getSpans(0, getText().length(),
            DrawableRecipientChip.class);
    final List<DrawableRecipientChip> chipsToRemove = new ArrayList<DrawableRecipientChip>();
    for (final DrawableRecipientChip chip : chips)
        if (chip.getDataId() == entry.getDataId())
            chipsToRemove.add(chip);
    for (final DrawableRecipientChip chip : chipsToRemove)
        removeChip(chip);
}

正如我之前所写的,要获取当前在视图中的contactIds 列表,请使用 "getContactIds()" 。另一种选择是:

/** returns a collection of all of the chips' items. key is the contact id, and the value is the recipient itself */
public Map<Long, RecipientEntry> getChosenRecipients() {
    final Map<Long, RecipientEntry> result = new HashMap<Long, RecipientEntry>();
    final DrawableRecipientChip[] chips = getSortedRecipients();
    if (chips != null)
        for (final DrawableRecipientChip chip : chips) {
            // if(result.)
            final long contactId = chip.getContactId();
            if (!result.containsKey(contactId))
                result.put(contactId, chip.getEntry());
        }
    return result;
}

也许我应该在 Github 上发布代码。

我现在唯一想念的是芯片本身的好听众:添加、移除和更换芯片时。在大多数情况下,我可以检测到它,但当用户按下退格键并移除芯片时则检测不到。


编辑:还添加了监听器。现在我在搜索联系人时发现了一个错误。它似乎搜索正常的英文字母,就好像它们是电话号码一样。


编辑:我决定在 GitHub 上放一个示例和一个库,在这里。希望尽快用更多有用的功能更新它。

我真的很高兴为代码做出任何贡献。

于 2014-04-06T09:46:45.350 回答
1

这个库似乎允许您配置它搜索的内容,同时还匹配 Material Design 外观。它似乎也基于谷歌的芯片库。我在调查类似问题时碰巧发现了它。

https://github.com/klinker41/android-chips

于 2015-08-21T21:32:19.533 回答