283 lines
7.1 KiB
Vue
283 lines
7.1 KiB
Vue
<template>
|
|
<label class="field-wrap" :for="id" :class="`field-wrap-${size}`">
|
|
<span v-if="label" class="field-label">
|
|
{{ label }}
|
|
<span v-show="required" class="text-red-800"> *</span>
|
|
</span>
|
|
<div class="real-field-wrap" :class="`size-${size}`">
|
|
<input :name="name" :type="type" :value="transformedValue" @input="onInput" @change="onChange" :disabled="disabled" :placeholder="placeholder" @focus="onFocus" @blur="onBlur">
|
|
<div v-if="hint" class="info-wrap">
|
|
<div v-tooltip="hint">
|
|
<sprite src="info-button" class="info-button"></sprite>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</label>
|
|
</template>
|
|
|
|
<script>
|
|
import wNumb from 'wnumb';
|
|
|
|
var numb = {
|
|
natural: wNumb({
|
|
mark: ',',
|
|
thousand: '.',
|
|
decimals: 0,
|
|
decoder(a) {
|
|
return a * 100;
|
|
},
|
|
encoder(a) {
|
|
return a / 100;
|
|
}
|
|
}),
|
|
naturalRaw: wNumb({
|
|
mark: '',
|
|
thousand: '',
|
|
decimals: 0,
|
|
decoder(a) {
|
|
return a * 100;
|
|
},
|
|
encoder(a) {
|
|
return a / 100;
|
|
}
|
|
}),
|
|
naturalDetailRaw: wNumb({
|
|
mark: '',
|
|
thousand: '',
|
|
decimals: 0,
|
|
decoder(a) {
|
|
return a * 10000;
|
|
},
|
|
encoder(a) {
|
|
return a / 10000;
|
|
}
|
|
}),
|
|
area: wNumb({
|
|
mark: ',',
|
|
thousand: '.',
|
|
decimals: 2,
|
|
decoder(a) {
|
|
return a * 100;
|
|
},
|
|
encoder(a) {
|
|
return a / 100;
|
|
}
|
|
}),
|
|
areaDetail: wNumb({
|
|
mark: ',',
|
|
thousand: '.',
|
|
decimals: 4,
|
|
decoder(a) {
|
|
return a * 10000;
|
|
},
|
|
encoder(a) {
|
|
return a / 10000;
|
|
}
|
|
}),
|
|
twoDecimalRaw: wNumb({
|
|
mark: ',',
|
|
thousand: '',
|
|
decimals: 2,
|
|
decoder(a) {
|
|
return a * 100;
|
|
},
|
|
encoder(a) {
|
|
return a / 100;
|
|
}
|
|
}),
|
|
fourDecimalRaw: wNumb({
|
|
mark: ',',
|
|
thousand: '',
|
|
decimals: 4,
|
|
decoder(a) {
|
|
return a * 10000;
|
|
},
|
|
encoder(a) {
|
|
return a / 10000;
|
|
}
|
|
})
|
|
};
|
|
|
|
var transformers = {
|
|
none: {
|
|
display: {
|
|
to(v) { return v; },
|
|
from(v) { return v; }
|
|
},
|
|
edit: {
|
|
to(v) { return v; },
|
|
from(v) { return v; }
|
|
}
|
|
},
|
|
natural: {
|
|
display: {
|
|
to(v) { return isNaN(parseInt(v)) ? '' : numb.natural.to(v); },
|
|
from(v) { return v === '' ? null : numb.natural.from(v); }
|
|
},
|
|
edit: {
|
|
to(v) { return isNaN(parseInt(v)) ? '' : numb.naturalRaw.to(v); },
|
|
from(v) { return v === '' ? null : numb.naturalRaw.from(v); }
|
|
}
|
|
},
|
|
area: {
|
|
display: {
|
|
to(v) { return v === null ? '' : numb.area.to(v); },
|
|
from(v) { return v === '' ? null : numb.area.from(v); }
|
|
},
|
|
edit: {
|
|
to(v) {
|
|
if (v === null) { return ''; }
|
|
if (Math.round(v / 100) * 100 === v) { return numb.naturalRaw.to(v); }
|
|
return numb.twoDecimalRaw.to(v);
|
|
},
|
|
from(v) {
|
|
if (v === '') { return null; }
|
|
if (v.indexOf(',') === -1) { return numb.naturalRaw.from(v); }
|
|
|
|
return numb.twoDecimalRaw.from(v);
|
|
}
|
|
}
|
|
},
|
|
currency: {
|
|
display: {
|
|
to(v) { return v === null ? '' : numb.area.to(v); },
|
|
from(v) { return v === '' ? null : numb.area.from(v); }
|
|
},
|
|
edit: {
|
|
to(v) {
|
|
if (v === null) { return ''; }
|
|
if (Math.round(v / 100) * 100 === v) { return numb.naturalRaw.to(v); }
|
|
return numb.twoDecimalRaw.to(v);
|
|
},
|
|
from(v) {
|
|
if (v === '') { return null; }
|
|
if (v.indexOf(',') === -1) { return numb.naturalRaw.from(v); }
|
|
|
|
return numb.twoDecimalRaw.from(v);
|
|
}
|
|
}
|
|
},
|
|
currencyDetail: {
|
|
display: {
|
|
to(v) { return v === null ? '' : numb.areaDetail.to(v); },
|
|
from(v) { return v === '' ? null : numb.areaDetail.from(v); }
|
|
},
|
|
edit: {
|
|
to(v) {
|
|
if (v === null) { return ''; }
|
|
if (Math.round(v / 10000) * 10000 === v) { return numb.naturalDetailRaw.to(v); }
|
|
return numb.fourDecimalRaw.to(v);
|
|
},
|
|
from(v) {
|
|
if (v === '') { return null; }
|
|
if (v.indexOf(',') === -1) { return numb.naturalDetailRaw.from(v); }
|
|
|
|
return numb.fourDecimalRaw.from(v);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
export default {
|
|
data: function() {
|
|
return {
|
|
focus: false
|
|
};
|
|
},
|
|
props: {
|
|
placeholder: {
|
|
default: function() {
|
|
return '';
|
|
}
|
|
},
|
|
default: {},
|
|
mode: {
|
|
default: function() { return 'none'; }
|
|
},
|
|
required: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
inset: {
|
|
default: function() {
|
|
return null;
|
|
}
|
|
},
|
|
size: {
|
|
default: function() {
|
|
return 'base';
|
|
}
|
|
},
|
|
id: {
|
|
required: true
|
|
},
|
|
hint: {
|
|
default: null
|
|
},
|
|
value: {
|
|
default: undefined
|
|
},
|
|
mask: {
|
|
default: undefined
|
|
},
|
|
label: {
|
|
default: false
|
|
},
|
|
type: {
|
|
required: false,
|
|
default: function() { return 'text'; }
|
|
},
|
|
disabled: {
|
|
default: false,
|
|
type: Boolean
|
|
},
|
|
name: {},
|
|
},
|
|
methods: {
|
|
onFocus() {
|
|
this.focus = true;
|
|
},
|
|
onBlur() {
|
|
this.focus = false;
|
|
},
|
|
onChange(v) {
|
|
if (this.mode !== 'none') {
|
|
this.transformedValue = v.target.value;
|
|
}
|
|
},
|
|
onInput(v) {
|
|
if (this.mode === 'none') {
|
|
this.transformedValue = v.target.value;
|
|
}
|
|
}
|
|
},
|
|
computed: {
|
|
transformedValue: {
|
|
get() {
|
|
return transformers[this.mode][this.focus ? 'edit' : 'display'].to(this.value);
|
|
},
|
|
set(v) {
|
|
this.$emit('input', transformers[this.mode][this.focus ? 'edit' : 'display'].from(v));
|
|
}
|
|
},
|
|
insetClass() {
|
|
if (this.inset === '') { return 'bg-inset'; }
|
|
if (this.inset === undefined) { return null; }
|
|
|
|
return `bg-${this.inset}`;
|
|
}
|
|
},
|
|
created() {
|
|
if (typeof this.value === 'undefined') {
|
|
this.$emit('input', this.default === undefined ? '' : this.default);
|
|
}
|
|
}
|
|
};
|
|
</script>
|
|
|
|
<style scope>
|
|
.bg-inset {
|
|
background: linear-gradient(to bottom, hsl(247.5, 66.7%, 97.6%) 0%, hsl(247.5, 66.7%, 97.6%) 41%, hsl(0deg 0% 100%) 41%, hsl(180deg 0% 100%) 100%);
|
|
}
|
|
</style>
|