web: Add timestamp display format to measure configuration.
This commit is contained in:
parent
cf69ff2fa1
commit
e9de9aebf3
8
web/package-lock.json
generated
8
web/package-lock.json
generated
@ -1059,6 +1059,14 @@
|
|||||||
"@types/lodash": "*"
|
"@types/lodash": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"@types/lodash.omit": {
|
||||||
|
"version": "4.5.6",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/lodash.omit/-/lodash.omit-4.5.6.tgz",
|
||||||
|
"integrity": "sha512-KXPpOSNX2h0DAG2w7ajpk7TXvWF28ZHs5nJhOJyP0BQHkehgr948RVsToItMme6oi0XJkp19CbuNXkIX8FiBlQ==",
|
||||||
|
"requires": {
|
||||||
|
"@types/lodash": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"@types/minimatch": {
|
"@types/minimatch": {
|
||||||
"version": "3.0.3",
|
"version": "3.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz",
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
"@types/lodash.assign": "^4.2.6",
|
"@types/lodash.assign": "^4.2.6",
|
||||||
"@types/lodash.findindex": "^4.6.6",
|
"@types/lodash.findindex": "^4.6.6",
|
||||||
"@types/lodash.merge": "^4.6.6",
|
"@types/lodash.merge": "^4.6.6",
|
||||||
|
"@types/lodash.omit": "^4.5.6",
|
||||||
"apexcharts": "^3.15.6",
|
"apexcharts": "^3.15.6",
|
||||||
"axios": "^0.18.1",
|
"axios": "^0.18.1",
|
||||||
"js-cookie": "^2.2.1",
|
"js-cookie": "^2.2.1",
|
||||||
|
@ -2,9 +2,11 @@
|
|||||||
<fieldset>
|
<fieldset>
|
||||||
<div>
|
<div>
|
||||||
<label for=measureType>Type</label>
|
<label for=measureType>Type</label>
|
||||||
|
<span v-if=measureExists>{{value.type}}</span>
|
||||||
<select
|
<select
|
||||||
:disabled=disabled
|
:disabled=disabled
|
||||||
name=measureType
|
name=measureType
|
||||||
|
v-if="!measureExists"
|
||||||
v-model=value.type>
|
v-model=value.type>
|
||||||
<option value=simple>Simple</option>
|
<option value=simple>Simple</option>
|
||||||
<option value=text>Text</option>
|
<option value=text>Text</option>
|
||||||
@ -14,6 +16,25 @@
|
|||||||
<label for=measureIsVisible>Show by default.</label>
|
<label for=measureIsVisible>Show by default.</label>
|
||||||
<input type=checkbox v-model=value.isVisible :disabled=disabled />
|
<input type=checkbox v-model=value.isVisible :disabled=disabled />
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for=timestampDisplayFormat>Timestamp Format</label>
|
||||||
|
<select
|
||||||
|
v-on:change=formatSelectionChanged
|
||||||
|
:disabled=disabled
|
||||||
|
v-model=selectedFormat
|
||||||
|
name=timestampDisplayFormat>
|
||||||
|
<option v-for="fmtStr in formatStrings"
|
||||||
|
:value=fmtStr>{{now.format(fmtStr)}}</option>
|
||||||
|
<option value="custom">Custom</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div v-if="selectedFormat === 'custom'">
|
||||||
|
<label for=timestampCustomDisplayFormat>
|
||||||
|
Custom Timestamp Format
|
||||||
|
(<a href="https://momentjs.com/docs/#/displaying/format/">see formatting options</a>)
|
||||||
|
</label>
|
||||||
|
<input type=text v-model=value.timestampDisplayFormat />
|
||||||
|
</div>
|
||||||
<TextMeasureConfigForm v-model=value v-show="value.type === 'text'" :disabled=disabled />
|
<TextMeasureConfigForm v-model=value v-show="value.type === 'text'" :disabled=disabled />
|
||||||
</fieldset>
|
</fieldset>
|
||||||
</template>
|
</template>
|
||||||
|
@ -2,6 +2,7 @@ import { Component, Emit, Prop, Vue, Watch } from 'vue-property-decorator';
|
|||||||
import { logService } from '@/services/logging';
|
import { logService } from '@/services/logging';
|
||||||
import { Measure, MeasureConfig } from '@/models';
|
import { Measure, MeasureConfig } from '@/models';
|
||||||
import TextMeasureConfigForm from './TextMeasureConfigForm.vue';
|
import TextMeasureConfigForm from './TextMeasureConfigForm.vue';
|
||||||
|
import moment from 'moment';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@ -11,12 +12,41 @@ import TextMeasureConfigForm from './TextMeasureConfigForm.vue';
|
|||||||
export class MeasureConfigForm extends Vue {
|
export class MeasureConfigForm extends Vue {
|
||||||
@Prop({}) public value!: MeasureConfig;
|
@Prop({}) public value!: MeasureConfig;
|
||||||
@Prop({}) public disabled: boolean = false;
|
@Prop({}) public disabled: boolean = false;
|
||||||
|
@Prop({}) public measureExists: boolean = false;
|
||||||
|
|
||||||
|
public now = moment();
|
||||||
|
public formatStrings = [
|
||||||
|
'l',
|
||||||
|
'L',
|
||||||
|
'll',
|
||||||
|
'LL',
|
||||||
|
'lll',
|
||||||
|
'LLL',
|
||||||
|
'llll',
|
||||||
|
'LLLL',
|
||||||
|
'Y-MM-DD',
|
||||||
|
'Y-MM-DDTHH:mm',
|
||||||
|
'Y-MM-DDTHH:mm:ss',
|
||||||
|
'Y-MM-DDTHH:mm:ss.SSSZZ',
|
||||||
|
'MM/DD',
|
||||||
|
'MMM Do',
|
||||||
|
'HH:mm',
|
||||||
|
'hh:mmA'
|
||||||
|
];
|
||||||
|
|
||||||
|
private selectedFormat: string = 'l';
|
||||||
|
|
||||||
@Watch('value', { immediate: true, deep: true })
|
@Watch('value', { immediate: true, deep: true })
|
||||||
@Emit('input')
|
@Emit('input')
|
||||||
private onConfigChanged(newVal: MeasureConfig, oldVal: MeasureConfig) {
|
private onConfigChanged(newVal: MeasureConfig, oldVal: MeasureConfig) {
|
||||||
return newVal;
|
return newVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private formatSelectionChanged() {
|
||||||
|
if (this.selectedFormat !== 'custom') {
|
||||||
|
this.value.timestampDisplayFormat = this.selectedFormat;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default MeasureConfigForm;
|
export default MeasureConfigForm;
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component, Prop, Vue } from 'vue-property-decorator';
|
||||||
import { Measure, MeasureConfig, MeasureType, Measurement, MeasurementMeta } from '@/models';
|
import { Measure, MeasureConfig, MeasureType, Measurement, MeasurementMeta } from '@/models';
|
||||||
import moment from 'moment';
|
|
||||||
import assign from 'lodash.assign';
|
import assign from 'lodash.assign';
|
||||||
import { library } from '@fortawesome/fontawesome-svg-core';
|
import { library } from '@fortawesome/fontawesome-svg-core';
|
||||||
import { faPencilAlt } from '@fortawesome/free-solid-svg-icons';
|
import { faPencilAlt } from '@fortawesome/free-solid-svg-icons';
|
||||||
import { byTimestampComparator } from '@/util';
|
import { byTimestampComparator, formatTS } from '@/util';
|
||||||
|
|
||||||
library.add(faPencilAlt);
|
library.add(faPencilAlt);
|
||||||
|
|
||||||
@ -13,8 +12,6 @@ export class SimpleDetails extends Vue {
|
|||||||
@Prop() private measure!: Measure<MeasureConfig>;
|
@Prop() private measure!: Measure<MeasureConfig>;
|
||||||
@Prop() private measurements!: Array<Measurement<MeasurementMeta>>;
|
@Prop() private measurements!: Array<Measurement<MeasurementMeta>>;
|
||||||
|
|
||||||
// private newMeasurement;
|
|
||||||
private moment = moment;
|
|
||||||
private chartOptions = {
|
private chartOptions = {
|
||||||
markers: { size: 6 },
|
markers: { size: 6 },
|
||||||
noData: { text: 'no data',
|
noData: { text: 'no data',
|
||||||
@ -37,7 +34,7 @@ export class SimpleDetails extends Vue {
|
|||||||
private get measurementTableData() {
|
private get measurementTableData() {
|
||||||
return (this.measurements || []).map((m) => {
|
return (this.measurements || []).map((m) => {
|
||||||
return assign({}, m, {
|
return assign({}, m, {
|
||||||
tsDisplay: moment(m.timestamp).format('MMM Do, HH:mm'),
|
tsDisplay: formatTS(this.measure, m),
|
||||||
tsSort: m.timestamp.toISOString()
|
tsSort: m.timestamp.toISOString()
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
1
web/src/models.d.ts
vendored
1
web/src/models.d.ts
vendored
@ -17,6 +17,7 @@ export interface LoginSubmit {
|
|||||||
export interface MeasureConfig {
|
export interface MeasureConfig {
|
||||||
type: MeasureType;
|
type: MeasureType;
|
||||||
isVisible: boolean;
|
isVisible: boolean;
|
||||||
|
timestampDisplayFormat: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface TextMeasureConfig extends MeasureConfig {
|
export interface TextMeasureConfig extends MeasureConfig {
|
||||||
|
@ -2,7 +2,8 @@
|
|||||||
|
|
||||||
button,
|
button,
|
||||||
.btn,
|
.btn,
|
||||||
.btn-action {
|
.btn-action,
|
||||||
|
.btn-icon {
|
||||||
border: 0;
|
border: 0;
|
||||||
border-radius: .25em;
|
border-radius: .25em;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@ -14,14 +15,27 @@ button,
|
|||||||
a { text-decoration: none; }
|
a { text-decoration: none; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn, .btn-icon { color: $fg-primary; }
|
||||||
|
|
||||||
|
.btn-icon {
|
||||||
|
|
||||||
|
border-radius: 1em;
|
||||||
|
padding: .5em;
|
||||||
|
margin: 0 .5em;
|
||||||
|
|
||||||
|
&:hover, &:focus {
|
||||||
|
background-color: darken($bg-primary, 20%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.btn-action {
|
.btn-action {
|
||||||
background-color: $color2;
|
background-color: $color2;
|
||||||
color: $color3;
|
color: $color3;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|
||||||
&:hover {
|
&:hover, &:focus {
|
||||||
background-color: darken($color2, 5%);
|
background-color: lighten($color2, 20%);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,25 @@
|
|||||||
import { Measurement, MeasurementMeta } from '@/models';
|
import { Measure, MeasureConfig, Measurement, MeasurementMeta } from '@/models';
|
||||||
|
import moment from 'moment';
|
||||||
|
|
||||||
export function byTimestampComparator<T extends MeasurementMeta>(
|
export function byTimestampComparator<T extends MeasurementMeta>(
|
||||||
a: Measurement<T>,
|
a: Measurement<T>,
|
||||||
b: Measurement<T>): number {
|
b: Measurement<T>): number {
|
||||||
return a.timestamp.getTime() - b.timestamp.getTime();
|
return a.timestamp.getTime() - b.timestamp.getTime();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function formatTS(
|
||||||
|
m: Measure<MeasureConfig>,
|
||||||
|
mm: Measurement<MeasurementMeta>
|
||||||
|
): string {
|
||||||
|
return moment(mm.timestamp).format(
|
||||||
|
m.config.timestampDisplayFormat || 'MMM Do');
|
||||||
|
}
|
||||||
|
|
||||||
|
export function slugify(s: string): string {
|
||||||
|
return s
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/[^\w\s\-]/g, '')
|
||||||
|
.replace(/\s+/g, '-');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,8 +1 @@
|
|||||||
@import '~@/styles/vars';
|
@import '~@/styles/vars';
|
||||||
|
|
||||||
.header-action .btn {
|
|
||||||
color: inherit;
|
|
||||||
margin-left: auto;
|
|
||||||
margin-right: 2rem;
|
|
||||||
background-color: none;
|
|
||||||
}
|
|
||||||
|
@ -19,7 +19,8 @@ export class NewMeasure extends Vue {
|
|||||||
id: '',
|
id: '',
|
||||||
config: {
|
config: {
|
||||||
type: 'simple' as MeasureType,
|
type: 'simple' as MeasureType,
|
||||||
isVisible: true
|
isVisible: true,
|
||||||
|
timestampDisplayFormat: 'l'
|
||||||
},
|
},
|
||||||
description: '',
|
description: '',
|
||||||
name: '',
|
name: '',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user