|
@ -15,6 +15,7 @@ |
|
|
// 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, render, Component } from "../lib/htm/preact.js" |
|
|
import { html, render, Component } from "../lib/htm/preact.js" |
|
|
import { Spinner } from "./spinner.js" |
|
|
import { Spinner } from "./spinner.js" |
|
|
|
|
|
import { SearchBox } from "./search-box.js" |
|
|
import * as widgetAPI from "./widget-api.js" |
|
|
import * as widgetAPI from "./widget-api.js" |
|
|
import * as frequent from "./frequently-used.js" |
|
|
import * as frequent from "./frequently-used.js" |
|
|
|
|
|
|
|
@ -37,12 +38,20 @@ export const parseQuery = str => Object.fromEntries( |
|
|
|
|
|
|
|
|
const supportedThemes = ["light", "dark", "black"] |
|
|
const supportedThemes = ["light", "dark", "black"] |
|
|
|
|
|
|
|
|
|
|
|
const defaultState = { |
|
|
|
|
|
packs: [], |
|
|
|
|
|
filtering: { |
|
|
|
|
|
searchTerm: "", |
|
|
|
|
|
packs: [], |
|
|
|
|
|
}, |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
class App extends Component { |
|
|
class App extends Component { |
|
|
constructor(props) { |
|
|
constructor(props) { |
|
|
super(props) |
|
|
super(props) |
|
|
this.defaultTheme = parseQuery(location.search.substr(1)).theme |
|
|
this.defaultTheme = parseQuery(location.search.substr(1)).theme |
|
|
this.state = { |
|
|
this.state = { |
|
|
packs: [], |
|
|
|
|
|
|
|
|
packs: defaultState.packs, |
|
|
loading: true, |
|
|
loading: true, |
|
|
error: null, |
|
|
error: null, |
|
|
stickersPerRow: parseInt(localStorage.mauStickersPerRow || "4"), |
|
|
stickersPerRow: parseInt(localStorage.mauStickersPerRow || "4"), |
|
@ -53,6 +62,7 @@ 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" |
|
@ -65,6 +75,7 @@ class App extends Component { |
|
|
this.imageObserver = null |
|
|
this.imageObserver = null |
|
|
this.packListRef = null |
|
|
this.packListRef = null |
|
|
this.navRef = null |
|
|
this.navRef = null |
|
|
|
|
|
this.searchStickers = this.searchStickers.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) |
|
@ -89,6 +100,28 @@ class App extends Component { |
|
|
localStorage.mauFrequentlyUsedStickerCache = JSON.stringify(stickers.map(sticker => [sticker.id, sticker])) |
|
|
localStorage.mauFrequentlyUsedStickerCache = JSON.stringify(stickers.map(sticker => [sticker.id, sticker])) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
searchStickers(e) { |
|
|
|
|
|
const sanitizeString = s => s.toLowerCase().trim() |
|
|
|
|
|
const searchTerm = sanitizeString(e.target.value) |
|
|
|
|
|
|
|
|
|
|
|
const allPacks = [this.state.frequentlyUsed, ...this.state.packs] |
|
|
|
|
|
const packsWithFilteredStickers = allPacks.map(pack => ({ |
|
|
|
|
|
...pack, |
|
|
|
|
|
stickers: pack.stickers.filter(sticker => |
|
|
|
|
|
sanitizeString(sticker.body).includes(searchTerm) || |
|
|
|
|
|
sanitizeString(sticker.id).includes(searchTerm) |
|
|
|
|
|
), |
|
|
|
|
|
})) |
|
|
|
|
|
|
|
|
|
|
|
this.setState({ |
|
|
|
|
|
filtering: { |
|
|
|
|
|
...this.state.filtering, |
|
|
|
|
|
searchTerm, |
|
|
|
|
|
packs: packsWithFilteredStickers.filter(({ stickers }) => !!stickers.length), |
|
|
|
|
|
}, |
|
|
|
|
|
}) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
setStickersPerRow(val) { |
|
|
setStickersPerRow(val) { |
|
|
localStorage.mauStickersPerRow = val |
|
|
localStorage.mauStickersPerRow = val |
|
|
document.documentElement.style.setProperty("--stickers-per-row", localStorage.mauStickersPerRow) |
|
|
document.documentElement.style.setProperty("--stickers-per-row", localStorage.mauStickersPerRow) |
|
@ -111,7 +144,10 @@ class App extends Component { |
|
|
reloadPacks() { |
|
|
reloadPacks() { |
|
|
this.imageObserver.disconnect() |
|
|
this.imageObserver.disconnect() |
|
|
this.sectionObserver.disconnect() |
|
|
this.sectionObserver.disconnect() |
|
|
this.setState({ packs: [] }) |
|
|
|
|
|
|
|
|
this.setState({ |
|
|
|
|
|
packs: defaultState.packs, |
|
|
|
|
|
filtering: defaultState.filtering, |
|
|
|
|
|
}) |
|
|
this._loadPacks(true) |
|
|
this._loadPacks(true) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@ -225,6 +261,9 @@ 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 packs = filterActive ? this.state.filtering.packs : [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>` |
|
|
} else if (this.state.error) { |
|
|
} else if (this.state.error) { |
|
@ -235,15 +274,17 @@ class App extends Component { |
|
|
} else if (this.state.packs.length === 0) { |
|
|
} else if (this.state.packs.length === 0) { |
|
|
return html`<main class="empty ${theme}"><h1>No packs found 😿</h1></main>` |
|
|
return html`<main class="empty ${theme}"><h1>No packs found 😿</h1></main>` |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return html`<main class="has-content ${theme}">
|
|
|
return html`<main class="has-content ${theme}">
|
|
|
<nav onWheel=${this.navScroll} ref=${elem => this.navRef = elem}> |
|
|
<nav onWheel=${this.navScroll} ref=${elem => this.navRef = elem}> |
|
|
<${NavBarItem} pack=${this.state.frequentlyUsed} iconOverride="recent" /> |
|
|
<${NavBarItem} pack=${this.state.frequentlyUsed} iconOverride="recent" /> |
|
|
${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} /> |
|
|
<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}> |
|
|
<${Pack} pack=${this.state.frequentlyUsed} send=${this.sendSticker} /> |
|
|
|
|
|
${this.state.packs.map(pack => html`<${Pack} id=${pack.id} pack=${pack} send=${this.sendSticker} />`)} |
|
|
|
|
|
|
|
|
${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} />`)} |
|
|
<${Settings} app=${this}/> |
|
|
<${Settings} app=${this}/> |
|
|
</div> |
|
|
</div> |
|
|
</main>` |
|
|
</main>` |
|
|