Skip to content

Commit bed001b

Browse files
marinaaisaDobromir Hristov
and
Dobromir Hristov
authored
Filter input component with tag selection (swiftlang#49)
* feat: add filter input component * refactor: minor cleanup and bug fixes * fix: tests and small component issues * fix: add tests and more smaller fixes * refactor: allow more styling extendability * chore: update license header year * feat: add disabled mode to input * feat: add FilterIcon as default icon for FilterInput * fix: import clipboard util * fix: it focuses the input if `focusInputWhenCreated` is on and input has content when component is created * fix: suggestedTags watcher should be immediate Co-authored-by: Dobromir Hristov <[email protected]>
1 parent 914871d commit bed001b

15 files changed

+4056
-2
lines changed

src/components/Filter/FilterInput.vue

+580
Large diffs are not rendered by default.

src/components/Filter/Tag.vue

+206
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,206 @@
1+
<!--
2+
This source file is part of the Swift.org open source project
3+
4+
Copyright (c) 2022 Apple Inc. and the Swift project authors
5+
Licensed under Apache License v2.0 with Runtime Library Exception
6+
7+
See https://swift.org/LICENSE.txt for license information
8+
See https://swift.org/CONTRIBUTORS.txt for Swift project authors
9+
-->
10+
11+
<template>
12+
<li class="tag" role="presentation">
13+
<button
14+
ref="button"
15+
:class="{ 'focus': isActiveTag }"
16+
role="option"
17+
:aria-selected="ariaSelected"
18+
aria-roledescription="tag"
19+
@focus="$emit('focus', { event: $event, tagName: name })"
20+
@click.prevent="$emit('click', { event: $event, tagName: name })"
21+
@dblclick.prevent="!keyboardIsVirtual && deleteTag()"
22+
@keydown.exact="$emit('keydown', { event: $event, tagName: name })"
23+
@keydown.shift.exact="$emit('keydown', { event: $event, tagName: name })"
24+
@keydown.shift.meta.exact="$emit('keydown', { event: $event, tagName: name })"
25+
@keydown.meta.exact="$emit('keydown', { event: $event, tagName: name })"
26+
@keydown.ctrl.exact="$emit('keydown', { event: $event, tagName: name })"
27+
@keydown.delete.prevent="deleteTag"
28+
@mousedown.prevent="focusButton"
29+
@copy="handleCopy"
30+
>
31+
<span v-if="!isRemovableTag" class="visuallyhidden">
32+
Add tag -
33+
</span>
34+
{{ name }}
35+
<span v-if="isRemovableTag" class="visuallyhidden">
36+
– Tag. Select to remove from list.
37+
</span>
38+
</button>
39+
</li>
40+
</template>
41+
<script>
42+
import { prepareDataForHTMLClipboard } from 'docc-render/utils/clipboard';
43+
44+
export default {
45+
name: 'Tag',
46+
props: {
47+
name: {
48+
type: String,
49+
required: true,
50+
},
51+
isFocused: {
52+
type: Boolean,
53+
default: () => false,
54+
},
55+
isRemovableTag: {
56+
type: Boolean,
57+
default: false,
58+
},
59+
isActiveTag: {
60+
type: Boolean,
61+
default: false,
62+
},
63+
activeTags: {
64+
type: Array,
65+
required: false,
66+
},
67+
keyboardIsVirtual: {
68+
type: Boolean,
69+
default: false,
70+
},
71+
},
72+
watch: {
73+
isFocused(newVal) {
74+
if (newVal) {
75+
this.focusButton();
76+
}
77+
},
78+
},
79+
mounted() {
80+
// initialize global clipboard listeners
81+
document.addEventListener('copy', this.handleCopy);
82+
document.addEventListener('cut', this.handleCut);
83+
document.addEventListener('paste', this.handlePaste);
84+
85+
this.$once('hook:beforeDestroy', () => {
86+
document.removeEventListener('copy', this.handleCopy);
87+
document.removeEventListener('cut', this.handleCut);
88+
document.removeEventListener('paste', this.handlePaste);
89+
});
90+
},
91+
methods: {
92+
isCurrentlyActiveElement() {
93+
return document.activeElement === this.$refs.button;
94+
},
95+
/**
96+
* Handles copy event
97+
* @param {ClipboardEvent} event
98+
*/
99+
handleCopy(event) {
100+
// handle only if the current focused item is the button
101+
if (!this.isCurrentlyActiveElement()) return;
102+
// stop the event.
103+
event.preventDefault();
104+
// copy as JSON
105+
let tags = [];
106+
if (this.activeTags.length > 0) {
107+
tags = this.activeTags;
108+
} else {
109+
tags = [this.name];
110+
}
111+
event.clipboardData.setData('text/html', prepareDataForHTMLClipboard({ tags }));
112+
// copy as plain text
113+
event.clipboardData.setData('text/plain', tags.join(' '));
114+
},
115+
handleCut(event) {
116+
if (!this.isCurrentlyActiveElement() || !this.isRemovableTag) return;
117+
this.handleCopy(event);
118+
this.deleteTag(event);
119+
},
120+
/**
121+
* Handles pasting into the page, when the focused element,
122+
* is the button of the current Tag instance
123+
* @param {ClipboardEvent} event
124+
*/
125+
handlePaste(event) {
126+
if (!this.isCurrentlyActiveElement() || !this.isRemovableTag) return;
127+
// stop the `paste` event.
128+
event.preventDefault();
129+
// delete the current tag, as we are pasting over it
130+
this.deleteTag(event);
131+
// emit up the event data, for the `FilterInput` to handle
132+
this.$emit('paste-content', event);
133+
},
134+
deleteTag(event) {
135+
this.$emit('delete-tag', { tagName: this.name, event });
136+
this.$emit('prevent-blur');
137+
},
138+
/**
139+
* Handles clicking on tags.
140+
* Works for Mouse clicks and VO clicks.
141+
* @param {MouseEvent} event
142+
*/
143+
focusButton(event = {}) {
144+
if (!this.keyboardIsVirtual) {
145+
this.$refs.button.focus();
146+
}
147+
// if the mouse click has no buttons clicked, its coming from VO
148+
if (event.buttons === 0 && this.isFocused) {
149+
this.deleteTag(event);
150+
}
151+
},
152+
},
153+
computed: {
154+
ariaSelected: ({ isActiveTag, isRemovableTag }) => {
155+
if (!isRemovableTag) return null;
156+
return isActiveTag ? 'true' : 'false';
157+
},
158+
},
159+
};
160+
</script>
161+
162+
<style scoped lang="scss">
163+
@import 'docc-render/styles/_core.scss';
164+
165+
.tag {
166+
display: inline-block;
167+
padding-right: rem(10px);
168+
169+
&:focus {
170+
outline: none;
171+
}
172+
173+
button {
174+
color: var(--color-figure-gray);
175+
background-color: var(--color-fill-tertiary);
176+
@include font-styles(body-reduced-tight);
177+
border-radius: rem(14px);
178+
padding: rem(4px) rem(10px);
179+
white-space: nowrap;
180+
border: 1px solid transparent;
181+
182+
@media (hover: hover) { // Prevent hover state to get stuck on iOS Safari
183+
&:hover {
184+
transition: background-color 0.2s, color 0.2s;
185+
background-color: var(--color-fill-blue);
186+
color: white;
187+
}
188+
}
189+
190+
// We only want to make active the tags when they are clicked (focus) to prevent
191+
// ghost active states when deleting tags. https://stackoverflow.com/questions/1677990/
192+
&:focus:active {
193+
background-color: var(--color-fill-blue);
194+
color: white;
195+
}
196+
197+
&:focus, &.focus {
198+
@include focus-shadow-form-element();
199+
}
200+
201+
@include on-keyboard-focus() {
202+
@include focus-shadow-form-element();
203+
}
204+
}
205+
}
206+
</style>

0 commit comments

Comments
 (0)