farvist 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +324 -0
- package/assets/farvist.js +187 -0
- package/assets/icons/farvist-icons.svg +49 -0
- package/assets/logo.svg +11 -0
- package/assets/og-image.svg +34 -0
- package/assets/patterns/blob.svg +9 -0
- package/assets/patterns/dots.svg +3 -0
- package/assets/patterns/grid.svg +3 -0
- package/assets/patterns/mesh.svg +20 -0
- package/dist/farvist.css +8967 -0
- package/dist/farvist.min.css +1 -0
- package/package.json +53 -0
- package/scss/abstracts/_functions.scss +85 -0
- package/scss/abstracts/_index.scss +9 -0
- package/scss/abstracts/_mixins.scss +78 -0
- package/scss/abstracts/_variables.scss +252 -0
- package/scss/base/_reset.scss +120 -0
- package/scss/base/_root.scss +87 -0
- package/scss/base/_typography.scss +102 -0
- package/scss/components/_accordion.scss +64 -0
- package/scss/components/_alerts.scss +63 -0
- package/scss/components/_avatar.scss +76 -0
- package/scss/components/_badges.scss +54 -0
- package/scss/components/_breadcrumb.scss +44 -0
- package/scss/components/_buttons.scss +163 -0
- package/scss/components/_callout.scss +37 -0
- package/scss/components/_cards.scss +74 -0
- package/scss/components/_chip.scss +61 -0
- package/scss/components/_dropdown.scss +90 -0
- package/scss/components/_empty-state.scss +37 -0
- package/scss/components/_forms.scss +125 -0
- package/scss/components/_icon.scss +25 -0
- package/scss/components/_list-group.scss +60 -0
- package/scss/components/_modal.scss +95 -0
- package/scss/components/_navbar.scss +67 -0
- package/scss/components/_pagination.scss +51 -0
- package/scss/components/_progress.scss +60 -0
- package/scss/components/_range.scss +58 -0
- package/scss/components/_rating.scss +26 -0
- package/scss/components/_segmented.scss +51 -0
- package/scss/components/_skeleton.scss +52 -0
- package/scss/components/_spinner.scss +60 -0
- package/scss/components/_stat.scss +32 -0
- package/scss/components/_stepper.scss +78 -0
- package/scss/components/_switch.scss +65 -0
- package/scss/components/_table.scss +51 -0
- package/scss/components/_tabs.scss +64 -0
- package/scss/components/_timeline.scss +77 -0
- package/scss/components/_toast.scss +79 -0
- package/scss/components/_tooltip.scss +45 -0
- package/scss/farvist.scss +60 -0
- package/scss/layout/_grid.scss +83 -0
- package/scss/utilities/_backgrounds.scss +110 -0
- package/scss/utilities/_effects.scss +118 -0
- package/scss/utilities/_spacing.scss +76 -0
- package/scss/utilities/_utilities.scss +213 -0
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Forms
|
|
3
|
+
// Glass inputs with a neon focus ring. Validation + input-group helpers too.
|
|
4
|
+
// =============================================================================
|
|
5
|
+
|
|
6
|
+
@use 'sass:map';
|
|
7
|
+
@use '../abstracts' as *;
|
|
8
|
+
|
|
9
|
+
.form-label {
|
|
10
|
+
display: inline-block;
|
|
11
|
+
margin-bottom: spacer(2);
|
|
12
|
+
font-weight: map.get($font-weights, 'medium');
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
.form-control,
|
|
16
|
+
.form-select {
|
|
17
|
+
display: block;
|
|
18
|
+
width: 100%;
|
|
19
|
+
padding: spacer(2) spacer(3);
|
|
20
|
+
font-size: map.get($font-sizes, 'base');
|
|
21
|
+
line-height: $line-height-base;
|
|
22
|
+
color: var(--fv-body-color);
|
|
23
|
+
background-color: var(--fv-glass-bg);
|
|
24
|
+
backdrop-filter: blur(var(--fv-glass-blur)) saturate(var(--fv-glass-saturate));
|
|
25
|
+
-webkit-backdrop-filter: blur(var(--fv-glass-blur)) saturate(var(--fv-glass-saturate));
|
|
26
|
+
border: $border-width solid var(--fv-glass-border);
|
|
27
|
+
border-radius: var(--fv-border-radius);
|
|
28
|
+
transition: border-color $transition-base, box-shadow $transition-base;
|
|
29
|
+
|
|
30
|
+
&::placeholder {
|
|
31
|
+
color: var(--fv-muted);
|
|
32
|
+
opacity: 0.8;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
&:focus {
|
|
36
|
+
outline: 0;
|
|
37
|
+
border-color: var(--fv-primary);
|
|
38
|
+
box-shadow: var(--fv-focus-ring);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
&:disabled {
|
|
42
|
+
opacity: 0.55;
|
|
43
|
+
cursor: not-allowed;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
.form-select {
|
|
48
|
+
appearance: none;
|
|
49
|
+
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3E%3Cpath fill='none' stroke='%2395a3c4' stroke-width='2' d='M2 5l6 6 6-6'/%3E%3C/svg%3E");
|
|
50
|
+
background-repeat: no-repeat;
|
|
51
|
+
background-position: right 0.75rem center;
|
|
52
|
+
background-size: 1rem;
|
|
53
|
+
padding-right: spacer(7);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
textarea.form-control {
|
|
57
|
+
min-height: 6rem;
|
|
58
|
+
resize: vertical;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
.form-text {
|
|
62
|
+
margin-top: spacer(1);
|
|
63
|
+
font-size: map.get($font-sizes, 'sm');
|
|
64
|
+
color: var(--fv-muted);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
.form-check {
|
|
68
|
+
display: flex;
|
|
69
|
+
align-items: center;
|
|
70
|
+
gap: spacer(2);
|
|
71
|
+
margin-bottom: spacer(2);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
.form-check-input {
|
|
75
|
+
width: 1rem;
|
|
76
|
+
height: 1rem;
|
|
77
|
+
accent-color: var(--fv-primary);
|
|
78
|
+
cursor: pointer;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
.form-check-label {
|
|
82
|
+
cursor: pointer;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Stitch controls together horizontally.
|
|
86
|
+
.input-group {
|
|
87
|
+
display: flex;
|
|
88
|
+
align-items: stretch;
|
|
89
|
+
|
|
90
|
+
.form-control {
|
|
91
|
+
flex: 1 1 auto;
|
|
92
|
+
width: 1%;
|
|
93
|
+
min-width: 0;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
.input-group-text {
|
|
98
|
+
display: flex;
|
|
99
|
+
align-items: center;
|
|
100
|
+
padding: spacer(2) spacer(3);
|
|
101
|
+
color: var(--fv-muted);
|
|
102
|
+
background-color: var(--fv-glass-bg-strong);
|
|
103
|
+
border: $border-width solid var(--fv-glass-border);
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
// Validation states.
|
|
107
|
+
.is-valid {
|
|
108
|
+
border-color: var(--fv-success) !important;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
.is-invalid {
|
|
112
|
+
border-color: var(--fv-danger) !important;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
.valid-feedback {
|
|
116
|
+
margin-top: spacer(1);
|
|
117
|
+
font-size: map.get($font-sizes, 'sm');
|
|
118
|
+
color: var(--fv-success);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
.invalid-feedback {
|
|
122
|
+
margin-top: spacer(1);
|
|
123
|
+
font-size: map.get($font-sizes, 'sm');
|
|
124
|
+
color: var(--fv-danger);
|
|
125
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Icon
|
|
3
|
+
// Sizing/coloring for inline SVG icons. The icons inherit `currentColor`, so
|
|
4
|
+
// they pick up text color, gradients (via .text-gradient on a parent) and glow.
|
|
5
|
+
// <svg class="icon"><use href="assets/icons/farvist-icons.svg#i-bell"/></svg>
|
|
6
|
+
// =============================================================================
|
|
7
|
+
|
|
8
|
+
@use '../abstracts' as *;
|
|
9
|
+
|
|
10
|
+
.icon {
|
|
11
|
+
display: inline-block;
|
|
12
|
+
width: 1em;
|
|
13
|
+
height: 1em;
|
|
14
|
+
vertical-align: -0.125em;
|
|
15
|
+
flex-shrink: 0;
|
|
16
|
+
fill: none;
|
|
17
|
+
stroke: currentColor;
|
|
18
|
+
stroke-width: 2;
|
|
19
|
+
stroke-linecap: round;
|
|
20
|
+
stroke-linejoin: round;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
.icon-sm { width: 0.85em; height: 0.85em; }
|
|
24
|
+
.icon-lg { width: 1.5em; height: 1.5em; }
|
|
25
|
+
.icon-xl { width: 2em; height: 2em; }
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · List group
|
|
3
|
+
// <div class="list-group">
|
|
4
|
+
// <a class="list-group-item is-active" href="#">Inbox</a>
|
|
5
|
+
// <a class="list-group-item" href="#">Drafts</a>
|
|
6
|
+
// <div class="list-group-item">Plain row</div>
|
|
7
|
+
// </div>
|
|
8
|
+
// =============================================================================
|
|
9
|
+
|
|
10
|
+
@use 'sass:map';
|
|
11
|
+
@use '../abstracts' as *;
|
|
12
|
+
|
|
13
|
+
.list-group {
|
|
14
|
+
display: flex;
|
|
15
|
+
flex-direction: column;
|
|
16
|
+
border-radius: map.get($radii, 'lg');
|
|
17
|
+
overflow: hidden;
|
|
18
|
+
@include glass;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
.list-group-item {
|
|
22
|
+
display: flex;
|
|
23
|
+
align-items: center;
|
|
24
|
+
gap: spacer(3);
|
|
25
|
+
padding: spacer(3) spacer(4);
|
|
26
|
+
color: var(--fv-body-color);
|
|
27
|
+
background-color: transparent;
|
|
28
|
+
border: 0;
|
|
29
|
+
border-bottom: $border-width solid var(--fv-glass-border);
|
|
30
|
+
text-align: left;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
.list-group-item:last-child {
|
|
34
|
+
border-bottom: 0;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Interactive variants (anchors / buttons).
|
|
38
|
+
a.list-group-item,
|
|
39
|
+
button.list-group-item {
|
|
40
|
+
width: 100%;
|
|
41
|
+
text-decoration: none;
|
|
42
|
+
cursor: pointer;
|
|
43
|
+
transition: background-color $transition-base;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
a.list-group-item:hover,
|
|
47
|
+
button.list-group-item:hover {
|
|
48
|
+
background-color: var(--fv-glass-bg);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
a.list-group-item:focus-visible,
|
|
52
|
+
button.list-group-item:focus-visible {
|
|
53
|
+
outline: 0;
|
|
54
|
+
box-shadow: var(--fv-focus-ring);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.list-group-item.is-active {
|
|
58
|
+
color: var(--fv-body-color);
|
|
59
|
+
background-color: var(--fv-glass-bg-strong);
|
|
60
|
+
}
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Modal (native <dialog>)
|
|
3
|
+
// Markup:
|
|
4
|
+
// <button data-fv-open="#demo">Open</button>
|
|
5
|
+
// <dialog class="modal" id="demo">
|
|
6
|
+
// <div class="modal-header">
|
|
7
|
+
// <h3 class="modal-title">Title</h3>
|
|
8
|
+
// <button class="modal-close" data-fv-dismiss aria-label="Close">×</button>
|
|
9
|
+
// </div>
|
|
10
|
+
// <div class="modal-body">…</div>
|
|
11
|
+
// <div class="modal-footer">…</div>
|
|
12
|
+
// </dialog>
|
|
13
|
+
// farvist.js wires data-fv-open / data-fv-dismiss + backdrop-click close.
|
|
14
|
+
// =============================================================================
|
|
15
|
+
|
|
16
|
+
@use 'sass:map';
|
|
17
|
+
@use '../abstracts' as *;
|
|
18
|
+
|
|
19
|
+
.modal {
|
|
20
|
+
width: calc(100% - #{spacer(5)});
|
|
21
|
+
max-width: 32rem;
|
|
22
|
+
padding: 0;
|
|
23
|
+
color: var(--fv-body-color);
|
|
24
|
+
border-radius: map.get($radii, 'xl');
|
|
25
|
+
@include glass($strong: true);
|
|
26
|
+
box-shadow: var(--fv-shadow-xl);
|
|
27
|
+
|
|
28
|
+
&::backdrop {
|
|
29
|
+
background-color: rgba(6, 9, 18, 0.6);
|
|
30
|
+
backdrop-filter: blur(4px);
|
|
31
|
+
-webkit-backdrop-filter: blur(4px);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
&[open] {
|
|
35
|
+
animation: fv-modal-in 0.2s ease;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
.modal-header {
|
|
40
|
+
display: flex;
|
|
41
|
+
align-items: center;
|
|
42
|
+
justify-content: space-between;
|
|
43
|
+
gap: spacer(3);
|
|
44
|
+
padding: spacer(4) spacer(5);
|
|
45
|
+
border-bottom: $border-width solid var(--fv-glass-border);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.modal-title {
|
|
49
|
+
margin: 0;
|
|
50
|
+
font-size: map.get($font-sizes, 'xl');
|
|
51
|
+
font-weight: map.get($font-weights, 'bold');
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
.modal-body {
|
|
55
|
+
padding: spacer(5);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
.modal-footer {
|
|
59
|
+
display: flex;
|
|
60
|
+
align-items: center;
|
|
61
|
+
justify-content: flex-end;
|
|
62
|
+
gap: spacer(2);
|
|
63
|
+
padding: spacer(4) spacer(5);
|
|
64
|
+
border-top: $border-width solid var(--fv-glass-border);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
.modal-close {
|
|
68
|
+
display: inline-flex;
|
|
69
|
+
align-items: center;
|
|
70
|
+
justify-content: center;
|
|
71
|
+
width: 2rem;
|
|
72
|
+
height: 2rem;
|
|
73
|
+
font-size: 1.5rem;
|
|
74
|
+
line-height: 1;
|
|
75
|
+
color: var(--fv-muted);
|
|
76
|
+
background: transparent;
|
|
77
|
+
border: 0;
|
|
78
|
+
border-radius: map.get($radii, 'md');
|
|
79
|
+
cursor: pointer;
|
|
80
|
+
transition: color $transition-base, background-color $transition-base;
|
|
81
|
+
|
|
82
|
+
&:hover {
|
|
83
|
+
color: var(--fv-body-color);
|
|
84
|
+
background-color: var(--fv-glass-bg);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
@keyframes fv-modal-in {
|
|
89
|
+
from { opacity: 0; transform: translateY(1rem) scale(0.98); }
|
|
90
|
+
to { opacity: 1; transform: none; }
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
@media (prefers-reduced-motion: reduce) {
|
|
94
|
+
.modal[open] { animation: none; }
|
|
95
|
+
}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Navbar
|
|
3
|
+
// A frosted-glass navigation bar. Sticky + backdrop blur reads beautifully over
|
|
4
|
+
// scrolling content.
|
|
5
|
+
// =============================================================================
|
|
6
|
+
|
|
7
|
+
@use 'sass:map';
|
|
8
|
+
@use '../abstracts' as *;
|
|
9
|
+
|
|
10
|
+
.navbar {
|
|
11
|
+
display: flex;
|
|
12
|
+
align-items: center;
|
|
13
|
+
gap: spacer(4);
|
|
14
|
+
padding: spacer(3) spacer(5);
|
|
15
|
+
@include glass($strong: true);
|
|
16
|
+
border-top: 0;
|
|
17
|
+
border-right: 0;
|
|
18
|
+
border-left: 0;
|
|
19
|
+
border-radius: 0;
|
|
20
|
+
box-shadow: var(--fv-shadow-md);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
.navbar-brand {
|
|
24
|
+
display: inline-flex;
|
|
25
|
+
align-items: center;
|
|
26
|
+
gap: spacer(2);
|
|
27
|
+
font-size: map.get($font-sizes, 'xl');
|
|
28
|
+
font-weight: map.get($font-weights, 'bold');
|
|
29
|
+
letter-spacing: -0.02em;
|
|
30
|
+
color: var(--fv-body-color);
|
|
31
|
+
text-decoration: none;
|
|
32
|
+
|
|
33
|
+
&:hover {
|
|
34
|
+
text-decoration: none;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
.navbar-nav {
|
|
39
|
+
display: flex;
|
|
40
|
+
align-items: center;
|
|
41
|
+
gap: spacer(1);
|
|
42
|
+
margin-left: auto;
|
|
43
|
+
margin-bottom: 0;
|
|
44
|
+
padding-left: 0;
|
|
45
|
+
list-style: none;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.nav-link {
|
|
49
|
+
display: block;
|
|
50
|
+
padding: spacer(2) spacer(3);
|
|
51
|
+
color: var(--fv-muted);
|
|
52
|
+
border-radius: map.get($radii, 'md');
|
|
53
|
+
text-decoration: none;
|
|
54
|
+
transition: color $transition-base, background-color $transition-base;
|
|
55
|
+
|
|
56
|
+
&:hover {
|
|
57
|
+
color: var(--fv-body-color);
|
|
58
|
+
background-color: var(--fv-glass-bg);
|
|
59
|
+
text-decoration: none;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
&.active {
|
|
63
|
+
color: var(--fv-body-color);
|
|
64
|
+
font-weight: map.get($font-weights, 'semibold');
|
|
65
|
+
background-color: var(--fv-glass-bg);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Pagination
|
|
3
|
+
// =============================================================================
|
|
4
|
+
|
|
5
|
+
@use 'sass:map';
|
|
6
|
+
@use '../abstracts' as *;
|
|
7
|
+
|
|
8
|
+
.pagination {
|
|
9
|
+
display: inline-flex;
|
|
10
|
+
flex-wrap: wrap;
|
|
11
|
+
gap: spacer(1);
|
|
12
|
+
padding: 0;
|
|
13
|
+
margin: 0;
|
|
14
|
+
list-style: none;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
.page-link {
|
|
18
|
+
display: inline-flex;
|
|
19
|
+
align-items: center;
|
|
20
|
+
justify-content: center;
|
|
21
|
+
min-width: 2.4rem;
|
|
22
|
+
height: 2.4rem;
|
|
23
|
+
padding: 0 spacer(2);
|
|
24
|
+
color: var(--fv-body-color);
|
|
25
|
+
text-decoration: none;
|
|
26
|
+
background-color: var(--fv-glass-bg);
|
|
27
|
+
border: $border-width solid var(--fv-glass-border);
|
|
28
|
+
border-radius: map.get($radii, 'md');
|
|
29
|
+
transition: background-color $transition-base, border-color $transition-base;
|
|
30
|
+
|
|
31
|
+
&:hover {
|
|
32
|
+
background-color: var(--fv-glass-bg-strong);
|
|
33
|
+
text-decoration: none;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
&:focus-visible {
|
|
37
|
+
outline: 0;
|
|
38
|
+
box-shadow: var(--fv-focus-ring);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
.page-item.active .page-link {
|
|
43
|
+
color: #ffffff;
|
|
44
|
+
background-image: var(--fv-gradient-primary);
|
|
45
|
+
border-color: transparent;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
.page-item.disabled .page-link {
|
|
49
|
+
opacity: 0.45;
|
|
50
|
+
pointer-events: none;
|
|
51
|
+
}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Progress
|
|
3
|
+
// Glass track with a gradient fill. Striped/animated + per-color variants.
|
|
4
|
+
// =============================================================================
|
|
5
|
+
|
|
6
|
+
@use 'sass:map';
|
|
7
|
+
@use '../abstracts' as *;
|
|
8
|
+
|
|
9
|
+
.progress {
|
|
10
|
+
position: relative;
|
|
11
|
+
display: flex;
|
|
12
|
+
height: 0.625rem;
|
|
13
|
+
overflow: hidden;
|
|
14
|
+
background-color: var(--fv-glass-bg);
|
|
15
|
+
border: $border-width solid var(--fv-glass-border);
|
|
16
|
+
border-radius: map.get($radii, 'pill');
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
.progress-bar {
|
|
20
|
+
width: 0;
|
|
21
|
+
height: 100%;
|
|
22
|
+
background-image: var(--fv-gradient-primary);
|
|
23
|
+
border-radius: inherit;
|
|
24
|
+
transition: width 0.4s ease;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
.progress-lg { height: 1rem; }
|
|
28
|
+
.progress-sm { height: 0.35rem; }
|
|
29
|
+
|
|
30
|
+
// Solid color variants override the gradient fill.
|
|
31
|
+
@each $name, $clr in $theme-colors {
|
|
32
|
+
.progress-#{$name} .progress-bar {
|
|
33
|
+
background-image: none;
|
|
34
|
+
background-color: $clr;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Diagonal stripes (optionally animated).
|
|
39
|
+
.progress-striped .progress-bar {
|
|
40
|
+
background-image: linear-gradient(
|
|
41
|
+
45deg,
|
|
42
|
+
rgba(255, 255, 255, 0.18) 25%, transparent 25%,
|
|
43
|
+
transparent 50%, rgba(255, 255, 255, 0.18) 50%,
|
|
44
|
+
rgba(255, 255, 255, 0.18) 75%, transparent 75%, transparent
|
|
45
|
+
);
|
|
46
|
+
background-size: 1rem 1rem;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
.progress-animated .progress-bar {
|
|
50
|
+
animation: fv-progress-stripes 1s linear infinite;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
@keyframes fv-progress-stripes {
|
|
54
|
+
from { background-position: 1rem 0; }
|
|
55
|
+
to { background-position: 0 0; }
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
@media (prefers-reduced-motion: reduce) {
|
|
59
|
+
.progress-animated .progress-bar { animation: none; }
|
|
60
|
+
}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Range input
|
|
3
|
+
// <input type="range" class="form-range" min="0" max="100" value="60" />
|
|
4
|
+
// (Vendor pseudo-elements MUST be in separate rules — a browser drops the whole
|
|
5
|
+
// selector list if it sees one it doesn't recognise.)
|
|
6
|
+
// =============================================================================
|
|
7
|
+
|
|
8
|
+
@use 'sass:map';
|
|
9
|
+
@use '../abstracts' as *;
|
|
10
|
+
|
|
11
|
+
.form-range {
|
|
12
|
+
width: 100%;
|
|
13
|
+
height: 1.5rem;
|
|
14
|
+
padding: 0;
|
|
15
|
+
background: transparent;
|
|
16
|
+
-webkit-appearance: none;
|
|
17
|
+
appearance: none;
|
|
18
|
+
cursor: pointer;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
.form-range:focus { outline: 0; }
|
|
22
|
+
|
|
23
|
+
// Track
|
|
24
|
+
.form-range::-webkit-slider-runnable-track {
|
|
25
|
+
height: 0.4rem;
|
|
26
|
+
background-color: var(--fv-glass-bg-strong);
|
|
27
|
+
border: $border-width solid var(--fv-glass-border);
|
|
28
|
+
border-radius: map.get($radii, 'pill');
|
|
29
|
+
}
|
|
30
|
+
.form-range::-moz-range-track {
|
|
31
|
+
height: 0.4rem;
|
|
32
|
+
background-color: var(--fv-glass-bg-strong);
|
|
33
|
+
border: $border-width solid var(--fv-glass-border);
|
|
34
|
+
border-radius: map.get($radii, 'pill');
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Thumb
|
|
38
|
+
.form-range::-webkit-slider-thumb {
|
|
39
|
+
-webkit-appearance: none;
|
|
40
|
+
width: 1.1rem;
|
|
41
|
+
height: 1.1rem;
|
|
42
|
+
margin-top: -0.42rem;
|
|
43
|
+
background-image: var(--fv-gradient-primary);
|
|
44
|
+
border: 0;
|
|
45
|
+
border-radius: 50%;
|
|
46
|
+
box-shadow: var(--fv-shadow-sm);
|
|
47
|
+
}
|
|
48
|
+
.form-range::-moz-range-thumb {
|
|
49
|
+
width: 1.1rem;
|
|
50
|
+
height: 1.1rem;
|
|
51
|
+
background-image: var(--fv-gradient-primary);
|
|
52
|
+
border: 0;
|
|
53
|
+
border-radius: 50%;
|
|
54
|
+
box-shadow: var(--fv-shadow-sm);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.form-range:focus-visible::-webkit-slider-thumb { box-shadow: var(--fv-focus-ring); }
|
|
58
|
+
.form-range:focus-visible::-moz-range-thumb { box-shadow: var(--fv-focus-ring); }
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Rating
|
|
3
|
+
// <span class="rating" role="img" aria-label="Rated 4 out of 5">
|
|
4
|
+
// <svg class="icon"><use href="...#i-star"/></svg> x4
|
|
5
|
+
// <svg class="icon rating-empty"><use href="...#i-star"/></svg>
|
|
6
|
+
// </span>
|
|
7
|
+
// =============================================================================
|
|
8
|
+
|
|
9
|
+
@use '../abstracts' as *;
|
|
10
|
+
|
|
11
|
+
.rating {
|
|
12
|
+
display: inline-flex;
|
|
13
|
+
align-items: center;
|
|
14
|
+
gap: 0.1rem;
|
|
15
|
+
color: var(--fv-warning);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
.rating .icon {
|
|
19
|
+
width: 1.1em;
|
|
20
|
+
height: 1.1em;
|
|
21
|
+
fill: currentColor;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
.rating-empty {
|
|
25
|
+
color: var(--fv-glass-border);
|
|
26
|
+
}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Segmented control (radio-based, no JS)
|
|
3
|
+
// <div class="segmented">
|
|
4
|
+
// <input type="radio" name="range" id="seg-d" checked><label for="seg-d">Day</label>
|
|
5
|
+
// <input type="radio" name="range" id="seg-w"><label for="seg-w">Week</label>
|
|
6
|
+
// </div>
|
|
7
|
+
// =============================================================================
|
|
8
|
+
|
|
9
|
+
@use 'sass:map';
|
|
10
|
+
@use '../abstracts' as *;
|
|
11
|
+
|
|
12
|
+
.segmented {
|
|
13
|
+
display: inline-flex;
|
|
14
|
+
gap: 0.2rem;
|
|
15
|
+
padding: 0.25rem;
|
|
16
|
+
background-color: var(--fv-glass-bg);
|
|
17
|
+
border: $border-width solid var(--fv-glass-border);
|
|
18
|
+
border-radius: map.get($radii, 'lg');
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
.segmented input {
|
|
22
|
+
position: absolute;
|
|
23
|
+
width: 1px;
|
|
24
|
+
height: 1px;
|
|
25
|
+
opacity: 0;
|
|
26
|
+
pointer-events: none;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
.segmented label {
|
|
30
|
+
padding: spacer(1) spacer(4);
|
|
31
|
+
font-size: map.get($font-sizes, 'sm');
|
|
32
|
+
font-weight: map.get($font-weights, 'medium');
|
|
33
|
+
color: var(--fv-muted);
|
|
34
|
+
border-radius: map.get($radii, 'md');
|
|
35
|
+
cursor: pointer;
|
|
36
|
+
transition: color $transition-base, background-color $transition-base;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
.segmented label:hover {
|
|
40
|
+
color: var(--fv-body-color);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
.segmented input:checked + label {
|
|
44
|
+
color: var(--fv-body-color);
|
|
45
|
+
background-color: var(--fv-glass-bg-strong);
|
|
46
|
+
box-shadow: var(--fv-shadow-sm);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
.segmented input:focus-visible + label {
|
|
50
|
+
box-shadow: var(--fv-focus-ring);
|
|
51
|
+
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
// =============================================================================
|
|
2
|
+
// Farvist · Components · Skeleton
|
|
3
|
+
// Shimmering placeholders for loading states.
|
|
4
|
+
// =============================================================================
|
|
5
|
+
|
|
6
|
+
@use 'sass:map';
|
|
7
|
+
@use '../abstracts' as *;
|
|
8
|
+
|
|
9
|
+
.skeleton {
|
|
10
|
+
display: block;
|
|
11
|
+
background-image: linear-gradient(
|
|
12
|
+
100deg,
|
|
13
|
+
var(--fv-glass-bg) 30%,
|
|
14
|
+
var(--fv-glass-bg-strong) 50%,
|
|
15
|
+
var(--fv-glass-bg) 70%
|
|
16
|
+
);
|
|
17
|
+
background-size: 200% 100%;
|
|
18
|
+
border-radius: map.get($radii, 'md');
|
|
19
|
+
animation: fv-skeleton 1.4s ease-in-out infinite;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
.skeleton-text {
|
|
23
|
+
height: 0.8em;
|
|
24
|
+
margin-bottom: spacer(2);
|
|
25
|
+
border-radius: map.get($radii, 'sm');
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
.skeleton-title {
|
|
29
|
+
width: 50%;
|
|
30
|
+
height: 1.4em;
|
|
31
|
+
margin-bottom: spacer(3);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
.skeleton-avatar {
|
|
35
|
+
width: 3rem;
|
|
36
|
+
height: 3rem;
|
|
37
|
+
border-radius: 50%;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
.skeleton-btn {
|
|
41
|
+
width: 7rem;
|
|
42
|
+
height: 2.6rem;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
@keyframes fv-skeleton {
|
|
46
|
+
from { background-position: 200% 0; }
|
|
47
|
+
to { background-position: -200% 0; }
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
@media (prefers-reduced-motion: reduce) {
|
|
51
|
+
.skeleton { animation: none; }
|
|
52
|
+
}
|