mirror of
https://github.com/scinote-eln/scinote-web.git
synced 2025-10-17 09:16:10 +08:00
265 lines
8.3 KiB
Vue
265 lines
8.3 KiB
Vue
<template>
|
|
<div class="results-wrapper">
|
|
<ResultsToolbar
|
|
ref="resultsToolbar"
|
|
:sort="sort"
|
|
:results="results"
|
|
:canCreate="canCreate == 'true'"
|
|
:archived="archived == 'true'"
|
|
:active_url="active_url"
|
|
:archived_url="archived_url"
|
|
:headerSticked="headerSticked"
|
|
:moduleName="moduleName"
|
|
:resultsCollapsed="resultsCollapsed"
|
|
@setSort="setSort"
|
|
@setFilters="setFilters"
|
|
@newResult="createResult"
|
|
@expandAll="expandResults"
|
|
@collapseAll="collapseResults"
|
|
class="my-4"
|
|
/>
|
|
<div
|
|
:class="{ 'tw-hidden': loadingOverlay }"
|
|
class="results-list">
|
|
<Result v-for="result in results" :key="result.id"
|
|
ref="results"
|
|
:result="result"
|
|
:resultToReload="resultToReload"
|
|
:activeDragResult="activeDragResult"
|
|
:userSettingsUrl="userSettingsUrl"
|
|
:protocolId="protocolId"
|
|
@result:update="updateResult"
|
|
@result:elements:loaded="resultToReload = null; elementsLoaded++"
|
|
@result:move_element="reloadResult"
|
|
@result:attachments:loaded="resultToReload = null; attachmentsLoaded++"
|
|
@result:move_attachment="reloadResult"
|
|
@result:duplicated="resetPageAndReload"
|
|
@result:archived="removeResult"
|
|
@result:deleted="removeResult"
|
|
@result:restored="removeResult"
|
|
@result:drag_enter="dragEnter"
|
|
@result:collapsed="checkResultsState"
|
|
/>
|
|
</div>
|
|
<div v-if="loadingOverlay" class="text-center h-20 flex items-center justify-center">
|
|
<div class="sci-loader"></div>
|
|
</div>
|
|
<clipboardPasteModal v-if="showClipboardPasteModal"
|
|
:image="pasteImages"
|
|
:objects="results"
|
|
:objectType="'result'"
|
|
:selectedObjectId="firstObjectInViewport()"
|
|
@files="uploadFilesToResult"
|
|
@cancel="showClipboardPasteModal = false"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import axios from '../../packs/custom_axios.js';
|
|
import ResultsToolbar from './results_toolbar.vue';
|
|
import Result from './result.vue';
|
|
|
|
import stackableHeadersMixin from '../mixins/stackableHeadersMixin';
|
|
import moduleNameObserver from '../mixins/moduleNameObserver';
|
|
|
|
import clipboardPasteModal from '../shared/content/attachments/clipboard_paste_modal.vue';
|
|
import AssetPasteMixin from '../shared/content/attachments/mixins/paste.js';
|
|
|
|
export default {
|
|
name: 'Results',
|
|
components: { ResultsToolbar, Result, clipboardPasteModal },
|
|
mixins: [stackableHeadersMixin, moduleNameObserver, AssetPasteMixin],
|
|
props: {
|
|
url: { type: String, required: true },
|
|
canCreate: { type: String, required: true },
|
|
archived: { type: String, required: true },
|
|
active_url: { type: String, required: true },
|
|
archived_url: { type: String, required: true },
|
|
userSettingsUrl: { type: String, required: false },
|
|
changeStatesUrl: { type: String, required: false },
|
|
protocolId: { type: Number, required: false }
|
|
},
|
|
data() {
|
|
return {
|
|
results: [],
|
|
sort: null,
|
|
filters: {},
|
|
resultToReload: null,
|
|
nextPageUrl: null,
|
|
loadingPage: false,
|
|
activeDragResult: null,
|
|
userSettingsUrl: null,
|
|
resultsCollapsed: false,
|
|
anchorId: null,
|
|
elementsLoaded: 0,
|
|
attachmentsLoaded: 0,
|
|
loadingOverlay: false
|
|
};
|
|
},
|
|
created() {
|
|
const urlParams = new URLSearchParams(window.location.search);
|
|
this.anchorId = urlParams.get('result_id');
|
|
|
|
if (this.anchorId) {
|
|
this.loadingOverlay = true;
|
|
}
|
|
},
|
|
watch: {
|
|
elementsLoaded() {
|
|
if (this.anchorId) {
|
|
this.scrollToResult();
|
|
}
|
|
},
|
|
attachmentsLoaded() {
|
|
if (this.anchorId) {
|
|
this.scrollToResult();
|
|
}
|
|
}
|
|
},
|
|
mounted() {
|
|
this.userSettingsUrl = document.querySelector('meta[name="user-settings-url"]').getAttribute('content');
|
|
window.addEventListener('scroll', this.infiniteScrollLoad, false);
|
|
window.addEventListener('scroll', this.initStackableHeaders, false);
|
|
this.nextPageUrl = this.url;
|
|
this.loadResults();
|
|
this.initStackableHeaders();
|
|
},
|
|
beforeUnmount() {
|
|
window.removeEventListener('scroll', this.infiniteScrollLoad, false);
|
|
window.removeEventListener('scroll', this.initStackableHeaders, false);
|
|
},
|
|
methods: {
|
|
scrollToResult() {
|
|
if (this.elementsLoaded === this.results.length && this.attachmentsLoaded === this.results.length) {
|
|
if (this.anchorId) {
|
|
const result = this.$refs.results.find((child) => child.result?.id === this.anchorId);
|
|
if (result) {
|
|
this.loadingOverlay = false;
|
|
this.$nextTick(() => {
|
|
result.$refs.resultContainer.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
|
this.anchorId = null;
|
|
});
|
|
}
|
|
}
|
|
|
|
if (!this.nextPageUrl) {
|
|
this.loadingOverlay = false;
|
|
this.anchorId = null;
|
|
}
|
|
}
|
|
},
|
|
getHeader() {
|
|
return this.$refs.resultsToolbar.$refs.resultsHeaderToolbar;
|
|
},
|
|
reloadResult(result) {
|
|
this.resultToReload = result;
|
|
},
|
|
resetPageAndReload() {
|
|
this.nextPageUrl = this.url;
|
|
this.results = [];
|
|
this.$nextTick(() => {
|
|
this.loadResults();
|
|
});
|
|
},
|
|
infiniteScrollLoad() {
|
|
if (window.scrollY + window.innerHeight >= document.body.scrollHeight - 20) {
|
|
this.loadResults();
|
|
}
|
|
},
|
|
loadResults() {
|
|
if (this.nextPageUrl === null || this.loadingPage) return;
|
|
|
|
if (window.scrollY + window.innerHeight >= document.body.scrollHeight - 20) {
|
|
this.loadingPage = true;
|
|
const params = this.sort ? { ...this.filters, sort: this.sort } : { ...this.filters };
|
|
axios.get(this.nextPageUrl, { params }).then((response) => {
|
|
this.results = this.results.concat(response.data.data);
|
|
this.sort = response.data.meta.sort;
|
|
this.nextPageUrl = response.data.links.next;
|
|
this.loadingPage = false;
|
|
|
|
if (this.anchorId) {
|
|
const result = this.results.find((e) => e.id === this.anchorId);
|
|
if (!result) {
|
|
this.loadResults();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},
|
|
setSort(sort) {
|
|
this.sort = sort;
|
|
this.resetPageAndReload();
|
|
},
|
|
setFilters(filters) {
|
|
this.filters = filters;
|
|
this.resetPageAndReload();
|
|
},
|
|
createResult() {
|
|
axios.post(
|
|
`${this.url}`,
|
|
{
|
|
headers: {
|
|
Accept: 'application/json'
|
|
}
|
|
}
|
|
).then(
|
|
(response) => {
|
|
this.results = [{ newResult: true, ...response.data.data }, ...this.results];
|
|
window.scrollTo(0, 0);
|
|
}
|
|
);
|
|
},
|
|
checkResultsState() {
|
|
this.resultsCollapsed = this.$refs.results.every((result) => result.isCollapsed);
|
|
},
|
|
collapseResults() {
|
|
$('.result-wrapper .collapse').collapse('hide');
|
|
this.updateResultStateSettings(true);
|
|
this.$refs.results.forEach((result) => { result.isCollapsed = true; });
|
|
this.resultsCollapsed = true;
|
|
},
|
|
expandResults() {
|
|
$('.result-wrapper .collapse').collapse('show');
|
|
this.updateResultStateSettings(false);
|
|
this.$refs.results.forEach((result) => { result.isCollapsed = false; });
|
|
this.resultsCollapsed = false;
|
|
},
|
|
updateResultStateSettings(newState) {
|
|
const data = {};
|
|
|
|
if (newState) {
|
|
data.collapsed = true;
|
|
} else {
|
|
data.expanded = true;
|
|
}
|
|
|
|
axios.post(this.changeStatesUrl, data);
|
|
},
|
|
updateResult(id, attributes) {
|
|
const resultIndex = this.results.findIndex((result) => result.id === id);
|
|
this.results[resultIndex].attributes = {
|
|
...this.results[resultIndex].attributes,
|
|
...attributes
|
|
};
|
|
},
|
|
removeResult(result_id) {
|
|
this.results = this.results.filter((r) => r.id != result_id);
|
|
},
|
|
dragEnter(id) {
|
|
this.activeDragResult = id;
|
|
},
|
|
uploadFilesToResult(file, resultId) {
|
|
this.$refs.results.find((child) => child.result?.id == resultId).uploadFiles(file);
|
|
},
|
|
firstObjectInViewport() {
|
|
const result = $('.result-wrapper:not(.locked)').toArray().find((element) => {
|
|
const { top, bottom } = element.getBoundingClientRect();
|
|
return bottom > 0 && top < window.innerHeight;
|
|
});
|
|
return result ? result.dataset.id : null;
|
|
}
|
|
}
|
|
};
|
|
</script>
|