/* global Sortable */ import boardNewIssue from './board_new_issue'; import boardCard from './board_card.vue'; import eventHub from '../eventhub'; import loadingIcon from '../../vue_shared/components/loading_icon.vue'; const Store = gl.issueBoards.BoardsStore; export default { name: 'BoardList', props: { disabled: { type: Boolean, required: true, }, list: { type: Object, required: true, }, issues: { type: Array, required: true, }, loading: { type: Boolean, required: true, }, issueLinkBase: { type: String, required: true, }, rootPath: { type: String, required: true, }, }, data() { return { scrollOffset: 250, filters: Store.state.filters, showCount: false, showIssueForm: false, }; }, components: { boardCard, boardNewIssue, loadingIcon, }, methods: { listHeight() { return this.$refs.list.getBoundingClientRect().height; }, scrollHeight() { return this.$refs.list.scrollHeight; }, scrollTop() { return this.$refs.list.scrollTop + this.listHeight(); }, scrollToTop() { this.$refs.list.scrollTop = 0; }, loadNextPage() { const getIssues = this.list.nextPage(); const loadingDone = () => { this.list.loadingMore = false; }; if (getIssues) { this.list.loadingMore = true; getIssues .then(loadingDone) .catch(loadingDone); } }, toggleForm() { this.showIssueForm = !this.showIssueForm; }, onScroll() { if (!this.loadingMore && (this.scrollTop() > this.scrollHeight() - this.scrollOffset)) { this.loadNextPage(); } }, }, watch: { filters: { handler() { this.list.loadingMore = false; this.$refs.list.scrollTop = 0; }, deep: true, }, issues() { this.$nextTick(() => { if (this.scrollHeight() <= this.listHeight() && this.list.issuesSize > this.list.issues.length) { this.list.page += 1; this.list.getIssues(false) .catch(() => { // TODO: handle request error }); } if (this.scrollHeight() > Math.ceil(this.listHeight())) { this.showCount = true; } else { this.showCount = false; } }); }, }, created() { eventHub.$on(`hide-issue-form-${this.list.id}`, this.toggleForm); eventHub.$on(`scroll-board-list-${this.list.id}`, this.scrollToTop); }, mounted() { const options = gl.issueBoards.getBoardSortableDefaultOptions({ scroll: document.querySelectorAll('.boards-list')[0], group: 'issues', disabled: this.disabled, filter: '.board-list-count, .is-disabled', dataIdAttr: 'data-issue-id', onStart: (e) => { const card = this.$refs.issue[e.oldIndex]; card.showDetail = false; Store.moving.list = card.list; Store.moving.issue = Store.moving.list.findIssue(+e.item.dataset.issueId); gl.issueBoards.onStart(); }, onAdd: (e) => { gl.issueBoards.BoardsStore .moveIssueToList(Store.moving.list, this.list, Store.moving.issue, e.newIndex); this.$nextTick(() => { e.item.remove(); }); }, onUpdate: (e) => { const sortedArray = this.sortable.toArray().filter(id => id !== '-1'); gl.issueBoards.BoardsStore .moveIssueInList(this.list, Store.moving.issue, e.oldIndex, e.newIndex, sortedArray); }, onMove(e) { return !e.related.classList.contains('board-list-count'); }, }); this.sortable = Sortable.create(this.$refs.list, options); // Scroll event on list to load more this.$refs.list.addEventListener('scroll', this.onScroll); }, beforeDestroy() { eventHub.$off(`hide-issue-form-${this.list.id}`, this.toggleForm); eventHub.$off(`scroll-board-list-${this.list.id}`, this.scrollToTop); this.$refs.list.removeEventListener('scroll', this.onScroll); }, template: `
`, };