2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 08:08:37 +00:00
books/src/pages/DatabaseSelector.vue
2022-09-18 19:37:26 +05:30

361 lines
9.3 KiB
Vue

<template>
<div
class="flex-1 flex justify-center items-center bg-gray-25"
:class="{
'pointer-events-none': loadingDatabase,
'window-drag': platform !== 'Windows',
}"
>
<div
class="w-full w-form shadow-lg rounded-lg border relative bg-white"
style="height: 700px"
>
<!-- Welcome to Frappe Books -->
<div class="px-4 py-4">
<h1 class="text-2xl font-semibold select-none">
{{ t`Welcome to Frappe Books` }}
</h1>
<p class="text-gray-600 text-base select-none">
{{
t`Create a new file or select an existing one from your computer`
}}
</p>
</div>
<hr />
<!-- New File (Blue Icon) -->
<div
@click="newDatabase"
class="px-4 h-row-largest flex flex-row items-center gap-4 p-2"
:class="creatingDemo ? '' : 'hover:bg-gray-50 cursor-pointer'"
>
<div class="w-8 h-8 rounded-full bg-blue-500 relative flex-center">
<feather-icon name="plus" class="text-white w-5 h-5" />
</div>
<div>
<p class="font-medium">
{{ t`New File` }}
</p>
<p class="text-sm text-gray-600">
{{ t`Create a new file and store it in your computer.` }}
</p>
</div>
</div>
<!-- Existing File (Green Icon) -->
<div
@click="existingDatabase"
class="px-4 h-row-largest flex flex-row items-center gap-4 p-2"
:class="creatingDemo ? '' : 'hover:bg-gray-50 cursor-pointer'"
>
<div class="w-8 h-8 rounded-full bg-green-500 relative flex-center">
<feather-icon name="upload" class="w-4 h-4 text-white" />
</div>
<div>
<p class="font-medium">
{{ t`Existing File` }}
</p>
<p class="text-sm text-gray-600">
{{ t`Load an existing .db file from your computer.` }}
</p>
</div>
</div>
<hr />
<!-- File List -->
<div class="overflow-y-auto" style="max-height: 340px">
<div
class="h-row-largest px-4 flex gap-4 items-center"
:class="creatingDemo ? '' : 'hover:bg-gray-50 cursor-pointer'"
v-for="(file, i) in files"
:key="file.dbPath"
@click="selectFile(file)"
>
<div
class="
w-8
h-8
rounded-full
flex
justify-center
items-center
bg-gray-200
text-gray-500
font-semibold
flex-shrink-0
text-base
"
>
{{ i + 1 }}
</div>
<div class="w-full">
<p class="font-medium">
{{ file.companyName }}
</p>
<div
class="text-sm text-gray-600 flex justify-between overflow-x-auto"
>
<p class="whitespace-nowrap mr-2">
{{ formatDate(file.modified) }}
</p>
<p class="text-right" v-if="fyo.store.isDevelopment">
{{ file.dbPath }}
</p>
</div>
</div>
<button
class="
ml-auto
p-2
hover:bg-red-200
rounded-full
w-8
h-8
text-gray-600
hover:text-red-400
"
@click.stop="() => deleteDb(i)"
>
<feather-icon name="x" class="w-4 h-4" />
</button>
</div>
</div>
<hr v-if="files?.length" />
<!-- Language Selector -->
<div
class="
w-full
flex
justify-between
items-center
absolute
p-4
text-gray-900
"
style="top: 100%; transform: translateY(-100%)"
>
<LanguageSelector v-show="!creatingDemo" class="text-sm w-28" />
<button
class="
text-sm
bg-gray-100
hover:bg-gray-200
rounded
px-4
py-1.5
w-28
h-8
"
@click="createDemo"
:disabled="creatingDemo"
>
{{ creatingDemo ? t`Please Wait` : t`Create Demo` }}
</button>
</div>
</div>
<Loading
v-if="creatingDemo"
:open="creatingDemo"
:show-x="false"
:full-width="true"
:percent="creationPercent"
:message="creationMessage"
/>
<!-- Base Count Selection when Dev -->
<Modal :open-modal="openModal" @closemodal="openModal = false">
<div class="p-4 text-gray-900">
<h2 class="text-xl font-semibold select-none">Set Base Count</h2>
<p class="text-base mt-2">
Base Count is a lower bound on the number of entries made when
creating the dummy instance.
</p>
<div class="flex my-12 justify-center items-baseline gap-4 text-base">
<label for="basecount" class="text-gray-600">Base Count</label>
<input
type="number"
name="basecount"
v-model="baseCount"
class="
bg-gray-100
focus:bg-gray-200
rounded-md
px-2
py-1
outline-none
"
/>
</div>
<div class="flex justify-between">
<Button @click="openModal = false">Cancel</Button>
<Button
@click="
() => {
openModal = false;
startDummyInstanceSetup();
}
"
type="primary"
>Create</Button
>
</div>
</div>
</Modal>
</div>
</template>
<script>
import { setupDummyInstance } from 'dummy';
import { ipcRenderer } from 'electron';
import { t } from 'fyo';
import { DateTime } from 'luxon';
import Button from 'src/components/Button.vue';
import LanguageSelector from 'src/components/Controls/LanguageSelector.vue';
import FeatherIcon from 'src/components/FeatherIcon.vue';
import Loading from 'src/components/Loading.vue';
import Modal from 'src/components/Modal.vue';
import { fyo } from 'src/initFyo';
import { deleteDb, getSavePath } from 'src/utils/ipcCalls';
import { updateConfigFiles } from 'src/utils/misc';
import { showMessageDialog } from 'src/utils/ui';
import { IPC_ACTIONS } from 'utils/messages';
export default {
name: 'DatabaseSelector',
emits: ['file-selected'],
data() {
return {
openModal: false,
baseCount: 100,
creationMessage: '',
creationPercent: 0,
creatingDemo: false,
loadingDatabase: false,
files: [],
};
},
async mounted() {
await this.setFiles();
if (fyo.store.isDevelopment) {
window.ds = this;
}
},
methods: {
formatDate(isoDate) {
return DateTime.fromISO(isoDate).toRelative();
},
async deleteDb(i) {
const file = this.files[i];
const vm = this;
await showMessageDialog({
message: t`Delete ${file.companyName}?`,
detail: t`Database file: ${file.dbPath}`,
buttons: [
{
label: this.t`Yes`,
async action() {
await deleteDb(file.dbPath);
await vm.setFiles();
},
},
{
label: this.t`No`,
action() {},
},
],
});
},
async createDemo() {
if (!fyo.store.isDevelopment) {
this.startDummyInstanceSetup();
} else {
this.openModal = true;
}
},
async startDummyInstanceSetup() {
const { filePath, canceled } = await getSavePath('demo', 'db');
if (canceled || !filePath) {
return;
}
this.creatingDemo = true;
await setupDummyInstance(
filePath,
fyo,
1,
this.baseCount,
(message, percent) => {
this.creationMessage = message;
this.creationPercent = percent;
}
);
updateConfigFiles(fyo);
await fyo.purgeCache();
await this.setFiles();
this.creatingDemo = false;
},
async setFiles() {
this.files = await ipcRenderer.invoke(IPC_ACTIONS.GET_DB_LIST);
},
async newDatabase() {
if (this.creatingDemo) {
return;
}
const { filePath, canceled } = await getSavePath('books', 'db');
if (canceled || !filePath) {
return;
}
this.connectToDatabase(filePath, true);
},
async existingDatabase() {
if (this.creatingDemo) {
return;
}
const filePath = (
await ipcRenderer.invoke(IPC_ACTIONS.GET_OPEN_FILEPATH, {
title: this.t`Select file`,
properties: ['openFile'],
filters: [{ name: 'SQLite DB File', extensions: ['db'] }],
})
)?.filePaths?.[0];
this.connectToDatabase(filePath);
},
async selectFile(file) {
if (this.creatingDemo) {
return;
}
await this.connectToDatabase(file.dbPath);
},
async connectToDatabase(filePath, isNew) {
if (!filePath) {
return;
}
if (isNew) {
this.$emit('file-selected', filePath, isNew);
return;
}
this.$emit('file-selected', filePath, !!isNew);
},
},
components: {
LanguageSelector,
Loading,
FeatherIcon,
Modal,
Button,
},
};
</script>