mirror of
https://github.com/frappe/books.git
synced 2025-01-25 08:08:37 +00:00
f484c2b4aa
- set lighter colour for disabled input - if readonly not tabable
376 lines
9.7 KiB
Vue
376 lines
9.7 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)"
|
|
:title="t`${file.companyName} stored at ${file.dbPath}`"
|
|
>
|
|
<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">
|
|
<div class="flex justify-between overflow-x-auto items-baseline">
|
|
<h2 class="font-medium">
|
|
{{ file.companyName }}
|
|
</h2>
|
|
<p class="whitespace-nowrap text-sm text-gray-600">
|
|
{{ formatDate(file.modified) }}
|
|
</p>
|
|
</div>
|
|
<p
|
|
class="
|
|
text-sm text-gray-600
|
|
overflow-x-auto
|
|
no-scrollbar
|
|
whitespace-nowrap
|
|
"
|
|
>
|
|
{{ truncate(file.dbPath) }}
|
|
</p>
|
|
</div>
|
|
<button
|
|
class="
|
|
ms-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 w-form">
|
|
<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: {
|
|
truncate(value) {
|
|
if (value.length < 72) {
|
|
return value;
|
|
}
|
|
|
|
return '...' + value.slice(value.length - 72);
|
|
},
|
|
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))?.sort(
|
|
(a, b) => Date.parse(b.modified) - Date.parse(a.modified)
|
|
);
|
|
},
|
|
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>
|