handle resetting search field

This commit is contained in:
Kévin Cocchi 2022-02-06 12:17:29 +01:00
parent 99ced8878a
commit 41f0432e8d
4 changed files with 39 additions and 24 deletions

1
web/res/reset.svg Normal file
View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 16 16"><path d="M16 8A8 8 0 1 1 0 8a8 8 0 0 1 16 0zM5.354 4.646a.5.5 0 1 0-.708.708L7.293 8l-2.647 2.646a.5.5 0 0 0 .708.708L8 8.707l2.646 2.647a.5.5 0 0 0 .708-.708L8.707 8l2.647-2.646a.5.5 0 0 0-.708-.708L8 7.293 5.354 4.646z"/></svg>

After

Width:  |  Height:  |  Size: 313 B

View File

@ -41,10 +41,7 @@ const supportedThemes = ["light", "dark", "black"]
const defaultState = { const defaultState = {
packs: [], packs: [],
filtering: { searchTerm: null,
searchTerm: "",
packs: [],
},
} }
class App extends Component { class App extends Component {
@ -52,7 +49,7 @@ class App extends Component {
super(props) super(props)
this.defaultTheme = params.get("theme") this.defaultTheme = params.get("theme")
this.state = { this.state = {
packs: defaultState.packs, ...defaultState,
loading: true, loading: true,
error: null, error: null,
stickersPerRow: parseInt(localStorage.mauStickersPerRow || "4"), stickersPerRow: parseInt(localStorage.mauStickersPerRow || "4"),
@ -63,7 +60,6 @@ class App extends Component {
stickerIDs: frequent.get(), stickerIDs: frequent.get(),
stickers: [], stickers: [],
}, },
filtering: defaultState.filtering,
} }
if (!supportedThemes.includes(this.state.theme)) { if (!supportedThemes.includes(this.state.theme)) {
this.state.theme = "light" this.state.theme = "light"
@ -77,6 +73,7 @@ class App extends Component {
this.packListRef = null this.packListRef = null
this.navRef = null this.navRef = null
this.searchStickers = this.searchStickers.bind(this) this.searchStickers = this.searchStickers.bind(this)
this.resetSearch = this.resetSearch.bind(this)
this.sendSticker = this.sendSticker.bind(this) this.sendSticker = this.sendSticker.bind(this)
this.navScroll = this.navScroll.bind(this) this.navScroll = this.navScroll.bind(this)
this.reloadPacks = this.reloadPacks.bind(this) this.reloadPacks = this.reloadPacks.bind(this)
@ -101,7 +98,18 @@ class App extends Component {
localStorage.mauFrequentlyUsedStickerCache = JSON.stringify(stickers.map(sticker => [sticker.id, sticker])) localStorage.mauFrequentlyUsedStickerCache = JSON.stringify(stickers.map(sticker => [sticker.id, sticker]))
} }
// Search
resetSearch() {
this.setState({ searchTerm: defaultState.searchTerm })
}
searchStickers(e) { searchStickers(e) {
if (e.key === "Escape") {
this.resetSearch()
return
}
const sanitizeString = s => s.toLowerCase().trim() const sanitizeString = s => s.toLowerCase().trim()
const searchTerm = sanitizeString(e.target.value) const searchTerm = sanitizeString(e.target.value)
@ -115,11 +123,8 @@ class App extends Component {
})) }))
this.setState({ this.setState({
filtering: { filteredPacks: packsWithFilteredStickers.filter(({ stickers }) => !!stickers.length),
...this.state.filtering,
searchTerm, searchTerm,
packs: packsWithFilteredStickers.filter(({ stickers }) => !!stickers.length),
},
}) })
} }
@ -145,10 +150,8 @@ class App extends Component {
reloadPacks() { reloadPacks() {
this.imageObserver.disconnect() this.imageObserver.disconnect()
this.sectionObserver.disconnect() this.sectionObserver.disconnect()
this.setState({ this.setState({ packs: defaultState.packs })
packs: defaultState.packs, this.resetSearch()
filtering: defaultState.filtering,
})
this._loadPacks(true) this._loadPacks(true)
} }
@ -258,6 +261,7 @@ class App extends Component {
const sticker = this.stickersByID.get(id) const sticker = this.stickersByID.get(id)
frequent.add(id) frequent.add(id)
this.updateFrequentlyUsed() this.updateFrequentlyUsed()
this.resetSearch()
widgetAPI.sendSticker(sticker) widgetAPI.sendSticker(sticker)
} }
@ -267,8 +271,8 @@ class App extends Component {
render() { render() {
const theme = `theme-${this.state.theme}` const theme = `theme-${this.state.theme}`
const filterActive = !!this.state.filtering.searchTerm const filterActive = !!this.state.searchTerm
const packs = filterActive ? this.state.filtering.packs : [this.state.frequentlyUsed, ...this.state.packs] const packs = filterActive ? this.state.filteredPacks : [this.state.frequentlyUsed, ...this.state.packs]
if (this.state.loading) { if (this.state.loading) {
return html`<main class="spinner ${theme}"><${Spinner} size=${80} green /></main>` return html`<main class="spinner ${theme}"><${Spinner} size=${80} green /></main>`
@ -287,7 +291,7 @@ class App extends Component {
${this.state.packs.map(pack => html`<${NavBarItem} id=${pack.id} pack=${pack}/>`)} ${this.state.packs.map(pack => html`<${NavBarItem} id=${pack.id} pack=${pack}/>`)}
<${NavBarItem} pack=${{ id: "settings", title: "Settings" }} iconOverride="settings" /> <${NavBarItem} pack=${{ id: "settings", title: "Settings" }} iconOverride="settings" />
</nav> </nav>
<${SearchBox} onKeyUp=${this.searchStickers} /> <${SearchBox} onKeyUp=${this.searchStickers} resetSearch=${this.resetSearch} value=${this.state.searchTerm} />
<div class="pack-list ${isMobileSafari ? "ios-safari-hack" : ""}" ref=${elem => this.packListRef = elem}> <div class="pack-list ${isMobileSafari ? "ios-safari-hack" : ""}" ref=${elem => this.packListRef = elem}>
${filterActive && packs.length === 0 ? html`<div class="search-empty"><h1>No stickers match your search</h1></div>` : null} ${filterActive && packs.length === 0 ? html`<div class="search-empty"><h1>No stickers match your search</h1></div>` : null}
${packs.map(pack => html`<${Pack} id=${pack.id} pack=${pack} send=${this.sendSticker} />`)} ${packs.map(pack => html`<${Pack} id=${pack.id} pack=${pack} send=${this.sendSticker} />`)}

View File

@ -15,12 +15,19 @@
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
import { html } from "../lib/htm/preact.js" import { html } from "../lib/htm/preact.js"
export const SearchBox = ({ onKeyUp, placeholder = 'Find stickers' }) => { export const SearchBox = ({ onKeyUp, resetSearch, value }) => {
const component = html` const className = `icon-display ${value ? 'reset-click-zone' : null}`
const title = value ? 'Click to reset' : null
const onClick = value ? resetSearch : null
const iconToDisplay = `icon-${!value ? 'search' : 'reset'}`
return html`
<div class="search-box"> <div class="search-box">
<input type="text" placeholder=${placeholder} onKeyUp=${onKeyUp} /> <input value=${value} placeholder="Find stickers" onKeyUp=${onKeyUp} />
<span class="icon icon-search" /> <div class=${className} title=${title} onClick=${onClick}>
<span class="icon ${iconToDisplay}" />
</div>
</div> </div>
` `
return component
} }

View File

@ -92,9 +92,12 @@ main.theme-black
&.icon-recent &.icon-recent
--icon-image: url(../res/recent.svg) --icon-image: url(../res/recent.svg)
&.icon.icon-search &.icon-search
--icon-image: url(../res/search.svg) --icon-image: url(../res/search.svg)
&.icon-reset
--icon-image: url(../res/reset.svg)
nav nav
display: flex display: flex
overflow-x: auto overflow-x: auto