mirror of
				https://github.com/maunium/stickerpicker
				synced 2025-10-26 09:48:00 +00:00 
			
		
		
		
	Compare commits
	
		
			No commits in common. "99ced8878aaf89a47dd8bdb592025ae2160d6f07" and "ec8eeeeaf526a122008aace7c71e5472bce832e6" have entirely different histories.
		
	
	
		
			99ced8878a
			...
			ec8eeeeaf5
		
	
		
							
								
								
									
										1
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								setup.py
									
									
									
									
									
								
							| @ -45,7 +45,6 @@ setuptools.setup( | ||||
|         "Programming Language :: Python :: 3.6", | ||||
|         "Programming Language :: Python :: 3.7", | ||||
|         "Programming Language :: Python :: 3.8", | ||||
|         "Programming Language :: Python :: 3.9", | ||||
|     ], | ||||
|     entry_points={"console_scripts": [ | ||||
|         "sticker-import=sticker.stickerimport:cmd", | ||||
|  | ||||
| @ -42,7 +42,6 @@ if TYPE_CHECKING: | ||||
|         url: str | ||||
|         info: MediaInfo | ||||
|         id: str | ||||
|         msgtype: str | ||||
| else: | ||||
|     MediaInfo = None | ||||
|     StickerInfo = None | ||||
|  | ||||
| @ -13,7 +13,6 @@ | ||||
| # | ||||
| # You should have received a copy of the GNU Affero General Public License | ||||
| # along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
| from functools import partial | ||||
| from io import BytesIO | ||||
| import os.path | ||||
| import json | ||||
| @ -22,7 +21,6 @@ from PIL import Image | ||||
| 
 | ||||
| from . import matrix | ||||
| 
 | ||||
| open_utf8 = partial(open, encoding='UTF-8') | ||||
| 
 | ||||
| def convert_image(data: bytes) -> (bytes, int, int): | ||||
|     image: Image.Image = Image.open(BytesIO(data)).convert("RGBA") | ||||
| @ -43,7 +41,7 @@ def convert_image(data: bytes) -> (bytes, int, int): | ||||
| def add_to_index(name: str, output_dir: str) -> None: | ||||
|     index_path = os.path.join(output_dir, "index.json") | ||||
|     try: | ||||
|         with open_utf8(index_path) as index_file: | ||||
|         with open(index_path) as index_file: | ||||
|             index_data = json.load(index_file) | ||||
|     except (FileNotFoundError, json.JSONDecodeError): | ||||
|         index_data = {"packs": []} | ||||
| @ -51,7 +49,7 @@ def add_to_index(name: str, output_dir: str) -> None: | ||||
|         index_data["homeserver_url"] = matrix.homeserver_url | ||||
|     if name not in index_data["packs"]: | ||||
|         index_data["packs"].append(name) | ||||
|         with open_utf8(index_path, "w") as index_file: | ||||
|         with open(index_path, "w") as index_file: | ||||
|             json.dump(index_data, index_file, indent="  ") | ||||
|         print(f"Added {name} to {index_path}") | ||||
| 
 | ||||
| @ -76,5 +74,4 @@ def make_sticker(mxc: str, width: int, height: int, size: int, | ||||
|                 "mimetype": "image/png", | ||||
|             }, | ||||
|         }, | ||||
|         "msgtype": "m.sticker", | ||||
|     } | ||||
|  | ||||
| @ -93,7 +93,7 @@ async def main(args: argparse.Namespace) -> None: | ||||
|     dirname = os.path.basename(os.path.abspath(args.path)) | ||||
|     meta_path = os.path.join(args.path, "pack.json") | ||||
|     try: | ||||
|         with util.open_utf8(meta_path) as pack_file: | ||||
|         with open(meta_path) as pack_file: | ||||
|             pack = json.load(pack_file) | ||||
|             print(f"Loaded existing pack meta from {meta_path}") | ||||
|     except FileNotFoundError: | ||||
| @ -112,14 +112,14 @@ async def main(args: argparse.Namespace) -> None: | ||||
|         if sticker: | ||||
|             pack["stickers"].append(sticker) | ||||
| 
 | ||||
|     with util.open_utf8(meta_path, "w") as pack_file: | ||||
|     with open(meta_path, "w") as pack_file: | ||||
|         json.dump(pack, pack_file) | ||||
|     print(f"Wrote pack to {meta_path}") | ||||
| 
 | ||||
