mirror of
https://github.com/Fabio286/antares.git
synced 2025-06-05 21:59:22 +02:00
Compare commits
9 Commits
v0.7.28-be
...
v0.7.29-be
Author | SHA1 | Date | |
---|---|---|---|
010147b553 | |||
da8cc39157 | |||
37d44c95ee | |||
4df4c6197d | |||
|
0506b653d7 | ||
5fd9fe48a2 | |||
b6a7124f33 | |||
|
5855ab0921 | ||
c2b602785a |
20
CHANGELOG.md
20
CHANGELOG.md
@@ -2,6 +2,26 @@
|
||||
|
||||
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
||||
|
||||
### [0.7.29-beta.0](https://github.com/antares-sql/antares/compare/v0.7.28...v0.7.29-beta.0) (2024-09-23)
|
||||
|
||||
|
||||
### Features
|
||||
|
||||
* cancel button when waiting to connect database, closes [#830](https://github.com/antares-sql/antares/issues/830) ([b6a7124](https://github.com/antares-sql/antares/commit/b6a7124f33397a2ae7da654b5867f6982ac5810e))
|
||||
* update czech translation ([0506b65](https://github.com/antares-sql/antares/commit/0506b653d74d8cd5e848bc2ec4d29d4b0247c880))
|
||||
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* mismatch between table field columns and results with duplicate fields, fixes [#848](https://github.com/antares-sql/antares/issues/848) ([da8cc39](https://github.com/antares-sql/antares/commit/da8cc39157a4b507d3d377ee1e888b8f8a52b7c5))
|
||||
|
||||
|
||||
### Improvements
|
||||
|
||||
* **UI:** hide edit/delete functions in readonly mode ([37d44c9](https://github.com/antares-sql/antares/commit/37d44c95ee559f3ee1345e91fca5e2c1e86c5fbf))
|
||||
|
||||
### [0.7.28](https://github.com/antares-sql/antares/compare/v0.7.28-beta.0...v0.7.28) (2024-08-20)
|
||||
|
||||
### [0.7.28-beta.0](https://github.com/antares-sql/antares/compare/v0.7.27...v0.7.28-beta.0) (2024-08-06)
|
||||
|
||||
|
||||
|
10499
package-lock.json
generated
10499
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "antares",
|
||||
"productName": "Antares",
|
||||
"version": "0.7.28-beta.0",
|
||||
"version": "0.7.29-beta.0",
|
||||
"description": "A modern, fast and productivity driven SQL client with a focus in UX.",
|
||||
"license": "MIT",
|
||||
"repository": "https://github.com/antares-sql/antares.git",
|
||||
|
@@ -18,7 +18,7 @@ export type Importer = MySQLImporter | PostgreSQLImporter
|
||||
|
||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||
export interface IpcResponse<T = any> {
|
||||
status: 'success' | 'error';
|
||||
status: 'success' | 'error' | 'abort';
|
||||
response?: T;
|
||||
}
|
||||
|
||||
|
@@ -5,11 +5,21 @@ import { SslOptions } from 'mysql2';
|
||||
|
||||
import { ClientsFactory } from '../libs/ClientsFactory';
|
||||
import { validateSender } from '../libs/misc/validateSender';
|
||||
const isAborting: Record<string, boolean> = {};
|
||||
|
||||
export default (connections: Record<string, antares.Client>) => {
|
||||
ipcMain.handle('test-connection', async (event, conn: antares.ConnectionParams) => {
|
||||
if (!validateSender(event.senderFrame)) return { status: 'error', response: 'Unauthorized process' };
|
||||
|
||||
let isLocalAborted = false;
|
||||
const abortChecker = setInterval(() => { // Intercepts abort request
|
||||
if (isAborting[conn.uid]) {
|
||||
isAborting[conn.uid] = false;
|
||||
isLocalAborted = true;
|
||||
clearInterval(abortChecker);
|
||||
}
|
||||
}, 50);
|
||||
|
||||
const params = {
|
||||
host: conn.host,
|
||||
port: +conn.port,
|
||||
@@ -65,19 +75,27 @@ export default (connections: Record<string, antares.Client>) => {
|
||||
client: conn.client,
|
||||
params
|
||||
});
|
||||
await connection.connect();
|
||||
|
||||
if (conn.client === 'firebird')
|
||||
connection.raw('SELECT rdb$get_context(\'SYSTEM\', \'DB_NAME\') FROM rdb$database');
|
||||
else
|
||||
await connection.select('1+1').run();
|
||||
await connection.connect();
|
||||
if (isLocalAborted) {
|
||||
connection.destroy();
|
||||
return;
|
||||
}
|
||||
|
||||
await connection.ping();
|
||||
|
||||
connection.destroy();
|
||||
clearInterval(abortChecker);
|
||||
|
||||
return { status: 'success' };
|
||||
}
|
||||
catch (err) {
|
||||
return { status: 'error', response: err.toString() };
|
||||
clearInterval(abortChecker);
|
||||
|
||||
if (!isLocalAborted)
|
||||
return { status: 'error', response: err.toString() };
|
||||
else
|
||||
return { status: 'abort', response: 'Connection aborted' };
|
||||
}
|
||||
});
|
||||
|
||||
@@ -88,6 +106,15 @@ export default (connections: Record<string, antares.Client>) => {
|
||||
ipcMain.handle('connect', async (event, conn: antares.ConnectionParams) => {
|
||||
if (!validateSender(event.senderFrame)) return { status: 'error', response: 'Unauthorized process' };
|
||||
|
||||
let isLocalAborted = false;
|
||||
const abortChecker = setInterval(() => { // Intercepts abort request
|
||||
if (isAborting[conn.uid]) {
|
||||
isAborting[conn.uid] = false;
|
||||
isLocalAborted = true;
|
||||
clearInterval(abortChecker);
|
||||
}
|
||||
}, 50);
|
||||
|
||||
const params = {
|
||||
host: conn.host,
|
||||
port: +conn.port,
|
||||
@@ -150,18 +177,36 @@ export default (connections: Record<string, antares.Client>) => {
|
||||
});
|
||||
|
||||
await connection.connect();
|
||||
if (isLocalAborted) {
|
||||
connection.destroy();
|
||||
return { status: 'abort', response: 'Connection aborted' };
|
||||
}
|
||||
|
||||
const structure = await connection.getStructure(new Set());
|
||||
if (isLocalAborted) {
|
||||
connection.destroy();
|
||||
return { status: 'abort', response: 'Connection aborted' };
|
||||
}
|
||||
|
||||
connections[conn.uid] = connection;
|
||||
clearInterval(abortChecker);
|
||||
|
||||
return { status: 'success', response: structure };
|
||||
}
|
||||
catch (err) {
|
||||
return { status: 'error', response: err.toString() };
|
||||
clearInterval(abortChecker);
|
||||
|
||||
if (!isLocalAborted)
|
||||
return { status: 'error', response: err.toString() };
|
||||
else
|
||||
return { status: 'abort', response: 'Connection aborted' };
|
||||
}
|
||||
});
|
||||
|
||||
ipcMain.on('abort-connection', (event, uid) => {
|
||||
isAborting[uid] = true;
|
||||
});
|
||||
|
||||
ipcMain.handle('disconnect', (event, uid) => {
|
||||
if (!validateSender(event.senderFrame)) return { status: 'error', response: 'Unauthorized process' };
|
||||
|
||||
|
@@ -109,6 +109,10 @@ export class FirebirdSQLClient extends BaseClient {
|
||||
return firebird.pool(this._poolSize, { ...this._params, blobAsText: true });
|
||||
}
|
||||
|
||||
ping () {
|
||||
return this.raw('SELECT rdb$get_context(\'SYSTEM\', \'DB_NAME\') FROM rdb$database');
|
||||
}
|
||||
|
||||
destroy () {
|
||||
if (this._poolSize)
|
||||
return (this._connection as firebird.ConnectionPool).destroy();
|
||||
|
@@ -214,6 +214,10 @@ export class MySQLClient extends BaseClient {
|
||||
}
|
||||
}
|
||||
|
||||
ping () {
|
||||
return this.select('1+1').run();
|
||||
}
|
||||
|
||||
destroy () {
|
||||
this._connection.end();
|
||||
clearInterval(this._keepaliveTimer);
|
||||
|
@@ -243,6 +243,10 @@ export class PostgreSQLClient extends BaseClient {
|
||||
return connection;
|
||||
}
|
||||
|
||||
ping () {
|
||||
return this.select('1+1').run();
|
||||
}
|
||||
|
||||
destroy () {
|
||||
this._connection.end();
|
||||
clearInterval(this._keepaliveTimer);
|
||||
|
@@ -35,6 +35,10 @@ export class SQLiteClient extends BaseClient {
|
||||
});
|
||||
}
|
||||
|
||||
ping () {
|
||||
return this.select('1+1').run();
|
||||
}
|
||||
|
||||
destroy () {
|
||||
this._connection.close();
|
||||
}
|
||||
|
@@ -386,20 +386,35 @@
|
||||
</div>
|
||||
</div>
|
||||
<div class="panel-footer">
|
||||
<button
|
||||
id="connection-test"
|
||||
class="btn btn-gray mr-2 d-flex"
|
||||
:class="{'loading': isTesting}"
|
||||
:disabled="isBusy"
|
||||
@click="startTest"
|
||||
<div
|
||||
@mouseenter="setCancelTestButtonVisibility(true)"
|
||||
@mouseleave="setCancelTestButtonVisibility(false)"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiLightningBolt"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.testConnection') }}
|
||||
</button>
|
||||
<button
|
||||
v-if="showTestCancel && isTesting"
|
||||
class="btn btn-gray mr-2 cancellable"
|
||||
:title="t('general.cancel')"
|
||||
@click="abortConnection()"
|
||||
>
|
||||
<BaseIcon icon-name="mdiWindowClose" :size="24" />
|
||||
<span class="d-invisible pr-1">{{ t('connection.testConnection') }}</span>
|
||||
</button>
|
||||
<button
|
||||
v-else
|
||||
id="connection-test"
|
||||
class="btn btn-gray mr-2 d-flex"
|
||||
:class="{'loading': isTesting}"
|
||||
:disabled="isBusy"
|
||||
@click="startTest"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiLightningBolt"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.testConnection') }}
|
||||
</button>
|
||||
</div>
|
||||
<button
|
||||
id="connection-save"
|
||||
class="btn btn-primary mr-2 d-flex"
|
||||
@@ -494,6 +509,8 @@ const firstInput: Ref<HTMLInputElement> = ref(null);
|
||||
const isConnecting = ref(false);
|
||||
const isTesting = ref(false);
|
||||
const isAsking = ref(false);
|
||||
const showTestCancel = ref(false);
|
||||
const abortController: Ref<AbortController> = ref(new AbortController());
|
||||
const selectedTab = ref('general');
|
||||
|
||||
const clientCustomizations = computed(() => {
|
||||
@@ -516,6 +533,10 @@ const setDefaults = () => {
|
||||
connection.value.database = clientCustomizations.value.defaultDatabase;
|
||||
};
|
||||
|
||||
const setCancelTestButtonVisibility = (val: boolean) => {
|
||||
showTestCancel.value = val;
|
||||
};
|
||||
|
||||
const startTest = async () => {
|
||||
isTesting.value = true;
|
||||
|
||||
@@ -526,7 +547,7 @@ const startTest = async () => {
|
||||
const res = await Connection.makeTest(connection.value);
|
||||
if (res.status === 'error')
|
||||
addNotification({ status: 'error', message: res.response.message || res.response.toString() });
|
||||
else
|
||||
else if (res.status === 'success')
|
||||
addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') });
|
||||
}
|
||||
catch (err) {
|
||||
@@ -537,13 +558,21 @@ const startTest = async () => {
|
||||
}
|
||||
};
|
||||
|
||||
const abortConnection = (): void => {
|
||||
abortController.value.abort();
|
||||
Connection.abortConnection(connection.value.uid);
|
||||
isTesting.value = false;
|
||||
isConnecting.value = false;
|
||||
abortController.value = new AbortController();
|
||||
};
|
||||
|
||||
const continueTest = async (credentials: { user: string; password: string }) => { // if "Ask for credentials" is true
|
||||
isAsking.value = false;
|
||||
const params = Object.assign({}, connection.value, credentials);
|
||||
|
||||
try {
|
||||
if (isConnecting.value) {
|
||||
await connectWorkspace(params);
|
||||
await connectWorkspace(params, { signal: abortController.value.signal }).catch(() => undefined);
|
||||
isConnecting.value = false;
|
||||
}
|
||||
else {
|
||||
|
@@ -387,20 +387,35 @@
|
||||
</div>
|
||||
</div>
|
||||
<div class="panel-footer">
|
||||
<button
|
||||
id="connection-test"
|
||||
class="btn btn-gray mr-2 d-flex"
|
||||
:class="{'loading': isTesting}"
|
||||
:disabled="isBusy"
|
||||
@click="startTest"
|
||||
<div
|
||||
@mouseenter="setCancelTestButtonVisibility(true)"
|
||||
@mouseleave="setCancelTestButtonVisibility(false)"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiLightningBolt"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.testConnection') }}
|
||||
</button>
|
||||
<button
|
||||
v-if="showTestCancel && isTesting"
|
||||
class="btn btn-gray mr-2 cancellable"
|
||||
:title="t('general.cancel')"
|
||||
@click="abortConnection()"
|
||||
>
|
||||
<BaseIcon icon-name="mdiWindowClose" :size="24" />
|
||||
<span class="d-invisible pr-1">{{ t('connection.testConnection') }}</span>
|
||||
</button>
|
||||
<button
|
||||
v-else
|
||||
id="connection-test"
|
||||
class="btn btn-gray mr-2 d-flex"
|
||||
:class="{'loading': isTesting}"
|
||||
:disabled="isBusy"
|
||||
@click="startTest"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiLightningBolt"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.testConnection') }}
|
||||
</button>
|
||||
</div>
|
||||
<button
|
||||
id="connection-save"
|
||||
class="btn btn-primary mr-2 d-flex"
|
||||
@@ -414,20 +429,35 @@
|
||||
/>
|
||||
{{ t('general.save') }}
|
||||
</button>
|
||||
<button
|
||||
id="connection-connect"
|
||||
class="btn btn-success d-flex"
|
||||
:class="{'loading': isConnecting}"
|
||||
:disabled="isBusy"
|
||||
@click="startConnection"
|
||||
<div
|
||||
@mouseenter="setCancelConnectButtonVisibility(true)"
|
||||
@mouseleave="setCancelConnectButtonVisibility(false)"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiConnection"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.connect') }}
|
||||
</button>
|
||||
<button
|
||||
v-if="showConnectCancel && isConnecting"
|
||||
class="btn btn-success cancellable"
|
||||
:title="t('general.cancel')"
|
||||
@click="abortConnection()"
|
||||
>
|
||||
<BaseIcon icon-name="mdiWindowClose" :size="24" />
|
||||
<span class="d-invisible pr-1">{{ t('connection.connect') }}</span>
|
||||
</button>
|
||||
<button
|
||||
v-else
|
||||
id="connection-connect"
|
||||
class="btn btn-success d-flex"
|
||||
:class="{'loading': isConnecting}"
|
||||
:disabled="isBusy"
|
||||
@click="startConnection"
|
||||
>
|
||||
<BaseIcon
|
||||
icon-name="mdiConnection"
|
||||
:size="24"
|
||||
class="mr-1"
|
||||
/>
|
||||
{{ t('connection.connect') }}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<ModalAskCredentials
|
||||
@@ -476,6 +506,9 @@ const localConnection: Ref<ConnectionParams & { pgConnString: string }> = ref(nu
|
||||
const isConnecting = ref(false);
|
||||
const isTesting = ref(false);
|
||||
const isAsking = ref(false);
|
||||
const showTestCancel = ref(false);
|
||||
const showConnectCancel = ref(false);
|
||||
const abortController: Ref<AbortController> = ref(new AbortController());
|
||||
const selectedTab = ref('general');
|
||||
|
||||
const clientCustomizations = computed(() => {
|
||||
@@ -501,7 +534,7 @@ const startConnection = async () => {
|
||||
if (localConnection.value.ask)
|
||||
isAsking.value = true;
|
||||
else {
|
||||
await connectWorkspace(localConnection.value);
|
||||
await connectWorkspace(localConnection.value, { signal: abortController.value.signal }).catch(() => undefined);
|
||||
isConnecting.value = false;
|
||||
}
|
||||
};
|
||||
@@ -516,7 +549,7 @@ const startTest = async () => {
|
||||
const res = await Connection.makeTest(localConnection.value);
|
||||
if (res.status === 'error')
|
||||
addNotification({ status: 'error', message: res.response.message || res.response.toString() });
|
||||
else
|
||||
else if (res.status === 'success')
|
||||
addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') });
|
||||
}
|
||||
catch (err) {
|
||||
@@ -527,20 +560,36 @@ const startTest = async () => {
|
||||
}
|
||||
};
|
||||
|
||||
const setCancelTestButtonVisibility = (val: boolean) => {
|
||||
showTestCancel.value = val;
|
||||
};
|
||||
|
||||
const setCancelConnectButtonVisibility = (val: boolean) => {
|
||||
showConnectCancel.value = val;
|
||||
};
|
||||
|
||||
const abortConnection = (): void => {
|
||||
abortController.value.abort();
|
||||
Connection.abortConnection(localConnection.value.uid);
|
||||
isTesting.value = false;
|
||||
isConnecting.value = false;
|
||||
abortController.value = new AbortController();
|
||||
};
|
||||
|
||||
const continueTest = async (credentials: {user: string; password: string }) => { // if "Ask for credentials" is true
|
||||
isAsking.value = false;
|
||||
const params = Object.assign({}, localConnection.value, credentials);
|
||||
try {
|
||||
if (isConnecting.value) {
|
||||
const params = Object.assign({}, props.connection, credentials);
|
||||
await connectWorkspace(params);
|
||||
await connectWorkspace(params, { signal: abortController.value.signal }).catch(() => undefined);
|
||||
isConnecting.value = false;
|
||||
}
|
||||
else {
|
||||
const res = await Connection.makeTest(params);
|
||||
if (res.status === 'error')
|
||||
addNotification({ status: 'error', message: res.response.message || res.response.toString() });
|
||||
else
|
||||
else if (res.status === 'success')
|
||||
addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') });
|
||||
}
|
||||
}
|
||||
|
@@ -16,7 +16,7 @@
|
||||
/> {{ t('general.run') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="selectedMisc.type === 'trigger' && customizations.triggerEnableDisable"
|
||||
v-if="selectedMisc.type === 'trigger' && customizations.triggerEnableDisable && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="toggleTrigger"
|
||||
>
|
||||
@@ -36,7 +36,7 @@
|
||||
</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="selectedMisc.type === 'scheduler'"
|
||||
v-if="selectedMisc.type === 'scheduler' && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="toggleScheduler"
|
||||
>
|
||||
@@ -63,7 +63,11 @@
|
||||
:size="18"
|
||||
/> {{ t('general.copyName') }}</span>
|
||||
</div>
|
||||
<div class="context-element" @click="showDeleteModal">
|
||||
<div
|
||||
v-if="!connection.readonly"
|
||||
class="context-element"
|
||||
@click="showDeleteModal"
|
||||
>
|
||||
<span class="d-flex">
|
||||
<BaseIcon
|
||||
class="text-light mt-1 mr-1"
|
||||
@@ -117,6 +121,7 @@ import Routines from '@/ipc-api/Routines';
|
||||
import Schedulers from '@/ipc-api/Schedulers';
|
||||
import Triggers from '@/ipc-api/Triggers';
|
||||
import { copyText } from '@/libs/copyText';
|
||||
import { useConnectionsStore } from '@/stores/connections';
|
||||
import { useNotificationsStore } from '@/stores/notifications';
|
||||
import { useWorkspacesStore } from '@/stores/workspaces';
|
||||
|
||||
@@ -132,6 +137,7 @@ const emit = defineEmits(['close-context', 'reload']);
|
||||
|
||||
const { addNotification } = useNotificationsStore();
|
||||
const workspacesStore = useWorkspacesStore();
|
||||
const { getConnectionByUid } = useConnectionsStore();
|
||||
|
||||
const { getSelected: selectedWorkspace } = storeToRefs(workspacesStore);
|
||||
|
||||
@@ -154,6 +160,7 @@ const workspace = computed(() => {
|
||||
const customizations = computed(() => {
|
||||
return getWorkspace(selectedWorkspace.value).customizations;
|
||||
});
|
||||
const connection = computed(() => getConnectionByUid(selectedWorkspace.value));
|
||||
|
||||
const deleteMessage = computed(() => {
|
||||
switch (props.selectedMisc.type) {
|
||||
|
@@ -3,7 +3,7 @@
|
||||
:context-event="contextEvent"
|
||||
@close-context="closeContext"
|
||||
>
|
||||
<div class="context-element">
|
||||
<div v-if="!connection.readonly" class="context-element">
|
||||
<span class="d-flex">
|
||||
<BaseIcon
|
||||
class="text-light mt-1 mr-1"
|
||||
@@ -135,7 +135,7 @@
|
||||
/> {{ t('database.export') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="workspace.customizations.schemaImport"
|
||||
v-if="workspace.customizations.schemaImport && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="initImport"
|
||||
>
|
||||
@@ -147,7 +147,7 @@
|
||||
/> {{ t('database.import') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="workspace.customizations.schemaEdit"
|
||||
v-if="workspace.customizations.schemaEdit && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="showEditModal"
|
||||
>
|
||||
@@ -159,7 +159,7 @@
|
||||
/> {{ t('database.editSchema') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="workspace.customizations.schemaDrop"
|
||||
v-if="workspace.customizations.schemaDrop && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="showDeleteModal"
|
||||
>
|
||||
@@ -219,6 +219,7 @@ import ModalImportSchema from '@/components/ModalImportSchema.vue';
|
||||
import Application from '@/ipc-api/Application';
|
||||
import Schema from '@/ipc-api/Schema';
|
||||
import { copyText } from '@/libs/copyText';
|
||||
import { useConnectionsStore } from '@/stores/connections';
|
||||
import { useNotificationsStore } from '@/stores/notifications';
|
||||
import { useSchemaExportStore } from '@/stores/schemaExport';
|
||||
import { useWorkspacesStore } from '@/stores/workspaces';
|
||||
@@ -248,7 +249,9 @@ const workspacesStore = useWorkspacesStore();
|
||||
const schemaExportStore = useSchemaExportStore();
|
||||
const { showExportModal } = schemaExportStore;
|
||||
|
||||
const connectionsStore = useConnectionsStore();
|
||||
const { getSelected: selectedWorkspace } = storeToRefs(workspacesStore);
|
||||
const { getConnectionByUid } = connectionsStore;
|
||||
|
||||
const {
|
||||
getWorkspace,
|
||||
@@ -261,6 +264,7 @@ const isEditModal = ref(false);
|
||||
const isImportSchemaModal = ref(false);
|
||||
|
||||
const workspace = computed(() => getWorkspace(selectedWorkspace.value));
|
||||
const connection = computed(() => getConnectionByUid(selectedWorkspace.value));
|
||||
|
||||
const openCreateTableTab = () => {
|
||||
emit('open-create-table-tab');
|
||||
|
@@ -60,7 +60,7 @@
|
||||
/> {{ t('application.settings') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="selectedTable && selectedTable.type === 'table' && customizations.tableDuplicate"
|
||||
v-if="selectedTable && selectedTable.type === 'table' && customizations.tableDuplicate && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="duplicateTable"
|
||||
>
|
||||
@@ -72,7 +72,7 @@
|
||||
/> {{ t('database.duplicateTable') }}</span>
|
||||
</div>
|
||||
<div
|
||||
v-if="selectedTable && selectedTable.type === 'table'"
|
||||
v-if="selectedTable && selectedTable.type === 'table' && !connection.readonly"
|
||||
class="context-element"
|
||||
@click="showEmptyModal"
|
||||
>
|
||||
@@ -83,7 +83,11 @@
|
||||
:size="18"
|
||||
/> {{ t('database.emptyTable') }}</span>
|
||||
</div>
|
||||
<div class="context-element" @click="showDeleteModal">
|
||||
<div
|
||||
v-if="!connection.readonly"
|
||||
class="context-element"
|
||||
@click="showDeleteModal"
|
||||
>
|
||||
<span class="d-flex">
|
||||
<BaseIcon
|
||||
class="text-light mt-1 mr-1"
|
||||
@@ -151,6 +155,7 @@ import BaseContextMenu from '@/components/BaseContextMenu.vue';
|
||||
import BaseIcon from '@/components/BaseIcon.vue';
|
||||
import Tables from '@/ipc-api/Tables';
|
||||
import { copyText } from '@/libs/copyText';
|
||||
import { useConnectionsStore } from '@/stores/connections';
|
||||
import { useNotificationsStore } from '@/stores/notifications';
|
||||
import { useSchemaExportStore } from '@/stores/schemaExport';
|
||||
import { useWorkspacesStore } from '@/stores/workspaces';
|
||||
@@ -168,6 +173,7 @@ const emit = defineEmits(['close-context', 'duplicate-table', 'reload', 'delete-
|
||||
const { addNotification } = useNotificationsStore();
|
||||
const workspacesStore = useWorkspacesStore();
|
||||
const { showExportModal } = useSchemaExportStore();
|
||||
const { getConnectionByUid } = useConnectionsStore();
|
||||
|
||||
const { getSelected: selectedWorkspace } = storeToRefs(workspacesStore);
|
||||
|
||||
@@ -185,6 +191,7 @@ const forceTruncate = ref(false);
|
||||
|
||||
const workspace = computed(() => getWorkspace(selectedWorkspace.value));
|
||||
const customizations = computed(() => workspace.value && workspace.value.customizations ? workspace.value.customizations : null);
|
||||
const connection = computed(() => getConnectionByUid(selectedWorkspace.value));
|
||||
|
||||
const showTableExportModal = () => {
|
||||
showExportModal(props.selectedSchema, props.selectedTable.name);
|
||||
|
@@ -38,7 +38,7 @@
|
||||
<div class="thead">
|
||||
<div class="tr">
|
||||
<div
|
||||
v-for="(field, index) in fields"
|
||||
v-for="(field, index) in filteredFields"
|
||||
:key="index"
|
||||
class="th c-hand"
|
||||
:title="`${field.type} ${fieldLength(field) ? `(${fieldLength(field)})` : ''}`"
|
||||
@@ -383,6 +383,11 @@ const sortedResults = computed(() => {
|
||||
|
||||
const resultsWithRows = computed(() => props.results.filter(result => result.rows.length));
|
||||
const fields = computed(() => resultsWithRows.value.length ? resultsWithRows.value[resultsetIndex.value].fields : []);
|
||||
const filteredFields = computed(() => fields.value.reduce((acc, cur) => {
|
||||
if (acc.findIndex(f => JSON.stringify(f) === JSON.stringify(cur)))
|
||||
acc.push(cur);
|
||||
return acc;
|
||||
}, [] as TableField[]));
|
||||
const keyUsage = computed(() => resultsWithRows.value.length ? resultsWithRows.value[resultsetIndex.value].keys : []);
|
||||
|
||||
const fieldsObj = computed(() => {
|
||||
|
@@ -1,3 +1,13 @@
|
||||
|
||||
/**
|
||||
* [TRANSLATION UPDATE HELPER]
|
||||
* - Open a terminal in antares folder and run `npm run translation:check short-code` replacing short-code with the one you are updating.
|
||||
* - The command will output which terms are missing or not translated from english.
|
||||
* - Open antares folder with your editor of choice.
|
||||
* - Go to antares/src/renderer/i18n/ and open the locale file you want to translate.
|
||||
* - Add and translate missing terms and consider whether to translate untranslated terms.
|
||||
*/
|
||||
|
||||
export const csCZ = {
|
||||
general: { // General purpose terms
|
||||
edit: 'Upravit',
|
||||
@@ -18,7 +28,7 @@ export const csCZ = {
|
||||
download: 'Stáhnout',
|
||||
add: 'Přidat',
|
||||
data: 'Data',
|
||||
properties: 'Vlastnosti', // Lawondyss: Not used
|
||||
properties: 'Vlastnosti',
|
||||
name: 'Název',
|
||||
clear: 'Vyčistit',
|
||||
options: 'Možnosti',
|
||||
@@ -39,6 +49,7 @@ export const csCZ = {
|
||||
new: 'Nové',
|
||||
select: 'Vybrat',
|
||||
change: 'Změnit',
|
||||
include: 'Včetně',
|
||||
includes: 'Zahrnout',
|
||||
completed: 'Dokončeno',
|
||||
aborted: 'Zrušeno',
|
||||
@@ -46,15 +57,15 @@ export const csCZ = {
|
||||
enable: 'Zapnuto',
|
||||
disable: 'Vypnout',
|
||||
contributors: 'Přispěvatelé',
|
||||
pin: 'Připnout', // Lawondyss: Not used
|
||||
unpin: 'Odepnout', // Lawondyss: Not used
|
||||
folder: 'Složka | Složky', // Lawondyss: Used only 1
|
||||
pin: 'Připnout',
|
||||
unpin: 'Odepnout',
|
||||
folder: 'Složka | Složky',
|
||||
none: 'Nic',
|
||||
singleQuote: 'Apostrofy',
|
||||
doubleQuote: 'Uvozovky',
|
||||
deleteConfirm: 'Skutečně chcete smazat',
|
||||
uploadFile: 'Nahrát soubor',
|
||||
format: 'Formátovat', // Format code
|
||||
format: 'Formátovat',
|
||||
history: 'Historie',
|
||||
filter: 'Filtrovat',
|
||||
manualValue: 'Vlastní hodnota',
|
||||
@@ -64,9 +75,16 @@ export const csCZ = {
|
||||
actionSuccessful: '{action} úspěšný',
|
||||
outputFormat: 'Formát výstupu',
|
||||
singleFile: 'Jediný {ext} soubor',
|
||||
zipCompressedFile: 'ZIP komprimovaný {ext} soubor'
|
||||
zipCompressedFile: 'ZIP komprimovaný {ext} soubor',
|
||||
copyName: 'Kopírovat název',
|
||||
search: 'Hledat',
|
||||
title: 'Titulek',
|
||||
archive: 'Archivovat', // verb
|
||||
undo: 'Zpět',
|
||||
moveTo: 'Přesunout do'
|
||||
},
|
||||
connection: { // Database connection
|
||||
connection: 'Připojení',
|
||||
connectionName: 'Název',
|
||||
hostName: 'Host',
|
||||
client: 'Klient',
|
||||
@@ -75,9 +93,9 @@ export const csCZ = {
|
||||
password: 'Heslo',
|
||||
credentials: 'Pověření',
|
||||
connect: 'Připojit',
|
||||
connected: 'Připojeno', // Lawondyss: Not used
|
||||
connected: 'Připojeno',
|
||||
disconnect: 'Odpojit',
|
||||
disconnected: 'Odpojeno', // Lawondyss: Not used
|
||||
disconnected: 'Odpojeno',
|
||||
ssl: 'SSL',
|
||||
enableSsl: 'Použít SSL',
|
||||
privateKey: 'Soukromý klíč',
|
||||
@@ -90,28 +108,30 @@ export const csCZ = {
|
||||
enableSsh: 'Použít SSH',
|
||||
connectionString: 'String připojení',
|
||||
addConnection: 'Add connection',
|
||||
createConnection: 'Vytvořit připojení', // Lawondyss: Not used
|
||||
createConnection: 'Vytvořit připojení',
|
||||
createNewConnection: 'Vytvořit nové připojení',
|
||||
askCredentials: 'Vyžadovat přihlášení',
|
||||
testConnection: 'Vyzkoušet',
|
||||
editConnection: 'Upravit připojení', // Lawondyss: Not used
|
||||
editConnection: 'Upravit připojení',
|
||||
deleteConnection: 'Smazat připojení',
|
||||
connectionSuccessfullyMade: 'Spojení úspěšně navázáno!',
|
||||
readOnlyMode: 'Pouze pro čtení',
|
||||
allConnections: 'Všechna připojení',
|
||||
searchForConnections: 'Hledat připojení'
|
||||
searchForConnections: 'Hledat připojení',
|
||||
keepAliveInterval: 'Keep alive interval',
|
||||
singleConnection: 'Jediné spojení'
|
||||
},
|
||||
database: { // Database related terms
|
||||
schema: 'Schéma',
|
||||
type: 'Typ',
|
||||
insert: 'Vložit', // Lawondyss: Not used
|
||||
insert: 'Vložit',
|
||||
indexes: 'Indexy',
|
||||
foreignKeys: 'Cizí klíče',
|
||||
length: 'Délka',
|
||||
unsigned: 'Unsigned',
|
||||
default: 'Výchozí',
|
||||
comment: 'Komentář',
|
||||
key: 'Klíč | Klíče', // Lawondyss: Used only 2
|
||||
key: 'Klíč | Klíče',
|
||||
order: 'Pořadí',
|
||||
expression: 'Výraz',
|
||||
autoIncrement: 'Auto Increment',
|
||||
@@ -119,8 +139,9 @@ export const csCZ = {
|
||||
field: 'Sloupec | Sloupce',
|
||||
approximately: 'Přibližně',
|
||||
total: 'Celkem',
|
||||
table: 'Tabulka | Tabulky', // Lawondyss: Used only without argument
|
||||
view: 'Pohled | Pohledy', // Lawondyss: Used only without argument
|
||||
table: 'Tabulka | Tabulky',
|
||||
view: 'Pohled | Pohledy',
|
||||
materializedview: 'Materializovaný pohled',
|
||||
definer: 'Definér',
|
||||
algorithm: 'Algoritmus',
|
||||
trigger: 'Trigger | Triggery',
|
||||
@@ -147,25 +168,25 @@ export const csCZ = {
|
||||
row: 'Řádek | Řádky',
|
||||
cell: 'Buňka | Buňky',
|
||||
triggerFunction: 'Trigger funkce | Trigger funkce',
|
||||
routine: 'Routina | Routiny', // Lawondyss: Not used
|
||||
routine: 'Routina | Routiny',
|
||||
drop: 'Drop',
|
||||
commit: 'Commit',
|
||||
rollback: 'Rollback',
|
||||
ddl: 'DDL',
|
||||
collation: 'Porovnání',
|
||||
resultsTable: 'Tabulka výsledků',
|
||||
unableEditFieldWithoutPrimary: 'Nelze upravit buňku bez primárního klíče ve výsledku', // Lawondyss: Not used
|
||||
editCell: 'Upravit buňku', // Lawondyss: Not used
|
||||
unableEditFieldWithoutPrimary: 'Nelze upravit buňku bez primárního klíče ve výsledku',
|
||||
editCell: 'Upravit buňku',
|
||||
deleteRows: 'Smazat řádek | Smazat {count} řádků',
|
||||
confirmToDeleteRows: 'Skutečně chcete smazat řádek? | Skutečně chcete smazat {count} řádků?',
|
||||
addNewRow: 'Přidat nový řádek', // Lawondyss: Not used
|
||||
addNewRow: 'Přidat nový řádek',
|
||||
numberOfInserts: 'Počet opakování',
|
||||
affectedRows: 'Ovlivněno řádků',
|
||||
createNewDatabase: 'Vytvořit novou databázi', // Lawondyss: Not used
|
||||
databaseName: 'Název databáze', // Lawondyss: Not use
|
||||
createNewDatabase: 'Vytvořit novou databázi',
|
||||
databaseName: 'Název databáze',
|
||||
serverDefault: 'Porovnání serveru',
|
||||
deleteDatabase: 'Smazat databázi', // Lawondyss: Not used
|
||||
editDatabase: 'Upravit databázi', // Lawondyss: Not used
|
||||
deleteDatabase: 'Smazat databázi',
|
||||
editDatabase: 'Upravit databázi',
|
||||
clearChanges: 'Zrušit změny',
|
||||
addNewField: 'Přidat nový sloupec',
|
||||
manageIndexes: 'Správa indexů',
|
||||
@@ -177,17 +198,18 @@ export const csCZ = {
|
||||
deleteField: 'Smazat sloupec',
|
||||
createNewIndex: 'Vytvořit nový index',
|
||||
addToIndex: 'Přidat do indexu',
|
||||
createNewTable: 'Vytvořit novou databázi', // Lawondyss: Not used
|
||||
createNewTable: 'Vytvořit novou databázi',
|
||||
emptyTable: 'Smazat obsah tabulky',
|
||||
duplicateTable: 'Duplikovat tabulku',
|
||||
deleteTable: 'Smazat tabulku',
|
||||
exportTable: 'Exportovat tabulku',
|
||||
emptyConfirm: 'Skutečně smazat obsah tabulky',
|
||||
thereAreNoIndexes: 'Nemá žádné indexy',
|
||||
thereAreNoForeign: 'Nemá žádné cizí klíče',
|
||||
createNewForeign: 'Vytvořit nový cizí klíč',
|
||||
referenceTable: 'Ref. tabulka',
|
||||
referenceField: 'Ref. sloupec',
|
||||
foreignFields: 'Cizí sloupce', // Lawondyss: Not used
|
||||
foreignFields: 'Cizí sloupce',
|
||||
invalidDefault: 'Neplatná výchozí hodnota',
|
||||
onDelete: 'Při smazání',
|
||||
selectStatement: 'Definice pohledu',
|
||||
@@ -195,7 +217,8 @@ export const csCZ = {
|
||||
sqlSecurity: 'SQL zabezpečení',
|
||||
updateOption: 'Volba aktualizace',
|
||||
deleteView: 'Smazat pohled',
|
||||
createNewView: 'Vytvořit nový pohled', // Lawondyss: Not used
|
||||
createNewView: 'Vytvořit nový pohled',
|
||||
createNewMaterializedView: 'Vytvořit nový materializovaný pohled',
|
||||
deleteTrigger: 'Smazat trigger',
|
||||
createNewTrigger: 'Vytvořit nový trigger',
|
||||
currentUser: 'Současný uživatel',
|
||||
@@ -212,7 +235,7 @@ export const csCZ = {
|
||||
createNewScheduler: 'Vytvořit nový scheduler',
|
||||
deleteScheduler: 'Smazat scheduler',
|
||||
preserveOnCompletion: 'Uchovat po dokončení',
|
||||
tableFiller: 'Vyplňovač tabulky', // Lawondyss: Not used
|
||||
tableFiller: 'Vyplňovač tabulky',
|
||||
fakeDataLanguage: 'Jazyk pro fake data',
|
||||
queryDuration: 'Doba trvání dotazu',
|
||||
setNull: 'Nastavit NULL',
|
||||
@@ -224,10 +247,11 @@ export const csCZ = {
|
||||
editSchema: 'Upravit schéma',
|
||||
deleteSchema: 'Smazat schema',
|
||||
noSchema: 'Bez schématu',
|
||||
runQuery: 'Spustit dotaz', // Lawondyss: Not used
|
||||
runQuery: 'Spustit dotaz',
|
||||
thereAreNoTableFields: 'Nemá žádné sloupce',
|
||||
newTable: 'Nová tabulka',
|
||||
newView: 'Nový pohled',
|
||||
newMaterializedView: 'Nový materializovaný pohled',
|
||||
newTrigger: 'Nový trigger',
|
||||
newRoutine: 'Nová routina',
|
||||
newFunction: 'Nová funkce',
|
||||
@@ -244,17 +268,17 @@ export const csCZ = {
|
||||
writingTableExport: 'Zapisuji data tabulky {table}',
|
||||
checkAllTables: 'Vybrat všechny tabulky',
|
||||
uncheckAllTables: 'Vybrat žádnou tabulku',
|
||||
killQuery: 'Zabít dotaz', // Lawondyss: Not used
|
||||
insertRow: 'Vložit řádek | Vložit řádky', // Lawondyss: Used only 2
|
||||
killQuery: 'Zabít dotaz',
|
||||
insertRow: 'Vložit řádek | Vložit řádky',
|
||||
commitMode: 'Způsob commitování',
|
||||
autoCommit: 'Auto commit',
|
||||
manualCommit: 'Ruční commit',
|
||||
importQueryErrors: 'Varování: došlo k chybě {n} | Varování: došlo k chybám {n}', // Lawondyss: Used without n argument
|
||||
executedQueries: '{n} dotaz spuštěn | {n} dotazy spuštěny', // Lawondyss: Used withoum n argument
|
||||
importQueryErrors: 'Varování: došlo k chybě {n} | Varování: došlo k chybám {n}',
|
||||
executedQueries: '{n} dotaz spuštěn | {n} dotazy spuštěny',
|
||||
disableFKChecks: 'Vypnout kontrolu cizích klíčů',
|
||||
formatQuery: 'Formátovat dotaz', // Lawondyss: Not used, probably duplicate to general.format
|
||||
queryHistory: 'Historie dotazů', // Lawondyss: Not used, probably duplicate to general.history
|
||||
clearQuery: 'Clear query', // Lawondyss: Not used, probably duplicate to general.clear
|
||||
formatQuery: 'Formátovat dotaz',
|
||||
queryHistory: 'Historie dotazů',
|
||||
clearQuery: 'Clear query',
|
||||
fillCell: 'Vyplnit buňku',
|
||||
executeSelectedQuery: 'Spustit vybraný dotaz',
|
||||
noResultsPresent: 'Nejsou k dispozici žádné výsledky',
|
||||
@@ -262,12 +286,11 @@ export const csCZ = {
|
||||
targetTable: 'Cílová tabulka',
|
||||
switchDatabase: 'Přepnout databázi',
|
||||
searchForElements: 'Vyhledávání prvků',
|
||||
searchForSchemas: 'Vyhledávání schémat'
|
||||
searchForSchemas: 'Vyhledávání schémat',
|
||||
savedQueries: 'Uložit dotazy'
|
||||
},
|
||||
application: { // Application related terms
|
||||
settings: 'Nastavení',
|
||||
scratchpad: 'Zápisník',
|
||||
disableScratchpad: 'Vypnout zápisník',
|
||||
console: 'Konzole',
|
||||
general: 'Obecné',
|
||||
themes: 'Motivy',
|
||||
@@ -275,7 +298,7 @@ export const csCZ = {
|
||||
about: 'Informace',
|
||||
language: 'Jazyk',
|
||||
shortcuts: 'Zkratky',
|
||||
key: 'Klávesa | Klávesy', // Keyboard key // Lawondyss: Usedn only 2
|
||||
key: 'Klávesa | Klávesy', // Keyboard key
|
||||
event: 'Akce',
|
||||
light: 'Světlý',
|
||||
dark: 'Tmavý',
|
||||
@@ -283,13 +306,19 @@ export const csCZ = {
|
||||
application: 'Aplikace',
|
||||
editor: 'Editor',
|
||||
changelog: 'Changelog',
|
||||
small: 'Malé', // Lawondyss: Not used, probably obsolete font size settings
|
||||
medium: 'Střední', // Lawondyss: Not used, probably obsolete font size settings
|
||||
large: 'Velké', // Lawondyss: Not used, probably obsolete font size settings
|
||||
small: 'Malé',
|
||||
medium: 'Střední',
|
||||
large: 'Velké',
|
||||
appearance: 'Vzhled',
|
||||
color: 'Barva',
|
||||
label: 'Název',
|
||||
icon: 'Ikona',
|
||||
customIcon: 'Vlastní ikona',
|
||||
fileName: 'Soubor',
|
||||
choseFile: 'Vybrat soubor',
|
||||
data: 'Data',
|
||||
password: 'Heslo',
|
||||
required: 'Povinné',
|
||||
madeWithJS: 'Vytvořeno s 💛 a JavaScriptem!',
|
||||
checkForUpdates: 'Zkontrolovat aktualizace',
|
||||
noUpdatesAvailable: 'Žádné dostupné aktualizace',
|
||||
@@ -308,7 +337,7 @@ export const csCZ = {
|
||||
editorTheme: 'Motiv editoru',
|
||||
wrapLongLines: 'Zalamovat dlouhé řádky',
|
||||
markdownSupported: 'Podporován Markdown',
|
||||
plantATree: 'Zasaďte strom', // Lawondyss: Not used
|
||||
plantATree: 'Zasaďte strom',
|
||||
dataTabPageSize: 'Počet řádků na stránku',
|
||||
noOpenTabs: 'Žádné otevřené karty, vyberte z elementů vlevo nebo:',
|
||||
restorePreviousSession: 'Pamatovat si otevřené karty',
|
||||
@@ -333,15 +362,16 @@ export const csCZ = {
|
||||
saveContent: 'Uložit obsah',
|
||||
openAllConnections: 'Otevřít všechna připojení',
|
||||
openSettings: 'Otevřít nastavení',
|
||||
openScratchpad: 'Otevřít zápisník',
|
||||
runOrReload: 'Spustit dotaz',
|
||||
openFilter: 'Otevřít filtr',
|
||||
nextResultsPage: 'Další stránka výsledků',
|
||||
previousResultsPage: 'Předešlá stránka výsledků',
|
||||
editFolder: 'Upravit složku',
|
||||
folderName: 'Název složky',
|
||||
deleteFolder: 'Smazat složku', // Lawondyss: Not used
|
||||
editConnectionAppearance: 'Upravit vzhled připojení', // Lawondyss: Not used
|
||||
deleteFolder: 'Smazat složku',
|
||||
newFolder: 'Nová složka',
|
||||
outOfFolder: 'Mimo složku',
|
||||
editConnectionAppearance: 'Upravit vzhled připojení',
|
||||
defaultCopyType: 'Výchozí typ kopírování',
|
||||
showTableSize: 'Velikost tabulky v panelu',
|
||||
showTableSizeDescription: 'Pouze MySQL/MariaDB. Povolení této možnosti může ovlivnit výkon u schémat s mnoha tabulkami.',
|
||||
@@ -356,7 +386,33 @@ export const csCZ = {
|
||||
csvStringDelimiter: 'Obalit text',
|
||||
csvIncludeHeader: 'Včetně hlavičky',
|
||||
csvExportOptions: 'Možnosti CSV exportu',
|
||||
scratchPadDefaultValue: '# JAK PODPOŘIT ANTARES\n\n- [ ] Dát Antares hvězdičku [GitHub repo](https://github.com/antares-sql/antares)\n- [ ] Poslat názor či radu\n- [ ] Nahlásit chybu\n- [ ] Pokud se líbí, sdílet Antares s přáteli\n\n# O ZÁPISNÍKU\n\nToto je zápisník, který uchovává vaše **osobní poznámky**. Podporuje `markdown` formát, ale můžete použít obyčejný text.\nTento obsah je pouze ukázky, neváhejte ho smazat, abyste si udělali místo na poznámky.\n'
|
||||
exportData: 'Exportovat data',
|
||||
exportDataExplanation: 'Export uložených připojení v Antaresu. Budete požádáni o zadání hesla pro zašifrování exportovaného souboru.',
|
||||
importData: 'Importovat data',
|
||||
importDataExplanation: 'Importuje soubor .antares obsahující připojení. Je třeba zadat heslo definované při exportu.',
|
||||
includeConnectionPasswords: 'Včetně hesel připojení',
|
||||
includeFolders: 'Včetně složek',
|
||||
encryptionPassword: 'Heslo pro zašifrování souboru',
|
||||
encryptionPasswordError: 'Heslo musí mít alespoň 8 znaků.',
|
||||
ignoreDuplicates: 'Ignorovat duplicity',
|
||||
wrongImportPassword: 'Chybné heslo pro import',
|
||||
wrongFileFormat: 'Chybný formát souboru',
|
||||
dataImportSuccess: 'Data úspěšně importována',
|
||||
note: 'Poznámka',
|
||||
thereAreNoNotesYet: 'Zatím tu nejsou žádné poznámky',
|
||||
addNote: 'Přidat poznámku',
|
||||
editNote: 'Upravit poznámku',
|
||||
saveAsNote: 'Uložit jako poznámku',
|
||||
showArchivedNotes: 'Zobrazit archivované poznámky',
|
||||
hideArchivedNotes: 'Skrýt archivované poznámky',
|
||||
tag: 'Tag', // Note tag
|
||||
saveFile: 'Uložit soubor',
|
||||
saveFileAs: 'Uložit do nového souboru',
|
||||
openFile: 'Otevřít soubor',
|
||||
openNotes: 'Otevřít poznámky',
|
||||
debugConsole: 'Debug konzole', // <- console tab name
|
||||
executedQueries: 'Log dotazů', // <- console tab name
|
||||
sizeLimitError: 'Maximální velikost {size} překročena'
|
||||
},
|
||||
faker: { // Faker.js methods, used in random generated content
|
||||
address: 'Address',
|
||||
|
@@ -15,6 +15,10 @@ export default class {
|
||||
return ipcRenderer.invoke('connect', unproxify(newParams));
|
||||
}
|
||||
|
||||
static abortConnection (uid: string): void {
|
||||
ipcRenderer.send('abort-connection', uid);
|
||||
}
|
||||
|
||||
static checkConnection (uid: string): Promise<boolean> {
|
||||
return ipcRenderer.invoke('check-connection', uid);
|
||||
}
|
||||
|
@@ -147,7 +147,7 @@ export const useWorkspacesStore = defineStore('workspaces', {
|
||||
else
|
||||
this.selectedWorkspace = uid;
|
||||
},
|
||||
async connectWorkspace (connection: ConnectionParams & { pgConnString?: string }, mode?: string) {
|
||||
async connectWorkspace (connection: ConnectionParams & { pgConnString?: string }, args?: {mode?: string; signal?: AbortSignal}) {
|
||||
this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid
|
||||
? {
|
||||
...workspace,
|
||||
@@ -155,112 +155,136 @@ export const useWorkspacesStore = defineStore('workspaces', {
|
||||
breadcrumbs: {},
|
||||
loadedSchemas: new Set(),
|
||||
database: connection.database,
|
||||
connectionStatus: mode === 'switch' ? 'connected' : 'connecting'
|
||||
connectionStatus: args?.mode === 'switch' ? 'connected' : 'connecting'
|
||||
}
|
||||
: workspace);
|
||||
|
||||
const connectionsStore = useConnectionsStore();
|
||||
const notificationsStore = useNotificationsStore();
|
||||
const settingsStore = useSettingsStore();
|
||||
|
||||
try {
|
||||
const { status, response } = await Connection.connect(connection);
|
||||
|
||||
if (status === 'error') {
|
||||
notificationsStore.addNotification({ status, message: response });
|
||||
return new Promise((resolve, reject) => {
|
||||
const abortHandler = () => {
|
||||
this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid
|
||||
? {
|
||||
...workspace,
|
||||
structure: [],
|
||||
breadcrumbs: {},
|
||||
loadedSchemas: new Set(),
|
||||
connectionStatus: 'failed'
|
||||
connectionStatus: 'disconnected'
|
||||
}
|
||||
: workspace);
|
||||
}
|
||||
else {
|
||||
let clientCustomizations: Customizations;
|
||||
const { updateLastConnection } = connectionsStore;
|
||||
return reject(new Error('Connection aborted by user'));
|
||||
};
|
||||
|
||||
updateLastConnection(connection.uid);
|
||||
args?.signal?.addEventListener('abort', abortHandler);
|
||||
|
||||
switch (connection.client) {
|
||||
case 'mysql':
|
||||
case 'maria':
|
||||
clientCustomizations = customizations.mysql;
|
||||
break;
|
||||
case 'pg':
|
||||
clientCustomizations = customizations.pg;
|
||||
break;
|
||||
case 'sqlite':
|
||||
clientCustomizations = customizations.sqlite;
|
||||
break;
|
||||
case 'firebird':
|
||||
clientCustomizations = customizations.firebird;
|
||||
break;
|
||||
}
|
||||
const dataTypes = clientCustomizations.dataTypes;
|
||||
const indexTypes = clientCustomizations.indexTypes;
|
||||
(async () => {
|
||||
try {
|
||||
const { status, response } = await Connection.connect(connection);
|
||||
|
||||
const { status, response: version } = await Schema.getVersion(connection.uid);
|
||||
if (status === 'error') {
|
||||
notificationsStore.addNotification({ status, message: response });
|
||||
this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid
|
||||
? {
|
||||
...workspace,
|
||||
structure: [],
|
||||
breadcrumbs: {},
|
||||
loadedSchemas: new Set(),
|
||||
connectionStatus: 'failed'
|
||||
}
|
||||
: workspace);
|
||||
|
||||
if (status === 'error')
|
||||
notificationsStore.addNotification({ status, message: version });
|
||||
|
||||
// Check if Maria or MySQL
|
||||
const isMySQL = version.name.includes('MySQL');
|
||||
const isMaria = version.name.includes('Maria');
|
||||
|
||||
if (isMySQL && connection.client !== 'mysql') {
|
||||
const connProxy = Object.assign({}, connection);
|
||||
connProxy.client = 'mysql';
|
||||
connectionsStore.editConnection(connProxy);
|
||||
}
|
||||
else if (isMaria && connection.client === 'mysql') {
|
||||
const connProxy = Object.assign({}, connection);
|
||||
connProxy.client = 'maria';
|
||||
connectionsStore.editConnection(connProxy);
|
||||
}
|
||||
|
||||
const cachedTabs: WorkspaceTab[] = settingsStore.restoreTabs ? persistentStore.get(connection.uid, []) as WorkspaceTab[] : [];
|
||||
|
||||
if (cachedTabs.length) {
|
||||
tabIndex[connection.uid] = cachedTabs.reduce((acc: number, curr) => {
|
||||
if (curr.index > acc) acc = curr.index;
|
||||
return acc;
|
||||
}, null);
|
||||
}
|
||||
|
||||
const selectedTab = cachedTabs.length
|
||||
? connection.database
|
||||
? cachedTabs.filter(tab => tab.type === 'query' || tab.database === connection.database)[0]?.uid
|
||||
: cachedTabs[0].uid
|
||||
: null;
|
||||
|
||||
this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid
|
||||
? {
|
||||
...workspace,
|
||||
client: connection.client,
|
||||
dataTypes,
|
||||
indexTypes,
|
||||
customizations: clientCustomizations,
|
||||
structure: response,
|
||||
connectionStatus: 'connected',
|
||||
tabs: cachedTabs,
|
||||
selectedTab,
|
||||
version
|
||||
return reject(new Error(response));
|
||||
}
|
||||
: workspace);
|
||||
else if (status === 'abort')
|
||||
return reject(new Error('Connection aborted by user'));
|
||||
else {
|
||||
let clientCustomizations: Customizations;
|
||||
const { updateLastConnection } = connectionsStore;
|
||||
|
||||
this.refreshCollations(connection.uid);
|
||||
this.refreshVariables(connection.uid);
|
||||
this.refreshEngines(connection.uid);
|
||||
this.refreshUsers(connection.uid);
|
||||
}
|
||||
}
|
||||
catch (err) {
|
||||
notificationsStore.addNotification({ status: 'error', message: err.stack });
|
||||
}
|
||||
updateLastConnection(connection.uid);
|
||||
|
||||
switch (connection.client) {
|
||||
case 'mysql':
|
||||
case 'maria':
|
||||
clientCustomizations = customizations.mysql;
|
||||
break;
|
||||
case 'pg':
|
||||
clientCustomizations = customizations.pg;
|
||||
break;
|
||||
case 'sqlite':
|
||||
clientCustomizations = customizations.sqlite;
|
||||
break;
|
||||
case 'firebird':
|
||||
clientCustomizations = customizations.firebird;
|
||||
break;
|
||||
}
|
||||
const dataTypes = clientCustomizations.dataTypes;
|
||||
const indexTypes = clientCustomizations.indexTypes;
|
||||
|
||||
const { status, response: version } = await Schema.getVersion(connection.uid);
|
||||
|
||||
if (status === 'error')
|
||||
notificationsStore.addNotification({ status, message: version });
|
||||
|
||||
// Check if Maria or MySQL
|
||||
const isMySQL = version.name.includes('MySQL');
|
||||
const isMaria = version.name.includes('Maria');
|
||||
|
||||
if (isMySQL && connection.client !== 'mysql') {
|
||||
const connProxy = Object.assign({}, connection);
|
||||
connProxy.client = 'mysql';
|
||||
connectionsStore.editConnection(connProxy);
|
||||
}
|
||||
else if (isMaria && connection.client === 'mysql') {
|
||||
const connProxy = Object.assign({}, connection);
|
||||
connProxy.client = 'maria';
|
||||
connectionsStore.editConnection(connProxy);
|
||||
}
|
||||
|
||||
const cachedTabs: WorkspaceTab[] = settingsStore.restoreTabs ? persistentStore.get(connection.uid, []) as WorkspaceTab[] : [];
|
||||
|
||||
if (cachedTabs.length) {
|
||||
tabIndex[connection.uid] = cachedTabs.reduce((acc: number, curr) => {
|
||||
if (curr.index > acc) acc = curr.index;
|
||||
return acc;
|
||||
}, null);
|
||||
}
|
||||
|
||||
const selectedTab = cachedTabs.length
|
||||
? connection.database
|
||||
? cachedTabs.filter(tab => tab.type === 'query' || tab.database === connection.database)[0]?.uid
|
||||
: cachedTabs[0].uid
|
||||
: null;
|
||||
|
||||
this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid
|
||||
? {
|
||||
...workspace,
|
||||
client: connection.client,
|
||||
dataTypes,
|
||||
indexTypes,
|
||||
customizations: clientCustomizations,
|
||||
structure: response,
|
||||
connectionStatus: 'connected',
|
||||
tabs: cachedTabs,
|
||||
selectedTab,
|
||||
version
|
||||
}
|
||||
: workspace);
|
||||
|
||||
args?.signal?.removeEventListener('abort', abortHandler);
|
||||
this.refreshCollations(connection.uid);
|
||||
this.refreshVariables(connection.uid);
|
||||
this.refreshEngines(connection.uid);
|
||||
this.refreshUsers(connection.uid);
|
||||
resolve(true);
|
||||
}
|
||||
}
|
||||
catch (err) {
|
||||
notificationsStore.addNotification({ status: 'error', message: err.stack });
|
||||
}
|
||||
})();
|
||||
});
|
||||
},
|
||||
async refreshStructure (uid: string) {
|
||||
const notificationsStore = useNotificationsStore();
|
||||
@@ -405,7 +429,7 @@ export const useWorkspacesStore = defineStore('workspaces', {
|
||||
},
|
||||
async switchConnection (connection: ConnectionParams & { pgConnString?: string }) {
|
||||
await Connection.disconnect(connection.uid);
|
||||
return this.connectWorkspace(connection, 'switch');
|
||||
return this.connectWorkspace(connection, { mode: 'switch' });
|
||||
},
|
||||
addWorkspace (uid: string) {
|
||||
const workspace: Workspace = {
|
||||
|
Reference in New Issue
Block a user