Merge branch 'sss-objects' into shigusegubu-themes3

This commit is contained in:
Henry Jameson 2025-03-26 15:58:15 +02:00
commit 55d29065f0
7 changed files with 95 additions and 38 deletions

View file

@ -0,0 +1 @@
Synchronized mutes, advanced mute control (regexp, expiry, naming)

View file

@ -1,4 +1,4 @@
import { throttle } from 'lodash'
import { throttle, cloneDeep } from 'lodash'
import { mapState, mapActions } from 'pinia'
import { useServerSideStorageStore } from 'src/stores/serverSideStorage'
import { v4 as uuidv4 } from 'uuid';
@ -14,12 +14,19 @@ import SharedComputedObject from '../helpers/shared_computed_object.js'
const FilteringTab = {
data () {
console.log(cloneDeep(useServerSideStorageStore().prefsStorage.simple.muteFilters))
return {
replyVisibilityOptions: ['all', 'following', 'self'].map(mode => ({
key: mode,
value: mode,
label: this.$t(`settings.reply_visibility_${mode}`)
}))
})),
muteFiltersDraftObject: cloneDeep(useServerSideStorageStore().prefsStorage.simple.muteFilters),
muteFiltersDraftDirty: Object.fromEntries(
Object.entries(
useServerSideStorageStore().prefsStorage.simple.muteFilters
).map(([k]) => [k, false])
)
}
},
components: {
@ -38,12 +45,13 @@ const FilteringTab = {
muteFilters: store => Object.entries(store.prefsStorage.simple.muteFilters),
muteFiltersObject: store => store.prefsStorage.simple.muteFilters
}
)
),
muteFiltersDraft () {
return Object.entries(this.muteFiltersDraftObject)
}
},
methods: {
...mapActions(useServerSideStorageStore, ['unsetPreference']),
pushServerSideStorage: throttle(() => useServerSideStorageStore().pushServerSideStorage(), 500),
setPreference: throttle(x => useServerSideStorageStore().setPreference(x), 500),
...mapActions(useServerSideStorageStore, ['setPreference', 'unsetPreference', 'pushServerSideStorage']),
getDatetimeLocal (timestamp) {
const date = new Date(timestamp)
const datetime = [
@ -84,22 +92,25 @@ const FilteringTab = {
}
const newId = uuidv4()
this.muteFiltersDraftObject[newId] = filter
this.setPreference({ path: 'simple.muteFilters.' + newId , value: filter })
this.pushServerSideStorage()
},
copyFilter (id) {
const filter = { ...this.muteFiltersObject[id] }
const filter = { ...this.muteFiltersDraftObject[id] }
const newId = uuidv4()
this.muteFiltersDraftObject[newId] = filter
this.setPreference({ path: 'simple.muteFilters.' + newId , value: filter })
this.pushServerSideStorage()
},
deleteFilter (id) {
delete this.muteFiltersDraftObject[id]
this.unsetPreference({ path: 'simple.muteFilters.' + id , value: null })
this.pushServerSideStorage()
},
updateFilter(id, field, value) {
const filter = { ...this.muteFiltersObject[id] }
const filter = { ...this.muteFiltersDraftObject[id] }
if (field === 'expires-never') {
// filter[field] = value
if (!value) {
@ -115,8 +126,15 @@ const FilteringTab = {
} else {
filter[field] = value
}
this.setPreference({ path: 'simple.muteFilters.' + id , value: filter })
this.muteFiltersDraftObject[id] = filter
this.muteFiltersDraftDirty[id] = true
console.log(this.muteFiltersDraftDirty)
},
saveFilter(id) {
this.setPreference({ path: 'simple.muteFilters.' + id , value: this.muteFiltersDraftObject[id] })
this.pushServerSideStorage()
this.muteFiltersDraftDirty[id] = false
console.log(this.muteFiltersDraftDirty)
}
},
// Updating nested properties

View file

@ -76,7 +76,7 @@
<div
class="mute-filter"
:style="{ order: filter[1].order }"
v-for="filter in muteFilters"
v-for="filter in muteFiltersDraft"
:key="filter[0]"
>
<div class="filter-name">
@ -201,6 +201,16 @@
>
{{ $t('settings.filter.delete') }}
</button>
{{ ' ' }}
<button
class="save-button button-default"
:class="{ disabled: !muteFiltersDraftDirty[filter[0]] }"
:disabled="!muteFiltersDraftDirty[filter[0]]"
type="button"
@click="saveFilter(filter[0])"
>
{{ $t('settings.filter.save') }}
</button>
</div>
</div>
<div class="mute-filter">

View file

@ -422,6 +422,7 @@
"expires": "Expires",
"expired": "Expired",
"copy": "Copy filter",
"save": "Save filter",
"delete": "Remove filter",
"new": "Create filter",
"regexp_error": "Invalid Regular Expression",

View file

@ -0,0 +1,41 @@
export const CONFIG_MIGRATION = 1
import { v4 as uuidv4 } from 'uuid';
// for future use
/*
const simpleDeclaration = {
store: 'server-side',
migration(serverside, rootState) {
serverside.setPreference({ path: 'simple.' + field, value: rootState.config[oldField ?? field] })
}
}
*/
export const declarations = [
{
field: 'muteFilters',
store: 'server-side',
migrationFlag: 'configMigration',
migrationNum: 1,
description: 'Mute filters, wordfilter/regexp/etc',
valueType: 'complex',
migration (serverside, rootState) {
rootState.config.muteWords.forEach((word, order) => {
const uniqueId = uuidv4()
serverside.setPreference({
path: 'simple.muteFilters.' + uniqueId,
value: {
type: 'word',
value: word,
name: word,
enabled: true,
expires: null,
hide: false,
order
}
})
})
}
}
]

View file

@ -1,5 +1,4 @@
import { compact, map, each, mergeWith, last, concat, uniq, isArray } from 'lodash'
import { v4 as uuidv4 } from 'uuid';
import backendInteractorService from '../services/backend_interactor_service/backend_interactor_service.js'
import { windowWidth, windowHeight } from '../services/window_utils/window_utils'
@ -9,7 +8,9 @@ import { registerPushNotifications, unregisterPushNotifications } from '../servi
import { useInterfaceStore } from 'src/stores/interface.js'
import { useOAuthStore } from 'src/stores/oauth.js'
import { useServerSideStorageStore, CONFIG_MIGRATION } from 'src/stores/serverSideStorage'
import { useServerSideStorageStore } from 'src/stores/serverSideStorage'
import { declarations } from 'src/modules/config_declaration'
// TODO: Unify with mergeOrAdd in statuses.js
export const mergeOrAdd = (arr, obj, item) => {
@ -637,33 +638,17 @@ const users = {
/**/
const { configMigration } = useServerSideStorageStore().flagStorage
// Wordfilter migration
if (configMigration < 1) {
// Convert existing wordfilter into synced one
store.rootState.config.muteWords.forEach((word, order) => {
const uniqueId = uuidv4()
useServerSideStorageStore().setPreference({
path: 'simple.muteFilters.' + uniqueId,
value: {
type: 'word',
value: word,
name: word,
enabled: true,
expires: null,
hide: false,
order
}
})
declarations
.filter(x => {
return x.store === 'server-side' &&
(x.migrationNum ?? x.migrationNum > configMigration)
})
.toSorted((a, b) => a.configMigration - b.configMigration)
.forEach(value => {
value.migration(useServerSideStorageStore(), store.rootState)
useServerSideStorageStore().setFlag({ flag: 'configMigration', value: value.migrationNum })
useServerSideStorageStore().pushServerSideStorage()
})
}
if (configMigration < CONFIG_MIGRATION) {
// Update the flag
useServerSideStorageStore().setFlag({ flag: 'configMigration', value: CONFIG_MIGRATION })
useServerSideStorageStore().pushServerSideStorage()
}
if (user.token) {
dispatch('setWsToken', user.token)

View file

@ -5,6 +5,7 @@ export const muteFilterHits = (muteFilters, status) => {
return muteFilters.toSorted((a,b) => b.order - a.order).map(filter => {
const { hide, expires, name, value, type, enabled} = filter
if (!enabled) return false
if (value === '') return false
if (expires !== null && expires < Date.now()) return false
switch (type) {
case 'word': {