|     if args.add_to_index: | ||||
|         picker_file_name = f"{pack['id']}.json" | ||||
|         picker_pack_path = os.path.join(args.add_to_index, picker_file_name) | ||||
|         with util.open_utf8(picker_pack_path, "w") as pack_file: | ||||
|         with open(picker_pack_path, "w") as pack_file: | ||||
|             json.dump(pack, pack_file) | ||||
|         print(f"Copied pack to {picker_pack_path}") | ||||
|         util.add_to_index(picker_file_name, args.add_to_index) | ||||
|  | ||||
| @ -19,12 +19,12 @@ import json | ||||
| index_path = "../web/packs/index.json" | ||||
| 
 | ||||
| try: | ||||
|     with util.open_utf8(index_path) as index_file: | ||||
|     with open(index_path) as index_file: | ||||
|         index_data = json.load(index_file) | ||||
| except (FileNotFoundError, json.JSONDecodeError): | ||||
|     index_data = {"packs": []} | ||||
| 
 | ||||
| with util.open_utf8(sys.argv[-1]) as file: | ||||
| with open(sys.argv[-1]) as file: | ||||
|     data = json.load(file) | ||||
| 
 | ||||
| for pack in data["assets"]: | ||||
| @ -45,12 +45,12 @@ for pack in data["assets"]: | ||||
|     } | ||||
|     filename = f"scalar-{pack['name'].replace(' ', '_')}.json" | ||||
|     pack_path = f"web/packs/{filename}" | ||||
|     with util.open_utf8(pack_path, "w") as pack_file: | ||||
|     with open(pack_path, "w") as pack_file: | ||||
|         json.dump(pack_data, pack_file) | ||||
|     print(f"Wrote {title} to {pack_path}") | ||||
|     if filename not in index_data["packs"]: | ||||
|         index_data["packs"].append(filename) | ||||
| 
 | ||||
