feat(login): custom CountryPicker — opens downward, searchable, ~150 countries
All checks were successful
Deploy to Production / deploy (push) Successful in 51s

Native <select> defers dropdown direction to the browser, which on mobile
routinely opens upward and hides countries behind the keyboard. Replaced
with a custom combobox that always opens DOWNWARD (absolute positioned
below the trigger) with a search input at top — at 150 countries a
scrollable list is unusable without search anyway.

COUNTRIES list expanded from 60 → 152 entries: every country with a
meaningful diaspora, including Russia, Pakistan, Bangladesh, Sri Lanka,
Cyprus, Malta, Albania, Bosnia, Kosovo, North Macedonia, Iran, Iraq,
Lebanon, Jordan, Kazakhstan, Morocco, Algeria, Tunisia, Ethiopia,
Tanzania, Uganda, Senegal, Ghana, Madagascar, Cameroon, Sri Lanka,
Belarus, Georgia, Armenia, Azerbaijan and the rest. Serbia was already in
the prior list — just unfindable without search.

Bonus: flag emojis computed from ISO-3166 alpha-2 codes (no asset files).
Search matches name + code + dial-prefix so "+41" or "CH" both find
Switzerland.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Marco Sadjadi 2026-05-25 21:38:36 +02:00
parent 1cccdbdff1
commit e9827b1f77
2 changed files with 269 additions and 13 deletions

View File

