You've already forked nginx-proxy-manager
							
							
				mirror of
				https://github.com/NginxProxyManager/nginx-proxy-manager.git
				synced 2025-11-04 04:11:42 +03:00 
			
		
		
		
	Certificates react work
- renewal and download - table columns rendering - searching - deleting
This commit is contained in:
		@@ -1,12 +1,22 @@
 | 
			
		||||
import { IconSearch } from "@tabler/icons-react";
 | 
			
		||||
import { useState } from "react";
 | 
			
		||||
import Alert from "react-bootstrap/Alert";
 | 
			
		||||
import { deleteCertificate, downloadCertificate } from "src/api/backend";
 | 
			
		||||
import { LoadingPage } from "src/components";
 | 
			
		||||
import { useCertificates } from "src/hooks";
 | 
			
		||||
import { T } from "src/locale";
 | 
			
		||||
import { showCustomCertificateModal, showDNSCertificateModal, showHTTPCertificateModal } from "src/modals";
 | 
			
		||||
import {
 | 
			
		||||
	showCustomCertificateModal,
 | 
			
		||||
	showDeleteConfirmModal,
 | 
			
		||||
	showDNSCertificateModal,
 | 
			
		||||
	showHTTPCertificateModal,
 | 
			
		||||
	showRenewCertificateModal,
 | 
			
		||||
} from "src/modals";
 | 
			
		||||
import { showError, showObjectSuccess } from "src/notifications";
 | 
			
		||||
import Table from "./Table";
 | 
			
		||||
 | 
			
		||||
