Web developers could style form elemets with CSS pseudo class & pseudo elements and other modern CSS properties.
HTML structure
<div class="input-set">
<select name="select-name">
<optgroup label="Group A">
<option value="1">option 1</option>
<option value="2">option 2</option>
<option value="3">option 3</option>
</optgroup>
<optgroup label="Group B">
<option value="4">option 4</option>
<option value="5">option 5</option>
<option value="6">option 6</option>
</optgroup>
</select>
<label class="input-set__label">
<span class="input-set__label__span">label</span>
</label>
<em class="input-set__em"></em>
</div>
Style Customization
Apply accent-color to <input />
's style.
<style>
.input-set {
--icon-warn: url() no-repeat 50% 50%/cover;
--icon-arrow-down: url() no-repeat 50% 50%/24px 24px;
--background-color: var(--default-background-color, rgba(var(--white)));
--text-color: var(--default-text-color, rgba(var(--batcave)));
--border-color: var(--default-border-color, rgba(var(--dolphin)));
--err-color: var(--default-err-color, rgba(var(--solo-cup)));
--err-text-color: var(--default-err-text-color, rgba(var(--dolphin)));
--label-text-color: var(--default-label-text-color, rgba(var(--dolphin)));
--theme: var(--default-theme, rgba(var(--dory)));
--hover: var(--theme);
/* readyonly, disabled */
--background-color-disabled: var(--default-background-color-disabled, rgba(var(--grey-hair)));
--text-color-disabled: var(--default-text-color-disabled, rgba(var(--bob)));
--border-color-disabled: var(--default-border-color-disabled, rgba(var(--bob)));
--placeholder-text-color-normal: transparent;
--placeholder-text-color-active: var(--default-placeholder-text-color, rgba(var(--charcoal)));
--placeholder-text-color: var(--placeholder-text-color-normal);
--border-radius: var(--default-border-radius, 4px);
--input-block-size: var(--default-input-block-size, 48px);
--input-line-height: var(--default-input-line-height, 46px);
--padding-inline-start: var(--default-padding-inline-start, 12px);
--padding-inline-end: var(--default-padding-inline-end, 12px);
--padding-inline: var(--padding-inline-start) var(--padding-inline-end);
--border-size-normal: 1px;
--border-size-active: 2px;
--border-size: var(--border-size-normal);
--err-display: none;
}
/* select */
.input-set--select select {
padding-inline-end: calc(var(--padding-inline-end) * 2 + 24px);
}
.input-set--select::after{position:absolute;inset-inline-end:var(--padding-inline-end);inset-block-start:12px;inline-size:24px;block-size:24px;content:'';background:var(--icon-arrow-down);pointer-events:none;}
/* blank */
.input-set--blank {
border-radius: var(--border-radius);
box-sizing: border-box;
background-color: var(--background-color);
box-shadow: inset calc(var(--border-size) * -1) calc(var(--border-size) * -1) 0 var(--border-color), inset var(--border-size) var(--border-size) 0 var(--border-color);
outline: 0 none;
}
.input-set--blank:focus-within {
--border-color: var(--hover);
--border-size: var(--border-size-active);
}
.input-set > :is(input, select, textarea)::-webkit-input-placeholder{color:var(--placeholder-text-color);transition:color 100ms ease;will-change:color;}
.input-set > :is(input, select, textarea)::-moz-placeholder{color:var(--placeholder-text-color);transition:color 100ms ease;will-change:color;}
.input-set{position:relative;display:block;}
.input-set > :is(input, select, textarea) {
font-size:1em;
inline-size:100%;
padding-inline:var(--padding-inline);
block-size:var(--input-block-size);
border-radius:var(--border-radius);
box-sizing:border-box;
display:block;
appearance:none;
border:0 none;
resize:none;
outline: 0 none;
line-height: var(--input-line-height);
color: var(--text-color);
background-color: var(--background-color);
box-shadow: inset calc(var(--border-size) * -1) calc(var(--border-size) * -1) 0 var(--border-color), inset var(--border-size) var(--border-size) 0 var(--border-color);
transition: box-shadow 100ms ease;
}
.input-set > :is(input, select, textarea):focus{outline: 0 none;}
.input-set__label{position:absolute;inset-inline-start:6px;inset-block-start:16px;inline-size:fit-content;font-size:1em;color:var(--label-text-color);line-height:1;padding-inline:6px;display:block;pointer-events:none;transition:transform 100ms ease,color 100ms ease,background-color 100ms ease;will-change:transform,color,background-color;}
.input-set__label__span{position:relative;z-index:1;}
.input-set__label::after{position:absolute;inset-inline-start:0;inset-block-start:50%;inline-size:100%;block-size:3.53px;background-color:var(--background-color);content:'';}
.input-set__em{font-size:14px;line-height:1;color:var(--err-text-color);padding-inline-start:var(--padding-inline-start);margin-block-start:8px;display:var(--err-display);align-items:center;gap:6px;}
.input-set__em::before{inline-size:14px;block-size:14px;background:var(--icon-warn);content:'';}
.input-set--hide-label {
--placeholder-text-color-normal: var(--placeholder-text-color-active);
}
.input-set--hide-label .input-set__label{display:none;}
/* focus */
.input-set > :is(input, select, textarea):focus {
--border-color: var(--hover);
--border-size: var(--border-size-active);
}
.input-set > :is(input, select, textarea):focus::-webkit-input-placeholder {
--placeholder-text-color: var(--placeholder-text-color-active);
}
.input-set > :is(input, select, textarea):focus::-moz-placeholder {
--placeholder-text-color: var(--placeholder-text-color-active);
}
.input-set > :is(input, select, textarea):focus ~ .input-set__label {
color: var(--hover);
transform: translateY(-24px) scale(0.85);
}
.input-set > :is(input, select, textarea):not(:placeholder-shown) ~ .input-set__label {
transform: translateY(-24px) scale(0.85);
}
/* invalid */
.input-set > :is(input, select, textarea):invalid~.input-set__em,
.input-set > :is(input, select, textarea)[invalid]~.input-set__em {
--err-display: flex;
}
.input-set > :is(input, select, textarea):invalid,
.input-set > :is(input, select, textarea)[invalid] {
--border-color: var(--err-color);
}
/* inert, readonly, disabled */
[inert] .input-set,
.input-set:has(:is(input, select, textarea)[readonly]),
.input-set:has(:is(input, select, textarea)[disabled]) {
--default-text-color: var(--text-color-disabled);
--default-border-color: var(--border-color-disabled);
--default-background-color: var(--background-color-disabled);
}
/* select */
.input-set--select select {
padding-inline-end: calc(var(--padding-inline-end) * 2 + 24px);
}
.input-set--select::after{position:absolute;inset-inline-end:var(--padding-inline-end);inset-block-start:50%;inline-size:24px;block-size:24px;content:'';margin-block-start:-12px;background:var(--icon-arrow-down);pointer-events:none;}
@media (hover: hover) {
.input-set:hover {
--border-color: var(--hover);
}
.input-set:hover > :is(input, select, textarea):not(:placeholder-shown)~.input-set__label {
--label-text-color: var(--hover);
}
}
@media (prefers-color-scheme: dark) {
.input-set {
--default-background-color: rgba(var(--batcave));
--default-text-color: rgba(var(--grey-hair));
--default-border-color: rgba(var(--charcoal));
--default-placeholder-text-color: rgba(var(--dolphin));
--default-label-text-color: rgba(var(--gandalf));
--default-placeholder-color: rgba(var(--dolphin));
--default-text-color-disabled: rgba(var(--shark));
--default-border-color-disabled: rgba(var(--charcoal));
--default-background-color-disabled: rgba(var(--batcave));
}
}
</style>
Events
Add input
and invalid
events to main form
to display error effect or not.
<script type="module">
const controller = new AbortController()
const { signal } = controller
const form = document.querySelector('form');
const transformErrorMessage = (inputEle) => {
let message = '';
const validity = inputEle.validity;
const {
validityDefault,
validityRangeOverflow,
validityRangeUnderflow,
validityValueMissing
} = inputEle.dataset;
switch (true) {
case validity.valueMissing:
message = validityValueMissing || '此為必填欄位';
break
case validity.rangeOverflow:
message = validityRangeOverflow || '內容已超過最大值';
break
case validity.rangeUnderflow:
message = validityRangeUnderflow || '內容低於最小值';
break
default:
// patternMismatch, badInput, customError, stepMismatch, tooLong, tooShort, typeMismatch
message = validityDefault || '格式錯誤,請再確認';
break
}
return message;
}
const onInput = (evt) => {
const select = evt.target.closest('select');
if (!select) {
return;
}
select.toggleAttribute('invalid', false);
};
const onInvalid = (evt) => {
evt.preventDefault();
const inputSet = evt.target.closest('.input-set');
const select = evt.target.closest('select');
if (!inputSet) {
return;
}
const errElement = inputSet.querySelector('.input-set__em');
errElement.textContent = transformErrorMessage(evt.target);
select.toggleAttribute('invalid', true);
}
// events
form.addEventListener('invalid', onInvalid, { signal, capture: true });
form.addEventListener('input', onInput, { signal });
</script>