2023-03-16 23:18:55 +02:00
|
|
|
import ModifiedIndicator from './modified_indicator.vue'
|
|
|
|
|
import ProfileSettingIndicator from './profile_setting_indicator.vue'
|
|
|
|
|
import DraftButtons from './draft_buttons.vue'
|
2025-12-01 23:56:49 +02:00
|
|
|
import { get, set, cloneDeep, isEqual } from 'lodash'
|
2023-03-16 23:18:55 +02:00
|
|
|
|
2023-03-12 14:32:13 +02:00
|
|
|
export default {
|
2023-03-16 23:18:55 +02:00
|
|
|
components: {
|
|
|
|
|
ModifiedIndicator,
|
|
|
|
|
DraftButtons,
|
|
|
|
|
ProfileSettingIndicator
|
|
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
props: {
|
2024-09-24 03:07:27 +03:00
|
|
|
modelValue: {
|
|
|
|
|
type: String,
|
|
|
|
|
default: null
|
|
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
path: {
|
2023-03-20 23:36:47 +02:00
|
|
|
type: [String, Array],
|
2024-09-24 03:07:27 +03:00
|
|
|
required: false
|
2023-03-12 14:32:13 +02:00
|
|
|
},
|
2025-12-10 19:57:34 +02:00
|
|
|
showDescription: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
required: false
|
|
|
|
|
},
|
2025-12-08 17:09:07 +02:00
|
|
|
descriptionPathOverride: {
|
|
|
|
|
type: [String, Array],
|
|
|
|
|
required: false
|
|
|
|
|
},
|
2025-12-03 23:05:46 +02:00
|
|
|
suggestions: {
|
|
|
|
|
type: [String, Array],
|
|
|
|
|
required: false
|
|
|
|
|
},
|
2025-11-27 18:18:03 +02:00
|
|
|
subgroup: {
|
|
|
|
|
type: String,
|
|
|
|
|
required: false
|
|
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
disabled: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
parentPath: {
|
2023-03-20 23:36:47 +02:00
|
|
|
type: [String, Array]
|
2023-03-12 14:32:13 +02:00
|
|
|
},
|
|
|
|
|
parentInvert: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
|
|
|
|
},
|
|
|
|
|
expert: {
|
|
|
|
|
type: [Number, String],
|
|
|
|
|
default: 0
|
|
|
|
|
},
|
|
|
|
|
source: {
|
|
|
|
|
type: String,
|
2023-03-21 22:46:40 +02:00
|
|
|
default: undefined
|
2023-03-16 23:18:55 +02:00
|
|
|
},
|
2025-12-03 12:19:20 +02:00
|
|
|
hideDraftButtons: { // this is for the weird backend hybrid (Boolean|String or Boolean|Number) settings
|
2025-12-07 23:11:54 +02:00
|
|
|
required: false,
|
2025-12-03 12:19:20 +02:00
|
|
|
type: Boolean
|
|
|
|
|
},
|
|
|
|
|
hideLabel: {
|
|
|
|
|
type: Boolean
|
|
|
|
|
},
|
2023-04-12 23:58:21 +03:00
|
|
|
hideDescription: {
|
|
|
|
|
type: Boolean
|
|
|
|
|
},
|
|
|
|
|
swapDescriptionAndLabel: {
|
|
|
|
|
type: Boolean
|
|
|
|
|
},
|
2025-12-03 23:05:46 +02:00
|
|
|
backendDescriptionPath: {
|
|
|
|
|
type: [String, Array]
|
|
|
|
|
},
|
2023-04-12 23:58:21 +03:00
|
|
|
overrideBackendDescription: {
|
|
|
|
|
type: Boolean
|
|
|
|
|
},
|
|
|
|
|
overrideBackendDescriptionLabel: {
|
2025-12-10 19:57:34 +02:00
|
|
|
type: [Boolean, String]
|
2023-04-12 23:58:21 +03:00
|
|
|
},
|
2023-03-16 23:18:55 +02:00
|
|
|
draftMode: {
|
|
|
|
|
type: Boolean,
|
2023-03-21 22:46:40 +02:00
|
|
|
default: undefined
|
2024-05-22 19:54:19 +03:00
|
|
|
},
|
|
|
|
|
timedApplyMode: {
|
|
|
|
|
type: Boolean,
|
|
|
|
|
default: false
|
2023-03-21 22:46:40 +02:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
inject: {
|
|
|
|
|
defaultSource: {
|
|
|
|
|
default: 'default'
|
|
|
|
|
},
|
|
|
|
|
defaultDraftMode: {
|
2023-03-16 23:18:55 +02:00
|
|
|
default: false
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
data () {
|
|
|
|
|
return {
|
2023-03-22 12:43:53 +02:00
|
|
|
localDraft: null
|
2023-03-16 23:18:55 +02:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
created () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.realDraftMode && (this.realSource !== 'admin' || this.path == null)) {
|
2025-12-01 20:09:05 +02:00
|
|
|
this.draft = cloneDeep(this.state)
|
2023-03-12 14:32:13 +02:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
computed: {
|
2023-03-22 12:43:53 +02:00
|
|
|
draft: {
|
|
|
|
|
get () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.realSource === 'admin' || this.path == null) {
|
2023-03-29 23:26:57 +03:00
|
|
|
return get(this.$store.state.adminSettings.draft, this.canonPath)
|
2023-03-22 12:43:53 +02:00
|
|
|
} else {
|
|
|
|
|
return this.localDraft
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
set (value) {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.realSource === 'admin' || this.path == null) {
|
2023-03-22 12:43:53 +02:00
|
|
|
this.$store.commit('updateAdminDraft', { path: this.canonPath, value })
|
|
|
|
|
} else {
|
|
|
|
|
this.localDraft = value
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
state () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) {
|
|
|
|
|
return this.modelValue
|
|
|
|
|
}
|
2023-03-29 23:26:57 +03:00
|
|
|
const value = get(this.configSource, this.canonPath)
|
2023-03-12 14:32:13 +02:00
|
|
|
if (value === undefined) {
|
|
|
|
|
return this.defaultState
|
|
|
|
|
} else {
|
|
|
|
|
return value
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-03-22 00:00:52 +02:00
|
|
|
visibleState () {
|
|
|
|
|
return this.realDraftMode ? this.draft : this.state
|
|
|
|
|
},
|
2023-03-21 22:46:40 +02:00
|
|
|
realSource () {
|
|
|
|
|
return this.source || this.defaultSource
|
|
|
|
|
},
|
|
|
|
|
realDraftMode () {
|
|
|
|
|
return typeof this.draftMode === 'undefined' ? this.defaultDraftMode : this.draftMode
|
|
|
|
|
},
|
2023-03-19 21:27:07 +02:00
|
|
|
backendDescription () {
|
2025-11-27 18:18:03 +02:00
|
|
|
return get(this.$store.state.adminSettings.descriptions, this.descriptionPath)
|
2023-03-19 21:27:07 +02:00
|
|
|
},
|
|
|
|
|
backendDescriptionLabel () {
|
2023-04-12 23:58:21 +03:00
|
|
|
if (this.realSource !== 'admin') return ''
|
2025-12-08 22:29:47 +02:00
|
|
|
if (this.overrideBackendDescriptionLabel !== '' && typeof this.overrideBackendDescriptionLabel === 'string') {
|
|
|
|
|
return this.overrideBackendDescriptionLabel
|
|
|
|
|
}
|
2023-04-12 23:58:21 +03:00
|
|
|
if (!this.backendDescription || this.overrideBackendDescriptionLabel) {
|
|
|
|
|
return this.$t([
|
|
|
|
|
'admin_dash',
|
|
|
|
|
'temp_overrides',
|
|
|
|
|
...this.canonPath.map(p => p.replace(/\./g, '_DOT_')),
|
|
|
|
|
'label'
|
|
|
|
|
].join('.'))
|
|
|
|
|
} else {
|
|
|
|
|
return this.swapDescriptionAndLabel
|
|
|
|
|
? this.backendDescription?.description
|
|
|
|
|
: this.backendDescription?.label
|
|
|
|
|
}
|
2023-03-19 21:27:07 +02:00
|
|
|
},
|
|
|
|
|
backendDescriptionDescription () {
|
2025-12-10 19:57:34 +02:00
|
|
|
if (this.description) return this.description
|
2023-04-12 23:58:21 +03:00
|
|
|
if (this.realSource !== 'admin') return ''
|
|
|
|
|
if (this.hideDescription) return null
|
|
|
|
|
if (!this.backendDescription || this.overrideBackendDescription) {
|
|
|
|
|
return this.$t([
|
|
|
|
|
'admin_dash',
|
|
|
|
|
'temp_overrides',
|
|
|
|
|
...this.canonPath.map(p => p.replace(/\./g, '_DOT_')),
|
|
|
|
|
'description'
|
|
|
|
|
].join('.'))
|
|
|
|
|
} else {
|
|
|
|
|
return this.swapDescriptionAndLabel
|
|
|
|
|
? this.backendDescription?.label
|
|
|
|
|
: this.backendDescription?.description
|
|
|
|
|
}
|
2023-03-19 21:27:07 +02:00
|
|
|
},
|
2023-03-20 23:36:47 +02:00
|
|
|
backendDescriptionSuggestions () {
|
2025-12-03 23:05:46 +02:00
|
|
|
return this.backendDescription?.suggestions || this.suggestions
|
2023-03-20 23:36:47 +02:00
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
shouldBeDisabled () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) {
|
|
|
|
|
return this.disabled
|
|
|
|
|
}
|
2025-12-01 20:09:05 +02:00
|
|
|
let parentValue = null
|
|
|
|
|
if (this.parentPath !== undefined && this.realSource === 'admin') {
|
|
|
|
|
if (this.realDraftMode) {
|
|
|
|
|
parentValue = get(this.$store.state.adminSettings.draft, this.parentPath)
|
|
|
|
|
} else {
|
|
|
|
|
parentValue = get(this.configSource, this.parentPath)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-03-12 14:32:13 +02:00
|
|
|
return this.disabled || (parentValue !== null ? (this.parentInvert ? parentValue : !parentValue) : false)
|
|
|
|
|
},
|
|
|
|
|
configSource () {
|
2023-03-21 22:46:40 +02:00
|
|
|
switch (this.realSource) {
|
2023-03-12 14:32:13 +02:00
|
|
|
case 'profile':
|
|
|
|
|
return this.$store.state.profileConfig
|
2023-03-14 21:50:43 +02:00
|
|
|
case 'admin':
|
|
|
|
|
return this.$store.state.adminSettings.config
|
2023-03-12 14:32:13 +02:00
|
|
|
default:
|
|
|
|
|
return this.$store.getters.mergedConfig
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
configSink () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) {
|
2024-10-09 23:53:30 +03:00
|
|
|
return (k, v) => this.$emit('update:modelValue', v)
|
2024-09-24 03:07:27 +03:00
|
|
|
}
|
2023-03-21 22:46:40 +02:00
|
|
|
switch (this.realSource) {
|
2023-03-12 14:32:13 +02:00
|
|
|
case 'profile':
|
|
|
|
|
return (k, v) => this.$store.dispatch('setProfileOption', { name: k, value: v })
|
2023-03-14 21:50:43 +02:00
|
|
|
case 'admin':
|
2023-03-16 23:18:55 +02:00
|
|
|
return (k, v) => this.$store.dispatch('pushAdminSetting', { path: k, value: v })
|
2023-03-12 14:32:13 +02:00
|
|
|
default:
|
2024-05-22 19:54:19 +03:00
|
|
|
if (this.timedApplyMode) {
|
|
|
|
|
return (k, v) => this.$store.dispatch('setOptionTemporarily', { name: k, value: v })
|
|
|
|
|
} else {
|
|
|
|
|
return (k, v) => this.$store.dispatch('setOption', { name: k, value: v })
|
|
|
|
|
}
|
2023-03-12 14:32:13 +02:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
defaultState () {
|
2023-03-21 22:46:40 +02:00
|
|
|
switch (this.realSource) {
|
2023-03-12 14:32:13 +02:00
|
|
|
case 'profile':
|
|
|
|
|
return {}
|
|
|
|
|
default:
|
|
|
|
|
return get(this.$store.getters.defaultConfig, this.path)
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-03-12 17:11:20 +02:00
|
|
|
isProfileSetting () {
|
2023-03-21 22:46:40 +02:00
|
|
|
return this.realSource === 'profile'
|
2023-03-12 14:32:13 +02:00
|
|
|
},
|
|
|
|
|
isChanged () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) return false
|
2023-03-21 22:46:40 +02:00
|
|
|
switch (this.realSource) {
|
2023-03-14 21:50:43 +02:00
|
|
|
case 'profile':
|
|
|
|
|
case 'admin':
|
2023-03-16 23:18:55 +02:00
|
|
|
return false
|
2023-03-14 21:50:43 +02:00
|
|
|
default:
|
|
|
|
|
return this.state !== this.defaultState
|
|
|
|
|
}
|
2023-03-12 14:32:13 +02:00
|
|
|
},
|
2023-03-22 12:43:53 +02:00
|
|
|
canonPath () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) return null
|
2023-03-22 12:43:53 +02:00
|
|
|
return Array.isArray(this.path) ? this.path : this.path.split('.')
|
|
|
|
|
},
|
2025-11-27 18:18:03 +02:00
|
|
|
descriptionPath () {
|
|
|
|
|
if (this.path == null) return null
|
2025-12-08 17:09:07 +02:00
|
|
|
if (this.descriptionPathOverride) return this.descriptionPathOverride
|
2025-11-27 18:18:03 +02:00
|
|
|
const path = Array.isArray(this.path) ? this.path : this.path.split('.')
|
|
|
|
|
if (this.subgroup) {
|
|
|
|
|
return [
|
2025-11-28 02:02:29 +02:00
|
|
|
...path.slice(0, path.length - 1),
|
2025-11-27 18:18:03 +02:00
|
|
|
':subgroup,' + this.subgroup,
|
|
|
|
|
...path.slice(path.length - 1)
|
|
|
|
|
]
|
|
|
|
|
}
|
|
|
|
|
return path
|
|
|
|
|
},
|
2023-03-16 23:18:55 +02:00
|
|
|
isDirty () {
|
2024-09-24 03:07:27 +03:00
|
|
|
if (this.path == null) return false
|
2023-03-22 12:43:53 +02:00
|
|
|
if (this.realSource === 'admin' && this.canonPath.length > 3) {
|
|
|
|
|
return false // should not show draft buttons for "grouped" values
|
|
|
|
|
} else {
|
2025-12-01 23:56:49 +02:00
|
|
|
return this.realDraftMode && !isEqual(this.draft, this.state)
|
2023-03-22 12:43:53 +02:00
|
|
|
}
|
2023-03-16 23:18:55 +02:00
|
|
|
},
|
|
|
|
|
canHardReset () {
|
2025-12-07 23:11:54 +02:00
|
|
|
return this.realSource === 'admin' && this.$store.state.adminSettings.modifiedPaths?.has(this.canonPath.join(' -> '))
|
2023-03-16 23:18:55 +02:00
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
matchesExpertLevel () {
|
|
|
|
|
return (this.expert || 0) <= this.$store.state.config.expertLevel > 0
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
methods: {
|
2023-03-16 23:18:55 +02:00
|
|
|
getValue (e) {
|
|
|
|
|
return e.target.value
|
|
|
|
|
},
|
2023-03-12 14:32:13 +02:00
|
|
|
update (e) {
|
2023-03-21 22:46:40 +02:00
|
|
|
if (this.realDraftMode) {
|
2023-03-16 23:18:55 +02:00
|
|
|
this.draft = this.getValue(e)
|
|
|
|
|
} else {
|
|
|
|
|
this.configSink(this.path, this.getValue(e))
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
commitDraft () {
|
2023-03-21 22:46:40 +02:00
|
|
|
if (this.realDraftMode) {
|
2023-03-16 23:18:55 +02:00
|
|
|
this.configSink(this.path, this.draft)
|
|
|
|
|
}
|
2023-03-12 14:32:13 +02:00
|
|
|
},
|
|
|
|
|
reset () {
|
2023-03-21 22:46:40 +02:00
|
|
|
if (this.realDraftMode) {
|
2023-05-24 11:44:41 +03:00
|
|
|
this.draft = cloneDeep(this.state)
|
2023-03-16 23:18:55 +02:00
|
|
|
} else {
|
2023-05-24 11:44:41 +03:00
|
|
|
set(this.$store.getters.mergedConfig, this.path, cloneDeep(this.defaultState))
|
2023-03-16 23:18:55 +02:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
hardReset () {
|
2023-03-21 22:46:40 +02:00
|
|
|
switch (this.realSource) {
|
2023-03-16 23:18:55 +02:00
|
|
|
case 'admin':
|
|
|
|
|
return this.$store.dispatch('resetAdminSetting', { path: this.path })
|
|
|
|
|
.then(() => { this.draft = this.state })
|
|
|
|
|
default:
|
|
|
|
|
console.warn('Hard reset not implemented yet!')
|
|
|
|
|
}
|
2023-03-12 14:32:13 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|