Skip to content

Commit b0bbab5

Browse files
authored
Merge pull request #1860 from cardstack/add-phone-input-component
Add phone input component
2 parents 7131849 + 1fce6bc commit b0bbab5

File tree

14 files changed

+453
-147
lines changed

14 files changed

+453
-147
lines changed

packages/boxel-ui/addon/package.json

+2
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,9 @@
4040
"@embroider/addon-shim": "^1.8.9",
4141
"@floating-ui/dom": "^1.6.3",
4242
"@glint/template": "1.3.0",
43+
"awesome-phonenumber": "^7.2.0",
4344
"classnames": "^2.3.2",
45+
"countries-list": "^3.1.1",
4446
"dayjs": "^1.11.7",
4547
"ember-basic-dropdown": "^8.0.0",
4648
"ember-css-url": "^1.0.0",

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

+2
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ import Modal from './components/modal/index.gts';
3838
import BoxelMultiSelect, {
3939
BoxelMultiSelectBasic,
4040
} from './components/multi-select/index.gts';
41+
import PhoneInput from './components/phone-input/index.gts';
4142
import Pill from './components/pill/index.gts';
4243
import ProgressBar from './components/progress-bar/index.gts';
4344
import ProgressRadial from './components/progress-radial/index.gts';
@@ -92,6 +93,7 @@ export {
9293
Menu,
9394
Message,
9495
Modal,
96+
PhoneInput,
9597
Pill,
9698
ProgressBar,
9799
ProgressRadial,

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,171 @@
1+
import { action } from '@ember/object';
2+
import Component from '@glimmer/component';
3+
import { tracked } from '@glimmer/tracking';
4+
import {
5+
getCountryCodeForRegionCode,
6+
getExample,
7+
getSupportedRegionCodes,
8+
parsePhoneNumber,
9+
} from 'awesome-phonenumber';
10+
import { type TCountryCode, countries, getEmojiFlag } from 'countries-list';
11+
import { debounce } from 'lodash';
12+
13+
import { type InputValidationState } from '../input/index.gts';
14+
import BoxelInputGroup from '../input-group/index.gts';
15+
16+
interface Signature {
17+
Args: {
18+
countryCode: string;
19+
onCountryCodeChange: (code: string) => void;
20+
onInput: (value: string) => void;
21+
value: string;
22+
};
23+
Blocks: {
24+
default: [];
25+
};
26+
Element: HTMLElement;
27+
}
28+
29+
interface CountryInfo {
30+
callingCode?: string;
31+
code: string;
32+
example?: {
33+
callingCode: string;
34+
nationalNumber: string;
35+
};
36+
flag?: string;
37+
name?: string;
38+
}
39+
40+
const getCountryInfo = (countryCode: string): CountryInfo | undefined => {
41+
let example = getExample(countryCode);
42+
let callingCode = getCountryCodeForRegionCode(countryCode);
43+
44+
let c = countries[countryCode as TCountryCode];
45+
if (c === undefined) {
46+
//here some country code may not be found due to the discrepancy between countries-list and libphonenumber-js library
47+
//Only scenario where this is true is the usage of "AC"
48+
//Most countries consider "AC" Ascension Island as part of "SH" Saint Helena
49+
return;
50+
}
51+
return {
52+
code: countryCode,
53+
callingCode: callingCode.toString(),
54+
name: c ? c.name : undefined,
55+
flag: getEmojiFlag(countryCode as TCountryCode),
56+
example: example
57+
? {
58+
callingCode: callingCode.toString(),
59+
nationalNumber: example.number?.international ?? '',
60+
}
61+
: undefined,
62+
};
63+
};
64+
65+
class PhoneInput extends Component<Signature> {
66+
@tracked items: Array<CountryInfo> = [];
67+
@tracked selectedItem: CountryInfo = getCountryInfo('US')!;
68+
@tracked validationState: InputValidationState = 'initial';
69+
@tracked input: string = this.args.value ?? '';
70+
71+
@action onSelectItem(item: CountryInfo): void {
72+
this.selectedItem = item;
73+
if (this.args.onCountryCodeChange) {
74+
this.args.onCountryCodeChange(item.callingCode ?? '');
75+
}
76+
if (this.input.length > 0) {
77+
const parsedPhoneNumber = parsePhoneNumber(this.input, {
78+
regionCode: this.selectedItem.code,
79+
});
80+
this.validationState = parsedPhoneNumber.valid ? 'valid' : 'invalid';
81+
}
82+
}
83+
84+
constructor(owner: unknown, args: Signature['Args']) {
85+
super(owner, args);
86+
this.items = getSupportedRegionCodes()
87+
.map((code) => {
88+
return getCountryInfo(code);
89+
})
90+
.filter((c) => c !== undefined) as CountryInfo[];
91+
92+
if (this.args.countryCode) {
93+
this.selectedItem = this.items.find(
94+
(item) => item.callingCode === this.args.countryCode,
95+
)!;
96+
}
97+
}
98+
99+
get placeholder(): string | undefined {
100+
if (this.selectedItem) {
101+
return this.selectedItem.example?.nationalNumber;
102+
}
103+
return undefined;
104+
}
105+
106+
@action onInput(v: string): void {
107+
this.debouncedInput(v);
108+
}
109+
110+
private debouncedInput = debounce((input: string) => {
111+
this.input = input;
112+
113+
if (input === '') {
114+
this.validationState = 'initial';
115+
return;
116+
}
117+
118+
const parsedPhoneNumber = parsePhoneNumber(input, {
119+
regionCode: this.selectedItem.code,
120+
});
121+
this.validationState = parsedPhoneNumber.valid ? 'valid' : 'invalid';
122+
//save when the state is valid
123+
if (this.validationState === 'valid') {
124+
this.args.onInput(this.input);
125+
}
126+
}, 300);
127+
128+
<template>
129+
<BoxelInputGroup
130+
@placeholder={{this.placeholder}}
131+
@state={{this.validationState}}
132+
@onInput={{this.onInput}}
133+
@value={{this.input}}
134+
>
135+
<:before as |Accessories|>
136+
<Accessories.Select
137+
@placeholder={{this.placeholder}}
138+
@selected={{this.selectedItem}}
139+
@onChange={{this.onSelectItem}}
140+
@options={{this.items}}
141+
@selectedItemComponent={{PhoneSelectedItem}}
142+
@searchEnabled={{true}}
143+
@searchField='name'
144+
@matchTriggerWidth={{false}}
145+
aria-label='Select an country calling code'
146+
as |item|
147+
>
148+
<div>{{item.flag}} {{item.name}} +{{item.callingCode}}</div>
149+
</Accessories.Select>
150+
</:before>
151+
</BoxelInputGroup>
152+
</template>
153+
}
154+
155+
export interface SelectedItemSignature {
156+
Args: {
157+
option: any;
158+
};
159+
Element: HTMLDivElement;
160+
}
161+
162+
class PhoneSelectedItem extends Component<SelectedItemSignature> {
163+
<template>
164+
<div>
165+
{{@option.flag}}
166+
+{{@option.callingCode}}
167+
</div>
168+
</template>
169+
}
170+
171+
export default PhoneInput;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
import { action } from '@ember/object';
2+
import Component from '@glimmer/component';
3+
import { tracked } from '@glimmer/tracking';
4+
import FreestyleUsage from 'ember-freestyle/components/freestyle/usage';
5+
6+
import PhoneInput from './index.gts';
7+
8+
export default class PhoneInputUsage extends Component {
9+
@tracked value = '';
10+
@tracked countryCode = '';
11+
12+
@action onInput(value: string): void {
13+
this.value = value;
14+
}
15+
16+
@action onCountryCodeChange(code: string): void {
17+
this.countryCode = code;
18+
}
19+
20+
<template>
21+
<FreestyleUsage @name='PhoneInput'>
22+
<:description>
23+
<p>
24+
PhoneInput is a component that allows users to input phone numbers
25+
with a dropdown select of country code and validation of the inputted
26+
numbers
27+
</p>
28+
</:description>
29+
<:example>
30+
<PhoneInput
31+
@value={{this.value}}
32+
@countryCode={{this.countryCode}}
33+
@onInput={{this.onInput}}
34+
@onCountryCodeChange={{this.onCountryCodeChange}}
35+
/>
36+
</:example>
37+
</FreestyleUsage>
38+
</template>
39+
}

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

+2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import MenuUsage from './components/menu/usage.gts';
2626
import MessageUsage from './components/message/usage.gts';
2727
import ModalUsage from './components/modal/usage.gts';
2828
import MultiSelectUsage from './components/multi-select/usage.gts';
29+
import PhoneInputUsage from './components/phone-input/usage.gts';
2930
import PillUsage from './components/pill/usage.gts';
3031
import ProgressBarUsage from './components/progress-bar/usage.gts';
3132
import ProgressRadialUsage from './components/progress-radial/usage.gts';
@@ -64,6 +65,7 @@ export const ALL_USAGE_COMPONENTS = [
6465
['Message', MessageUsage],
6566
['Modal', ModalUsage],
6667
['MultiSelect', MultiSelectUsage],
68+
['PhoneInput', PhoneInputUsage],
6769
['Pill', PillUsage],
6870
['ProgressBar', ProgressBarUsage],
6971
['ProgressRadial', ProgressRadialUsage],
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
{
2+
"data": {
3+
"type": "card",
4+
"attributes": {
5+
"contactPhone": {
6+
"value": "01355889283",
7+
"type": {
8+
"index": 2,
9+
"label": "Work"
10+
}
11+
},
12+
"title": null,
13+
"description": null,
14+
"thumbnailURL": null
15+
},
16+
"meta": {
17+
"adoptsFrom": {
18+
"module": "../phone-number",
19+
"name": "CardWithContactPhoneNumber"
20+
}
21+
}
22+
}
23+
}

packages/experiments-realm/Contact/a01fc5c9-d70d-4b9c-aae4-384cf2b79b25.json

+16-8
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,24 @@
99
"primaryEmail": "david@gmail.com",
1010
"secondaryEmail": "david23232@gmail.com",
1111
"phoneMobile": {
12-
"type": "office",
13-
"country": 1,
14-
"area": 415,
15-
"number": 123456
12+
"phoneNumber": {
13+
"number": "1158524828",
14+
"countryCode": "60"
15+
},
16+
"type": {
17+
"index": 0,
18+
"label": "Mobile"
19+
}
1620
},
1721
"phoneOffice": {
18-
"type": null,
19-
"country": null,
20-
"area": null,
21-
"number": null
22+
"phoneNumber": {
23+
"number": null,
24+
"countryCode": null
25+
},
26+
"type": {
27+
"index": null,
28+
"label": null
29+
}
2230
},
2331
"socialLinks": [
2432
{

packages/experiments-realm/ExperimentsFieldsPreview/de720f57-964b-4a09-8d52-80cd8bb4b739.json

+3-5
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,8 @@
2626
"dan@email.com"
2727
],
2828
"phone": {
29-
"type": null,
30-
"country": null,
31-
"area": null,
32-
"number": null
29+
"number": "1159292211",
30+
"countryCode": "60"
3331
},
3432
"percentage": 100.159393,
3533
"currency": {
@@ -105,4 +103,4 @@
105103
}
106104
}
107105
}
108-
}
106+
}

