Declarative pin code component for Svelte.
A pin code is a short sequence of characters (usually numeric) used for verification. It differs from a password in that it is typically ephemeral and is not predetermined by the user.
Try it in the Svelte REPL.
Yarn
yarn add -D svelte-pincode
NPM
npm i -D svelte-pincode
Bind to either the code
or value
prop.
- code (
string[]
): Array of input values. An empty string represents an undefined value - value (
string
):code
joined as a string
<script>
import { Pincode, PincodeInput } from "svelte-pincode";
let code = [];
let value = '';
</script>
<Pincode bind:code bind:value>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<div>code: <code>{JSON.stringify(code)}</code></div>
<div>value: <code>{JSON.stringify(value)}</code></div>
Set selectTextOnFocus
to true
for the input value text to be selected when focused.
<script>
let input;
</script>
<Pincode selectTextOnFocus>
<PincodeInput bind:ref={input} value="0" />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<br />
<button on:click={() => input.focus()}>
Focus input
</button>
By default, this component accepts alphanumeric values.
Set type
to "numeric"
to only allow numbers.
<Pincode type="numeric">
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
Define intitial input values by using the code
prop or value
prop on PincodeInput
.
<script>
let pincode = ["1", "", "3"];
</script>
<Pincode bind:code={pincode}>
<PincodeInput />
<PincodeInput value="2" />
<PincodeInput />
<PincodeInput />
</Pincode>
<div>code: <code>{JSON.stringify(pincode)}</code></div>
Actual validation is left to the consumer.
This example illustrates how you can validate the code once all inputs have a value by binding to the complete
prop.
<script>
const correctCode = "1234";
let inputValue = '';
let complete = false;
$: success = complete && inputValue === correctCode;
$: error = complete && !success;
</script>
<Pincode bind:complete bind:value={inputValue}>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<div class:complete class:success class:error>
{#if !complete}
Enter {correctCode.length - inputValue.length} more digit(s)...
{/if}
{#if success}Correct code!{/if}
{#if error}Incorrect code.{/if}
</div>
As an alternative to the complete
prop, you can listen to the dispatched "complete" event:
<Pincode
on:complete="{(e) => {
console.log(e.detail); // { code: string[]; value: string; }
}}"
/>
code
can be set programmatically.
Type in some initial values and then try setting or clearing the code using the buttons.
<script>
let passcode = [];
</script>
<Pincode bind:code={passcode}>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<div>code: <code>{JSON.stringify(passcode)}</code></div>
<button on:click={() => passcode = ['1', '2', '3', '4']}>
Set code
</button>
<button on:click={() => passcode = ['', '', '', '']}>
Clear code
</button>
To programmatically focus the first input, invoke the focusFirstInput
method on a component reference.
<script>
let ref;
</script>
<Pincode bind:this={ref}>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<br />
<button on:click={ref.focusFirstInput}>
Focus first input
</button>
To focus the next input with no value, invoke the focusNextEmptyInput
method.
<script>
let pincodeRef;
</script>
<Pincode code={["9", "9"]} bind:this={pincodeRef}>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<br />
<button on:click={pincodeRef.focusNextEmptyInput}>
Focus next empty input
</button>
To focus the last input, invoke the focusLastInput
method.
<script>
let passcodeRef;
</script>
<Pincode bind:this={passcodeRef}>
<PincodeInput />
<PincodeInput />
<PincodeInput />
<PincodeInput />
</Pincode>
<br />
<button on:click={passcodeRef.focusLastInput}>
Focus last input
</button>
This component is minimally styled; override the default styles by targeting the data-pincode
selector:
/** Pincode **/
:global([data-pincode]) {
display: inline-flex;
border: 1px solid #e0e0e0;
}
/** PincodeInput */
:global([data-pincode] input) {
width: 3rem;
padding: 0.5rem 1rem;
margin: 0;
border: 0;
border-radius: 0;
text-align: center;
}
:global([data-pincode] input:focus) {
z-index: 1;
}
:global([data-pincode] input:not(:last-of-type)) {
border-right: 1px solid #e0e0e0;
}
Alternatively, you can use the unstyled components located in the svelte/src/unstyled
folder.
<script>
import Pincode from "svelte-pincode/src/unstyled/Pincode.svelte";
import PincodeInput from "svelte-pincode/src/unstyled/PincodeInput.svelte";
</script>
Prop name | Value | Default |
---|---|---|
code | string[] |
[] |
value | string |
"" |
type | "alphanumeric" or "numeric" |
"alphanumeric" |
selectTextOnFocus | boolean |
false |
focusFirstInput
focusNextEmptyInput
focusLastInput
- on:complete: fired when all inputs have a value
<Pincode
on:complete="{(e) => {
console.log(e.detail); // { code: string[]; value: string; }
}}"
/>
Prop name | Value |
---|---|
id | string (default: "input" + Math.random().toString(36) ) |
ref | HTMLInputElement (default: null ) |
- on:focus
- on:blur
- on:keydown
Svelte version 3.31 or greater is required to use this component with TypeScript.
TypeScript definitions are located in the types folder.