1 Commits

Author SHA1 Message Date
BOHA
dc9720a67a 1.3.0 2026-03-27 10:17:05 +01:00
7 changed files with 32 additions and 50 deletions

4
package-lock.json generated
View File

@@ -1,12 +1,12 @@
{
"name": "app-ts",
"version": "1.3.3",
"version": "1.3.0",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"name": "app-ts",
"version": "1.3.3",
"version": "1.3.0",
"license": "ISC",
"dependencies": {
"@dnd-kit/core": "^6.3.1",

View File

@@ -1,6 +1,6 @@
{
"name": "app-ts",
"version": "1.3.3",
"version": "1.3.0",
"description": "",
"main": "dist/server.js",
"scripts": {

View File

@@ -975,21 +975,18 @@ export default function InvoiceDetail() {
// ─── Edit mode: PDF export ───
const handleViewPdf = async (_lang = "cs") => {
const newWindow = window.open("", "_blank");
setPdfLoading(true);
try {
const response = await apiFetch(`${API_BASE}/invoices/${id}/file`);
if (!response.ok) {
newWindow?.close();
alert.error("PDF soubor nenalezen — uložte fakturu pro vygenerování");
return;
}
const blob = await response.blob();
const url = URL.createObjectURL(blob);
if (newWindow) newWindow.location.href = url;
window.open(url, "_blank");
setTimeout(() => URL.revokeObjectURL(url), 60000);
} catch {
newWindow?.close();
alert.error("Chyba připojení");
} finally {
setPdfLoading(false);

View File

@@ -356,7 +356,7 @@ export default function OfferDetail() {
setForm((prev) => ({
...prev,
currency:
prev.currency === "CZK"
prev.currency === "EUR"
? companySettings.default_currency || "CZK"
: prev.currency,
vat_rate:
@@ -768,25 +768,19 @@ export default function OfferDetail() {
const handlePdf = async () => {
if (!isEdit || pdfLoading) return;
const newWindow = window.open("", "_blank");
setPdfLoading(true);
try {
const response = await apiFetch(`${API_BASE}/offers/${id}/file`);
if (response.status === 401) {
newWindow?.close();
return;
}
if (response.status === 401) return;
if (!response.ok) {
newWindow?.close();
alert.error("PDF soubor nenalezen — uložte nabídku pro vygenerování");
return;
}
const blob = await response.blob();
const url = URL.createObjectURL(blob);
if (newWindow) newWindow.location.href = url;
window.open(url, "_blank");
setTimeout(() => URL.revokeObjectURL(url), 60000);
} catch {
newWindow?.close();
alert.error("Chyba při generování PDF");
} finally {
setPdfLoading(false);

View File

@@ -221,28 +221,26 @@ export default function Offers() {
const handlePdf = async (quotation: Quotation) => {
if (pdfLoading) return;
const newWindow = window.open("", "_blank");
setPdfLoading(quotation.id);
try {
const response = await apiFetch(
`${API_BASE}/offers/${quotation.id}/file`,
);
if (response.status === 401) {
newWindow?.close();
return;
}
const response = await apiFetch(`${API_BASE}/offers-pdf/${quotation.id}`);
if (response.status === 401) return;
if (!response.ok) {
newWindow?.close();
alert.error("PDF soubor nenalezen — otevřete nabídku a uložte ji");
alert.error("Nepodařilo se vygenerovat PDF");
return;
}
const blob = await response.blob();
const url = URL.createObjectURL(blob);
if (newWindow) newWindow.location.href = url;
setTimeout(() => URL.revokeObjectURL(url), 60000);
const html = await response.text();
const w = window.open("", "_blank");
if (w) {
w.document.open();
w.document.write(html);
w.document.close();
w.onload = () => w.print();
} else {
alert.error("Prohlížeč zablokoval vyskakovací okno");
}
} catch {
newWindow?.close();
alert.error("Chyba připojení");
alert.error("Chyba při generování PDF");
} finally {
setPdfLoading(null);
}
@@ -755,7 +753,7 @@ export default function Offers() {
<button
onClick={() => handlePdf(q)}
className="admin-btn-icon"
title="Zobrazit nabídku"
title="PDF"
disabled={pdfLoading === q.id}
>
{pdfLoading === q.id ? (

View File

@@ -503,22 +503,19 @@ export default function ReceivedInvoices({
};
const openFile = async (inv: ReceivedInvoice) => {
const newWindow = window.open("", "_blank");
try {
const response = await apiFetch(
`${API_BASE}/received-invoices/${inv.id}/file`,
);
if (!response.ok) {
newWindow?.close();
alert.error("Nepodařilo se načíst soubor");
return;
}
const blob = await response.blob();
const url = URL.createObjectURL(blob);
if (newWindow) newWindow.location.href = url;
window.open(url, "_blank");
setTimeout(() => URL.revokeObjectURL(url), 60000);
} catch {
newWindow?.close();
alert.error("Chyba připojení");
}
};

View File

@@ -68,7 +68,7 @@ export default async function companySettingsRoutes(
const settings = await prisma.company_settings.findFirst({
select: { [column]: true },
});
const buf = settings?.[column] as unknown as Buffer | null;
const buf = settings?.[column] as Buffer | null;
if (!buf) return error(reply, "Logo nenalezeno", 404);
let mime = "image/png";
@@ -203,15 +203,9 @@ export default async function companySettingsRoutes(
max_requests_per_minute: true,
available_vat_rates: true,
available_currencies: true,
smtp_from: true,
smtp_from_name: true,
offer_number_pattern: true,
order_number_pattern: true,
invoice_number_pattern: true,
},
});
}
if (!settings) return error(reply, "Nastavení nenalezeno", 500);
// Check if logo exists
const logoCheck = await prisma.company_settings.findFirst({
@@ -225,8 +219,9 @@ export default async function companySettingsRoutes(
settings.custom_fields as string | null,
);
// eslint-disable-next-line @typescript-eslint/no-var-requires
const pkg = require("../../../package.json") as { version: string };
const pkg = await import("../../../package.json", {
assert: { type: "json" },
});
let available_vat_rates: number[] = [0, 10, 12, 15, 21];
try {
@@ -260,7 +255,7 @@ export default async function companySettingsRoutes(
available_currencies,
has_logo,
has_logo_dark,
app_version: pkg.version,
app_version: pkg.default.version,
});
});
@@ -269,8 +264,9 @@ export default async function companySettingsRoutes(
"/system-info",
{ preHandler: requirePermission("settings.manage") },
async (request, reply) => {
// eslint-disable-next-line @typescript-eslint/no-var-requires
const pkg = require("../../../package.json") as { version: string };
const pkg = await import("../../../package.json", {
assert: { type: "json" },
});
const uptimeSec = process.uptime();
const days = Math.floor(uptimeSec / 86400);
const hours = Math.floor((uptimeSec % 86400) / 3600);
@@ -303,7 +299,7 @@ export default async function companySettingsRoutes(
const projectNas = new NasFileManager();
return success(reply, {
app_version: pkg.version,
app_version: pkg.default.version,
node_version: process.version,
platform: `${os.type()} ${os.release()}`,
uptime: uptimeStr,