packages/experiments-realm/crm/contact.gts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import StringField from 'https://cardstack.com/base/string';
2-
import { PhoneField } from '../phone';
2+
import { ContactPhoneNumber } from '../phone-number';
33
import { EmailField } from '../email';
44
import { ContactLinkField } from '../fields/contact-link';
55
import {
@@ -701,8 +701,8 @@ export class Contact extends CardDef {
701701
@field department = contains(StringField);
702702
@field primaryEmail = contains(EmailField);
703703
@field secondaryEmail = contains(EmailField);
704-
@field phoneMobile = contains(PhoneField);
705-
@field phoneOffice = contains(PhoneField);
704+
@field phoneMobile = contains(ContactPhoneNumber);
705+
@field phoneOffice = contains(ContactPhoneNumber);
706706
@field socialLinks = containsMany(SocialLinkField);
707707
@field statusTag = contains(StatusTagField); //this is an empty field that gets computed in subclasses
708708

packages/experiments-realm/experiments_fields_preview.gts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import { FeaturedImageField } from './fields/featured-image';
22
import { ContactLinkField } from './fields/contact-link';
33
import { EmailField } from './email';
4-
import { PhoneField } from './phone';
4+
import { PhoneField } from './phone-number';
55
import { UrlField } from './url';
66
import { WebsiteField } from './website';
77
import { Address as AddressField } from './address';

0 commit comments

Comments
 (0)