| with util.open_utf8(index_path, "w") as index_file: | ||||
| with open(index_path, "w") as index_file: | ||||
|     json.dump(index_data, index_file, indent="  ") | ||||
| print(f"Updated {index_path}") | ||||
|  | ||||
| @ -71,7 +71,7 @@ async def reupload_pack(client: TelegramClient, pack: StickerSetFull, output_dir | ||||
| 
 | ||||
|     already_uploaded = {} | ||||
|     try: | ||||
|         with util.open_utf8(pack_path) as pack_file: | ||||
|         with open(pack_path) as pack_file: | ||||
|             existing_pack = json.load(pack_file) | ||||
|             already_uploaded = {int(sticker["net.maunium.telegram.sticker"]["id"]): sticker | ||||
|                                 for sticker in existing_pack["stickers"]} | ||||
| @ -99,7 +99,7 @@ async def reupload_pack(client: TelegramClient, pack: StickerSetFull, output_dir | ||||
|                 doc["body"] = sticker.emoticon | ||||
|             doc["net.maunium.telegram.sticker"]["emoticons"].append(sticker.emoticon) | ||||
| 
 | ||||
|     with util.open_utf8(pack_path, "w") as pack_file: | ||||
|     with open(pack_path, "w") as pack_file: | ||||
|         json.dump({ | ||||
|             "title": pack.set.title, | ||||
|             "id": f"tg-{pack.set.id}", | ||||
| @ -138,12 +138,11 @@ async def main(args: argparse.Namespace) -> None: | ||||
|     if args.list: | ||||
|         stickers: AllStickers = await client(GetAllStickersRequest(hash=0)) | ||||
|         index = 1 | ||||
|         width = len(str(len(stickers.sets))) | ||||
|         width = len(str(stickers.sets)) | ||||
|         print("Your saved sticker packs:") | ||||
|         for saved_pack in stickers.sets: | ||||
|             print(f"{index:>{width}}. {saved_pack.title} " | ||||
|                   f"(t.me/addstickers/{saved_pack.short_name})") | ||||
|             index += 1 | ||||
|     elif args.pack[0]: | ||||
|         input_packs = [] | ||||
|         for pack_url in args.pack[0]: | ||||
|  | ||||
| @ -1,23 +0,0 @@ | ||||
| const { install, printStats } = require("esinstall") | ||||
| 
 | ||||
| install( | ||||
|   [{ | ||||
|     specifier: "htm/preact", | ||||
|     all: false, | ||||
|     default: false, | ||||
|     namespace: false, | ||||
|     named: ["html", "render", "Component"], | ||||
|   }], | ||||
|   { | ||||
|     dest: "./lib", | ||||
|     sourceMap: false, | ||||
|     treeshake: true, | ||||
|     verbose: true, | ||||
|   } | ||||
| ).then(data => { | ||||
|   const oldPrefix = "web_modules/" | ||||
|   const newPrefix = "lib/" | ||||
|   const spaces = " ".repeat(oldPrefix.length - newPrefix.length) | ||||
|   console.log("Installation complete") | ||||
|   console.log(printStats(data.stats).replace(oldPrefix, newPrefix + spaces)) | ||||
| }) | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -4,16 +4,28 @@ | ||||
|   "description": "A fast and simple Matrix sticker picker widget", | ||||
|   "repository": "https://github.com/maunium/stickerpicker", | ||||
|   "author": "Tulir Asokan <tulir@maunium.net>", | ||||
|   "license": "AGPL-3.0-or-later", | ||||
|   "license": "MPL-2.0", | ||||
|   "private": true, | ||||
|   "scripts": { | ||||
|     "esinstall": "node ./esinstall.js", | ||||
|     "sass": "sass --no-source-map --style=compressed style/" | ||||
|     "snowpack": "snowpack", | ||||
|     "sass": "node-sass -o style style/*.sass --output-style compressed" | ||||
|   }, | ||||
|   "snowpack": { | ||||
|     "install": [ | ||||
|       "htm/preact" | ||||
|     ], | ||||
|     "installOptions": { | ||||
|       "sourceMap": false, | ||||
|       "dest": "lib", | ||||
|       "treeshake": true | ||||
|     } | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "htm": "^3.1.0", | ||||
|     "preact": "^10.5.14", | ||||
|     "esinstall": "^1.1.7", | ||||
|     "sass": "^1.42.1" | ||||
|     "htm": "^3.0.4", | ||||
|     "preact": "^10.4.8", | ||||
|     "snowpack": "^2.10.3" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "node-sass": "^4.14.1" | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -1 +0,0 @@ | ||||
| <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24"><path d="M9.145 18.29c-5.042 0-9.145-4.102-9.145-9.145s4.103-9.145 9.145-9.145 9.145 4.103 9.145 9.145-4.102 9.145-9.145 9.145zm0-15.167c-3.321 0-6.022 2.702-6.022 6.022s2.702 6.022 6.022 6.022 6.023-2.702 6.023-6.022-2.702-6.022-6.023-6.022zm9.263 12.443c-.817 1.176-1.852 2.188-3.046 2.981l5.452 5.453 3.014-3.013-5.42-5.421z"/></svg> | ||||
| Before Width: | Height: | Size: 419 B | 
| @ -15,19 +15,12 @@ | ||||
| // along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | ||||
| import { html, render, Component } from "../lib/htm/preact.js" | ||||
| import { Spinner } from "./spinner.js" | ||||
| import { SearchBox } from "./search-box.js" | ||||
| import * as widgetAPI from "./widget-api.js" | ||||
| import * as frequent from "./frequently-used.js" | ||||
| 
 | ||||
| // The base URL for fetching packs. The app will first fetch ${PACK_BASE_URL}/index.json,
 | ||||
| // then ${PACK_BASE_URL}/${packFile} for each packFile in the packs object of the index.json file.
 | ||||
| const PACKS_BASE_URL = "packs" | ||||
| 
 | ||||
| let INDEX = `${PACKS_BASE_URL}/index.json` | ||||
| const params = new URLSearchParams(document.location.search) | ||||
| if (params.has('config')) { | ||||
| 	INDEX = params.get("config") | ||||
| } | ||||
| // This is updated from packs/index.json
 | ||||
| let HOMESERVER_URL = "https://matrix-client.matrix.org" | ||||
| 
 | ||||
| @ -37,22 +30,19 @@ const makeThumbnailURL = mxc => `${HOMESERVER_URL}/_matrix/media/r0/thumbnail/${ | ||||
| // This is also used to fix scrolling to sections on Element iOS
 | ||||
| const isMobileSafari = navigator.userAgent.match(/(iPod|iPhone|iPad)/) && navigator.userAgent.match(/AppleWebKit/) | ||||
| 
 | ||||
| const supportedThemes = ["light", "dark", "black"] | ||||
| export const parseQuery = str => Object.fromEntries( | ||||
| 	str.split("&") | ||||
| 		.map(part => part.split("=")) | ||||
| 		.map(([key, value = ""]) => [key, value])) | ||||
| 
 | ||||
| const defaultState = { | ||||
| 	packs: [], | ||||
| 	filtering: { | ||||
| 		searchTerm: "", | ||||
| 		packs: [], | ||||
| 	}, | ||||
| } | ||||
| const supportedThemes = ["light", "dark", "black"] | ||||
| 
 | ||||
| class App extends Component { | ||||
| 	constructor(props) { | ||||
| 		super(props) | ||||
| 		this.defaultTheme = params.get("theme") | ||||
| 		this.defaultTheme = parseQuery(location.search.substr(1)).theme | ||||
| 		this.state = { | ||||
| 			packs: defaultState.packs, | ||||
| 			packs: [], | ||||
| 			loading: true, | ||||
| 			error: null, | ||||
| 			stickersPerRow: parseInt(localStorage.mauStickersPerRow || "4"), | ||||
| @ -63,7 +53,6 @@ class App extends Component { | ||||
| 				stickerIDs: frequent.get(), | ||||
| 				stickers: [], | ||||
| 			}, | ||||
| 			filtering: defaultState.filtering, | ||||
| 		} | ||||
| 		if (!supportedThemes.includes(this.state.theme)) { | ||||
| 			this.state.theme = "light" | ||||
| @ -76,7 +65,6 @@ class App extends Component { | ||||
| 		this.imageObserver = null | ||||
| 		this.packListRef = null | ||||
| 		this.navRef = null | ||||
| 		this.searchStickers = this.searchStickers.bind(this) | ||||
| 		this.sendSticker = this.sendSticker.bind(this) | ||||
| 		this.navScroll = this.navScroll.bind(this) | ||||
| 		this.reloadPacks = this.reloadPacks.bind(this) | ||||
| @ -101,28 +89,6 @@ class App extends Component { | ||||
| 		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) { | ||||
| 		localStorage.mauStickersPerRow = val | ||||
| 		document.documentElement.style.setProperty("--stickers-per-row", localStorage.mauStickersPerRow) | ||||
| @ -145,16 +111,13 @@ class App extends Component { | ||||
| 	reloadPacks() { | ||||
| 		this.imageObserver.disconnect() | ||||
| 		this.sectionObserver.disconnect() | ||||
| 		this.setState({ | ||||
| 			packs: defaultState.packs, | ||||
| 			filtering: defaultState.filtering, | ||||
| 		}) | ||||
| 		this.setState({ packs: [] }) | ||||
| 		this._loadPacks(true) | ||||
| 	} | ||||
| 
 | ||||
| 	_loadPacks(disableCache = false) { | ||||
| 		const cache = disableCache ? "no-cache" : undefined | ||||
| 		fetch(INDEX, { cache }).then(async indexRes => { | ||||
| 		fetch(`${PACKS_BASE_URL}/index.json`, { cache }).then(async indexRes => { | ||||
| 			if (indexRes.status >= 400) { | ||||
| 				this.setState({ | ||||
| 					loading: false, | ||||
| @ -166,12 +129,7 @@ class App extends Component { | ||||
| 			HOMESERVER_URL = indexData.homeserver_url || HOMESERVER_URL | ||||
| 			// TODO only load pack metadata when scrolled into view?
 | ||||
| 			for (const packFile of indexData.packs) { | ||||
| 				let packRes | ||||
| 				if (packFile.startsWith("https://") || packFile.startsWith("http://")) { | ||||
| 					packRes = await fetch(packFile, { cache }) | ||||
| 				} else { | ||||
| 					packRes = await fetch(`${PACKS_BASE_URL}/${packFile}`, { cache }) | ||||
| 				} | ||||
| 				const packRes = await fetch(`${PACKS_BASE_URL}/${packFile}`, { cache }) | ||||
| 				const packData = await packRes.json() | ||||
| 				for (const sticker of packData.stickers) { | ||||
| 					this.stickersByID.set(sticker.id, sticker) | ||||
| @ -267,9 +225,6 @@ class App extends Component { | ||||
| 
 | ||||
| 	render() { | ||||
| 		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) { | ||||
| 			return html`<main class="spinner ${theme}"><${Spinner} size=${80} green /></main>` | ||||
| 		} else if (this.state.error) { | ||||
| @ -280,17 +235,15 @@ class App extends Component { | ||||
| 		} else if (this.state.packs.length === 0) { | ||||
| 			return html`<main class="empty ${theme}"><h1>No packs found 😿</h1></main>` | ||||
| 		} | ||||
| 
 | ||||
| 		return html`<main class="has-content ${theme}">
 | ||||
| 			<nav onWheel=${this.navScroll} ref=${elem => this.navRef = elem}> | ||||
| 				<${NavBarItem} pack=${this.state.frequentlyUsed} iconOverride="recent" /> | ||||
| 				${this.state.packs.map(pack => html`<${NavBarItem} id=${pack.id} pack=${pack}/>`)} | ||||
| 				<${NavBarItem} pack=${{ id: "settings", title: "Settings" }} iconOverride="settings" /> | ||||
| 			</nav> | ||||
| 			<${SearchBox} onKeyUp=${this.searchStickers} /> | ||||
| 			<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} | ||||
| 				${packs.map(pack => html`<${Pack} id=${pack.id} pack=${pack} send=${this.sendSticker} />`)} | ||||
| 				<${Pack} pack=${this.state.frequentlyUsed} send=${this.sendSticker} /> | ||||
| 				${this.state.packs.map(pack => html`<${Pack} id=${pack.id} pack=${pack} send=${this.sendSticker} />`)} | ||||
| 				<${Settings} app=${this}/> | ||||
| 			</div> | ||||
| 		</main>` | ||||
|  | ||||
| @ -1,26 +0,0 @@ | ||||
| // maunium-stickerpicker - A fast and simple Matrix sticker picker widget.
 | ||||
| // Copyright (C) 2020 Tulir Asokan
 | ||||
| //
 | ||||
| // This program is free software: you can redistribute it and/or modify
 | ||||
| // it under the terms of the GNU Affero General Public License as published by
 | ||||
| // the Free Software Foundation, either version 3 of the License, or
 | ||||
| // (at your option) any later version.
 | ||||
| //
 | ||||
| // This program is distributed in the hope that it will be useful,
 | ||||
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | ||||
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | ||||
| // GNU Affero General Public License for more details.
 | ||||
| //
 | ||||
| // You should have received a copy of the GNU Affero General Public License
 | ||||
| // along with this program.  If not, see <https://www.gnu.org/licenses/>.
 | ||||
| import { html } from "../lib/htm/preact.js" | ||||
| 
 | ||||
| export const SearchBox = ({ onKeyUp, placeholder = 'Find stickers' }) => { | ||||
| 	const component = html` | ||||
| 		<div class="search-box"> | ||||
| 			<input type="text" placeholder=${placeholder} onKeyUp=${onKeyUp} /> | ||||
| 			<span class="icon icon-search" /> | ||||
| 		</div> | ||||
| 	` | ||||
| 	return component | ||||
| } | ||||
| @ -1 +1 @@ | ||||
| *{font-family:sans-serif}body{margin:0}h1{font-size:1rem}:root{--stickers-per-row: 4;--sticker-size: calc(100vw / var(--stickers-per-row))}main{color:var(--text-color)}main.spinner{margin-top:5rem}main.error,main.empty{margin:2rem}main.empty{text-align:center}main.has-content{position:fixed;top:0;left:0;right:0;bottom:0;display:grid;grid-template-rows:calc(12vw + 2px) min-content auto}main.theme-light{--highlight-color: #eee;--search-box-color: var(--highlight-color);--text-color: black;background-color:#fff}main.theme-dark{--highlight-color: #444;--search-box-color: #383e4b;--text-color: white;background-color:#22262e}main.theme-black{--highlight-color: #222;--search-box-color: var(--highlight-color);--text-color: white;background-color:#000}.icon{width:100%;height:100%;background-color:var(--text-color);mask-size:contain;-webkit-mask-size:contain;mask-image:var(--icon-image);-webkit-mask-image:var(--icon-image)}.icon.icon-settings{--icon-image: url(../res/settings.svg)}.icon.icon-recent{--icon-image: url(../res/recent.svg)}.icon.icon.icon-search{--icon-image: url(../res/search.svg)}nav{display:flex;overflow-x:auto}nav>a{border-bottom:2px solid transparent}nav>a.visible{border-bottom-color:green}nav>a>div.sticker{width:12vw;height:12vw}div.pack-list,nav{scrollbar-width:none}div.pack-list::-webkit-scrollbar,nav::-webkit-scrollbar{display:none}div.pack-list{overflow-y:auto}div.pack-list.ios-safari-hack{position:fixed;top:calc(calc(12vw + 2px) + calc(2 * 0.7rem + 2 * 0.5rem + 1rem));bottom:0;left:0;right:0;-webkit-overflow-scrolling:touch}div.search-empty{margin:1.2rem;text-align:center}section.stickerpack{margin-top:.75rem}section.stickerpack>div.sticker-list{display:flex;flex-wrap:wrap}section.stickerpack>h1{margin:0 0 0 .75rem}div.sticker{display:flex;padding:4px;cursor:pointer;position:relative;width:var(--sticker-size);height:var(--sticker-size);box-sizing:border-box}div.sticker:hover{background-color:var(--highlight-color)}div.sticker>img{display:none;width:100%;object-fit:contain}div.sticker>img.visible{display:initial}div.sticker>.icon{width:70%;height:70%;margin:15%}div.search-box{position:relative;display:flex}div.search-box>input[type=text]{flex-grow:1;background-color:var(--search-box-color);outline:none;border:none;border-radius:.25rem;height:1rem;padding:.7rem;padding-right:calc(1rem + 0.7rem);margin:.5rem;font-size:1rem;color:var(--text-color)}div.search-box>span.icon{display:flex;position:absolute;top:calc(50% - 1rem / 2);right:1rem;width:1rem;height:1rem;box-sizing:border-box}div.settings-list{display:flex;flex-direction:column}div.settings-list>*{margin:.5rem}div.settings-list button{padding:.5rem;border-radius:.25rem}div.settings-list input{width:100%} | ||||
| *{font-family:sans-serif}body{margin:0}h1{font-size:1rem}:root{--stickers-per-row: 4;--sticker-size: calc(100vw / var(--stickers-per-row))}main{color:var(--text-color)}main.spinner{margin-top:5rem}main.error,main.empty{margin:2rem}main.empty{text-align:center}main.has-content{position:fixed;top:0;left:0;right:0;bottom:0;display:grid;grid-template-rows:calc(12vw + 2px) auto}main.theme-light{--highlight-color: #eee;--text-color: black;background-color:white}main.theme-dark{--highlight-color: #444;--text-color: white;background-color:#22262e}main.theme-black{--highlight-color: #222;--text-color: white;background-color:black}.icon{width:100%;height:100%;background-color:var(--text-color);mask-size:contain;-webkit-mask-size:contain;mask-image:var(--icon-image);-webkit-mask-image:var(--icon-image)}.icon.icon-settings{--icon-image: url(../res/settings.svg)}.icon.icon-recent{--icon-image: url(../res/recent.svg)}nav{display:flex;overflow-x:auto}nav>a{border-bottom:2px solid transparent}nav>a.visible{border-bottom-color:green}nav>a>div.sticker{width:12vw;height:12vw}div.pack-list,nav{scrollbar-width:none}div.pack-list::-webkit-scrollbar,nav::-webkit-scrollbar{display:none}div.pack-list{overflow-y:auto}div.pack-list.ios-safari-hack{position:fixed;top:calc(12vw + 2px);bottom:0;left:0;right:0;-webkit-overflow-scrolling:touch}section.stickerpack{margin-top:.75rem}section.stickerpack>div.sticker-list{display:flex;flex-wrap:wrap}section.stickerpack>h1{margin:0 0 0 .75rem}div.sticker{display:flex;padding:4px;cursor:pointer;position:relative;width:var(--sticker-size);height:var(--sticker-size);box-sizing:border-box}div.sticker:hover{background-color:var(--highlight-color)}div.sticker>img{display:none;width:100%;object-fit:contain}div.sticker>img.visible{display:initial}div.sticker>.icon{width:70%;height:70%;margin:15%}div.settings-list{display:flex;flex-direction:column}div.settings-list>*{margin:.5rem}div.settings-list button{padding:.5rem;border-radius:.25rem}div.settings-list input{width:100%} | ||||
|  | ||||
| @ -32,12 +32,6 @@ $nav-bottom-highlight: 2px | ||||
| $nav-height: calc(#{$nav-sticker-size} + #{$nav-bottom-highlight}) | ||||
| $nav-height-inverse: calc(-#{$nav-sticker-size} - #{$nav-bottom-highlight}) | ||||
| 
 | ||||
| $search-box-icon-size: 1rem | ||||
| $search-box-input-height: 1rem | ||||
| $search-box-input-padding: .7rem | ||||
| $search-box-input-margin: .5rem | ||||
| $search-box-height: calc(2 * #{$search-box-input-padding} + 2 * #{$search-box-input-margin} + #{$search-box-input-height}) | ||||
| 
 | ||||
| main | ||||
|   color: var(--text-color) | ||||
| 
 | ||||
| @ -56,24 +50,22 @@ main | ||||
|     left: 0 | ||||
|     right: 0 | ||||
|     bottom: 0 | ||||
| 
 | ||||
|     display: grid | ||||
|     grid-template-rows: $nav-height min-content auto | ||||
|     grid-template-rows: $nav-height auto | ||||
| 
 | ||||
| main.theme-light | ||||
|   --highlight-color: #eee | ||||
|   --search-box-color: var(--highlight-color) | ||||
|   --text-color: black | ||||
|   background-color: white | ||||
| 
 | ||||
| main.theme-dark | ||||
|   --highlight-color: #444 | ||||
|   --search-box-color: #383e4b | ||||
|   --text-color: white | ||||
|   background-color: #22262e | ||||
| 
 | ||||
| main.theme-black | ||||
|   --highlight-color: #222 | ||||
|   --search-box-color: var(--highlight-color) | ||||
|   --text-color: white | ||||
|   background-color: black | ||||
| 
 | ||||
| @ -92,9 +84,6 @@ main.theme-black | ||||
|   &.icon-recent | ||||
|     --icon-image: url(../res/recent.svg) | ||||
| 
 | ||||
|   &.icon.icon-search | ||||
|     --icon-image: url(../res/search.svg) | ||||
| 
 | ||||
| nav | ||||
|   display: flex | ||||
|   overflow-x: auto | ||||
| @ -120,16 +109,12 @@ div.pack-list | ||||
| 
 | ||||
| div.pack-list.ios-safari-hack | ||||
|   position: fixed | ||||
|   top: calc(#{$nav-height} + #{$search-box-height}) | ||||
|   top: $nav-height | ||||
|   bottom: 0 | ||||
|   left: 0 | ||||
|   right: 0 | ||||
|   -webkit-overflow-scrolling: touch | ||||
| 
 | ||||
| div.search-empty | ||||
|   margin: 1.2rem | ||||
|   text-align: center | ||||
| 
 | ||||
| section.stickerpack | ||||
|   margin-top: .75rem | ||||
| 
 | ||||
| @ -165,32 +150,6 @@ div.sticker | ||||
|     height: 70% | ||||
|     margin: 15% | ||||
| 
 | ||||
| div.search-box | ||||
|   position: relative | ||||
|   display: flex | ||||
| 
 | ||||
|   >input[type="text"] | ||||
|     flex-grow: 1 | ||||
|     background-color: var(--search-box-color) | ||||
|     outline: none | ||||
|     border: none | ||||
|     border-radius: .25rem | ||||
|     height: $search-box-input-height | ||||
|     padding: $search-box-input-padding | ||||
|     padding-right: calc(#{$search-box-icon-size} + #{$search-box-input-padding}) | ||||
|     margin: $search-box-input-margin | ||||
|     font-size: 1rem | ||||
|     color: var(--text-color) | ||||
| 
 | ||||
|   >span.icon | ||||
|     display: flex | ||||
|     position: absolute | ||||
|     top: calc(50% - #{$search-box-icon-size} / 2) | ||||
|     right: $search-box-icon-size | ||||
|     width: $search-box-icon-size | ||||
|     height: $search-box-icon-size | ||||
|     box-sizing: border-box | ||||
| 
 | ||||
| div.settings-list | ||||
|   display: flex | ||||
|   flex-direction: column | ||||
|  | ||||
| @ -1 +1 @@ | ||||
| .sk-center-wrapper{width:100%;display:flex;justify-content:space-around}.sk-chase{position:relative;animation:sk-chase 2.5s infinite linear both}.sk-chase.green>.sk-chase-dot:before{background-color:#00c853}.sk-chase>.sk-chase-dot{width:100%;height:100%;position:absolute;left:0;top:0;animation:sk-chase-dot 2s infinite ease-in-out both}.sk-chase>.sk-chase-dot:before{content:"";display:block;width:25%;height:25%;border-radius:100%;animation:sk-chase-dot-before 2s infinite ease-in-out both;background-color:#fff}.sk-chase>.sk-chase-dot:nth-child(1){animation-delay:-1.1s}.sk-chase>.sk-chase-dot:nth-child(2){animation-delay:-1s}.sk-chase>.sk-chase-dot:nth-child(3){animation-delay:-0.9s}.sk-chase>.sk-chase-dot:nth-child(4){animation-delay:-0.8s}.sk-chase>.sk-chase-dot:nth-child(5){animation-delay:-0.7s}.sk-chase>.sk-chase-dot:nth-child(6){animation-delay:-0.6s}.sk-chase>.sk-chase-dot:nth-child(1):before{animation-delay:-1.1s}.sk-chase>.sk-chase-dot:nth-child(2):before{animation-delay:-1s}.sk-chase>.sk-chase-dot:nth-child(3):before{animation-delay:-0.9s}.sk-chase>.sk-chase-dot:nth-child(4):before{animation-delay:-0.8s}.sk-chase>.sk-chase-dot:nth-child(5):before{animation-delay:-0.7s}.sk-chase>.sk-chase-dot:nth-child(6):before{animation-delay:-0.6s}@keyframes sk-chase{100%{transform:rotate(360deg)}}@keyframes sk-chase-dot{80%,100%{transform:rotate(360deg)}}@keyframes sk-chase-dot-before{50%{transform:scale(0.4)}100%,0%{transform:scale(1)}} | ||||
| .sk-center-wrapper{width:100%;display:flex;justify-content:space-around}.sk-chase{position:relative;animation:sk-chase 2.5s infinite linear both}.sk-chase.green>.sk-chase-dot:before{background-color:#00C853}.sk-chase>.sk-chase-dot{width:100%;height:100%;position:absolute;left:0;top:0;animation:sk-chase-dot 2.0s infinite ease-in-out both}.sk-chase>.sk-chase-dot:before{content:'';display:block;width:25%;height:25%;border-radius:100%;animation:sk-chase-dot-before 2.0s infinite ease-in-out both;background-color:#FFF}.sk-chase>.sk-chase-dot:nth-child(1){animation-delay:-1.1s}.sk-chase>.sk-chase-dot:nth-child(2){animation-delay:-1.0s}.sk-chase>.sk-chase-dot:nth-child(3){animation-delay:-0.9s}.sk-chase>.sk-chase-dot:nth-child(4){animation-delay:-0.8s}.sk-chase>.sk-chase-dot:nth-child(5){animation-delay:-0.7s}.sk-chase>.sk-chase-dot:nth-child(6){animation-delay:-0.6s}.sk-chase>.sk-chase-dot:nth-child(1):before{animation-delay:-1.1s}.sk-chase>.sk-chase-dot:nth-child(2):before{animation-delay:-1.0s}.sk-chase>.sk-chase-dot:nth-child(3):before{animation-delay:-0.9s}.sk-chase>.sk-chase-dot:nth-child(4):before{animation-delay:-0.8s}.sk-chase>.sk-chase-dot:nth-child(5):before{animation-delay:-0.7s}.sk-chase>.sk-chase-dot:nth-child(6):before{animation-delay:-0.6s}@keyframes sk-chase{100%{transform:rotate(360deg)}}@keyframes sk-chase-dot{80%,100%{transform:rotate(360deg)}}@keyframes sk-chase-dot-before{50%{transform:scale(0.4)}100%,0%{transform:scale(1)}} | ||||
|  | ||||
							
								
								
									
										2366
									
								
								web/yarn.lock
									
									
									
									
									
								
							
							
						
						
									
										2366
									
								
								web/yarn.lock
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user