Skip to content
Snippets Groups Projects
MkInput.vue 5.49 KiB
Newer Older
syuilo's avatar
syuilo committed
<template>
syuilo's avatar
syuilo committed
<div class="matxzzsk">
	<div class="label" @click="focus"><slot name="label"></slot></div>
	<div class="input" :class="{ inline, disabled, focused }">
syuilo's avatar
syuilo committed
		<div ref="prefixEl" class="prefix"><slot name="prefix"></slot></div>
syuilo's avatar
syuilo committed
			v-model="v"
syuilo's avatar
syuilo committed
			v-adaptive-border
syuilo's avatar
syuilo committed
			:type="type"
syuilo's avatar
syuilo committed
			:disabled="disabled"
			:required="required"
			:readonly="readonly"
			:placeholder="placeholder"
			:pattern="pattern"
			:autocomplete="autocomplete"
			:spellcheck="spellcheck"
			:step="step"
syuilo's avatar
syuilo committed
			:list="id"
syuilo's avatar
syuilo committed
			@focus="focused = true"
			@blur="focused = false"
			@keydown="onKeydown($event)"
			@input="onInput"
		>
syuilo's avatar
syuilo committed
		<datalist v-if="datalist" :id="id">
syuilo's avatar
syuilo committed
			<option v-for="data in datalist" :value="data"/>
		</datalist>
syuilo's avatar
syuilo committed
		<div ref="suffixEl" class="suffix"><slot name="suffix"></slot></div>
syuilo's avatar
syuilo committed
	</div>
syuilo's avatar
syuilo committed
	<div class="caption"><slot name="caption"></slot></div>
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
	<MkButton v-if="manualSave && changed" primary class="save" @click="updated"><i class="ti ti-check"></i> {{ i18n.ts.save }}</MkButton>
syuilo's avatar
syuilo committed
</div>
syuilo's avatar
syuilo committed
</template>

<script lang="ts" setup>
import { onMounted, onUnmounted, nextTick, ref, shallowRef, watch, computed, toRefs } from 'vue';
syuilo's avatar
syuilo committed
import { debounce } from 'throttle-debounce';
import MkButton from '@/components/MkButton.vue';
import { useInterval } from '@/scripts/use-interval';
syuilo's avatar
syuilo committed
import { i18n } from '@/i18n';
syuilo's avatar
syuilo committed

const props = defineProps<{
	modelValue: string | number;
syuilo's avatar
syuilo committed
	type?: 'text' | 'number' | 'password' | 'email' | 'url' | 'date' | 'time' | 'search';
	required?: boolean;
	readonly?: boolean;
	disabled?: boolean;
	pattern?: string;
	placeholder?: string;
	autofocus?: boolean;
	autocomplete?: boolean;
	spellcheck?: boolean;
	step?: any;
	datalist?: string[];
	inline?: boolean;
	debounce?: boolean;
	manualSave?: boolean;
	small?: boolean;
	large?: boolean;
}>();

const emit = defineEmits<{
	(ev: 'change', _ev: KeyboardEvent): void;
	(ev: 'keydown', _ev: KeyboardEvent): void;
	(ev: 'enter'): void;
	(ev: 'update:modelValue', value: string | number): void;
}>();

const { modelValue, type, autofocus } = toRefs(props);
const v = ref(modelValue.value);
const id = Math.random().toString(); // TODO: uuid?
const focused = ref(false);
const changed = ref(false);
const invalid = ref(false);
const filled = computed(() => v.value !== '' && v.value != null);
const inputEl = shallowRef<HTMLElement>();
const prefixEl = shallowRef<HTMLElement>();
const suffixEl = shallowRef<HTMLElement>();
const height =
syuilo's avatar
syuilo committed
	props.small ? 33 :
	props.large ? 39 :
	36;

const focus = () => inputEl.value.focus();
const onInput = (ev: KeyboardEvent) => {
	changed.value = true;
	emit('change', ev);
};
const onKeydown = (ev: KeyboardEvent) => {
	if (ev.isComposing || ev.key === 'Process' || ev.keyCode === 229) return;

	emit('keydown', ev);

	if (ev.code === 'Enter') {
		emit('enter');
	}
};

const updated = () => {
	changed.value = false;
	if (type.value === 'number') {
		emit('update:modelValue', parseFloat(v.value));
	} else {
		emit('update:modelValue', v.value);
	}
};
syuilo's avatar
syuilo committed

const debouncedUpdated = debounce(1000, updated);
syuilo's avatar
syuilo committed

