Compare commits

5 Commits

Author SHA1 Message Date
BOHA
495fdf6da2 1.3.2 2026-03-27 10:42:29 +01:00
BOHA
7d29f40ab2 fix: offers table PDF button opens blob from NAS instead of print page
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-27 10:42:29 +01:00
BOHA
6b9f1dee87 1.3.1 2026-03-27 10:33:15 +01:00
BOHA
687dcb9371 fix: OfferDetail uses default currency from system settings
The useEffect checked prev.currency === "EUR" but initial default was
changed to "CZK", so the settings default was never applied.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-27 10:33:15 +01:00
BOHA
9c49015968 1.3.0 2026-03-27 10:25:40 +01:00
5 changed files with 27 additions and 27 deletions

4
package-lock.json generated
View File

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

View File

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

View File

@@ -356,7 +356,7 @@ export default function OfferDetail() {
setForm((prev) => ({ setForm((prev) => ({
...prev, ...prev,
currency: currency:
prev.currency === "EUR" prev.currency === "CZK"
? companySettings.default_currency || "CZK" ? companySettings.default_currency || "CZK"
: prev.currency, : prev.currency,
vat_rate: vat_rate:

View File

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

View File

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