Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

autocomplete: Support @-wildcard in user-mention autocomplete #889

Open
wants to merge 6 commits into
base: main
Choose a base branch
from

Conversation

sm-sayedi
Copy link
Collaborator

@sm-sayedi sm-sayedi commented Aug 16, 2024

Screenshot

Screen recording

Wildcard.mention.mp4

Fixes: #234

Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks @sm-sayedi! A few high-level comments below.

@@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,8H4A2,2 0 0,0 2,10V14A2,2 0 0,0 4,16H5V20A1,1 0 0,0 6,21H8A1,1 0 0,0 9,20V16H12L17,20V4L12,8M15,15.6L13,14H4V10H13L15,8.4V15.6M21.5,12C21.5,13.71 20.54,15.26 19,16V8C20.53,8.75 21.5,10.3 21.5,12Z" /></svg>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What was the origin of this icon — is it from a particular place in Figma?

(I don't see a bullhorn.svg in Zulip web, which would be the other usual source.)

See git log --stat assets/icons/ for example previous commit messages.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That icon was from materialdesignicons.com. Updated the icon in this revision to match it with the web. The web uses a Font Awesome icon.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool. That sounds like a case of this issue, then:

Given Vlad's reply there, we'll probably want some other icon for this. I think the most efficient way to resolve that is to start a thread in #mobile and ask Vlad there what icon he'd like us to use.

He did recently finish a design that's related (I still need to update the relevant issue to reflect that):
https://www.figma.com/design/1JTNtYo9memgW7vV6d0ygq/Zulip-Mobile?node-id=3859-2936&t=z2va3CAInaEg1eWy-0
but looking there, it doesn't have an example for a wildcard, only for a group or an individual user. So go ahead and ask in chat, and include a link to that Figma node for context.

Comment on lines 202 to 205
/// The full name of the wildcard to be shown in autocomplete suggestions.
///
/// Ex: "all (Notify channel)" or "everyone (Notify recipients)".
final String fullDisplayName;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems more about our UI than about the Zulip server API. So let's find a home for it outside of lib/api/.

I think a good solution would look similar to MentionAutocompleteResult and its subclasses. Instead of a User being an object that knows how to behave as an @-mention autocomplete candidate, we'd have an object that describes an @-mention autocomplete candidate and refers to a User to do so.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In fact probably a good solution would be for these "autocomplete candidate" objects to be the same objects as the result objects.

I'll try doing a quick refactor in that direction.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(→ #889 (comment) below)

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems more about our UI than about the Zulip server API. So let's find a home for it outside of lib/api/.

Now that the model class is moved outside of lib/api, is it okay for fullDisplayName to still live in the class?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah. Try to find the arrangement that seems cleanest (and once the PR is ready, reviewers may have suggestions), but between lib/model/ and lib/widgets/ either way is basically fine for where that lives.

Map<String, Wildcard> wildcardsForNarrow(Narrow narrow) => Map.fromEntries(
_wildcardsForNarrow(narrow).map((w) => MapEntry(w.name, w)));

List<Wildcard> _wildcardsForNarrow(Narrow narrow) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should live in an autocomplete-specific file — store.dart, and the PerAccountStore class definition, are fairly crowded places, so anything that can naturally go somewhere more specific should do so.

Comment on lines 352 to 353
return [
Wildcard(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does this mirror the corresponding logic in zulip-mobile? (I haven't read this closely.) A link for comparison would be handy.

Copy link
Collaborator Author

@sm-sayedi sm-sayedi Aug 22, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It doesn't follow the zulip-mobile in the exact same way regarding the code structure. Here's the zulip-mobile implementation: https://github.com/zulip/zulip-mobile/blob/main/src/autocomplete/WildcardMentionItem.js. Should I provide this link in the code itself?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK. Let's try to make sure it gets the same results, even if the code ends up organized differently. (There should be test cases in zulip-mobile that will be helpful to translate over, to verify that.)

Then also include a link to that zulip-mobile implementation in either a code comment or commit message. For the link, make it a GitHub permalink:
#692 (comment)
#884 (comment)
That'll be helpful for reviewers, as a point of comparison.

@gnprice
Copy link
Member

gnprice commented Aug 19, 2024

In fact probably a good solution would be for these "autocomplete candidate" objects to be the same objects as the result objects.

I'll try doing a quick refactor in that direction.

OK, I started down that road (generating #895 along the way), and then realized it would mean we'd allocate one of those objects for every user, in MentionAutocompleteView._usersByRelevance and so MentionAutocompleteView.init. That doesn't sound great performance-wise when there are like 30k users in a realm.

Instead, let's handle the wildcard-vs-user distinction in the control flow. Take a look at #896, and build your changes atop that. Then the existing results for an @-mention are computed here:

  Future<List<MentionAutocompleteResult>?> computeResults() async {
    final results = <MentionAutocompleteResult>[];
    if (await filterCandidates(filter: _testUser,
          candidates: sortedUsers, results: results)) {
      return null;
    }
    return results;
  }

where that filterCandidates call does the main loop and accumulates results onto that results local. To add wildcard candidates, you can just add more code next to that, calling List.add on the same results local.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch from 76fd1ac to 1664a2d Compare August 22, 2024 20:36
@sm-sayedi
Copy link
Collaborator Author

Thanks @gnprice for the previous review. It was really helpful. Pushed a new revision (still draft). PTAL.

@alya
Copy link
Collaborator

alya commented Aug 22, 2024

We're currently thinking we should use the three-people icon for wildcard mentions, rather than a loudspeaker. Assuming we move forward with it for the web app, we should make the same change here.

Copy link
Member

@gnprice gnprice left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, I missed that this had a request for my review, sorry ­@sm-sayedi — being a draft and with neither of the review labels, it looked like it was awaiting another revision first.

In general any time there's a PR where you're awaiting review for more than a week or two, please feel free to ping on the PR thread or in chat. We generally try to turn around reviews within a day or two, so a wait of over a week is likely a sign that something fell through the cracks.

Comments below. Since this PR is still a draft, I looked only at high-level aspects of the structure.

_isChannelWildcardIncluded = false;
final results = <MentionAutocompleteResult>[];
// give priority to wildcard mentions
if (await filterCandidates(filter: _testWildcard,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The fancy filterCandidates machinery isn't needed for the wildcards, because there's only a handful of them. (It's useful when there are potentially thousands of candidates.) So this can be simplified to a direct loop through the wildcard candidates.

That also lets _inChannelWildcardIncluded be a local variable, which simplifies reasoning about it.

… Then I started writing that the logic might be further clarified by unrolling the loop, akin to the logic in zulip-mobile:
https://github.com/zulip/zulip-mobile/blob/715d60a5e87fe37032bce58bd72edb99208e15be/src/autocomplete/WildcardMentionItem.js#L113-L138

But I see we discussed this previously above:
#889 (comment)
and that you have the corresponding logic living instead in the widgets code, in ComposeAutocomplete._wildcards. That structure may be fine too. Should have a link to the zulip-mobile version, though, as discussed there.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I switched back to a similar logic of what zulip-mobile does. I think it fits our current code structure well.

final List<User> sortedUsers;

@override
Future<List<MentionAutocompleteResult>?> computeResults() async {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Moving this method to lower in the file is perfectly reasonable, but should happen as its own separate NFC prep commit. That way the substantive change stands out better.

Comment on lines 182 to 185
fullDisplayName: 'all (${isDmNarrow
? zulipLocalizations.notifyRecipients
: zulipLocalizations.notifyChannel(isChannelWildcardAvailable
? "channel" : "stream")})',
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it will be cleaner to move this fullDisplayName logic to where the result is actually being displayed, at build time.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also avoid string manipulation on translated strings. See the "general background" at the end of this section:
https://github.com/zulip/zulip-mobile/blob/main/docs/howto/translations.md#all-code-contributors-wiring-our-code-for-translations

Instead, put the two parts of this into separate widgets or TextSpans. That way they can also get different text styles.

case WildcardMentionAutocompleteResult(:var wildcardName):
avatar = const Icon(ZulipIcons.bullhorn, size: 29); // web uses 19px
print('wildcard name: $wildcardName');
label = _wildcards(context, store).singleWhere((w) => w.name == wildcardName).fullDisplayName;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's avoid doing a search like this, and instead have the MentionAutocompleteResult value include whatever information this needs.

(If we did do a search, then like with store.users[userId] above, we should make it so that we're just looking it up in an existing data structure rather than recomputing the whole list again.)

In fact I think I'd like to refactor UserMentionAutocompleteResult so that it carries the actual User object, rather than just a user ID. That change is orthogonal to this PR, though.

@gnprice gnprice force-pushed the issue-234-wildcard-@-mention-autocomplete branch from 1664a2d to a24bddb Compare November 8, 2024 00:01
@gnprice
Copy link
Member

gnprice commented Nov 8, 2024

Also just pushed a rebase of the branch. (I did the rebase in order to better review it, since some things had changed in main; having done that, I figure I might as well share the result.)

The main change is the rebase across e91694f (#908), which causes some of the changes here to disappear as having already happened.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 5 times, most recently from 078194e to b0a6651 Compare December 6, 2024 07:41
@sm-sayedi sm-sayedi marked this pull request as ready for review December 6, 2024 07:42
@sm-sayedi
Copy link
Collaborator Author

Thanks @gnprice for the initial reviews. The PR is now ready for the maintainer review.

Note: The first two commits are actually necessary for displaying wildcard mentions based on a permission. I first created those commits and then started the CZO discussion , in which it was decided to leave the permission part for a future issue. I am not sure if we want to keep these two commits for now!

Comment on lines 284 to 282
return Text.rich(TextSpan(text: '${wildcard.name} ', children: [
TextSpan(text: description, style: TextStyle(fontSize: 12,
color: Colors.black.withValues(alpha: 0.8)))]));
Copy link
Collaborator Author

@sm-sayedi sm-sayedi Dec 6, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Web uses 85% of the name (i.e., all) font size for the description (i.e., Notify channel). For us, 85% of 14 is almost 12.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch from b0a6651 to a0ab0ed Compare December 6, 2024 08:00
@sm-sayedi sm-sayedi added the maintainer review PR ready for review by Zulip maintainers label Dec 6, 2024
@sm-sayedi
Copy link
Collaborator Author

The CI is failing and it's related to icons. I am not sure why it is happening!

@gnprice
Copy link
Member

gnprice commented Dec 6, 2024

The error output is:

Running icons...

added 162 packages in 29s
Error: there were icon updates:
 M assets/icons/ZulipIcons.ttf

Seems like that font file isn't updated to match your other changes. The icons.dart file has instructions for updating for icons.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 3 times, most recently from 9c80cb0 to e873ea4 Compare December 6, 2024 17:06
@sm-sayedi
Copy link
Collaborator Author

sm-sayedi commented Dec 6, 2024

Just transferred the CI failure discussion to CZO: https://chat.zulip.org/#narrow/channel/516-mobile-dev-help/topic/CI.20failure.20related.20to.20Zulip.20icons.20-.20.23F889/near/1994794.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 2 times, most recently from 0af384f to 06918d3 Compare December 9, 2024 06:44
@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 3 times, most recently from 08789d6 to ee45cde Compare December 14, 2024 02:44
Copy link
Collaborator

@chrisbobbe chrisbobbe left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, @sm-sayedi! Comments below.

Note: The first two commits are actually necessary for displaying wildcard mentions based on a permission. I first created those commits and then started the CZO discussion , in which it was decided to leave the permission part for a future issue. I am not sure if we want to keep these two commits for now!

I think it's fine to leave those commits in. Checking that the new fields are added in the proper order isn't super trivial; we might as well keep the work we did for that. 🙂

Comment on lines 158 to 164
enum RealmWildcardMentionPolicy {
anyone(apiValue: 1),
members(apiValue: 2),
fullMembers(apiValue: 3),
orgAdmins(apiValue: 5),
nobody(apiValue: 6),
moderators(apiValue: 7);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How about:

  • anyone > everyone
  • orgAdmins > administrators

to be more similar to the "permission levels" list here: https://zulip.com/api/roles-and-permissions#permission-levels

@@ -375,6 +377,7 @@ class PerAccountStore extends ChangeNotifier with EmojiStore, ChannelStore, Mess
Uri? tryResolveUrl(String reference) => _tryResolveUrl(realmUrl, reference);

String get zulipVersion => account.zulipVersion;
final RealmWildcardMentionPolicy realmWildcardMentionPolicy;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This needs a // TODO(#668): update this realm setting, like the setting below it.

Comment on lines 57 to 62
case ChannelNarrow(:final streamId) || TopicNarrow(:final streamId):
final stream = eg.stream(streamId: streamId);
message = eg.streamMessage(stream: stream);
await store.addStream(stream);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the narrow is a topic narrow, we should create message with the narrow's topic, right?

@@ -33,7 +33,9 @@ import 'test_app.dart';
/// before the end of the test.
Future<Finder> setupToComposeInput(WidgetTester tester, {
List<User> users = const [],
Narrow? narrow,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

autocomplete_test: Make `setupToComposeInput` accept `narrow` param

commit-message nit: autocomplete test: for the prefix, like in other commits

final stream = eg.stream(streamId: streamId);
message = eg.streamMessage(stream: stream);
await store.addStream(stream);
default: throw AssertionError(); // never happens because of the assert above
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

How about throw StateError('unexpected narrow type'), without a comment? That way, even if the assert line gets changed for some reason in the future, this line stays coherent. (Here and in another switch below.)

Comment on lines 268 to 277
Wildcard.all => isDmNarrow
? localizations.notifyRecipients
: isChannelWildcardAvailable
? localizations.notifyChannel
: localizations.notifyStream,
Wildcard.everyone => isDmNarrow
? localizations.notifyRecipients
: isChannelWildcardAvailable
? localizations.notifyChannel
: localizations.notifyStream,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
Wildcard.all => isDmNarrow
? localizations.notifyRecipients
: isChannelWildcardAvailable
? localizations.notifyChannel
: localizations.notifyStream,
Wildcard.everyone => isDmNarrow
? localizations.notifyRecipients
: isChannelWildcardAvailable
? localizations.notifyChannel
: localizations.notifyStream,
Wildcard.all || Wildcard.everyone => isDmNarrow
? localizations.notifyRecipients
: isChannelWildcardAvailable
? localizations.notifyChannel
: localizations.notifyStream,

@@ -811,20 +887,27 @@ void main() {
RecentDmConversation(userIds: [1, 2], maxMessageId: 100),
]);

// Check the ranking of the full list of users.
// Check the ranking of the full list of user mentions.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// Check the ranking of the full list of user mentions.
// Check the ranking of the full list of mentions.

(because wildcard mentions aren't user mentions, right?)

// 3. Human vs. Bot users (human users come first).
// 4. Alphabetical order by name.
check(await getResults(topicNarrow, MentionAutocompleteQuery('')))
// 1. Channel and/or topic wildcards.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: How about:

Suggested change
// 1. Channel and/or topic wildcards.
// 1. Wildcards.

There aren't other kinds of wildcard results that would appear in the list. But if there were, I think they'd probably go in this position too.

@@ -615,6 +596,59 @@ class MentionAutocompleteView extends AutocompleteView<MentionAutocompleteQuery,
return userAName.compareTo(userBName); // TODO(i18n): add locale-aware sorting
}

List<WildcardMentionAutocompleteResult> get wildcardMentionResults {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I just checked the web app, and if the user wants a silent mention (they've typed "@_") then wildcard mentions are excluded. I guess that sounds right, since the wildcard options say "Notify channel" etc.; how about we follow web and do that?

@@ -438,7 +438,7 @@ void main() {
});
});

group('MentionAutocompleteView sorting users results', () {
group('MentionAutocompleteView sorting user mention results', () {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
group('MentionAutocompleteView sorting user mention results', () {
group('MentionAutocompleteView sorting mention results', () {

(since wildcard mentions aren't user mentions)

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch from ee45cde to 55c36c4 Compare January 12, 2025 18:21
@sm-sayedi
Copy link
Collaborator Author

sm-sayedi commented Jan 12, 2025

Thank you @chrisbobbe for the thorough review. Comments addressed, please have a look.

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 2 times, most recently from c21fe24 to 0b864f2 Compare January 15, 2025 19:11
@sm-sayedi sm-sayedi requested a review from chrisbobbe January 15, 2025 19:12
@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch 2 times, most recently from 9d2d7bc to 65a2ab7 Compare January 19, 2025 10:23
@chrisbobbe
Copy link
Collaborator

CI is failing, could you take a look please? 🙂

@sm-sayedi sm-sayedi force-pushed the issue-234-wildcard-@-mention-autocomplete branch from 65a2ab7 to 21568dc Compare January 22, 2025 04:51
@sm-sayedi
Copy link
Collaborator Author

Thanks for pointing it out. The CI failure was related to #1290; just rebased on top of the main. PTAL.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
maintainer review PR ready for review by Zulip maintainers
Projects
None yet
Development

Successfully merging this pull request may close these issues.

autocomplete: Support wildcard @-mentions
4 participants