forked from bfintal/Counter-Up2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
121 lines (98 loc) · 3.42 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
export const counterUp = ( el, options = {} ) => {
const {
action = 'start',
duration = 1000,
delay = 16,
} = options
// Allow people to use this as a stop method.
if ( action === 'stop' ) {
stopCountUp( el )
return
}
stopCountUp( el )
// If no number, don't do anything.
if ( ! /[0-9]/.test( el.innerHTML ) ) {
return
}
const nums = divideNumbers( el.innerHTML, {
duration: duration || el.getAttribute( 'data-duration' ),
delay: delay || el.getAttribute( 'data-delay' ),
} )
// Remember the contents.
el._countUpOrigInnerHTML = el.innerHTML
// Start counting.
el.innerHTML = nums[ 0 ] || ' ' // Use a non-breaking space to prevent layout shift.
el.style.visibility = 'visible'
// Function for displaying output with the set time and delay.
const output = function() {
el.innerHTML = nums.shift() || ' ' // Use a non-breaking space to prevent layout shift.
if ( nums.length ) {
clearTimeout( el.countUpTimeout )
el.countUpTimeout = setTimeout( output, delay )
} else {
el._countUpOrigInnerHTML = undefined
}
}
el.countUpTimeout = setTimeout( output, delay )
}
export default counterUp
const stopCountUp = el => {
clearTimeout( el.countUpTimeout )
if ( el._countUpOrigInnerHTML ) {
el.innerHTML = el._countUpOrigInnerHTML
el._countUpOrigInnerHTML = undefined
}
el.style.visibility = ''
}
export const divideNumbers = ( numToDivide, options = {} ) => {
const {
duration = 1000,
delay = 16,
} = options
// Number of times the number will change.
const divisions = duration / delay
// Split numbers and html tags.
const splitValues = numToDivide.toString().split( /(<[^>]+>|[0-9.][,.0-9]*[0-9]*)/ )
// Contains all numbers to be displayed.
const nums = []
// Set blank strings to ready the split values.
for ( let k = 0; k < divisions; k++ ) {
nums.push( '' )
}
// Loop through all numbers and html tags.
for ( let i = 0; i < splitValues.length; i++ ) {
// If number split it into smaller numbers and insert it to nums.
if ( /([0-9.][,.0-9]*[0-9]*)/.test( splitValues[ i ] ) && ! /<[^>]+>/.test( splitValues[ i ] ) ) {
let num = splitValues[ i ]
// Find all the occurances of . and ,
const symbols = [ ...num.matchAll( /[.,]/g ) ]
// Get all the locations of the characters so we can re-place them later on.
.map( m => ( { char: m[0], i: num.length - m.index - 1 } ) )
// Make sure we go through the characters from right to left
.sort( ( a, b ) => a.i - b.i )
// Remove commas and dots for computation purposes.
num = num.replace( /[.,]/g, '' )
// Start adding numbers from the end.
let k = nums.length - 1
// Create small numbers we'll the count over.
for ( let val = divisions; val >= 1; val-- ) {
let newNum = parseInt( num / divisions * val, 10 )
// Re-insert the symbols in the indices they were at.
newNum = symbols.reduce( ( num, { char, i } ) => {
return num.length <= i ? num // If we don't have enough numbers, don't insert the symbol.
: num.slice( 0, -i ) + (char ?? '') + num.slice( -i ) // Replace char for an empty string in case it's not defined.
}, newNum.toString() )
// Insert all small numbers.
nums[ k-- ] += newNum
}
} else {
// Insert all non-numbers in the same place.
for ( let k = 0; k < divisions; k++ ) {
nums[ k ] += splitValues[ i ]
}
}
}
// The last value of the element should be the original one.
nums[ nums.length ] = numToDivide.toString()
return nums
}