@veluai/velu 0.1.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/dist/cli.js +11 -0
- package/package.json +52 -0
- package/runtime/velu-ui/base.css +311 -0
- package/runtime/velu-ui/components/Accordion.jsx +64 -0
- package/runtime/velu-ui/components/ApiClient.jsx +121 -0
- package/runtime/velu-ui/components/ApiField.jsx +87 -0
- package/runtime/velu-ui/components/ApiPath.jsx +63 -0
- package/runtime/velu-ui/components/ApiSidebar.jsx +122 -0
- package/runtime/velu-ui/components/AskBar.jsx +71 -0
- package/runtime/velu-ui/components/Callout.jsx +114 -0
- package/runtime/velu-ui/components/Card.jsx +131 -0
- package/runtime/velu-ui/components/Chatbot.jsx +596 -0
- package/runtime/velu-ui/components/CodeBlock.jsx +375 -0
- package/runtime/velu-ui/components/Columns.jsx +56 -0
- package/runtime/velu-ui/components/Field.jsx +81 -0
- package/runtime/velu-ui/components/Image.jsx +163 -0
- package/runtime/velu-ui/components/MethodBadge.jsx +31 -0
- package/runtime/velu-ui/components/NavSelect.jsx +108 -0
- package/runtime/velu-ui/components/PageFeedback.jsx +219 -0
- package/runtime/velu-ui/components/PageFooter.jsx +213 -0
- package/runtime/velu-ui/components/PageHeader.jsx +414 -0
- package/runtime/velu-ui/components/PageNav.jsx +77 -0
- package/runtime/velu-ui/components/PoweredBy.jsx +51 -0
- package/runtime/velu-ui/components/Prompt.jsx +115 -0
- package/runtime/velu-ui/components/Search.jsx +366 -0
- package/runtime/velu-ui/components/Sidebar.jsx +191 -0
- package/runtime/velu-ui/components/Steps.jsx +65 -0
- package/runtime/velu-ui/components/ThemeToggle.jsx +48 -0
- package/runtime/velu-ui/components/Toc.jsx +537 -0
- package/runtime/velu-ui/components/TocBar.jsx +195 -0
- package/runtime/velu-ui/components/Tree.jsx +87 -0
- package/runtime/velu-ui/components/TryItBar.jsx +90 -0
- package/runtime/velu-ui/components/accordion.css +92 -0
- package/runtime/velu-ui/components/api.css +479 -0
- package/runtime/velu-ui/components/ask-bar.css +94 -0
- package/runtime/velu-ui/components/card.css +105 -0
- package/runtime/velu-ui/components/chatbot.css +617 -0
- package/runtime/velu-ui/components/code-block.css +263 -0
- package/runtime/velu-ui/components/docs-layout.css +775 -0
- package/runtime/velu-ui/components/field.css +82 -0
- package/runtime/velu-ui/components/image.css +237 -0
- package/runtime/velu-ui/components/nav-select.css +157 -0
- package/runtime/velu-ui/components/page-feedback.css +241 -0
- package/runtime/velu-ui/components/page-footer.css +130 -0
- package/runtime/velu-ui/components/page-header.css +520 -0
- package/runtime/velu-ui/components/page-nav.css +50 -0
- package/runtime/velu-ui/components/powered-by.css +66 -0
- package/runtime/velu-ui/components/prompt.css +99 -0
- package/runtime/velu-ui/components/search.css +307 -0
- package/runtime/velu-ui/components/sidebar.css +144 -0
- package/runtime/velu-ui/components/steps.css +77 -0
- package/runtime/velu-ui/components/theme-toggle.css +70 -0
- package/runtime/velu-ui/components/toc-bar.css +234 -0
- package/runtime/velu-ui/components/tree.css +49 -0
- package/runtime/velu-ui/index.js +45 -0
- package/runtime/velu-ui/lib/copyText.js +64 -0
- package/runtime/velu-ui/lib/lang-icons.jsx +156 -0
- package/runtime/velu-ui/lib/prism-langs.js +957 -0
- package/runtime/velu-ui/lib/prism-loader.js +74 -0
- package/runtime/velu-ui/lib/resolveIcon.jsx +29 -0
- package/runtime/velu-ui/lib/scrollIntoNearestView.js +66 -0
- package/runtime/velu-ui/mdx-components.jsx +85 -0
- package/runtime/velu-ui/primitives/Cluster.jsx +49 -0
- package/runtime/velu-ui/primitives/Stack.jsx +63 -0
- package/runtime/velu-ui/primitives/Switcher.jsx +57 -0
- package/runtime/velu-ui/primitives/stack.css +3 -0
- package/runtime/velu-ui/primitives/switcher.css +25 -0
- package/runtime/velu-ui/styles.css +43 -0
- package/runtime/velu-ui/tokens.css +4 -0
- package/schema/velu.schema.json +167 -0
- package/src/navigation.js +434 -0
- package/src/runtime/App.jsx +1473 -0
- package/src/runtime/client-entry.jsx +22 -0
- package/src/runtime/server-entry.jsx +16 -0
- package/src/template.html +48 -0
- package/templates/starter/ai-tools/claude-code.mdx +26 -0
- package/templates/starter/ai-tools/cursor.mdx +17 -0
- package/templates/starter/api-reference/endpoint/create.mdx +24 -0
- package/templates/starter/api-reference/endpoint/get.mdx +27 -0
- package/templates/starter/api-reference/introduction.mdx +28 -0
- package/templates/starter/development.mdx +19 -0
- package/templates/starter/essentials/code.mdx +28 -0
- package/templates/starter/essentials/images.mdx +29 -0
- package/templates/starter/essentials/markdown.mdx +25 -0
- package/templates/starter/essentials/navigation.mdx +39 -0
- package/templates/starter/essentials/settings.mdx +30 -0
- package/templates/starter/favicon.svg +6 -0
- package/templates/starter/index.mdx +31 -0
- package/templates/starter/quickstart.mdx +31 -0
- package/templates/starter/velu.json +33 -0
|
@@ -0,0 +1,241 @@
|
|
|
1
|
+
/* PageFeedback — "Was this page helpful?" row. All values are tokens;
|
|
2
|
+
light/dark via [data-theme]. Row layout (flex/gap/wrap) is the
|
|
3
|
+
<Cluster> primitive; this file owns only typography + the buttons. */
|
|
4
|
+
|
|
5
|
+
.velu-feedback__question {
|
|
6
|
+
font-size: var(--f-h6);
|
|
7
|
+
color: var(--text-color);
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
.velu-feedback__btn {
|
|
11
|
+
position: relative; /* anchor for the heart burst */
|
|
12
|
+
display: inline-flex;
|
|
13
|
+
align-items: center;
|
|
14
|
+
gap: var(--s-3);
|
|
15
|
+
padding-block: var(--s-3);
|
|
16
|
+
padding-inline: var(--s-1);
|
|
17
|
+
background: transparent;
|
|
18
|
+
border: var(--border-width) solid var(--border-color);
|
|
19
|
+
border-radius: var(--radius-md);
|
|
20
|
+
color: var(--text-color);
|
|
21
|
+
font: inherit;
|
|
22
|
+
font-size: var(--f-h6);
|
|
23
|
+
font-weight: var(--weight-medium);
|
|
24
|
+
line-height: var(--lh-h6);
|
|
25
|
+
cursor: pointer;
|
|
26
|
+
transition:
|
|
27
|
+
border-color 0.12s ease,
|
|
28
|
+
background 0.12s ease,
|
|
29
|
+
color 0.12s ease;
|
|
30
|
+
}
|
|
31
|
+
.velu-feedback__btn svg {
|
|
32
|
+
flex: none;
|
|
33
|
+
/* Resting icon matches the label (full-contrast text color), not the
|
|
34
|
+
muted grey — accent is reserved for hover / chosen. */
|
|
35
|
+
color: var(--text-color);
|
|
36
|
+
}
|
|
37
|
+
.velu-feedback__btn:hover {
|
|
38
|
+
border-color: var(--accent-color);
|
|
39
|
+
}
|
|
40
|
+
.velu-feedback__btn:hover svg {
|
|
41
|
+
color: var(--accent-color);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/* The currently-picked button — accent border + tint, accent icon.
|
|
45
|
+
The vote stays changeable: clicking the other button moves it here. */
|
|
46
|
+
.velu-feedback__btn--chosen {
|
|
47
|
+
border-color: var(--accent-color);
|
|
48
|
+
background: color-mix(in srgb, var(--accent-color) 10%, transparent);
|
|
49
|
+
color: var(--accent-color);
|
|
50
|
+
}
|
|
51
|
+
.velu-feedback__btn--chosen svg {
|
|
52
|
+
color: var(--accent-color);
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
.velu-feedback__thanks {
|
|
56
|
+
font-size: var(--f-h6);
|
|
57
|
+
color: var(--muted-color);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/* ── "No" improvement form ──────────────────────────────────────────── */
|
|
61
|
+
/* Vertical rhythm is the <Stack> primitive; this owns only the chrome. */
|
|
62
|
+
.velu-feedback__form {
|
|
63
|
+
padding-block-start: var(--s0);
|
|
64
|
+
border-block-start: var(--border-width) solid var(--border-color);
|
|
65
|
+
}
|
|
66
|
+
.velu-feedback__form-title {
|
|
67
|
+
margin: 0;
|
|
68
|
+
font-size: var(--f-h4);
|
|
69
|
+
line-height: var(--lh-h4);
|
|
70
|
+
font-weight: var(--weight-medium);
|
|
71
|
+
color: var(--text-color);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/* Reason radio rows — native <input> visually hidden, circle drawn by
|
|
75
|
+
the adjacent mark span (so :checked can style it). */
|
|
76
|
+
.velu-feedback__radio {
|
|
77
|
+
position: relative;
|
|
78
|
+
display: flex;
|
|
79
|
+
align-items: center;
|
|
80
|
+
gap: var(--s-1);
|
|
81
|
+
cursor: pointer;
|
|
82
|
+
}
|
|
83
|
+
.velu-feedback__radio input {
|
|
84
|
+
position: absolute;
|
|
85
|
+
inset-block-start: 0;
|
|
86
|
+
inset-inline-start: 0;
|
|
87
|
+
inline-size: var(--icon-size-lg);
|
|
88
|
+
block-size: var(--icon-size-lg);
|
|
89
|
+
margin: 0;
|
|
90
|
+
opacity: 0;
|
|
91
|
+
cursor: pointer;
|
|
92
|
+
}
|
|
93
|
+
.velu-feedback__radio-mark {
|
|
94
|
+
position: relative;
|
|
95
|
+
flex: none;
|
|
96
|
+
inline-size: var(--icon-size-lg);
|
|
97
|
+
block-size: var(--icon-size-lg);
|
|
98
|
+
border: var(--border-width) solid var(--border-color);
|
|
99
|
+
border-radius: 50%;
|
|
100
|
+
transition: border-color 0.12s ease;
|
|
101
|
+
}
|
|
102
|
+
.velu-feedback__radio input:checked + .velu-feedback__radio-mark {
|
|
103
|
+
border-color: var(--accent-color);
|
|
104
|
+
}
|
|
105
|
+
.velu-feedback__radio input:checked + .velu-feedback__radio-mark::after {
|
|
106
|
+
content: '';
|
|
107
|
+
position: absolute;
|
|
108
|
+
inset: 22%;
|
|
109
|
+
background: var(--accent-color);
|
|
110
|
+
border-radius: 50%;
|
|
111
|
+
}
|
|
112
|
+
.velu-feedback__radio input:focus-visible + .velu-feedback__radio-mark {
|
|
113
|
+
outline: var(--border-width) solid var(--accent-color);
|
|
114
|
+
outline-offset: 2px;
|
|
115
|
+
}
|
|
116
|
+
.velu-feedback__radio:hover .velu-feedback__radio-mark {
|
|
117
|
+
border-color: var(--accent-color);
|
|
118
|
+
}
|
|
119
|
+
.velu-feedback__radio-label {
|
|
120
|
+
font-size: var(--f-body);
|
|
121
|
+
color: var(--text-color);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
/* Optional free-text detail */
|
|
125
|
+
.velu-feedback__detail {
|
|
126
|
+
inline-size: 100%;
|
|
127
|
+
box-sizing: border-box;
|
|
128
|
+
padding: var(--s-1);
|
|
129
|
+
background: var(--page-bg);
|
|
130
|
+
border: var(--border-width) solid var(--border-color);
|
|
131
|
+
border-radius: var(--radius-md);
|
|
132
|
+
resize: vertical;
|
|
133
|
+
font: inherit;
|
|
134
|
+
font-size: var(--f-h6);
|
|
135
|
+
line-height: var(--lh-body);
|
|
136
|
+
color: var(--text-color);
|
|
137
|
+
}
|
|
138
|
+
.velu-feedback__detail::placeholder {
|
|
139
|
+
color: var(--muted-color);
|
|
140
|
+
}
|
|
141
|
+
.velu-feedback__detail:focus-visible {
|
|
142
|
+
outline: none;
|
|
143
|
+
border-color: var(--accent-color);
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/* Cancel / Submit actions */
|
|
147
|
+
.velu-feedback__action {
|
|
148
|
+
padding-block: var(--s-2);
|
|
149
|
+
padding-inline: var(--s0);
|
|
150
|
+
border-radius: var(--radius-md);
|
|
151
|
+
font: inherit;
|
|
152
|
+
font-size: var(--f-h6);
|
|
153
|
+
font-weight: var(--weight-medium);
|
|
154
|
+
cursor: pointer;
|
|
155
|
+
transition: border-color 0.12s ease, background 0.12s ease;
|
|
156
|
+
}
|
|
157
|
+
.velu-feedback__action--cancel {
|
|
158
|
+
background: transparent;
|
|
159
|
+
border: var(--border-width) solid var(--border-color);
|
|
160
|
+
color: var(--text-color);
|
|
161
|
+
}
|
|
162
|
+
.velu-feedback__action--cancel:hover {
|
|
163
|
+
border-color: var(--accent-color);
|
|
164
|
+
}
|
|
165
|
+
.velu-feedback__action--submit {
|
|
166
|
+
background: var(--accent-color);
|
|
167
|
+
border: var(--border-width) solid var(--accent-color);
|
|
168
|
+
color: #fff;
|
|
169
|
+
}
|
|
170
|
+
.velu-feedback__action--submit:hover {
|
|
171
|
+
background: color-mix(in srgb, var(--accent-color) 88%, #000);
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
/* ── Submitted confirmation — sits below the divider line ───────────── */
|
|
175
|
+
.velu-feedback__sent {
|
|
176
|
+
padding-block-start: var(--s0);
|
|
177
|
+
border-block-start: var(--border-width) solid var(--border-color);
|
|
178
|
+
}
|
|
179
|
+
.velu-feedback__sent-title {
|
|
180
|
+
font-size: var(--f-body);
|
|
181
|
+
font-weight: var(--weight-medium);
|
|
182
|
+
color: var(--text-color);
|
|
183
|
+
}
|
|
184
|
+
.velu-feedback__sent-note {
|
|
185
|
+
font-size: var(--f-h6);
|
|
186
|
+
color: var(--muted-color);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
/* ── Heart burst (on "Yes") ─────────────────────────────────────────── */
|
|
190
|
+
/* Zero-size anchor pinned to the top-center of the Yes button; hearts
|
|
191
|
+
are absolutely positioned relative to it and fan out from (0,0). */
|
|
192
|
+
.velu-feedback__hearts {
|
|
193
|
+
position: absolute;
|
|
194
|
+
inset-block-start: 0;
|
|
195
|
+
inset-inline-start: 50%;
|
|
196
|
+
inline-size: 0;
|
|
197
|
+
block-size: 0;
|
|
198
|
+
pointer-events: none;
|
|
199
|
+
}
|
|
200
|
+
.velu-feedback__heart {
|
|
201
|
+
position: absolute;
|
|
202
|
+
inset-block-start: 0;
|
|
203
|
+
inset-inline-start: 0;
|
|
204
|
+
inline-size: var(--s-1);
|
|
205
|
+
block-size: var(--s-1);
|
|
206
|
+
/* --hb-tx / --hb-ty / --hb-rot are set per-heart inline. */
|
|
207
|
+
animation: velu-feedback-hearts 0.8s ease-out forwards;
|
|
208
|
+
}
|
|
209
|
+
.velu-feedback__heart path {
|
|
210
|
+
fill: currentColor;
|
|
211
|
+
}
|
|
212
|
+
/* Festive shades, all derived from the brand accent. */
|
|
213
|
+
.velu-feedback__heart--0 {
|
|
214
|
+
color: var(--accent-color);
|
|
215
|
+
}
|
|
216
|
+
.velu-feedback__heart--1 {
|
|
217
|
+
color: color-mix(in srgb, var(--accent-color) 60%, #fff);
|
|
218
|
+
}
|
|
219
|
+
.velu-feedback__heart--2 {
|
|
220
|
+
color: color-mix(in srgb, var(--accent-color) 55%, var(--text-color));
|
|
221
|
+
}
|
|
222
|
+
.velu-feedback__heart--3 {
|
|
223
|
+
color: color-mix(in srgb, var(--accent-color) 82%, #fff);
|
|
224
|
+
}
|
|
225
|
+
@keyframes velu-feedback-hearts {
|
|
226
|
+
0% {
|
|
227
|
+
transform: translate(-50%, 0) rotate(0deg) scale(0.6);
|
|
228
|
+
opacity: 1;
|
|
229
|
+
}
|
|
230
|
+
100% {
|
|
231
|
+
transform: translate(calc(-50% + var(--hb-tx)), var(--hb-ty))
|
|
232
|
+
rotate(var(--hb-rot)) scale(1);
|
|
233
|
+
opacity: 0;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
@media (prefers-reduced-motion: reduce) {
|
|
238
|
+
.velu-feedback__hearts {
|
|
239
|
+
display: none;
|
|
240
|
+
}
|
|
241
|
+
}
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
/* PageFooter — docs site footer. Vertical rhythm + columns / socials
|
|
2
|
+
are composed from Stack / Switcher / Cluster primitives in the JSX;
|
|
3
|
+
this file owns only chrome (typography, link colors, icon sizing).
|
|
4
|
+
Light / dark via [data-theme] via the velu-ui tokens. */
|
|
5
|
+
|
|
6
|
+
.velu-footer {
|
|
7
|
+
padding-block: var(--s2);
|
|
8
|
+
padding-inline: var(--s3);
|
|
9
|
+
/* Matches the header — same subtle hairline (--surface-color
|
|
10
|
+
resolves to #f2f3f4 in light / #323334 in dark). */
|
|
11
|
+
border-block-start: var(--border-width) solid var(--surface-color);
|
|
12
|
+
color: var(--text-color);
|
|
13
|
+
/* Opaque so the footer visually eclipses the fixed sidebar / TOC
|
|
14
|
+
when the page scrolls into it (DocsPage gives the footer a higher
|
|
15
|
+
z-index than the asides). */
|
|
16
|
+
background: var(--page-bg);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/* ── Container — wraps the brand row + columns row, centered in the
|
|
20
|
+
footer. fit-content sizes it to the wider of the two rows; margin
|
|
21
|
+
auto centers that block within the footer's padding box. */
|
|
22
|
+
.velu-footer__container {
|
|
23
|
+
inline-size: fit-content;
|
|
24
|
+
max-inline-size: 100%;
|
|
25
|
+
margin-inline: auto;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/* Columns row — flex-wrap with split gaps. column-gap is fluid via
|
|
29
|
+
clamp(): scales with viewport width between a floor (--s4) and a
|
|
30
|
+
ceiling (calc(var(--s5) * 7)) so it shrinks gracefully on narrow
|
|
31
|
+
screens. row-gap is small (used only when columns wrap onto a new
|
|
32
|
+
line). */
|
|
33
|
+
.velu-footer__row--columns {
|
|
34
|
+
display: flex;
|
|
35
|
+
flex-wrap: wrap;
|
|
36
|
+
align-items: flex-start;
|
|
37
|
+
column-gap: clamp(var(--s4), 12vw, calc(var(--s5) * 7));
|
|
38
|
+
row-gap: var(--s2);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/* ── Brand row ──────────────────────────────────────────────────────── */
|
|
42
|
+
/* Brand sits on its own row, hugged to the inline-start (align-self:
|
|
43
|
+
flex-start stops the outer column-Stack from stretching it across
|
|
44
|
+
the full row width). */
|
|
45
|
+
.velu-footer__brand,
|
|
46
|
+
.velu-footer__brand-link {
|
|
47
|
+
align-self: flex-start;
|
|
48
|
+
color: var(--text-color);
|
|
49
|
+
}
|
|
50
|
+
.velu-footer__brand-link {
|
|
51
|
+
text-decoration: none;
|
|
52
|
+
color: inherit;
|
|
53
|
+
}
|
|
54
|
+
.velu-footer__mark {
|
|
55
|
+
flex: none;
|
|
56
|
+
inline-size: calc(var(--icon-size-lg) * (32 / 24));
|
|
57
|
+
block-size: var(--icon-size-lg);
|
|
58
|
+
color: var(--accent-color);
|
|
59
|
+
}
|
|
60
|
+
/* Outfit Light, per the brand spec — falls back to --font-sans. */
|
|
61
|
+
.velu-footer__wordmark {
|
|
62
|
+
font-family: var(--font-brand);
|
|
63
|
+
font-weight: var(--weight-light);
|
|
64
|
+
font-size: var(--f-h4);
|
|
65
|
+
line-height: 1;
|
|
66
|
+
letter-spacing: -0.01em;
|
|
67
|
+
color: var(--text-color);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/* ── Link columns ───────────────────────────────────────────────────── */
|
|
71
|
+
.velu-footer__column-title {
|
|
72
|
+
font-size: var(--f-h6);
|
|
73
|
+
font-weight: var(--weight-medium);
|
|
74
|
+
color: var(--text-color);
|
|
75
|
+
}
|
|
76
|
+
.velu-footer__list {
|
|
77
|
+
list-style: none;
|
|
78
|
+
margin: 0;
|
|
79
|
+
padding: 0;
|
|
80
|
+
}
|
|
81
|
+
/* Footer text is always full-contrast (--text-color) — no muted greys.
|
|
82
|
+
Links shift to the accent color on hover; no underline.
|
|
83
|
+
line-height: 1.4 overrides the body's 1.75 so menu links don't sit
|
|
84
|
+
in a tall box (kept the columns from looking spread out). */
|
|
85
|
+
.velu-footer__link {
|
|
86
|
+
display: inline-block;
|
|
87
|
+
font-size: var(--f-h6);
|
|
88
|
+
font-weight: var(--weight-light);
|
|
89
|
+
line-height: 1.4;
|
|
90
|
+
color: var(--text-color);
|
|
91
|
+
text-decoration: none;
|
|
92
|
+
transition: color 0.12s ease;
|
|
93
|
+
}
|
|
94
|
+
.velu-footer__link:hover {
|
|
95
|
+
color: var(--accent-color);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/* ── Social row ─────────────────────────────────────────────────────── */
|
|
99
|
+
/* 2em × 2em (32×32 at 16px base) buttons; the glyph fills ~1.25em so a
|
|
100
|
+
little breathing room remains around it. Each kind gets its OWN
|
|
101
|
+
brand color (set via the data-kind attribute below). Monochrome
|
|
102
|
+
brands (github, x) inherit --text-color so they theme correctly. */
|
|
103
|
+
.velu-footer__social {
|
|
104
|
+
display: inline-flex;
|
|
105
|
+
align-items: center;
|
|
106
|
+
justify-content: center;
|
|
107
|
+
flex: none;
|
|
108
|
+
inline-size: 2em;
|
|
109
|
+
block-size: 2em;
|
|
110
|
+
padding: 0;
|
|
111
|
+
color: var(--text-color);
|
|
112
|
+
text-decoration: none;
|
|
113
|
+
transition: opacity 0.12s ease;
|
|
114
|
+
}
|
|
115
|
+
.velu-footer__social:hover {
|
|
116
|
+
opacity: 0.78;
|
|
117
|
+
}
|
|
118
|
+
.velu-footer__social svg {
|
|
119
|
+
inline-size: var(--icon-size-lg);
|
|
120
|
+
block-size: var(--icon-size-lg);
|
|
121
|
+
}
|
|
122
|
+
/* Brand colors — YouTube red, LinkedIn blue. GitHub + X stay
|
|
123
|
+
--text-color (their canonical brand is monochrome and follows the
|
|
124
|
+
surrounding theme). */
|
|
125
|
+
.velu-footer__social[data-kind='youtube'] {
|
|
126
|
+
color: #ff0000;
|
|
127
|
+
}
|
|
128
|
+
.velu-footer__social[data-kind='linkedin'] {
|
|
129
|
+
color: #0a66c2;
|
|
130
|
+
}
|