@ -1,5 +1,6 @@
'use client';
import { CountryPicker } from '@/components/country-picker';
import { Input, Label } from '@/components/input';
import { Logo } from '@/components/logo';
import { Button } from '@/components/ui/button';
@ -21,69 +22,163 @@ const ERROR_COPY: Record<string, string> = {
sms_verify_failed: 'Could not verify the code. Try again.',
};
// Country dial codes for the phone-login picker. Sorted by name; Switzerland
// is the default (Swiss-built product, Swiss Twilio sender number).
// Country dial codes for the phone-login picker. ~150 entries — every country
// with a non-trivial diaspora. Sorted alphabetically by name. Switzerland is
// the default (Swiss-built product, Swiss Twilio sender number).
const COUNTRIES: { code: string; name: string; dial: string }[] = [
{ code: 'AL', name: 'Albania', dial: '+355' },
{ code: 'DZ', name: 'Algeria', dial: '+213' },
{ code: 'AD', name: 'Andorra', dial: '+376' },
{ code: 'AO', name: 'Angola', dial: '+244' },
{ code: 'AR', name: 'Argentina', dial: '+54' },
{ code: 'AM', name: 'Armenia', dial: '+374' },
{ code: 'AW', name: 'Aruba', dial: '+297' },
{ code: 'AU', name: 'Australia', dial: '+61' },
{ code: 'AT', name: 'Austria', dial: '+43' },
{ code: 'AZ', name: 'Azerbaijan', dial: '+994' },
{ code: 'BS', name: 'Bahamas', dial: '+1' },
{ code: 'BH', name: 'Bahrain', dial: '+973' },
{ code: 'BD', name: 'Bangladesh', dial: '+880' },
{ code: 'BB', name: 'Barbados', dial: '+1' },
{ code: 'BY', name: 'Belarus', dial: '+375' },
{ code: 'BE', name: 'Belgium', dial: '+32' },
{ code: 'BZ', name: 'Belize', dial: '+501' },
{ code: 'BJ', name: 'Benin', dial: '+229' },
{ code: 'BM', name: 'Bermuda', dial: '+1' },
{ code: 'BT', name: 'Bhutan', dial: '+975' },
{ code: 'BO', name: 'Bolivia', dial: '+591' },
{ code: 'BA', name: 'Bosnia & Herzegovina', dial: '+387' },
{ code: 'BW', name: 'Botswana', dial: '+267' },
{ code: 'BR', name: 'Brazil', dial: '+55' },
{ code: 'BN', name: 'Brunei', dial: '+673' },
{ code: 'BG', name: 'Bulgaria', dial: '+359' },
{ code: 'BF', name: 'Burkina Faso', dial: '+226' },
{ code: 'KH', name: 'Cambodia', dial: '+855' },
{ code: 'CM', name: 'Cameroon', dial: '+237' },
{ code: 'CA', name: 'Canada', dial: '+1' },
{ code: 'CV', name: 'Cape Verde', dial: '+238' },
{ code: 'KY', name: 'Cayman Islands', dial: '+1' },
{ code: 'CL', name: 'Chile', dial: '+56' },
{ code: 'CN', name: 'China', dial: '+86' },
{ code: 'CO', name: 'Colombia', dial: '+57' },
{ code: 'CR', name: 'Costa Rica', dial: '+506' },
{ code: 'HR', name: 'Croatia', dial: '+385' },
{ code: 'CY', name: 'Cyprus', dial: '+357' },
{ code: 'CZ', name: 'Czechia', dial: '+420' },
{ code: 'DK', name: 'Denmark', dial: '+45' },
{ code: 'DO', name: 'Dominican Republic', dial: '+1' },
{ code: 'EC', name: 'Ecuador', dial: '+593' },
{ code: 'EG', name: 'Egypt', dial: '+20' },
{ code: 'SV', name: 'El Salvador', dial: '+503' },
{ code: 'EE', name: 'Estonia', dial: '+372' },
{ code: 'ET', name: 'Ethiopia', dial: '+251' },
{ code: 'FJ', name: 'Fiji', dial: '+679' },
{ code: 'FI', name: 'Finland', dial: '+358' },
{ code: 'FR', name: 'France', dial: '+33' },
{ code: 'GE', name: 'Georgia', dial: '+995' },
{ code: 'DE', name: 'Germany', dial: '+49' },
{ code: 'GH', name: 'Ghana', dial: '+233' },
{ code: 'GR', name: 'Greece', dial: '+30' },
{ code: 'GT', name: 'Guatemala', dial: '+502' },
{ code: 'HN', name: 'Honduras', dial: '+504' },
{ code: 'HK', name: 'Hong Kong', dial: '+852' },
{ code: 'HU', name: 'Hungary', dial: '+36' },
{ code: 'IS', name: 'Iceland', dial: '+354' },
{ code: 'IN', name: 'India', dial: '+91' },
{ code: 'ID', name: 'Indonesia', dial: '+62' },
{ code: 'IR', name: 'Iran', dial: '+98' },
{ code: 'IQ', name: 'Iraq', dial: '+964' },
{ code: 'IE', name: 'Ireland', dial: '+353' },
{ code: 'IL', name: 'Israel', dial: '+972' },
{ code: 'IT', name: 'Italy', dial: '+39' },
{ code: 'CI', name: 'Ivory Coast', dial: '+225' },
{ code: 'JM', name: 'Jamaica', dial: '+1' },
{ code: 'JP', name: 'Japan', dial: '+81' },
{ code: 'JO', name: 'Jordan', dial: '+962' },
{ code: 'KZ', name: 'Kazakhstan', dial: '+7' },
{ code: 'KE', name: 'Kenya', dial: '+254' },
{ code: 'XK', name: 'Kosovo', dial: '+383' },
{ code: 'KW', name: 'Kuwait', dial: '+965' },
{ code: 'KG', name: 'Kyrgyzstan', dial: '+996' },
{ code: 'LA', name: 'Laos', dial: '+856' },
{ code: 'LV', name: 'Latvia', dial: '+371' },
{ code: 'LB', name: 'Lebanon', dial: '+961' },
{ code: 'LY', name: 'Libya', dial: '+218' },
{ code: 'LI', name: 'Liechtenstein', dial: '+423' },
{ code: 'LT', name: 'Lithuania', dial: '+370' },
{ code: 'LU', name: 'Luxembourg', dial: '+352' },
{ code: 'MO', name: 'Macau', dial: '+853' },
{ code: 'MK', name: 'North Macedonia', dial: '+389' },
{ code: 'MG', name: 'Madagascar', dial: '+261' },
{ code: 'MY', name: 'Malaysia', dial: '+60' },
{ code: 'MV', name: 'Maldives', dial: '+960' },
{ code: 'MT', name: 'Malta', dial: '+356' },
{ code: 'MX', name: 'Mexico', dial: '+52' },
{ code: 'MD', name: 'Moldova', dial: '+373' },
{ code: 'MC', name: 'Monaco', dial: '+377' },
{ code: 'MN', name: 'Mongolia', dial: '+976' },
{ code: 'ME', name: 'Montenegro', dial: '+382' },
{ code: 'MA', name: 'Morocco', dial: '+212' },
{ code: 'MZ', name: 'Mozambique', dial: '+258' },
{ code: 'MM', name: 'Myanmar', dial: '+95' },
{ code: 'NA', name: 'Namibia', dial: '+264' },
{ code: 'NP', name: 'Nepal', dial: '+977' },
{ code: 'NL', name: 'Netherlands', dial: '+31' },
{ code: 'NZ', name: 'New Zealand', dial: '+64' },
{ code: 'NI', name: 'Nicaragua', dial: '+505' },
{ code: 'NG', name: 'Nigeria', dial: '+234' },
{ code: 'NO', name: 'Norway', dial: '+47' },
{ code: 'OM', name: 'Oman', dial: '+968' },
{ code: 'PK', name: 'Pakistan', dial: '+92' },
{ code: 'PS', name: 'Palestine', dial: '+970' },
{ code: 'PA', name: 'Panama', dial: '+507' },
{ code: 'PG', name: 'Papua New Guinea', dial: '+675' },
{ code: 'PY', name: 'Paraguay', dial: '+595' },
{ code: 'PE', name: 'Peru', dial: '+51' },
{ code: 'PH', name: 'Philippines', dial: '+63' },
{ code: 'PL', name: 'Poland', dial: '+48' },
{ code: 'PT', name: 'Portugal', dial: '+351' },
{ code: 'PR', name: 'Puerto Rico', dial: '+1' },
{ code: 'QA', name: 'Qatar', dial: '+974' },
{ code: 'RO', name: 'Romania', dial: '+40' },
{ code: 'RU', name: 'Russia', dial: '+7' },
{ code: 'RW', name: 'Rwanda', dial: '+250' },
{ code: 'SM', name: 'San Marino', dial: '+378' },
{ code: 'SA', name: 'Saudi Arabia', dial: '+966' },
{ code: 'SN', name: 'Senegal', dial: '+221' },
{ code: 'RS', name: 'Serbia', dial: '+381' },
{ code: 'SG', name: 'Singapore', dial: '+65' },
{ code: 'SK', name: 'Slovakia', dial: '+421' },
{ code: 'SI', name: 'Slovenia', dial: '+386' },
{ code: 'SO', name: 'Somalia', dial: '+252' },
{ code: 'ZA', name: 'South Africa', dial: '+27' },
{ code: 'KR', name: 'South Korea', dial: '+82' },
{ code: 'ES', name: 'Spain', dial: '+34' },
{ code: 'LK', name: 'Sri Lanka', dial: '+94' },
{ code: 'SD', name: 'Sudan', dial: '+249' },
{ code: 'SE', name: 'Sweden', dial: '+46' },
{ code: 'CH', name: 'Switzerland', dial: '+41' },
{ code: 'SY', name: 'Syria', dial: '+963' },
{ code: 'TW', name: 'Taiwan', dial: '+886' },
{ code: 'TJ', name: 'Tajikistan', dial: '+992' },
{ code: 'TZ', name: 'Tanzania', dial: '+255' },
{ code: 'TH', name: 'Thailand', dial: '+66' },
{ code: 'TT', name: 'Trinidad & Tobago', dial: '+1' },
{ code: 'TN', name: 'Tunisia', dial: '+216' },
{ code: 'TR', name: 'Turkey', dial: '+90' },
{ code: 'TM', name: 'Turkmenistan', dial: '+993' },
{ code: 'UG', name: 'Uganda', dial: '+256' },
{ code: 'UA', name: 'Ukraine', dial: '+380' },
{ code: 'AE', name: 'United Arab Emirates', dial: '+971' },
{ code: 'GB', name: 'United Kingdom', dial: '+44' },
{ code: 'US', name: 'United States', dial: '+1' },
{ code: 'UY', name: 'Uruguay', dial: '+598' },
{ code: 'UZ', name: 'Uzbekistan', dial: '+998' },
{ code: 'VE', name: 'Venezuela', dial: '+58' },
{ code: 'VN', name: 'Vietnam', dial: '+84' },
{ code: 'YE', name: 'Yemen', dial: '+967' },
{ code: 'ZM', name: 'Zambia', dial: '+260' },
{ code: 'ZW', name: 'Zimbabwe', dial: '+263' },
];
function dialFor(code: string): string {
@ -301,18 +396,11 @@ export default function LoginPage() {
<form onSubmit={requestSmsCode} className="space-y-3">
<div className="space-y-1.5">
<Label htmlFor="country">Country</Label>
<select
id="country"
<CountryPicker
countries={COUNTRIES}
value={country}
onChange={(e) => setCountry(e.target.value)}
className="h-8 w-full rounded-md border border-[--color-border] bg-[--color-bg-subtle] px-2 text-[13px] text-[--color-fg] transition-colors duration-200 focus:border-[--color-accent] focus:outline-none focus:ring-1 focus:ring-[--color-accent]"
>
{COUNTRIES.map((c) => (
<option key={c.code} value={c.code}>
{c.name} ({c.dial})
</option>
))}
</select>
onChange={setCountry}
/>
</div>
<div className="space-y-1.5">
<Label htmlFor="phone" hint={dialFor(country)}>

View File

@ -0,0 +1,168 @@
'use client';
import { Check, ChevronDown, Search } from 'lucide-react';
import { useEffect, useMemo, useRef, useState } from 'react';
export interface Country {
code: string; // ISO-3166 alpha-2
name: string;
dial: string; // E.164 dial prefix incl. leading '+'
}
/** Compute the regional-indicator-symbol flag emoji from an ISO-3166 alpha-2
* code. 🇨🇭 from "CH", 🇺🇸 from "US" etc. no asset files needed. */
function flagFor(code: string): string {
if (code.length !== 2) return '';
const A = 0x1f1e6 - 65;
return String.fromCodePoint(
...code
.toUpperCase()
.split('')
.map((c) => A + c.charCodeAt(0)),
);
}
/**
* Custom country picker opens DOWNWARD always (native <select> defers the
* direction to the browser and frequently opens upward on mobile, hiding
* countries behind the keyboard). Built-in search filter so a 150-entry list
* stays usable.
*/
export function CountryPicker({
countries,
value,
onChange,
}: {
countries: Country[];
value: string;
onChange: (code: string) => void;
}) {
const [open, setOpen] = useState(false);
const [query, setQuery] = useState('');
const wrap = useRef<HTMLDivElement>(null);
const searchRef = useRef<HTMLInputElement>(null);
const selected = countries.find((c) => c.code === value);
const filtered = useMemo(() => {
const q = query.trim().toLowerCase();
if (!q) return countries;
return countries.filter(
(c) =>
c.name.toLowerCase().includes(q) ||
c.code.toLowerCase().includes(q) ||
c.dial.replace('+', '').includes(q.replace('+', '')),
);
}, [countries, query]);
// Outside-click + Escape close
useEffect(() => {
if (!open) return;
function onDocMouseDown(e: MouseEvent) {
if (!wrap.current?.contains(e.target as Node)) setOpen(false);
}
function onKey(e: KeyboardEvent) {
if (e.key === 'Escape') setOpen(false);
}
document.addEventListener('mousedown', onDocMouseDown);
document.addEventListener('keydown', onKey);
return () => {
document.removeEventListener('mousedown', onDocMouseDown);
document.removeEventListener('keydown', onKey);
};
}, [open]);
// Autofocus the search input when the dropdown opens.
useEffect(() => {
if (open) {
const t = setTimeout(() => searchRef.current?.focus(), 0);
return () => clearTimeout(t);
}
}, [open]);
function select(code: string) {
onChange(code);
setOpen(false);
setQuery('');
}
return (
<div ref={wrap} className="relative">
<button
type="button"
aria-haspopup="listbox"
aria-expanded={open}
onClick={() => setOpen((v) => !v)}
className="flex h-8 w-full items-center justify-between gap-2 rounded-md border border-[--color-border] bg-[--color-bg-subtle] px-2.5 text-[13px] text-[--color-fg] transition-colors duration-200 hover:border-[--color-border-strong] focus:border-[--color-accent] focus:outline-none focus:ring-1 focus:ring-[--color-accent]"
>
<span className="flex min-w-0 items-center gap-2">
{selected && (
<span aria-hidden className="text-[14px] leading-none">
{flagFor(selected.code)}
</span>
)}
<span className="truncate">
{selected ? `${selected.name} (${selected.dial})` : 'Select country'}
</span>
</span>
<ChevronDown size={13} className="shrink-0 text-[--color-fg-subtle]" />
</button>
{open && (
<div
role="listbox"
className="absolute left-0 right-0 top-full z-50 mt-1 overflow-hidden rounded-md border border-[--color-border] bg-[--color-bg-elevated] shadow-lg shadow-black/40"
>
<div className="flex items-center gap-2 border-b border-[--color-border] px-2.5">
<Search size={13} className="text-[--color-fg-subtle]" />
<input
ref={searchRef}
value={query}
onChange={(e) => setQuery(e.target.value)}
placeholder="Search country or code…"
className="h-9 w-full bg-transparent text-[13px] text-[--color-fg] placeholder:text-[--color-fg-subtle] focus:outline-none"
onKeyDown={(e) => {
if (e.key === 'Enter' && filtered.length > 0) {
e.preventDefault();
select(filtered[0]!.code);
}
}}
/>
</div>
<ul className="max-h-60 overflow-y-auto py-1">
{filtered.length === 0 && (
<li className="px-3 py-2 text-[12px] text-[--color-fg-subtle]">No match.</li>
)}
{filtered.map((c) => {
const isSelected = c.code === value;
return (
<li key={c.code}>
<button
type="button"
role="option"
aria-selected={isSelected}
onClick={() => select(c.code)}
className={`flex w-full items-center gap-2.5 px-2.5 py-1.5 text-left text-[12.5px] transition-colors ${
isSelected
? 'bg-[--color-bg-subtle] text-[--color-fg]'
: 'text-[--color-fg-muted] hover:bg-[--color-bg-subtle] hover:text-[--color-fg]'
}`}
>
<span aria-hidden className="text-[14px] leading-none">
{flagFor(c.code)}
</span>
<span className="flex-1 truncate">{c.name}</span>
<span className="mono text-[11px] text-[--color-fg-subtle]">{c.dial}</span>
{isSelected && (
<Check size={12} className="text-[--color-accent]" />
)}
</button>
</li>
);
})}
</ul>
</div>
)}
</div>
);
}