npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

svelte-tel-input

v3.5.2

Published

svelte-tel-input

Downloads

17,240

Readme

npm version

Svelte Tel Input

Lightweight svelte tel/phone input standardizer.

🔥 Check it out live here

Installation

Svelte Tel Input is distributed via npm.

npm install svelte-tel-input

Features

  • Support SSR/SSG.
  • Parse and validate phone number.You can store one exact format (E164), no matter how users type their phone numbers.
  • Format (specified to its country), to make it more readable.
  • Prevent non-digits typing into the input, except the leading + sign (and space optionally).
  • Handle copy-pasted phone numbers, it's sanitize non-digit characters except the leading + sign (and space optionally).
  • Automatic placeholder generation for the selected country.
  • International or National formatted phone numbers.

Usage

Advanced

Snippet would be too long - Example - REPL (StackBlitz)

Basic

Example - REPL (StackBlitz)

<script lang="ts">
  import { TelInput, normalizedCountries } from 'svelte-tel-input';
  import type { DetailedValue, CountryCode, E164Number } from 'svelte-tel-input/types';

  // Any Country Code Alpha-2 (ISO 3166)
  let selectedCountry: CountryCode | null = 'HU';

  // You must use E164 number format. It's guarantee the parsing and storing consistency.
  let value: E164Number | null = '+36301234567';

  // Validity
  let valid = true;

  // Optional - Extended details about the parsed phone number
  let detailedValue: DetailedValue | null = null;
</script>

<div class="wrapper">
  <select
    class="country-select {!valid ? 'invalid' : ''}"
    aria-label="Default select example"
    name="Country"
    bind:value={selectedCountry}
  >
    <option value={null} hidden={selectedCountry !== null}>Please select</option>
    {#each normalizedCountries as currentCountry (currentCountry.id)}
      <option
        value={currentCountry.iso2}
        selected={currentCountry.iso2 === selectedCountry}
        aria-selected={currentCountry.iso2 === selectedCountry}
      >
        {currentCountry.iso2} (+{currentCountry.dialCode})
      </option>
    {/each}
  </select>
  <TelInput
    bind:country={selectedCountry}
    bind:value
    bind:valid
    bind:detailedValue
    class="basic-tel-input {!valid ? 'invalid' : ''}"
  />
</div>

<style>
  .wrapper :global(.basic-tel-input) {
    height: 32px;
    padding-left: 12px;
    padding-right: 12px;
    border-radius: 6px;
    border: 1px solid;
    outline: none;
  }

  .wrapper :global(.country-select) {
    height: 36px;
    padding-left: 12px;
    padding-right: 12px;
    border-radius: 6px;
    border: 1px solid;
    outline: none;
  }

  .wrapper :global(.invalid) {
    border-color: red;
  }
</style>

Props

The default export of the library is the main TelInput component. It has the following props:

| Property name | Type | Default Value | Usage | | ------------- | ---------------------- | ------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | value | E164Number \| null | null | E164 is the international format of phone.numbers. This is the main entry point to store and/or load an existent phone number. | | country | CountryCode \| null | null | It's accept any Country Code Alpha-2 (ISO 3166). You can set manually (e.g: by the user via a select). The parser will inspect the entered phone number and if it detect a valid country calling code, then it's automatically set the country to according to the detected country calling code. E.g: +36 -> HU | | disabled | boolean | false | It's block the parser and prevent entering input. You must handle its styling on your own. | | valid | boolean | true | Indicates whether the entered tel number validity. | | detailedValue | DetailedValue \|null | null | All of the formatted results of the tel input. | | class | string | `` | You can pass down any classname to the component | | required | boolean \| null | null | Set the required attribute on the input element | | options | TelInputOptions | check below | Allow or disallow spaces in the input field | | id | string \| null | uid | HTMLInputElement's attribute | | name | string \| null | null | HTMLInputElement's attribute | | readonly | boolean \| null | null | HTMLInputElement's attribute | | size | number \| null | null | HTMLInputElement's attribute | | autocomplete | string \| null | null | HTMLInputElement's attribute | | |

Config options:

{
    // Generates country specific placeholder for the selected country.
    autoPlaceholder: true,
    // Allow or disallow spaces in the input field
    spaces: true,
    // If you have a parsed phone number and you change country manually from outside, then it's set the `valid` prop to false.
    invalidateOnCountryChange: false,
    // Formatted output `national` | `international`
    format: 'national'
}

Dispatched Events

The default export of the library is the main TelInput component. It has the following props:

| Event name | Type | | ------------------- | ---------------------- | | updateValue | E164Number \| null | | updateDetailedValue | DetailedValue \|null | | updateCountry | CountryCode \| null | | updateValid | boolean | | parseError | string |

Use case of the event driven behavior

<script lang="ts">
	// Imports, etc....
	let value: E164Number | null = null;
	const yourHandler = (e: CustomEvent<E164Number | null>) => {
        value = e.detail //
        // do stuff...
	};
</script>

<TelInput value={cachedValue ?? value} on:updateValue={yourHandler} ... />

Caveats

  • In order to reset value and/or country from outside (you must pass (or set if you binded) null for the property) have some side-effects:

    • Reseting the value will set (keep the country as is):
      • detailedValue to null
      • dispatch updateDetailedValue event
    • Reseting the country will set:
      • value to null
      • detailedValue to null
      • valid to true if invalidateOnCountryChange config option is false (@default false). Otherwise it will be false.
      • and dispatch updateValid, updateValue updateDetailedValue events
    • Reseting both value and country will set:
      • valid to true
      • detailedValue to null;
  • Let's assume you pass a US E164 number, which can be a partial E164, but long enough to determine the country and you pass DE country directly. The country will be updated to US, which is determined from the E164 in this example. If the E164 is not long enough to determine its country, then the country will stay what you passed to the component (DE).

Goals

  • Solve the problem that a users can enter the same phone number in different formats.
  • Storing a phone number in a standard format, that can be indexable and searchable in any database.
  • Should be accessible for the the browser. Eg. for a <a href="tel+36201234567 />.
  • The stored phone number format can be useable for any SMS gateway(e.g for 2FA) and if somebody can call the number from anywhere, it should work.

Dependencies

libphonenumber-js

Changelog

| Package | Changelog | | -------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------- | | @gyurielf/svelte-tel-input | Changelog |

Roadmap

  • [x] Add Changelog
  • [x] Add CI/CD
  • [x] Integrate libphonenumber
  • [x] Implement parser
  • [x] Add basics docs and examples
  • [x] Add advanced examples
  • [x] Generate placeholders autimatically
  • [x] Move to monorepo
  • [ ] Improve A11Y

See the open issues for a list of proposed features (and known issues).

Support

License

Distributed under the MIT License. See LICENSE.md for more information.