Compare commits

...

19 Commits

Author SHA1 Message Date
c040620185
chore: move margin outside of the popup component 2021-11-02 22:50:10 +01:00
cff2a0bf6f
chore: use <script setup> 2021-11-02 22:40:29 +01:00
fa25fbccd0
fix: close when clicked outside 2021-11-02 22:36:58 +01:00
e29a19f218
fix: restrict filter params in comparison 2021-11-02 22:30:55 +01:00
5bf288a066
feat: rename default slot to content 2021-11-02 22:19:17 +01:00
ec927d6564
feat: move active columns filter to new popup component 2021-11-02 22:17:35 +01:00
0fbe83c362
chore: put ref on popup wrapper div instead of slot 2021-11-02 22:12:55 +01:00
bc7c89d859
fix: lint 2021-11-02 22:09:21 +01:00
c4b9bc5942
chore: simplify using only some filter values 2021-11-02 22:04:00 +01:00
86efb07f09
feat: create new popup component to handle popups generally 2021-11-02 21:59:14 +01:00
9250f4e76b
fix: column active test 2021-11-02 21:11:14 +01:00
0a0c6a6450
fix: defaultParams function name 2021-11-02 21:03:55 +01:00
566fe77b02
fix: defaultParams function name 2021-11-02 20:56:53 +01:00
7207c452f3
chore: use const for default params 2021-11-02 20:27:37 +01:00
fe41d34bab
fix: setting done filter 2021-11-02 20:27:20 +01:00
df1a76d529
chore: simplify done filter check 2021-11-02 20:06:54 +01:00
dc02c09f19
chore: make this.value a computed property again 2021-11-02 20:02:19 +01:00
0d752c9d32
chore: only use factory function 2021-11-02 20:02:19 +01:00
f3d338c857
feat: add button to clear active filters 2021-11-02 20:02:19 +01:00
11 changed files with 219 additions and 165 deletions

View File

