From 3b20b6ce0038ddb25e06ecc5103d77e19e23e1c1 Mon Sep 17 00:00:00 2001 From: Terry Truong Date: Mon, 16 Jan 2023 10:45:13 +1100 Subject: Add loading indicator --- src/App.vue | 35 ++++++++++++++++++++++++++++++++--- src/components/LoadingModal.vue | 27 +++++++++++++++++++++++++++ src/components/SearchModal.vue | 4 +++- src/components/icon/LoaderIcon.vue | 13 +++++++++++++ 4 files changed, 75 insertions(+), 4 deletions(-) create mode 100644 src/components/LoadingModal.vue create mode 100644 src/components/icon/LoaderIcon.vue diff --git a/src/App.vue b/src/App.vue index b789d91..7b35c60 100644 --- a/src/App.vue +++ b/src/App.vue @@ -35,7 +35,8 @@ + @close="searchOpen = false" @search="onSearch" @info-click="onInfoClick" + @net-wait="primeLoadInd(SERVER_WAIT_MSG)" @net-get="endLoadInd"/> @@ -46,6 +47,9 @@ + + + @@ -58,6 +62,7 @@ import InfoModal from './components/InfoModal.vue'; import SearchModal from './components/SearchModal.vue'; import SettingsModal from './components/SettingsModal.vue'; import HelpModal from './components/HelpModal.vue'; +import LoadingModal from './components/LoadingModal.vue'; import IconButton from './components/IconButton.vue'; // Icons import HelpIcon from './components/icon/HelpIcon.vue'; @@ -259,7 +264,7 @@ async function onEventDisplay( if (store.reqImgs){ urlParams.append('imgonly', 'true'); } - let responseObj: EventResponseJson | null = await queryServer(urlParams); + let responseObj: EventResponseJson | null = await loadFromServer(urlParams, 2000); if (responseObj == null){ console.log('WARNING: Server gave null response to event query'); return; @@ -319,7 +324,7 @@ const infoModalData = ref(null as EventInfo | null); async function onInfoClick(eventTitle: string){ // Query server for event info let urlParams = new URLSearchParams({type: 'info', event: eventTitle}); - let responseObj: EventInfoJson | null = await queryServer(urlParams); + let responseObj: EventInfoJson | null = await loadFromServer(urlParams); if (responseObj != null){ infoModalData.value = jsonToEventInfo(responseObj); } else { @@ -354,6 +359,30 @@ function onSettingChg(option: string){ // For help modal const helpOpen = ref(false); +// For loading modal +const SERVER_WAIT_MSG = 'Loading data'; +const loadingMsg = ref(null as null | string); +const pendingLoadingRevealHdlr = ref(0); // Used to delay showing the loading modal +function primeLoadInd(msg: string, delay?: number){ // Sets up a loading message to display after a timeout + clearTimeout(pendingLoadingRevealHdlr.value); + pendingLoadingRevealHdlr.value = window.setTimeout(() => { + loadingMsg.value = msg; + }, delay == null ? 500 : delay); +} +function endLoadInd(){ // Cancels or closes a loading message + clearTimeout(pendingLoadingRevealHdlr.value); + pendingLoadingRevealHdlr.value = 0; + if (loadingMsg.value != null){ + loadingMsg.value = null; + } +} +async function loadFromServer(urlParams: URLSearchParams, delay?: number){ // Like queryServer() but uses loading modal + primeLoadInd(SERVER_WAIT_MSG, delay); + let responseObj = await queryServer(urlParams); + endLoadInd(); + return responseObj; +} + // For timeline reset const resetFlags: Ref = ref([]); function onReset(){ diff --git a/src/components/LoadingModal.vue b/src/components/LoadingModal.vue new file mode 100644 index 0000000..5d207c7 --- /dev/null +++ b/src/components/LoadingModal.vue @@ -0,0 +1,27 @@ + + + diff --git a/src/components/SearchModal.vue b/src/components/SearchModal.vue index ade86be..be8df51 100644 --- a/src/components/SearchModal.vue +++ b/src/components/SearchModal.vue @@ -47,7 +47,7 @@ const props = defineProps({ eventTree: {type: Object as PropType>, required: true}, titleToEvent: {type: Object as PropType>, required: true}, }); -const emit = defineEmits(['search', 'close', 'info-click']); +const emit = defineEmits(['search', 'close', 'info-click', 'net-wait', 'net-get']); // Search-suggestion data const searchSuggs = ref([] as string[]); @@ -179,7 +179,9 @@ async function resolveSearch(eventTitle: string){ if (store.reqImgs){ urlParams.append('imgonly', 'true'); } + emit('net-wait'); // Allows the parent component to show a loading-indicator let responseObj: EventInfoJson | null = await queryServer(urlParams); + emit('net-get'); if (responseObj != null){ let eventInfo = jsonToEventInfo(responseObj); if (store.reqImgs && eventInfo.event.imgId == null){ diff --git a/src/components/icon/LoaderIcon.vue b/src/components/icon/LoaderIcon.vue new file mode 100644 index 0000000..c2a0369 --- /dev/null +++ b/src/components/icon/LoaderIcon.vue @@ -0,0 +1,13 @@ + -- cgit v1.2.3