Compare commits
	
		
			16 Commits
		
	
	
		
			1.13.4
			...
			feature/cu
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 77413c7e53 | |||
| 72e5425c08 | |||
| 53f5fa3988 | |||
| 6ef6dc0078 | |||
| b89d4f248c | |||
| 444b1f5370 | |||
| 3709881176 | |||
| a00af08b3f | |||
| 9ef34359d8 | |||
| 4d79589903 | |||
| 1386b80d0c | |||
| 286bd61497 | |||
| 50b5e4e455 | |||
| 2c91f46375 | |||
| 0cb1193269 | |||
| 564a971c63 | 
							
								
								
									
										15
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -2,23 +2,8 @@ | ||||
|  | ||||
| All notable changes to this project will be documented in this file. Dates are displayed in UTC. | ||||
|  | ||||
| #### [1.13.4](https://git.odit.services/lfk/frontend/compare/1.13.3...1.13.4) | ||||
|  | ||||
| - feat(donationcreate): improved focus handling [`a827279`](https://git.odit.services/lfk/frontend/commit/a82727916345c7e713d4225c4771ef3f23d1392c) | ||||
| - chore(deps): remove unused [`3842d8b`](https://git.odit.services/lfk/frontend/commit/3842d8b1048ce12f0f70bf3d0530590470f0d200) | ||||
| - fix(donationcreate): clearing [`9298a0d`](https://git.odit.services/lfk/frontend/commit/9298a0dc922ee5ed5b7c9017c865ad4b68fca3c8) | ||||
| - feat(donationcreate): autofocus runner input on page load [`b9e2e65`](https://git.odit.services/lfk/frontend/commit/b9e2e653310c686bc06b9f27c38b49e9c6a3eaef) | ||||
| - fix(DonationCreate): remove duplicate spaces from getRunnerLabel [`30a26ef`](https://git.odit.services/lfk/frontend/commit/30a26ef3ed55d072cd9bf2aea1b200fadc2a05f1) | ||||
| - fix(donationcreate): improved resetAll [`7d9314f`](https://git.odit.services/lfk/frontend/commit/7d9314f05c58c1b50901f3797c0b461c4c79e5d2) | ||||
| - fix(DeleteDonationModal): cannot overflow [`ca066aa`](https://git.odit.services/lfk/frontend/commit/ca066aa7a7a8d7c46e0f59370b06636faf5736ca) | ||||
| - feat(donationcreate): full width [`b0063cd`](https://git.odit.services/lfk/frontend/commit/b0063cdead5f71c334c36e5587a58e957825dbcd) | ||||
| - feat(donationcreate): add runner id to select [`27e7bbb`](https://git.odit.services/lfk/frontend/commit/27e7bbb9d142fbea659e89fb2335cc6c567d14ce) | ||||
|  | ||||
| #### [1.13.3](https://git.odit.services/lfk/frontend/compare/1.13.2...1.13.3) | ||||
|  | ||||
| > 19 May 2025 | ||||
|  | ||||
| - chore(release): 1.13.3 [`2139b19`](https://git.odit.services/lfk/frontend/commit/2139b197ba672275e2a0b5ffbcf7fa43f80874e6) | ||||
| - Refactor code structure for improved readability and maintainability [`e3c6d5a`](https://git.odit.services/lfk/frontend/commit/e3c6d5a5c0eaac2c91432b0be37d6fa11e57f644) | ||||
| - refactor(donation): Refactor donor selection and add new donor creation functionality [`8c3f009`](https://git.odit.services/lfk/frontend/commit/8c3f0092d2735b1c85976f4e6955780b1035f68a) | ||||
| - fix(donation): Ensure all selections are cleared on reset [`4e1a944`](https://git.odit.services/lfk/frontend/commit/4e1a944a2d7d0d0666fb8d2181a9941d0f11957f) | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
|  | ||||
|   <body> | ||||
|     <span style="display: none; visibility: hidden" id="buildinfo" | ||||
|       >RELEASE_INFO-1.13.4-RELEASE_INFO</span | ||||
|       >RELEASE_INFO-1.13.3-RELEASE_INFO</span | ||||
|     > | ||||
|     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||
|     <script src="/env.js"></script> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "@odit/lfk-frontend", | ||||
|   "version": "1.13.4", | ||||
|   "version": "1.13.3", | ||||
|   "type": "module", | ||||
|   "scripts": { | ||||
|     "i18n-order": "node order.js", | ||||
|   | ||||
| @@ -81,7 +81,7 @@ | ||||
|                 /></svg | ||||
|               > | ||||
|             </div> | ||||
|             <div class="mt-3 sm:text-left max-h-[75vh]"> | ||||
|             <div class="mt-3 sm:text-left max-h-[75vh] overflow-y-auto"> | ||||
|               <h3 class="text-lg leading-6 font-medium text-gray-900"> | ||||
|                 {$_("please-confirm-the-deletion-of-donation")} | ||||
|               </h3> | ||||
|   | ||||
| @@ -5,8 +5,8 @@ | ||||
| 		DonorService, | ||||
| 		RunnerService, | ||||
| 	} from "@odit/lfk-client-js"; | ||||
|   import Select from "svelte-select"; | ||||
| 	import toast from "svelte-french-toast"; | ||||
| 	import VirtualSelect from "./VirtualSelect.svelte"; | ||||
| 	import { onMount } from "svelte"; | ||||
|  | ||||
| 	let runners = []; | ||||
| @@ -53,37 +53,32 @@ | ||||
| 	loadDonors(); | ||||
|  | ||||
| 	const getRunnerLabel = (option) => { | ||||
|     return [option.firstname,option.middlename,option.lastname].join(" ").replace("  "," ") + " [#"+option.id+"]"; | ||||
|   } | ||||
|  | ||||
|   const filterRunners = (label, filterText, option) => { | ||||
|     if (filterText.startsWith("#")) { | ||||
|       return option.value.id == parseInt(filterText.replace("#", "")); | ||||
|     } | ||||
| 		return ( | ||||
|       label.toLowerCase().includes(filterText.toLowerCase()) || | ||||
|       option.value.toString().startsWith(filterText.toLowerCase()) | ||||
| 			[option.firstname, option.middlename, option.lastname] | ||||
| 				.join(" ") | ||||
| 				.replace("  ", " ") + | ||||
| 			" [#" + | ||||
| 			option.id + | ||||
| 			"]" | ||||
| 		); | ||||
| 	}; | ||||
|  | ||||
| 	let selectRefRunner; | ||||
| 	let selectRefDonor; | ||||
|  | ||||
| 	function resetAll() { | ||||
| 		runnerinfo = { id: 0, firstname: "", lastname: "" }; | ||||
| 		donorinfo = { id: 0, firstname: "", lastname: "" }; | ||||
| 		amount = null; | ||||
| 		address_checked = false; | ||||
| 		donor_create_new = false; | ||||
|     const clears = document.querySelectorAll(".clearSelect"); | ||||
|     clears.forEach(c => { | ||||
|       c.click(); | ||||
|     }); | ||||
|     setTimeout(() => { | ||||
|       document.querySelector("#wrapper_runner_select input").focus(); | ||||
|     }, 50); | ||||
| 		selectRefRunner?.reset(); | ||||
| 		selectRefDonor?.reset(); | ||||
| 		document.querySelector("#jjqzqicxujrnnh1x3447x18x").focus(); | ||||
| 	} | ||||
|  | ||||
| 	onMount(() => { | ||||
|     document.querySelector("#wrapper_runner_select input").focus(); | ||||
|   }) | ||||
| 		document.querySelector("#jjqzqicxujrnnh1x3447x18x").focus(); | ||||
| 	}); | ||||
| </script> | ||||
|  | ||||
| <div class="p-4"> | ||||
| @@ -102,24 +97,32 @@ | ||||
| 				</div> | ||||
| 			{/if} | ||||
|  | ||||
|       <!-- Runner Selection --> | ||||
|       <div id="wrapper_runner_select"> | ||||
| 			<!--  --> | ||||
| 			<h4 class="text-xl font-semibold">{$_("runner")}</h4> | ||||
|         <Select | ||||
|           containerClasses="rounded-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||
|           itemFilter={(label, filterText, option) => | ||||
|             filterRunners(label, filterText, option)} | ||||
|           items={runners} | ||||
|           showChevron={true} | ||||
|           placeholder={$_("search-for-runner-by-name-or-id")} | ||||
|           noOptionsMessage={$_("no-runners-found")} | ||||
|           on:select={(selectedValue) => { | ||||
|             runnerinfo = selectedValue.detail.value; | ||||
|             document.querySelector("#donation_amount_eur").focus(); | ||||
| 			<VirtualSelect | ||||
| 				inputElementID="jjqzqicxujrnnh1x3447x18x" | ||||
| 				bind:this={selectRefRunner} | ||||
| 				on:onClear={() => { | ||||
| 					console.log("Cleared selection"); | ||||
| 				}} | ||||
| 				options={runners} | ||||
| 				filterFn={(item, searchTerm) => { | ||||
| 					if (searchTerm.startsWith("#")) { | ||||
| 						const id = parseInt(searchTerm.replace("#", "")); | ||||
| 						return item.value.id === id; | ||||
| 					} | ||||
| 					return item.label.toLowerCase().includes(searchTerm.toLowerCase()); | ||||
| 				}} | ||||
| 				bind:selected={runnerinfo} | ||||
| 				inputAriaLabel={$_("search-for-runner-by-name-or-id")} | ||||
| 				inputPlaceholder={$_("search-for-runner-by-name-or-id")} | ||||
| 				noOptionsText={$_("no-runners-found")} | ||||
| 				on:onSelected={(data) => { | ||||
| 					if (data.detail !== null) { | ||||
| 						document.querySelector("#donation_amount_eur").focus(); | ||||
| 					} | ||||
| 				}} | ||||
|           on:clear={() => (runnerinfo = { id: 0, firstname: "", lastname: "" })} | ||||
| 			/> | ||||
|       </div> | ||||
|  | ||||
| 			<!-- Amount Input --> | ||||
| 			<div> | ||||
| @@ -131,8 +134,7 @@ | ||||
| 						class:focus:border-red-500={!amount > 0} | ||||
| 						class:focus:ring-red-500={!amount > 0} | ||||
| 						bind:value={amount} | ||||
|             on:keydown={(e)=> | ||||
|             { | ||||
| 						on:keydown={(e) => { | ||||
| 							if (e.key === "Enter") { | ||||
| 								e.preventDefault(); | ||||
| 								document.querySelector("#button_existing_donor").focus(); | ||||
| @@ -160,13 +162,16 @@ | ||||
| 				<div class="mb-2"> | ||||
| 					<div class="flex border rounded-md overflow-hidden shadow-sm"> | ||||
| 						<button | ||||
|             on:keydown={(e)=> | ||||
|             { | ||||
| 							on:keydown={(e) => { | ||||
| 								if (e.key === "ArrowRight") { | ||||
| 									e.preventDefault(); | ||||
| 									document.querySelector("#button_new_donor").focus(); | ||||
| 									document.querySelector("#button_new_donor").click(); | ||||
| 								} | ||||
| 								if (e.key === "Enter") { | ||||
| 									e.preventDefault(); | ||||
| 									document.querySelector("#zt12c3udy3bme5bqobmqcif1").focus(); | ||||
| 								} | ||||
| 							}} | ||||
| 							id="button_existing_donor" | ||||
| 							class:bg-indigo-600={!donor_create_new} | ||||
| @@ -180,19 +185,25 @@ | ||||
| 							{$_("existing-donor")} | ||||
| 						</button> | ||||
| 						<button | ||||
|               on:keydown={(e)=> | ||||
|               { | ||||
| 							on:keydown={(e) => { | ||||
| 								if (e.key === "ArrowLeft") { | ||||
| 									e.preventDefault(); | ||||
| 									document.querySelector("#button_existing_donor").focus(); | ||||
| 									document.querySelector("#button_existing_donor").click(); | ||||
| 								} | ||||
| 								if (e.key === "Enter") { | ||||
| 									e.preventDefault(); | ||||
| 									document.querySelector("#button_new_donor").click(); | ||||
| 								} | ||||
| 							}} | ||||
| 							id="button_new_donor" | ||||
| 							class={`py-2 px-4 w-1/2 transition-colors ${donor_create_new ? "bg-indigo-600 text-white" : "bg-gray-100 text-gray-700"}`} | ||||
| 							on:click={() => { | ||||
| 								donor_create_new = true; | ||||
| 								donorinfo = { id: 0, firstname: "", lastname: "" }; | ||||
| 								setTimeout(() => { | ||||
| 									document.querySelector("#firstname").focus(); | ||||
| 								}, 50); | ||||
| 							}} | ||||
| 						> | ||||
| 							{$_("new-donor")} | ||||
| @@ -201,19 +212,31 @@ | ||||
| 				</div> | ||||
|  | ||||
| 				{#if !donor_create_new} | ||||
|           <Select | ||||
|             containerClasses="rounded-md mt-1 focus:ring-indigo-500 focus:border-indigo-500 block w-full shadow-sm sm:text-sm border-gray-300 border bg-gray-50 text-neutral-800 rounded-md p-2" | ||||
|             itemFilter={(label, filterText, option) => | ||||
|               filterRunners(label, filterText, option)} | ||||
|             items={donors} | ||||
|             showChevron={true} | ||||
|             placeholder={$_("search-for-donor")} | ||||
|             noOptionsMessage={$_("no-donors-found")} | ||||
|             on:select={(selectedValue) => { | ||||
|               donorinfo = selectedValue.detail.value; | ||||
| 					<VirtualSelect | ||||
| 						inputElementID="zt12c3udy3bme5bqobmqcif1" | ||||
| 						bind:this={selectRefDonor} | ||||
| 						on:onClear={() => { | ||||
| 							console.log("Cleared selection"); | ||||
| 						}} | ||||
| 						options={donors} | ||||
| 						filterFn={(item, searchTerm) => { | ||||
| 							return item.label | ||||
| 								.toLowerCase() | ||||
| 								.includes(searchTerm.toLowerCase()); | ||||
| 						}} | ||||
| 						bind:selected={donorinfo} | ||||
| 						inputAriaLabel={$_("search-for-donor")} | ||||
| 						inputPlaceholder={$_("search-for-donor")} | ||||
| 						noOptionsText={$_("no-donors-found")} | ||||
| 						on:onSelected={(data) => { | ||||
| 							console.log(data.detail); | ||||
| 							if (data.detail !== null) { | ||||
| 								document.querySelector("#submit_button").focus(); | ||||
| 								setTimeout(() => { | ||||
| 									document.querySelector("#submit_button").focus(); | ||||
| 								}, 100); | ||||
| 							} | ||||
| 						}} | ||||
|             on:clear={() => | ||||
|               (donorinfo = { id: 0, firstname: "", lastname: "" })} | ||||
| 					/> | ||||
| 				{:else} | ||||
| 					<div class="space-y-3"> | ||||
| @@ -228,6 +251,11 @@ | ||||
| 							<input | ||||
| 								type="text" | ||||
| 								id="firstname" | ||||
| 								on:keydown={(e) => { | ||||
| 									if (e.key === "Enter") { | ||||
| 										document.querySelector("#lastname").focus(); | ||||
| 									} | ||||
| 								}} | ||||
| 								bind:value={donorinfo.firstname} | ||||
| 								class="focus:ring-indigo-500 focus:border-indigo-500 flex-1 block w-full rounded-none rounded-l-md sm:text-sm border-neutral-300 border bg-neutral-50 text-neutral-800 p-2" | ||||
| 								placeholder={$_("first-name")} | ||||
|   | ||||
							
								
								
									
										357
									
								
								src/components/tools/VirtualSelect.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										357
									
								
								src/components/tools/VirtualSelect.svelte
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,357 @@ | ||||
| <script> | ||||
| 	import { createEventDispatcher, onMount, tick } from "svelte"; | ||||
|  | ||||
| 	// Generate a default unique ID | ||||
| 	function generateDefaultID() { | ||||
| 		return "virtual-select-" + Math.random().toString(36).slice(2); | ||||
| 	} | ||||
|  | ||||
| 	// Props | ||||
| 	export let options = []; | ||||
| 	export let selected = null; | ||||
| 	export let inputPlaceholder = "Search options..."; | ||||
| 	export let noOptionsText = "No options found"; | ||||
| 	export let inputAriaLabel = "Search and select an option"; | ||||
| 	export let toggleAriaLabel = "Toggle dropdown"; | ||||
| 	export let clearAriaLabel = "Clear selection"; | ||||
| 	export let filterFn = null; // Custom filter function | ||||
| 	export let autofocus = false; // Autofocus input | ||||
| 	export let inputElementID = generateDefaultID(); // Input element ID | ||||
|  | ||||
| 	// Internal state | ||||
| 	let searchTerm = ""; | ||||
| 	let filteredOptions = options; | ||||
| 	let isOpen = false; | ||||
| 	let container; | ||||
| 	let visibleItems = []; | ||||
| 	let startIndex = 0; | ||||
| 	let itemHeight = 40; // Fixed height for each option (in pixels) | ||||
| 	let visibleCount = 10; // Default number of items to render | ||||
| 	let focusedIndex = -1; // Track the focused option index (-1 means no focus) | ||||
| 	let inputElement; // Reference to input element | ||||
|  | ||||
| 	const dispatch = createEventDispatcher(); | ||||
|  | ||||
| 	// Filter options based on search term | ||||
| 	$: { | ||||
| 		filteredOptions = searchTerm | ||||
| 			? filterFn | ||||
| 				? options.filter((option) => filterFn(option, searchTerm)) | ||||
| 				: options.filter((option) => | ||||
| 						option.label.toLowerCase().includes(searchTerm.toLowerCase()) | ||||
| 					) | ||||
| 			: options; | ||||
| 		// Reset scroll and focus when filtered options change | ||||
| 		startIndex = 0; | ||||
| 		focusedIndex = -1; | ||||
| 		updateVisibleItems(); | ||||
| 	} | ||||
|  | ||||
| 	// Update visible items based on scroll position | ||||
| 	function updateVisibleItems() { | ||||
| 		if (!container) return; | ||||
| 		const scrollTop = container.scrollTop; | ||||
| 		startIndex = Math.floor(scrollTop / itemHeight); | ||||
| 		const endIndex = Math.min( | ||||
| 			startIndex + visibleCount, | ||||
| 			filteredOptions.length | ||||
| 		); | ||||
| 		visibleItems = filteredOptions.slice(startIndex, endIndex); | ||||
| 	} | ||||
|  | ||||
| 	// Handle scroll event | ||||
| 	function handleScroll() { | ||||
| 		updateVisibleItems(); | ||||
| 	} | ||||
|  | ||||
| 	// Calculate visible item count based on container height | ||||
| 	async function updateVisibleCount() { | ||||
| 		if (container) { | ||||
| 			await tick(); // Wait for DOM to render | ||||
| 			visibleCount = Math.ceil(container.clientHeight / itemHeight) + 2; // Buffer of 2 items | ||||
| 			updateVisibleItems(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Handle option selection | ||||
| 	function selectOption(option) { | ||||
| 		selected = option.value; | ||||
| 		isOpen = false; | ||||
| 		searchTerm = option.label; // Set searchTerm to the selected option's label | ||||
| 		focusedIndex = -1; | ||||
| 		dispatch("onSelected", option.value); | ||||
| 	} | ||||
|  | ||||
| 	// Handle clear selection | ||||
| 	function clearSelection() { | ||||
| 		selected = null; | ||||
| 		searchTerm = ""; | ||||
| 		focusedIndex = -1; | ||||
| 		dispatch("onSelected", null); | ||||
| 		dispatch("onClear"); | ||||
| 	} | ||||
|  | ||||
| 	// Reset component state | ||||
| 	export function reset() { | ||||
| 		selected = null; | ||||
| 		searchTerm = ""; | ||||
| 		isOpen = false; | ||||
| 		focusedIndex = -1; | ||||
| 		startIndex = 0; | ||||
| 		updateVisibleItems(); | ||||
| 		dispatch("onSelected", null); | ||||
| 		dispatch("onClear"); | ||||
| 	} | ||||
|  | ||||
| 	// Toggle dropdown | ||||
| 	async function toggleDropdown() { | ||||
| 		isOpen = !isOpen; | ||||
| 		if (isOpen) { | ||||
| 			forceVisibleItems(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Handle click outside to close dropdown | ||||
| 	function handleClickOutside(event) { | ||||
| 		if (!event.target.closest(".select-container")) { | ||||
| 			isOpen = false; | ||||
| 			focusedIndex = -1; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Handle input focus to open dropdown | ||||
| 	async function handleInputFocus() { | ||||
| 		// forceVisibleItems(); | ||||
| 	} | ||||
|  | ||||
| 	// Handle input typing to open dropdown | ||||
| 	async function handleInput() { | ||||
| 		forceVisibleItems(); | ||||
| 	} | ||||
|  | ||||
| 	async function forceVisibleItems() { | ||||
| 		isOpen = true; | ||||
| 		await updateVisibleCount(); // Ensure items render on focus | ||||
| 		// these 2 timeouts are a more or less tmp fix for rendering items when dropdown opens | ||||
| 		setTimeout(async () => { | ||||
| 			await updateVisibleCount(); // Ensure items render on focus | ||||
| 		}, 25); | ||||
| 		setTimeout(async () => { | ||||
| 			await updateVisibleCount(); // Ensure items render on focus | ||||
| 		}, 50); | ||||
| 	} | ||||
|  | ||||
| 	// Handle keyboard navigation | ||||
| 	function handleKeydown(event, index) { | ||||
| 		if (!isOpen) return; | ||||
|  | ||||
| 		if (event.key === "ArrowDown") { | ||||
| 			event.preventDefault(); | ||||
| 			if (focusedIndex < filteredOptions.length - 1) { | ||||
| 				focusedIndex += 1; | ||||
| 				scrollToFocusedItem(); | ||||
| 			} | ||||
| 		} else if (event.key === "ArrowUp") { | ||||
| 			event.preventDefault(); | ||||
| 			if (focusedIndex > 0) { | ||||
| 				focusedIndex -= 1; | ||||
| 				scrollToFocusedItem(); | ||||
| 			} else if (focusedIndex === -1 && filteredOptions.length > 0) { | ||||
| 				focusedIndex = 0; | ||||
| 				scrollToFocusedItem(); | ||||
| 			} | ||||
| 		} else if (event.key === "Enter" && index >= 0) { | ||||
| 			event.preventDefault(); | ||||
| 			selectOption(filteredOptions[index]); | ||||
| 		} else if (event.key === "Escape") { | ||||
| 			event.preventDefault(); | ||||
| 			isOpen = false; | ||||
| 			focusedIndex = -1; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Scroll to the focused item | ||||
| 	function scrollToFocusedItem() { | ||||
| 		if (!container || focusedIndex < 0) return; | ||||
|  | ||||
| 		const itemTop = focusedIndex * itemHeight; | ||||
| 		const itemBottom = itemTop + itemHeight; | ||||
| 		const containerTop = container.scrollTop; | ||||
| 		const containerBottom = containerTop + container.clientHeight; | ||||
|  | ||||
| 		if (itemTop < containerTop) { | ||||
| 			container.scrollTop = itemTop; | ||||
| 		} else if (itemBottom > containerBottom) { | ||||
| 			container.scrollTop = itemBottom - container.clientHeight; | ||||
| 		} | ||||
| 		updateVisibleItems(); | ||||
| 	} | ||||
|  | ||||
| 	// Initialize container size observer and autofocus fallback | ||||
| 	onMount(async () => { | ||||
| 		if (container) { | ||||
| 			const resizeObserver = new ResizeObserver(updateVisibleCount); | ||||
| 			resizeObserver.observe(container); | ||||
| 			return () => resizeObserver.disconnect(); | ||||
| 		} | ||||
| 		// Fallback autofocus with tick to ensure inputElement is bound | ||||
| 		if (autofocus && inputElement) { | ||||
| 			await tick(); | ||||
| 			inputElement.focus(); | ||||
| 		} | ||||
| 	}); | ||||
|  | ||||
| 	// Get display text for the input | ||||
| 	function getDisplayText() { | ||||
| 		if (!selected) return inputPlaceholder; | ||||
| 		const selectedOption = options.find((option) => option.value === selected); | ||||
| 		return selectedOption ? selectedOption.label : inputPlaceholder; | ||||
| 	} | ||||
| </script> | ||||
|  | ||||
| <svelte:window on:click={handleClickOutside} /> | ||||
|  | ||||
| <div class="select-container relative w-full"> | ||||
| 	<!-- Select element with inline search --> | ||||
| 	<div | ||||
| 		class="border rounded-md px-3 py-2 bg-white shadow-sm flex items-center gap-2" | ||||
| 		role="combobox" | ||||
| 		aria-expanded={isOpen} | ||||
| 	> | ||||
| 		<input | ||||
| 			autocomplete="off" | ||||
| 			type="text" | ||||
| 			id={inputElementID} | ||||
| 			bind:value={searchTerm} | ||||
| 			bind:this={inputElement} | ||||
| 			placeholder={getDisplayText()} | ||||
| 			class="w-full bg-transparent focus:outline-none {selected | ||||
| 				? 'text-black' | ||||
| 				: 'text-gray-700'}" | ||||
| 			{autofocus} | ||||
| 			on:focus={handleInputFocus} | ||||
| 			on:input={handleInput} | ||||
| 			on:keydown={(e) => { | ||||
| 				if (e.key === "Enter" && !isOpen) { | ||||
| 					toggleDropdown(); | ||||
| 				} else { | ||||
| 					handleKeydown(e, focusedIndex); | ||||
| 				} | ||||
| 			}} | ||||
| 			aria-label={inputAriaLabel} | ||||
| 		/> | ||||
| 		{#if selected} | ||||
| 			<button | ||||
| 				type="button" | ||||
| 				class="w-5 h-5 flex items-center justify-center text-gray-500 hover:text-gray-700" | ||||
| 				on:click={clearSelection} | ||||
| 				on:keydown={(e) => { | ||||
| 					if (e.key === "Enter" || e.key === " ") { | ||||
| 						e.preventDefault(); | ||||
| 						clearSelection(); | ||||
| 					} else if (e.key === "Escape") { | ||||
| 						e.preventDefault(); | ||||
| 						isOpen = false; | ||||
| 						focusedIndex = -1; | ||||
| 					} | ||||
| 				}} | ||||
| 				role="button" | ||||
| 				tabindex="0" | ||||
| 				aria-label={clearAriaLabel} | ||||
| 			> | ||||
| 				<svg | ||||
| 					class="w-4 h-4" | ||||
| 					fill="none" | ||||
| 					stroke="currentColor" | ||||
| 					viewBox="0 0 24 24" | ||||
| 				> | ||||
| 					<path | ||||
| 						stroke-linecap="round" | ||||
| 						stroke-linejoin="round" | ||||
| 						stroke-width="2" | ||||
| 						d="M6 18L18 6M6 6l12 12" | ||||
| 					/> | ||||
| 				</svg> | ||||
| 			</button> | ||||
| 		{/if} | ||||
| 		<svg | ||||
| 			class="w-4 h-4 text-gray-500 transform {isOpen ? 'rotate-180' : ''}" | ||||
| 			fill="none" | ||||
| 			stroke="currentColor" | ||||
| 			viewBox="0 0 24 24" | ||||
| 			on:click={toggleDropdown} | ||||
| 			role="button" | ||||
| 			tabindex="0" | ||||
| 			on:keydown={(e) => { | ||||
| 				if (e.key === "Enter") toggleDropdown(); | ||||
| 				else if (e.key === "Escape") { | ||||
| 					e.preventDefault(); | ||||
| 					isOpen = false; | ||||
| 					focusedIndex = -1; | ||||
| 				} | ||||
| 			}} | ||||
| 			aria-label={toggleAriaLabel} | ||||
| 		> | ||||
| 			<path | ||||
| 				stroke-linecap="round" | ||||
| 				stroke-linejoin="round" | ||||
| 				stroke | ||||
| 				Politeness="round" | ||||
| 				stroke-width="2" | ||||
| 				d="M19 9l-7 7-7-7" | ||||
| 			/> | ||||
| 		</svg> | ||||
| 	</div> | ||||
|  | ||||
| 	<!-- Dropdown --> | ||||
| 	{#if isOpen} | ||||
| 		<div | ||||
| 			class="absolute z-10 w-full mt-1 bg-white border rounded-md shadow-lg max-h-80 overflow-auto" | ||||
| 			bind:this={container} | ||||
| 			on:scroll={handleScroll} | ||||
| 			role="listbox" | ||||
| 		> | ||||
| 			{#if filteredOptions.length > 0} | ||||
| 				<!-- Virtualized list container --> | ||||
| 				<div style="height: {filteredOptions.length * itemHeight}px;"> | ||||
| 					<div style="transform: translateY({startIndex * itemHeight}px);"> | ||||
| 						{#each visibleItems as item, i (item.label + "-" + (startIndex + i))} | ||||
| 							<div | ||||
| 								class="px-3 py-2 hover:bg-blue-100 cursor-pointer {selected === | ||||
| 								item.value | ||||
| 									? 'bg-blue-50' | ||||
| 									: ''} {focusedIndex === startIndex + i | ||||
| 									? 'bg-blue-200 outline outline-2 outline-blue-500' | ||||
| 									: ''}" | ||||
| 								on:click={() => selectOption(item)} | ||||
| 								on:keydown={(e) => handleKeydown(e, startIndex + i)} | ||||
| 								role="option" | ||||
| 								tabindex="0" | ||||
| 								aria-selected={selected === item.value} | ||||
| 							> | ||||
| 								{item.label} | ||||
| 							</div> | ||||
| 						{/each} | ||||
| 					</div> | ||||
| 				</div> | ||||
| 			{:else} | ||||
| 				<div class="px-3 py-2 text-gray-500">{noOptionsText}</div> | ||||
| 			{/if} | ||||
| 		</div> | ||||
| 	{/if} | ||||
| </div> | ||||
|  | ||||
| <style> | ||||
| 	/* Ensure Tailwind classes handle additional styling */ | ||||
| 	:global(.select-container input:focus) { | ||||
| 		border-color: #3b82f6; /* Tailwind's blue-500 */ | ||||
| 	} | ||||
| 	:global([role="option"]:focus) { | ||||
| 		outline: 2px solid #3b82f6; | ||||
| 		outline-offset: -2px; | ||||
| 	} | ||||
| 	:global([role="button"]:focus) { | ||||
| 		outline: 2px solid #3b82f6; | ||||
| 		outline-offset: -2px; | ||||
| 	} | ||||
| </style> | ||||
| @@ -227,7 +227,7 @@ | ||||
|     "enabled_large": "Aktiviert", | ||||
|     "english": "Englisch", | ||||
|     "enter-payment": "Zahlung eingeben", | ||||
|     "error-creating-donation": "Fehler bei der Anlage", | ||||
|     "error-creating-donation": "Fehler beim Erstellen des Sponsorings", | ||||
|     "error-during-import": "Fehler beim Importieren", | ||||
|     "error-whyile-copying-to-clipboard": "Fehler beim Kopieren in die Zwischenablage", | ||||
|     "error_on_login": "😢Fehler beim Login", | ||||
|   | ||||
| @@ -227,6 +227,7 @@ | ||||
|     "enabled_large": "Enabled", | ||||
|     "english": "English", | ||||
|     "enter-payment": "Enter payment", | ||||
|     "error-creating-donation": "error creating the sponsoring", | ||||
|     "error-during-import": "Error during import", | ||||
|     "error-whyile-copying-to-clipboard": "Error while copying to clipboard", | ||||
|     "error_on_login": "Error on login", | ||||
|   | ||||
		Reference in New Issue
	
	Block a user