export default function TableWrapper() {
 | 
			
		||||
	const [search, setSearch] = useState("");
 | 
			
		||||
	const { isFetching, isLoading, isError, error, data } = useCertificates([
 | 
			
		||||
		"owner",
 | 
			
		||||
		"dead_hosts",
 | 
			
		||||
@@ -22,6 +32,31 @@ export default function TableWrapper() {
 | 
			
		||||
		return <Alert variant="danger">{error?.message || "Unknown error"}</Alert>;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	const handleDelete = async (id: number) => {
 | 
			
		||||
		await deleteCertificate(id);
 | 
			
		||||
		showObjectSuccess("certificate", "deleted");
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	const handleDownload = async (id: number) => {
 | 
			
		||||
		try {
 | 
			
		||||
			await downloadCertificate(id);
 | 
			
		||||
		} catch (err: any) {
 | 
			
		||||
			showError(err.message);
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	let filtered = null;
 | 
			
		||||
	if (search && data) {
 | 
			
		||||
		filtered = data?.filter(
 | 
			
		||||
			(item) =>
 | 
			
		||||
				item.domainNames.some((domain: string) => domain.toLowerCase().includes(search)) ||
 | 
			
		||||
				item.niceName.toLowerCase().includes(search),
 | 
			
		||||
		);
 | 
			
		||||
	} else if (search !== "") {
 | 
			
		||||
		// this can happen if someone deletes the last item while searching
 | 
			
		||||
		setSearch("");
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return (
 | 
			
		||||
		<div className="card mt-4">
 | 
			
		||||
			<div className="card-status-top bg-pink" />
 | 
			
		||||
@@ -33,66 +68,83 @@ export default function TableWrapper() {
 | 
			
		||||
								<T id="certificates" />
 | 
			
		||||
							</h2>
 | 
			
		||||
						</div>
 | 
			
		||||
						<div className="col-md-auto col-sm-12">
 | 
			
		||||
							<div className="ms-auto d-flex flex-wrap btn-list">
 | 
			
		||||
								<div className="input-group input-group-flat w-auto">
 | 
			
		||||
									<span className="input-group-text input-group-text-sm">
 | 
			
		||||
										<IconSearch size={16} />
 | 
			
		||||
									</span>
 | 
			
		||||
									<input
 | 
			
		||||
										id="advanced-table-search"
 | 
			
		||||
										type="text"
 | 
			
		||||
										className="form-control form-control-sm"
 | 
			
		||||
										autoComplete="off"
 | 
			
		||||
									/>
 | 
			
		||||
								</div>
 | 
			
		||||
								<div className="dropdown">
 | 
			
		||||
									<button
 | 
			
		||||
										type="button"
 | 
			
		||||
										className="btn btn-sm dropdown-toggle btn-pink mt-1"
 | 
			
		||||
										data-bs-toggle="dropdown"
 | 
			
		||||
									>
 | 
			
		||||
										<T id="object.add" tData={{ object: "certificate" }} />
 | 
			
		||||
									</button>
 | 
			
		||||
									<div className="dropdown-menu">
 | 
			
		||||
										<a
 | 
			
		||||
											className="dropdown-item"
 | 
			
		||||
											href="#"
 | 
			
		||||
											onClick={(e) => {
 | 
			
		||||
												e.preventDefault();
 | 
			
		||||
												showHTTPCertificateModal();
 | 
			
		||||
											}}
 | 
			
		||||
						{data?.length ? (
 | 
			
		||||
							<div className="col-md-auto col-sm-12">
 | 
			
		||||
								<div className="ms-auto d-flex flex-wrap btn-list">
 | 
			
		||||
									<div className="input-group input-group-flat w-auto">
 | 
			
		||||
										<span className="input-group-text input-group-text-sm">
 | 
			
		||||
											<IconSearch size={16} />
 | 
			
		||||
										</span>
 | 
			
		||||
										<input
 | 
			
		||||
											id="advanced-table-search"
 | 
			
		||||
											type="text"
 | 
			
		||||
											className="form-control form-control-sm"
 | 
			
		||||
											autoComplete="off"
 | 
			
		||||
											onChange={(e: any) => setSearch(e.target.value.toLowerCase().trim())}
 | 
			
		||||
										/>
 | 
			
		||||
									</div>
 | 
			
		||||
									<div className="dropdown">
 | 
			
		||||
										<button
 | 
			
		||||
											type="button"
 | 
			
		||||
											className="btn btn-sm dropdown-toggle btn-pink mt-1"
 | 
			
		||||
											data-bs-toggle="dropdown"
 | 
			
		||||
										>
 | 
			
		||||
											<T id="lets-encrypt-via-http" />
 | 
			
		||||
										</a>
 | 
			
		||||
										<a
 | 
			
		||||
											className="dropdown-item"
 | 
			
		||||
											href="#"
 | 
			
		||||
											onClick={(e) => {
 | 
			
		||||
												e.preventDefault();
 | 
			
		||||
												showDNSCertificateModal();
 | 
			
		||||
											}}
 | 
			
		||||
										>
 | 
			
		||||
											<T id="lets-encrypt-via-dns" />
 | 
			
		||||
										</a>
 | 
			
		||||
										<div className="dropdown-divider" />
 | 
			
		||||
										<a
 | 
			
		||||
											className="dropdown-item"
 | 
			
		||||
											href="#"
 | 
			
		||||
											onClick={(e) => {
 | 
			
		||||
												e.preventDefault();
 | 
			
		||||
												showCustomCertificateModal();
 | 
			
		||||
											}}
 | 
			
		||||
										>
 | 
			
		||||
											<T id="certificates.custom" />
 | 
			
		||||
										</a>
 | 
			
		||||
											<T id="object.add" tData={{ object: "certificate" }} />
 | 
			
		||||
										</button>
 | 
			
		||||
										<div className="dropdown-menu">
 | 
			
		||||
											<a
 | 
			
		||||
												className="dropdown-item"
 | 
			
		||||
												href="#"
 | 
			
		||||
												onClick={(e) => {
 | 
			
		||||
													e.preventDefault();
 | 
			
		||||
													showHTTPCertificateModal();
 | 
			
		||||
												}}
 | 
			
		||||
											>
 | 
			
		||||
												<T id="lets-encrypt-via-http" />
 | 
			
		||||
											</a>
 | 
			
		||||
											<a
 | 
			
		||||
												className="dropdown-item"
 | 
			
		||||
												href="#"
 | 
			
		||||
												onClick={(e) => {
 | 
			
		||||
													e.preventDefault();
 | 
			
		||||
													showDNSCertificateModal();
 | 
			
		||||
												}}
 | 
			
		||||
											>
 | 
			
		||||
												<T id="lets-encrypt-via-dns" />
 | 
			
		||||
											</a>
 | 
			
		||||
											<div className="dropdown-divider" />
 | 
			
		||||
											<a
 | 
			
		||||
												className="dropdown-item"
 | 
			
		||||
												href="#"
 | 
			
		||||
												onClick={(e) => {
 | 
			
		||||
													e.preventDefault();
 | 
			
		||||
													showCustomCertificateModal();
 | 
			
		||||
												}}
 | 
			
		||||
											>
 | 
			
		||||
												<T id="certificates.custom" />
 | 
			
		||||
											</a>
 | 
			
		||||
										</div>
 | 
			
		||||
									</div>
 | 
			
		||||
								</div>
 | 
			
		||||
							</div>
 | 
			
		||||
						</div>
 | 
			
		||||
						) : null}
 | 
			
		||||
					</div>
 | 
			
		||||
				</div>
 | 
			
		||||
				<Table data={data ?? []} isFetching={isFetching} />
 | 
			
		||||
				<Table
 | 
			
		||||
					data={filtered ?? data ?? []}
 | 
			
		||||
					isFiltered={!!search}
 | 
			
		||||
					isFetching={isFetching}
 | 
			
		||||
					onRenew={showRenewCertificateModal}
 | 
			
		||||
					onDownload={handleDownload}
 | 
			
		||||
					onDelete={(id: number) =>
 | 
			
		||||
						showDeleteConfirmModal({
 | 
			
		||||
							title: <T id="object.delete" tData={{ object: "certificate" }} />,
 | 
			
		||||
							onConfirm: () => handleDelete(id),
 | 
			
		||||
							invalidations: [["certificates"], ["certificate", id]],
 | 
			
		||||
							children: <T id="object.delete.content" tData={{ object: "certificate" }} />,
 | 
			
		||||
						})
 | 
			
		||||
					}
 | 
			
		||||
				/>
 | 
			
		||||
			</div>
 | 
			
		||||
		</div>
 | 
			
		||||
	);
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user