@ -219,10 +219,10 @@ describe('Lists', () => {
cy.get('.table-view .filter-container .items .button') cy.get('.table-view .filter-container .items .button')
.contains('Columns') .contains('Columns')
.click() .click()
cy.get('.table-view .filter-container .card .card-content .fancycheckbox .check') cy.get('.table-view .filter-container .card.columns-filter .card-content .fancycheckbox .check')
.contains('Priority') .contains('Priority')
.click() .click()
cy.get('.table-view .filter-container .card .card-content .fancycheckbox .check') cy.get('.table-view .filter-container .card.columns-filter .card-content .fancycheckbox .check')
.contains('Done') .contains('Done')
.click() .click()

View File

@ -1,37 +1,49 @@
<template> <template>
<transition name="fade"> <x-button
v-if="hasFilters"
type="secondary"
@click="clearFilters"
>
{{ $t('filters.clear') }}
</x-button>
<popup>
<template #trigger="{toggle}">
<x-button
@click.prevent.stop="toggle()"
type="secondary"
icon="filter"
>
{{ $t('filters.title') }}
</x-button>
</template>
<template #content="{isOpen}">
<filters <filters
v-if="visibleInternal"
v-model="value" v-model="value"
ref="filters" ref="filters"
class="filter-popup"
:class="{'is-open': isOpen}"
/> />
</transition> </template>
</popup>
</template> </template>
<script> <script>
import {closeWhenClickedOutside} from '@/helpers/closeWhenClickedOutside'
import Filters from '../../../components/list/partials/filters' import Filters from '../../../components/list/partials/filters'
import {getDefaultParams} from '../../tasks/mixins/taskList'
import Popup from '../../misc/popup'
export default { export default {
name: 'filter-popup', name: 'filter-popup',
components: { components: {
Popup,
Filters, Filters,
}, },
props: { props: {
modelValue: { modelValue: {
required: true, required: true,
}, },
visible: {
type: Boolean,
default: false,
},
}, },
emits: ['update:modelValue'], emits: ['update:modelValue'],
data() {
return {
visibleInternal: false,
}
},
computed: { computed: {
value: { value: {
get() { get() {
@ -41,34 +53,46 @@ export default {
this.$emit('update:modelValue', value) this.$emit('update:modelValue', value)
}, },
}, },
hasFilters() {
// this.value also contains the page parameter which we don't want to include in filters
// eslint-disable-next-line no-unused-vars
const {filter_by, filter_value, filter_comparator, filter_concat, s} = this.value
const def = {...getDefaultParams()}
const params = {filter_by, filter_value, filter_comparator, filter_concat, s}
const defaultParams = {
filter_by: def.filter_by,
filter_value: def.filter_value,
filter_comparator: def.filter_comparator,
filter_concat: def.filter_concat,
s: s ? def.s : undefined,
}
return JSON.stringify(params) !== JSON.stringify(defaultParams)
}, },
mounted() {
document.addEventListener('click', this.hidePopup)
},
beforeUnmount() {
document.removeEventListener('click', this.hidePopup)
}, },
watch: { watch: {
modelValue: { modelValue: {
handler(value) { handler(value) {
this.params = value this.value = value
}, },
immediate: true, immediate: true,
}, },
visible() {
this.visibleInternal = !this.visibleInternal
},
}, },
methods: { methods: {
hidePopup(e) { clearFilters() {
if (!this.visibleInternal) { this.value = {...getDefaultParams()}
return
}
closeWhenClickedOutside(e, this.$refs.filters.$el, () => {
this.visibleInternal = false
})
}, },
}, },
} }
</script> </script>
<style scoped lang="scss">
.filter-popup {
margin: 0 !important;
&.is-open {
margin: 2rem 0 1rem !important;
}
}
</style>

View File

@ -458,15 +458,7 @@ export default {
return return
} }
let foundDone = false this.filters.done = this.params.filter_by.some((f) => f === 'done') === false
this.params.filter_by.forEach((f, i) => {
if (f === 'done') {
foundDone = i
}
})
if (foundDone === false) {
this.filters.done = true
}
}, },
async prepareRelatedObjectFilter(kind, filterName = null, servicePrefix = null) { async prepareRelatedObjectFilter(kind, filterName = null, servicePrefix = null) {
if (filterName === null) { if (filterName === null) {

View File

@ -0,0 +1,54 @@
<template>
<slot name="trigger" :isOpen="open" :toggle="toggle"></slot>
<div class="popup" :class="{'is-open': open}" ref="popup">
<slot name="content" :isOpen="open"/>
</div>
</template>
<script setup>
import {closeWhenClickedOutside} from '@/helpers/closeWhenClickedOutside'
import {onBeforeUnmount, onMounted, ref} from 'vue'
const open = ref(false)
const popup = ref(null)
const toggle = () => {
open.value = !open.value
}
const hidePopup = e => {
if (!open.value) {
return
}
// we actually want to use popup.$el, not its value.
// eslint-disable-next-line vue/no-ref-as-operand
closeWhenClickedOutside(e, popup.value, () => {
open.value = false
})
}
onMounted(() => {
document.addEventListener('click', hidePopup)
})
onBeforeUnmount(() => {
document.removeEventListener('click', hidePopup)
})
</script>
<style scoped lang="scss">
.popup {
transition: opacity $transition;
opacity: 0;
height: 0;
overflow: hidden;
position: absolute;
top: 1rem;
&.is-open {
opacity: 1;
height: auto;
}
}
</style>

View File

@ -9,13 +9,13 @@
> >
{{ $t('filters.title') }} {{ $t('filters.title') }}
</x-button> </x-button>
</div>
<filter-popup <filter-popup
:visible="showTaskFilter" :visible="showTaskFilter"
v-model="params" v-model="params"
@update:modelValue="loadTasks()" @update:modelValue="loadTasks()"
/> />
</div> </div>
</div>
<div class="dates"> <div class="dates">
<template v-for="(y, yk) in days" :key="yk + 'year'"> <template v-for="(y, yk) in days" :key="yk + 'year'">
<div class="months"> <div class="months">

View File

@ -1,14 +1,14 @@
import TaskCollectionService from '@/services/taskCollection' import TaskCollectionService from '@/services/taskCollection'
// FIXME: merge with DEFAULT_PARAMS in filters.vue // FIXME: merge with DEFAULT_PARAMS in filters.vue
const DEFAULT_PARAMS = { export const getDefaultParams = () => ({
sort_by: ['position', 'id'], sort_by: ['position', 'id'],
order_by: ['asc', 'desc'], order_by: ['asc', 'desc'],
filter_by: ['done'], filter_by: ['done'],
filter_value: ['false'], filter_value: ['false'],
filter_comparator: ['equals'], filter_comparator: ['equals'],
filter_concat: 'and', filter_concat: 'and',
} })
/** /**
* This mixin provides a base set of methods and properties to get tasks on a list. * This mixin provides a base set of methods and properties to get tasks on a list.
@ -26,7 +26,7 @@ export default {
searchTerm: '', searchTerm: '',
showTaskFilter: false, showTaskFilter: false,
params: DEFAULT_PARAMS, params: {...getDefaultParams()},
} }
}, },
watch: { watch: {

View File

@ -344,6 +344,7 @@
}, },
"filters": { "filters": {
"title": "Filters", "title": "Filters",
"clear": "Clear Filters",
"attributes": { "attributes": {
"title": "Title", "title": "Title",
"titlePlaceholder": "The saved filter title goes here…", "titlePlaceholder": "The saved filter title goes here…",

View File

@ -47,10 +47,6 @@ $filter-container-top-link-share-list: -47px;
justify-content: space-between; justify-content: space-between;
margin-right: .5rem; margin-right: .5rem;
.button, .input {
height: $switch-view-height;
}
.field { .field {
transition: width $transition; transition: width $transition;
width: 100%; width: 100%;

View File

@ -2,19 +2,12 @@
<div class="kanban-view"> <div class="kanban-view">
<div class="filter-container" v-if="isSavedFilter"> <div class="filter-container" v-if="isSavedFilter">
<div class="items"> <div class="items">
<x-button
@click.prevent.stop="toggleFilterPopup"
icon="filter"
type="secondary"
>
{{ $t('filters.title') }}
</x-button>
</div>
<filter-popup <filter-popup
:visible="showFilters"
v-model="params" v-model="params"
@update:modelValue="loadBuckets"
/> />
</div> </div>
</div>
<div <div
:class="{ 'is-loading': loading && !oneTaskUpdating}" :class="{ 'is-loading': loading && !oneTaskUpdating}"
class="kanban kanban-bucket-container loader-container" class="kanban kanban-bucket-container loader-container"
@ -300,7 +293,6 @@ export default {
filter_comparator: [], filter_comparator: [],
filter_concat: 'and', filter_concat: 'and',
}, },
showFilters: false,
} }
}, },
created() { created() {
@ -359,10 +351,6 @@ export default {
}, },
methods: { methods: {
toggleFilterPopup() {
this.showFilters = !this.showFilters
},
loadBuckets() { loadBuckets() {
// Prevent trying to load buckets if the task popup view is active // Prevent trying to load buckets if the task popup view is active
if (this.$route.name !== 'list.kanban') { if (this.$route.name !== 'list.kanban') {

View File

@ -41,20 +41,12 @@
v-if="!showTaskSearch" v-if="!showTaskSearch"
/> />
</div> </div>
<x-button
@click.prevent.stop="showTaskFilter = !showTaskFilter"
type="secondary"
icon="filter"
>
{{ $t('filters.title') }}
</x-button>
</div>
<filter-popup <filter-popup
:visible="showTaskFilter"
v-model="params" v-model="params"
@update:modelValue="loadTasks()" @update:modelValue="loadTasks()"
/> />
</div> </div>
</div>
<card :padding="false" :has-content="false" class="has-overflow"> <card :padding="false" :has-content="false" class="has-overflow">
<template <template
@ -154,6 +146,7 @@ import FilterPopup from '@/components/list/partials/filter-popup.vue'
import {HAS_TASKS} from '@/store/mutation-types' import {HAS_TASKS} from '@/store/mutation-types'
import Nothing from '@/components/misc/nothing.vue' import Nothing from '@/components/misc/nothing.vue'
import Pagination from '@/components/misc/pagination.vue' import Pagination from '@/components/misc/pagination.vue'
import Popup from '@/components/misc/popup'
import draggable from 'vuedraggable' import draggable from 'vuedraggable'
import {calculateItemPosition} from '../../../helpers/calculateItemPosition' import {calculateItemPosition} from '../../../helpers/calculateItemPosition'
@ -197,6 +190,7 @@ export default {
taskList, taskList,
], ],
components: { components: {
Popup,
Nothing, Nothing,
FilterPopup, FilterPopup,
SingleTaskInList, SingleTaskInList,

View File

@ -2,23 +2,18 @@
<div :class="{'is-loading': taskCollectionService.loading}" class="table-view loader-container"> <div :class="{'is-loading': taskCollectionService.loading}" class="table-view loader-container">
<div class="filter-container"> <div class="filter-container">
<div class="items"> <div class="items">
<popup>
<template #trigger="{toggle}">
<x-button <x-button
@click.prevent.stop="() => {showActiveColumnsFilter = !showActiveColumnsFilter; showTaskFilter = false}" @click.prevent.stop="toggle()"
icon="th" icon="th"
type="secondary" type="secondary"
> >
{{ $t('list.table.columns') }} {{ $t('list.table.columns') }}
</x-button> </x-button>
<x-button </template>
@click.prevent.stop="() => {showTaskFilter = !showTaskFilter; showActiveColumnsFilter = false}" <template #content="{isOpen}">
icon="filter" <card class="columns-filter" :class="{'is-open': isOpen}">
type="secondary"
>
{{ $t('filters.title') }}
</x-button>
</div>
<transition name="fade">
<card v-if="showActiveColumnsFilter">
<fancycheckbox @change="saveTaskColumns" v-model="activeColumns.id">#</fancycheckbox> <fancycheckbox @change="saveTaskColumns" v-model="activeColumns.id">#</fancycheckbox>
<fancycheckbox @change="saveTaskColumns" v-model="activeColumns.done"> <fancycheckbox @change="saveTaskColumns" v-model="activeColumns.done">
{{ $t('task.attributes.done') }} {{ $t('task.attributes.done') }}
@ -57,13 +52,14 @@
{{ $t('task.attributes.createdBy') }} {{ $t('task.attributes.createdBy') }}
</fancycheckbox> </fancycheckbox>
</card> </card>
</transition> </template>
</popup>
<filter-popup <filter-popup
:visible="showTaskFilter"
v-model="params" v-model="params"
@update:modelValue="loadTasks()" @update:modelValue="loadTasks()"
/> />
</div> </div>
</div>
<card :padding="false" :has-content="false"> <card :padding="false" :has-content="false">
<div class="has-horizontal-overflow"> <div class="has-horizontal-overflow">
@ -200,10 +196,12 @@ import Sort from '../../../components/tasks/partials/sort'
import {saveListView} from '@/helpers/saveListView' import {saveListView} from '@/helpers/saveListView'
import FilterPopup from '@/components/list/partials/filter-popup.vue' import FilterPopup from '@/components/list/partials/filter-popup.vue'
import Pagination from '@/components/misc/pagination.vue' import Pagination from '@/components/misc/pagination.vue'
import Popup from '../../../components/misc/popup'
export default { export default {
name: 'Table', name: 'Table',
components: { components: {
Popup,
Done, Done,
FilterPopup, FilterPopup,
Sort, Sort,
@ -219,7 +217,6 @@ export default {
], ],
data() { data() {
return { return {
showActiveColumnsFilter: false,
activeColumns: { activeColumns: {
id: true, id: true,
done: true, done: true,
@ -323,4 +320,12 @@ export default {
} }
} }
} }
.columns-filter {
margin: 0 !important;
&.is-open {
margin: 2rem 0 1rem !important;
}
}
</style> </style>