Skip to content

Commit a77fef4

Browse files
committed
add phone input component
1 parent 6ca40c2 commit a77fef4

File tree

7 files changed

+201
-0
lines changed

7 files changed

+201
-0
lines changed

packages/boxel-ui/addon/package.json

+2
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
"@floating-ui/dom": "^1.6.3",
4242
"@glint/template": "1.3.0",
4343
"classnames": "^2.3.2",
44+
"countries-list": "^3.1.1",
4445
"dayjs": "^1.11.7",
4546
"ember-basic-dropdown": "^8.0.0",
4647
"ember-css-url": "^1.0.0",
@@ -59,6 +60,7 @@
5960
"ember-velcro": "^2.1.3",
6061
"file-loader": "^6.2.0",
6162
"focus-trap": "^7.4.3",
63+
"libphonenumber-js": "^1.11.15",
6264
"lodash": "^4.17.21",
6365
"pluralize": "^8.0.0",
6466
"tracked-built-ins": "^3.2.0",

packages/boxel-ui/addon/src/components.ts

+2
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ import BoxelMultiSelect, {
4141
import Pill from './components/pill/index.gts';
4242
import ProgressBar from './components/progress-bar/index.gts';
4343
import ProgressRadial from './components/progress-radial/index.gts';
44+
import PhoneInput from './components/phone-input/index.gts';
4445
import RadioInput from './components/radio-input/index.gts';
4546
import RealmIcon from './components/realm-icon/index.gts';
4647
import ResizablePanelGroup, {
@@ -95,6 +96,7 @@ export {
9596
Pill,
9697
ProgressBar,
9798
ProgressRadial,
99+
PhoneInput,
98100
RadioInput,
99101
RealmIcon,
100102
ResizablePanel,

packages/boxel-ui/addon/src/components/input-group/accessories/index.gts

+1
Original file line numberDiff line numberDiff line change
@@ -152,6 +152,7 @@ export const Select: TemplateOnlyComponent<SelectAccessorySignature> =
152152
@onChange={{@onChange}}
153153
@onBlur={{@onBlur}}
154154
@matchTriggerWidth={{@matchTriggerWidth}}
155+
@selectedItemComponent={{@selectedItemComponent}}
155156
data-test-boxel-input-group-select-accessory-trigger
156157
...attributes
157158
as |item|
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
import type { TemplateOnlyComponent } from '@ember/component/template-only';
2+
import { action } from '@ember/object';
3+
import Component from '@glimmer/component';
4+
import { tracked } from '@glimmer/tracking';
5+
import { type TCountryCode, countries, getEmojiFlag } from 'countries-list';
6+
import {
7+
type CountryCallingCode,
8+
type CountryCode,
9+
getCountries,
10+
getCountryCallingCode,
11+
getExampleNumber,
12+
isValidPhoneNumber,
13+
} from 'libphonenumber-js';
14+
// @ts-expect-error import not found
15+
import examples from 'libphonenumber-js/mobile/examples';
16+
import { debounce } from 'lodash';
17+
18+
import { type InputValidationState } from '../input/index.gts';
19+
import BoxelInputGroup from '../input-group/index.gts';
20+
21+
interface Signature {
22+
Args: {
23+
value: string;
24+
onInput: (value: string) => void;
25+
};
26+
Blocks: {
27+
default: [];
28+
};
29+
Element: HTMLElement;
30+
}
31+
32+
interface CountryInfo {
33+
callingCode?: CountryCallingCode;
34+
code: CountryCode;
35+
name?: string;
36+
flag?: string;
37+
example?: {
38+
callingCode: CountryCallingCode;
39+
nationalNumber: string;
40+
};
41+
}
42+
43+
const getCountryInfo = (countryCode: CountryCode): CountryInfo | undefined => {
44+
let example = getExampleNumber(countryCode, examples);
45+
let callingCode = getCountryCallingCode(countryCode);
46+
let c = countries[countryCode as TCountryCode];
47+
if (c === undefined) {
48+
return undefined;
49+
}
50+
return {
51+
code: countryCode,
52+
callingCode,
53+
name: c ? c.name : undefined,
54+
flag: getEmojiFlag(countryCode as TCountryCode),
55+
example: example
56+
? {
57+
callingCode,
58+
nationalNumber: example.format('NATIONAL'),
59+
}
60+
: undefined,
61+
};
62+
};
63+
64+
export default class PhoneInput extends Component<Signature> {
65+
@tracked items: Array<CountryInfo> = [];
66+
@tracked selectedItem: CountryInfo = getCountryInfo('US')!;
67+
@tracked validationState: InputValidationState = 'initial';
68+
@tracked input: string = this.args.value ?? '';
69+
70+
@action onSelectItem(item: CountryInfo): void {
71+
this.selectedItem = item;
72+
if (this.input.length > 0) {
73+
this.validationState = isValidPhoneNumber(
74+
this.input,
75+
this.selectedItem.code,
76+
)
77+
? 'valid'
78+
: 'invalid';
79+
}
80+
}
81+
82+
constructor(owner: unknown, args: any) {
83+
super(owner, args);
84+
this.items = getCountries()
85+
.map((code) => {
86+
return getCountryInfo(code);
87+
})
88+
.filter((c) => c !== undefined) as CountryInfo[];
89+
}
90+
91+
get placeholder(): string | undefined {
92+
if (this.selectedItem) {
93+
return this.selectedItem.example?.nationalNumber;
94+
}
95+
return undefined;
96+
}
97+
98+
get phoneNumber(): string {
99+
return `+${this.selectedItem.callingCode} `;
100+
}
101+
102+
@action onInput(v: string): void {
103+
this.debouncedInput(v);
104+
}
105+
106+
private debouncedInput = debounce((input: string) => {
107+
this.validationState = isValidPhoneNumber(input, this.selectedItem.code)
108+
? 'valid'
109+
: 'invalid';
110+
this.input = input;
111+
//save when the state is valid
112+
if (this.validationState === 'valid') {
113+
this.args.onInput(this.input);
114+
}
115+
}, 300);
116+
117+
<template>
118+
<BoxelInputGroup
119+
@placeholder={{this.placeholder}}
120+
@state={{this.validationState}}
121+
@onInput={{this.onInput}}
122+
@value={{this.input}}
123+
>
124+
<:before as |Accessories|>
125+
<Accessories.Select
126+
@placeholder={{this.placeholder}}
127+
@selected={{this.selectedItem}}
128+
@onChange={{this.onSelectItem}}
129+
@options={{this.items}}
130+
@selectedItemComponent={{PhoneSelectedItem}}
131+
@searchEnabled={{true}}
132+
@searchField='name'
133+
@matchTriggerWidth={{false}}
134+
aria-label='Select an country calling code'
135+
as |item|
136+
>
137+
<div>{{item.flag}} {{item.name}} +{{item.callingCode}}</div>
138+
</Accessories.Select>
139+
</:before>
140+
</BoxelInputGroup>
141+
</template>
142+
}
143+
144+
export interface SelectedItemSignature {
145+
Args: {
146+
option: any;
147+
};
148+
Element: HTMLElement;
149+
}
150+
151+
const PhoneSelectedItem: TemplateOnlyComponent<SelectedItemSignature> =
152+
<template><div>{{@option.flag}} +{{@option.callingCode}}</div></template>;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import Component from '@glimmer/component';
2+
import FreestyleUsage from 'ember-freestyle/components/freestyle/usage';
3+
import { tracked } from '@glimmer/tracking';
4+
import { action } from '@ember/object';
5+
import PhoneInput from './index.gts';
6+
7+
export default class PhoneInputUsage extends Component {
8+
@tracked value = '';
9+
10+
@action onInput(value: string): void {
11+
this.value = value;
12+
}
13+
14+
<template>
15+
<FreestyleUsage @name='PhoneInput'>
16+
<:description>
17+
<p>
18+
PhoneInput is a component that allows users to input phone numbers
19+
with a dropdown select of country code and validation of the inputted
20+
numbers
21+
</p>
22+
</:description>
23+
<:example>
24+
<PhoneInput @value={{this.value}} @onInput={{this.onInput}} />
25+
</:example>
26+
</FreestyleUsage>
27+
</template>
28+
}

packages/boxel-ui/addon/src/usage.ts

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import CardContainerUsage from './components/card-container/usage.gts';
1010
import CardContentContainerUsage from './components/card-content-container/usage.gts';
1111
import CardHeaderUsage from './components/card-header/usage.gts';
1212
import CircleSpinnerUsage from './components/circle-spinner/usage.gts';
13+
import PhoneInputUsage from './components/phone-input/usage.gts';
1314
import DateRangePickerUsage from './components/date-range-picker/usage.gts';
1415
import DragAndDropUsage from './components/drag-and-drop/usage.gts';
1516
import DropdownTriggerUsage from './components/dropdown/trigger/usage.gts';
@@ -48,6 +49,7 @@ export const ALL_USAGE_COMPONENTS = [
4849
['CardContentContainer', CardContentContainerUsage],
4950
['CardHeader', CardHeaderUsage],
5051
['CircleSpinner', CircleSpinnerUsage],
52+
['PhoneInput', PhoneInputUsage],
5153
['DateRangePicker', DateRangePickerUsage],
5254
['DragAndDrop', DragAndDropUsage],
5355
['DropdownTrigger', DropdownTriggerUsage],

pnpm-lock.yaml

+14
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)