watch(modelValue, newValue => {
	v.value = newValue;
});
syuilo's avatar
syuilo committed

watch(v, newValue => {
	if (!props.manualSave) {
		if (props.debounce) {
			debouncedUpdated();
		} else {
			updated();
		}
	}
syuilo's avatar
syuilo committed

	invalid.value = inputEl.value.validity.badInput;
});
// このコンポーネントが作成された時、非表示状態である場合がある
// 非表示状態だと要素の幅などは0になってしまうので、定期的に計算する
useInterval(() => {
	if (prefixEl.value) {
		if (prefixEl.value.offsetWidth) {
			inputEl.value.style.paddingLeft = prefixEl.value.offsetWidth + 'px';
		}
	}
	if (suffixEl.value) {
		if (suffixEl.value.offsetWidth) {
			inputEl.value.style.paddingRight = suffixEl.value.offsetWidth + 'px';
		}
	}
}, 100, {
	immediate: true,
	afterMounted: true,
});
syuilo's avatar
syuilo committed

onMounted(() => {
	nextTick(() => {
		if (autofocus.value) {
			focus();
		}
	});
syuilo's avatar
syuilo committed
});
</script>

syuilo's avatar
syuilo committed
.matxzzsk {
	> .label {
		font-size: 0.85em;
syuilo's avatar
syuilo committed
		padding: 0 0 8px 0;
syuilo's avatar
syuilo committed
		user-select: none;

		&:empty {
			display: none;
		}
	}

	> .caption {
syuilo's avatar
syuilo committed
		font-size: 0.85em;
		padding: 8px 0 0 0;
syuilo's avatar
syuilo committed
		color: var(--fgTransparentWeak);

		&:empty {
			display: none;
syuilo's avatar
syuilo committed
		}
	}

	> .input {
		position: relative;

		> input {
syuilo's avatar
syuilo committed
			appearance: none;
			-webkit-appearance: none;
syuilo's avatar
syuilo committed
			display: block;
			height: v-bind("height + 'px'");
syuilo's avatar
syuilo committed
			width: 100%;
			margin: 0;
syuilo's avatar
syuilo committed
			padding: 0 12px;
syuilo's avatar
syuilo committed
			font: inherit;
			font-weight: normal;
			font-size: 1em;
syuilo's avatar
syuilo committed
			color: var(--fg);
syuilo's avatar
syuilo committed
			background: var(--panel);
			border: solid 1px var(--panel);
syuilo's avatar
syuilo committed
			border-radius: 6px;
syuilo's avatar
syuilo committed
			outline: none;
			box-shadow: none;
			box-sizing: border-box;
syuilo's avatar
syuilo committed
			transition: border-color 0.1s ease-out;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
			&:hover {
syuilo's avatar
syuilo committed
				border-color: var(--inputBorderHover) !important;
syuilo's avatar
syuilo committed
			}
		}

		> .prefix,
		> .suffix {
syuilo's avatar
syuilo committed
			display: flex;
			align-items: center;
syuilo's avatar
syuilo committed
			position: absolute;
			z-index: 1;
			top: 0;
syuilo's avatar
syuilo committed
			padding: 0 12px;
syuilo's avatar
syuilo committed
			font-size: 1em;
			height: v-bind("height + 'px'");
syuilo's avatar
syuilo committed
			pointer-events: none;

			&:empty {
				display: none;
			}

			> * {
				display: inline-block;
				min-width: 16px;
				max-width: 150px;
syuilo's avatar
syuilo committed
				white-space: nowrap;
				text-overflow: ellipsis;
			}
		}

		> .prefix {
			left: 0;
syuilo's avatar
syuilo committed
			padding-right: 6px;
syuilo's avatar
syuilo committed
		}

		> .suffix {
			right: 0;
syuilo's avatar
syuilo committed
			padding-left: 6px;
syuilo's avatar
syuilo committed
		}

syuilo's avatar
syuilo committed
		&.inline {
			display: inline-block;
			margin: 0;
		}

		&.focused {
			> input {
syuilo's avatar
syuilo committed
				border-color: var(--accent) !important;
syuilo's avatar
syuilo committed
				//box-shadow: 0 0 0 4px var(--focus);
			}
		}
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
		&.disabled {
			opacity: 0.7;
syuilo's avatar
syuilo committed

syuilo's avatar
syuilo committed
			&, * {
				cursor: not-allowed !important;
			}
syuilo's avatar
syuilo committed
		}
	}
syuilo's avatar
syuilo committed

	> .save {
		margin: 8px 0 0 0;
	}
syuilo's avatar
syuilo committed
}
</style>