Optional Dependencies
Character Length Limit
TL;DR: Solution.
TimeredCounter
works by treating strings as numbers in a certain base and then converting them to decimal numbers. This approach is convenient for sampling numbers to create scrolling lists.
This method works in most cases, but problems arise when the numbers are too large or too small.
According to the ECMA specification, when number < 10^-5 || number > 10^21
, numbers are represented in scientific notation[1][2].
This causes a problem as numbers represented in scientific notation lose some precision.
1234567891012131415161
=> 1.2345678910121315e+21 // lost precision for the last 6 digits
Due to the implementation, this issue is more likely to occur with timered-counter-string
. The base is essentially the number of unique characters in the string, which can easily result in a number exceeding 10^21
.
Similarly, when using timered-counter-number
, if the value is less than 10^-5
, the number will also be represented in scientific notation.
How to Remove Character Length Limit
To solve this problem, we need to use a third-party high-precision calculation library like decimal.js. TimeredCounter
provides two adapters:
BuildInNumberAdapter
(default): UsesNumber
.DecimalJsAdapter
: Usesdecimal.js
, requires installing decimal.js.Of course, you can implement your own adapter by implementing the
NumberAdapter
interface.
In the example below, you can switch adapters to see the effect.
9007199254740991
is the largest integer that can be accurately represented in JavaScript. When the value exceeds this, precision is lost. You can try inputting:
9007199254740992
: 9007199254740991 + 19007199254741001
: 9007199254740991 + 109007199254741091
: 9007199254740991 + 100Observe the precision loss.
Click to view code
<script setup>
import { ref, watch } from "vue";
import { setNumberAdapter } from "timered-counter";
const number = ref(Number.MAX_SAFE_INTEGER.toString(10));
const adapters = [
{ label: "BuildInNumberAdapter", value: "number" },
// { label: "BuildInBigintAdapter", adapter: BuildInBigintAdapter() },
{ label: "DecimalJsAdapter", value: "decimal.js" },
];
const adapterIndex = ref(0);
watch(adapterIndex, (index) => setNumberAdapter(adapters[index].value));
const realRenderNumber = ref();
watch([number, adapterIndex], () => (realRenderNumber.value = number.value), {
immediate: true,
});
const counterRef = ref();
function handleAnimationEnd() {
realRenderNumber.value = counterRef.value?.getAttribute("aria-label");
}
function handleInput(e) {
number.value = e.target.value
.split("")
.filter((c) => (c >= "0" && c <= "9") || c === "-" || c === ".")
.join("");
}
</script>
<template>
<div class="text-center">
<timered-counter-number
ref="counterRef"
:key="adapters[adapterIndex].value"
:value="number"
@timered-counter-animation-end="handleAnimationEnd"
/>
</div>
<div
v-if="realRenderNumber !== number"
class="bg-[var(--vp-c-danger-soft)] p-2 rounded mt-4"
>
Mismatch: should be
<span class="text-[var(--vp-c-danger-1)]">{{ number }}</span> but rendered
as <span class="text-[var(--vp-c-danger-1)]">{{ realRenderNumber }}</span>
</div>
<hr />
<div class="flex gap-4">
<textarea
class="flex-auto border border-solid p-1"
@input="handleInput"
:value="number"
/>
<select
v-model="adapterIndex"
class="self-start border border-solid p-1 appearance-auto"
>
<option v-for="(adapter, index) in adapters" :key="index" :value="index">
{{ adapter.label }}
</option>
</select>
</div>
</template>
<style scoped></style>
Support Emoji Segmentation
For general emoji
, we can use the default configuration, but for some special emoji
, like 👨👩👧👦
, which are called Emoji ZWJ Sequence
and are composed of multiple emoji
.
In this case, it is difficult to correctly recognize strings containing such emoji
. We can use Intl.Segmenter or third-party libraries like grapheme-splitter. TimeredCounter
provides three adapters:
BuildInStringAdapter
(default): UsesString.split("")
to split strings.IntlSegmenterAdapter
: UsesIntl.Segmenter
, requires browser support.GraphemeSplitterAdapter
: Usesgrapheme-splitter
, requires installing grapheme-splitter.
In the example below, you can switch adapters to see the effect.
You can try inputting the following characters to see the effect.
⌚
: Basic emoji↔️
: Text character rendered as emoji👩
: Modifiable basic emoji👩🏿
: Modifiable basic emoji + emoji modifier🧑💻
: Emoji combination sequence
Click to view code
<script setup>
import { onMounted, ref, watch } from "vue";
import { setStringAdapter } from "timered-counter";
const string = ref("emoji 🎉 🧑💻");
const adapters = [
{ label: "BuildInStringAdapter", value: "string" },
{
label: "BuildInIntlSegmenterAdapter",
value: "intl-segmenter",
},
{ label: "GraphemeSplitterAdapter", value: "grapheme-splitter" },
];
const adapterIndex = ref(0);
watch(adapterIndex, (index) => setStringAdapter(adapters[index].value));
const realRenderString = ref();
watch([string, adapterIndex], () => (realRenderString.value = string.value), {
immediate: true,
});
const counterRef = ref();
function handleAnimationEnd() {
realRenderString.value = counterRef.value?.getAttribute("aria-label");
}
onMounted(() => setTimeout(() => handleAnimationEnd(), 1000));
</script>
<template>
<div class="text-center">
<timered-counter-string
ref="counterRef"
:value="string"
:key="adapters[adapterIndex].value"
@timered-counter-animation-end="handleAnimationEnd"
/>
</div>
<div
v-if="realRenderString !== string"
class="bg-[var(--vp-c-danger-soft)] p-2 rounded mt-4"
>
Mismatch: should be
<span class="text-[var(--vp-c-danger-1)]">{{ string }}</span> but rendered
as <span class="text-[var(--vp-c-danger-1)]">{{ realRenderString }}</span>
</div>
<hr />
<div class="flex gap-4">
<textarea class="flex-auto border border-solid p-1" v-model="string" />
<select
v-model="adapterIndex"
class="self-start border border-solid p-1 appearance-auto"
>
<option v-for="(adapter, index) in adapters" :key="index" :value="index">
{{ adapter.label }}
</option>
</select>
</div>
</template>
<style scoped></style>