mirror of
				https://github.com/NginxProxyManager/nginx-proxy-manager.git
				synced 2025-10-25 21:03:33 +00:00 
			
		
		
		
	Fixes to lang selection now apply immediately
Moved lang list to separate file Cleaned up some lang workarounds
This commit is contained in:
		| @@ -2,7 +2,7 @@ import React, { ReactNode } from "react"; | ||||
|  | ||||
| import cn from "classnames"; | ||||
|  | ||||
| export interface ButtonProps { | ||||
| export interface ButtonProps extends React.ButtonHTMLAttributes<any> { | ||||
| 	/** | ||||
| 	 * Child elements within | ||||
| 	 */ | ||||
| @@ -60,6 +60,7 @@ export const Button: React.FC<ButtonProps> = ({ | ||||
| 	href, | ||||
| 	target, | ||||
| 	onClick, | ||||
| 	...rest | ||||
| }) => { | ||||
| 	const classes = [ | ||||
| 		"btn", | ||||
| @@ -99,7 +100,8 @@ export const Button: React.FC<ButtonProps> = ({ | ||||
| 		<button | ||||
| 			className={cn(classes, className)} | ||||
| 			aria-label="Button" | ||||
| 			onClick={onClick}> | ||||
| 			onClick={onClick} | ||||
| 			{...rest}> | ||||
| 			{children} | ||||
| 		</button> | ||||
| 	); | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { useHealthState } from "context"; | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const FixedFooterWrapper = styled.div` | ||||
| @@ -29,10 +29,10 @@ function Footer({ fixed }: Props) { | ||||
| 										target="_blank" | ||||
| 										rel="noreferrer" | ||||
| 										className="link-secondary"> | ||||
| 										<FormattedMessage | ||||
| 											id="footer.userguide" | ||||
| 											defaultMessage="User Guide" | ||||
| 										/> | ||||
| 										{intl.formatMessage({ | ||||
| 											id: "footer.userguide", | ||||
| 											defaultMessage: "User Guide", | ||||
| 										})} | ||||
| 									</a> | ||||
| 								</li> | ||||
| 								<li className="list-inline-item"> | ||||
| @@ -41,10 +41,10 @@ function Footer({ fixed }: Props) { | ||||
| 										target="_blank" | ||||
| 										rel="noreferrer" | ||||
| 										className="link-secondary"> | ||||
| 										<FormattedMessage | ||||
| 											id="footer.changelog" | ||||
| 											defaultMessage="Change Log" | ||||
| 										/> | ||||
| 										{intl.formatMessage({ | ||||
| 											id: "footer.changelog", | ||||
| 											defaultMessage: "Change Log", | ||||
| 										})} | ||||
| 									</a> | ||||
| 								</li> | ||||
| 								<li className="list-inline-item"> | ||||
| @@ -53,10 +53,10 @@ function Footer({ fixed }: Props) { | ||||
| 										target="_blank" | ||||
| 										rel="noreferrer" | ||||
| 										className="link-secondary"> | ||||
| 										<FormattedMessage | ||||
| 											id="footer.github" | ||||
| 											defaultMessage="Github" | ||||
| 										/> | ||||
| 										{intl.formatMessage({ | ||||
| 											id: "footer.github", | ||||
| 											defaultMessage: "Github", | ||||
| 										})} | ||||
| 									</a> | ||||
| 								</li> | ||||
| 							</ul> | ||||
| @@ -64,20 +64,22 @@ function Footer({ fixed }: Props) { | ||||
| 						<div className="col-12 col-lg-auto mt-3 mt-lg-0"> | ||||
| 							<ul className="list-inline list-inline-dots mb-0"> | ||||
| 								<li className="list-inline-item"> | ||||
| 									<FormattedMessage | ||||
| 										id="footer.copyright" | ||||
| 										defaultMessage="Copyright © {year} jc21.com." | ||||
| 										values={{ year: new Date().getFullYear() }} | ||||
| 									/>{" "} | ||||
| 									{intl.formatMessage( | ||||
| 										{ | ||||
| 											id: "footer.copyright", | ||||
| 											defaultMessage: "Copyright © {year} jc21.com", | ||||
| 										}, | ||||
| 										{ year: new Date().getFullYear() }, | ||||
| 									)}{" "} | ||||
| 									<a | ||||
| 										className="link-secondary" | ||||
| 										href="https://preview.tabler.io/" | ||||
| 										target="_blank" | ||||
| 										rel="noreferrer"> | ||||
| 										<FormattedMessage | ||||
| 											id="footer.theme" | ||||
| 											defaultMessage="Theme by Tabler" | ||||
| 										/> | ||||
| 										{intl.formatMessage({ | ||||
| 											id: "footer.theme", | ||||
| 											defaultMessage: "Theme by Tabler", | ||||
| 										})} | ||||
| 									</a> | ||||
| 								</li> | ||||
| 								<li className="list-inline-item"> | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import React, { useState } from "react"; | ||||
| import React, { useEffect, useRef, useState } from "react"; | ||||
|  | ||||
| import { Button, Dropdown, Flag } from "components"; | ||||
| import { useLocaleState } from "context"; | ||||
| import { changeLocale, getFlagCodeForLocale, getLocale, intl } from "locale"; | ||||
| import { changeLocale, getFlagCodeForLocale, intl } from "locale"; | ||||
|  | ||||
| export interface LocalPickerProps { | ||||
| 	/** | ||||
| @@ -15,11 +15,16 @@ export const LocalePicker: React.FC<LocalPickerProps> = ({ | ||||
| 	onChange, | ||||
| 	...rest | ||||
| }) => { | ||||
| 	const dropRef = useRef(null); | ||||
| 	const { locale, setLocale } = useLocaleState(); | ||||
|  | ||||
| 	// const [locale, setLocale] = useState(getLocale()); | ||||
| 	const [localeShown, setLocaleShown] = useState(false); | ||||
|  | ||||
| 	const options = [ | ||||
| 		["us", "en-US"], | ||||
| 		["de", "de-DE"], | ||||
| 		["ir", "fa-IR"], | ||||
| 	]; | ||||
|  | ||||
| 	const handleOnChange = (e: any) => { | ||||
| 		changeLocale(e.currentTarget.rel); | ||||
| 		setLocale(e.currentTarget.rel); | ||||
| @@ -27,15 +32,25 @@ export const LocalePicker: React.FC<LocalPickerProps> = ({ | ||||
| 		onChange && onChange(locale); | ||||
| 	}; | ||||
|  | ||||
| 	const options = [ | ||||
| 		["us", "en-US"], | ||||
| 		["de", "de-DE"], | ||||
| 		["ir", "fa-IR"], | ||||
| 	]; | ||||
| 	const handleClickOutside = (event: any) => { | ||||
| 		if ( | ||||
| 			dropRef.current && | ||||
| 			// @ts-expect-error ts-migrate(2531) FIXME: Object is possibly 'null'. | ||||
| 			!dropRef.current.contains(event.target) | ||||
| 		) { | ||||
| 			setLocaleShown(false); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	useEffect(() => { | ||||
| 		document.addEventListener("mousedown", handleClickOutside); | ||||
| 		return () => document.removeEventListener("mousedown", handleClickOutside); | ||||
| 	}, []); | ||||
|  | ||||
| 	return ( | ||||
| 		<div className="dropdown" {...rest}> | ||||
| 		<div className="dropdown" {...rest} ref={dropRef}> | ||||
| 			<Button | ||||
| 				type="button" | ||||
| 				shape="ghost" | ||||
| 				onClick={(e: any) => { | ||||
| 					setLocaleShown(!localeShown); | ||||
|   | ||||
| @@ -1,7 +1,12 @@ | ||||
| import React, { lazy, Suspense } from "react"; | ||||
|  | ||||
| import { SiteWrapper, SuspenseLoader } from "components"; | ||||
| import { useAuthState, useHealthState, UserProvider } from "context"; | ||||
| import { | ||||
| 	useAuthState, | ||||
| 	useLocaleState, | ||||
| 	useHealthState, | ||||
| 	UserProvider, | ||||
| } from "context"; | ||||
| import { BrowserRouter, Switch, Route } from "react-router-dom"; | ||||
|  | ||||
| const AccessLists = lazy(() => import("pages/AccessLists")); | ||||
| @@ -17,6 +22,7 @@ const Users = lazy(() => import("pages/Users")); | ||||
| function Router() { | ||||
| 	const { health } = useHealthState(); | ||||
| 	const { authenticated } = useAuthState(); | ||||
| 	const { locale } = useLocaleState(); | ||||
| 	const Spinner = <SuspenseLoader />; | ||||
|  | ||||
| 	if (health.loading) { | ||||
| @@ -42,7 +48,7 @@ function Router() { | ||||
| 	return ( | ||||
| 		<UserProvider> | ||||
| 			<BrowserRouter> | ||||
| 				<SiteWrapper> | ||||
| 				<SiteWrapper key={`locale-${locale}`}> | ||||
| 					<Suspense fallback={Spinner}> | ||||
| 						<Switch> | ||||
| 							<Route path="/hosts"> | ||||
|   | ||||
| @@ -3,16 +3,17 @@ import { createIntl, createIntlCache } from "react-intl"; | ||||
| import langDe from "./lang/de.json"; | ||||
| import langEn from "./lang/en.json"; | ||||
| import langFa from "./lang/fa.json"; | ||||
| import langList from "./lang/lang-list.json"; | ||||
|  | ||||
| const loadMessages = (locale?: string) => { | ||||
| 	locale = locale || "en"; | ||||
| 	switch (locale.substr(0, 2)) { | ||||
| 		case "de": | ||||
| 			return Object.assign({}, langEn, langDe); | ||||
| 			return Object.assign({}, langList, langEn, langDe); | ||||
| 		case "fa": | ||||
| 			return Object.assign({}, langEn, langFa); | ||||
| 			return Object.assign({}, langList, langEn, langFa); | ||||
| 		default: | ||||
| 			return langEn; | ||||
| 			return Object.assign({}, langList, langEn); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -11,42 +11,12 @@ | ||||
| 	"column.description": { | ||||
| 		"defaultMessage": "Beschreibung" | ||||
| 	}, | ||||
| 	"column.id": { | ||||
| 		"defaultMessage": "ID" | ||||
| 	}, | ||||
| 	"column.name": { | ||||
| 		"defaultMessage": "Name" | ||||
| 	}, | ||||
| 	"dashboard.title": { | ||||
| 		"defaultMessage": "Dashboard" | ||||
| 	}, | ||||
| 	"footer.changelog": { | ||||
| 		"defaultMessage": "Änderungen" | ||||
| 	}, | ||||
| 	"footer.copyright": { | ||||
| 		"defaultMessage": "Copyright © {year} jc21.com." | ||||
| 	}, | ||||
| 	"footer.github": { | ||||
| 		"defaultMessage": "Github" | ||||
| 	}, | ||||
| 	"footer.theme": { | ||||
| 		"defaultMessage": "Theme by Tabler" | ||||
| 	}, | ||||
| 	"footer.userguide": { | ||||
| 		"defaultMessage": "Handbuch" | ||||
| 	}, | ||||
| 	"hosts.title": { | ||||
| 		"defaultMessage": "Hosts" | ||||
| 	}, | ||||
| 	"locale-de-DE": { | ||||
| 		"defaultMessage": "Deutsch" | ||||
| 	}, | ||||
| 	"locale-en-US": { | ||||
| 		"defaultMessage": "Englisch" | ||||
| 	}, | ||||
| 	"locale-fa-IR": { | ||||
| 		"defaultMessage": "Persisch" | ||||
| 	}, | ||||
| 	"login.login": { | ||||
| 		"defaultMessage": "Einloggen" | ||||
| 	}, | ||||
| @@ -68,18 +38,12 @@ | ||||
| 	"user.email": { | ||||
| 		"defaultMessage": "E-Mail" | ||||
| 	}, | ||||
| 	"user.name": { | ||||
| 		"defaultMessage": "Name" | ||||
| 	}, | ||||
| 	"user.nickname": { | ||||
| 		"defaultMessage": "Benutzername" | ||||
| 	}, | ||||
| 	"user.password": { | ||||
| 		"defaultMessage": "Passwort" | ||||
| 	}, | ||||
| 	"users.admin": { | ||||
| 		"defaultMessage": "Administrator" | ||||
| 	}, | ||||
| 	"users.standard": { | ||||
| 		"defaultMessage": "Normaler Benutzer" | ||||
| 	}, | ||||
|   | ||||
| @@ -38,15 +38,6 @@ | ||||
| 	"hosts.title": { | ||||
| 		"defaultMessage": "Hosts" | ||||
| 	}, | ||||
| 	"locale-de-DE": { | ||||
| 		"defaultMessage": "Deutsche" | ||||
| 	}, | ||||
| 	"locale-en-US": { | ||||
| 		"defaultMessage": "English" | ||||
| 	}, | ||||
| 	"locale-fa-IR": { | ||||
| 		"defaultMessage": "Persian" | ||||
| 	}, | ||||
| 	"login.login": { | ||||
| 		"defaultMessage": "Sign in" | ||||
| 	}, | ||||
|   | ||||
							
								
								
									
										11
									
								
								frontend/src/locale/src/lang-list.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								frontend/src/locale/src/lang-list.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| { | ||||
| 	"locale-de-DE": { | ||||
| 		"defaultMessage": "Deutsch" | ||||
| 	}, | ||||
| 	"locale-en-US": { | ||||
| 		"defaultMessage": "English" | ||||
| 	}, | ||||
| 	"locale-fa-IR": { | ||||
| 		"defaultMessage": "فارسی" | ||||
| 	} | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,10 +15,10 @@ function AccessLists() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage | ||||
| 							id="accesslists.title" | ||||
| 							defaultMessage="Access Lists" | ||||
| 						/> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "accesslists.title", | ||||
| 							defaultMessage: "Access Lists", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,7 +15,10 @@ function AuditLog() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage id="auditlog.title" defaultMessage="Audit Log" /> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "auditlog.title", | ||||
| 							defaultMessage: "Audit Log", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,10 +15,10 @@ function Certificates() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage | ||||
| 							id="certificates.title" | ||||
| 							defaultMessage="Certificates" | ||||
| 						/> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "certificates.title", | ||||
| 							defaultMessage: "Certificates", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,7 +15,10 @@ function Dashboard() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage id="dashboard.title" defaultMessage="Dashboard" /> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "dashboard.title", | ||||
| 							defaultMessage: "Dashboard", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,7 +15,10 @@ function Hosts() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage id="hosts.title" defaultMessage="Hosts" /> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "hosts.title", | ||||
| 							defaultMessage: "Hosts", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| import React, { useEffect, useRef, useState, ChangeEvent } from "react"; | ||||
|  | ||||
| import { Alert, Button } from "components"; | ||||
| import { LocalePicker } from "components"; | ||||
| import { useAuthState } from "context"; | ||||
| import { intl } from "locale"; | ||||
| import { FormattedMessage } from "react-intl"; | ||||
|  | ||||
| import logo from "../../img/logo-text-vertical-grey.png"; | ||||
|  | ||||
| @@ -50,10 +50,19 @@ function Login() { | ||||
| 				autoComplete="off" | ||||
| 				onSubmit={onSubmit}> | ||||
| 				<div className="card-body"> | ||||
| 					<div className="row mb-4"> | ||||
| 						<div className="col" /> | ||||
| 						<div className="col col-md-2"> | ||||
| 							<LocalePicker /> | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					{errorMessage ? <Alert type="danger">{errorMessage}</Alert> : null} | ||||
| 					<div className="mb-3"> | ||||
| 						<label className="form-label"> | ||||
| 							<FormattedMessage id="user.email" defaultMessage="Email" /> | ||||
| 							{intl.formatMessage({ | ||||
| 								id: "user.email", | ||||
| 								defaultMessage: "Email", | ||||
| 							})} | ||||
| 						</label> | ||||
| 						<input | ||||
| 							ref={emailRef} | ||||
| @@ -73,7 +82,10 @@ function Login() { | ||||
| 					</div> | ||||
| 					<div className="mb-2"> | ||||
| 						<label className="form-label"> | ||||
| 							<FormattedMessage id="user.password" defaultMessage="Password" /> | ||||
| 							{intl.formatMessage({ | ||||
| 								id: "user.password", | ||||
| 								defaultMessage: "Password", | ||||
| 							})} | ||||
| 						</label> | ||||
| 						<div className="input-group input-group-flat"> | ||||
| 							<input | ||||
| @@ -95,8 +107,15 @@ function Login() { | ||||
| 						</div> | ||||
| 					</div> | ||||
| 					<div className="form-footer"> | ||||
| 						<Button color="cyan" loading={loading} className="w-100"> | ||||
| 							<FormattedMessage id="login.login" defaultMessage="Sign in" /> | ||||
| 						<Button | ||||
| 							color="cyan" | ||||
| 							loading={loading} | ||||
| 							className="w-100" | ||||
| 							type="submit"> | ||||
| 							{intl.formatMessage({ | ||||
| 								id: "login.login", | ||||
| 								defaultMessage: "Sign in", | ||||
| 							})} | ||||
| 						</Button> | ||||
| 					</div> | ||||
| 				</div> | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import { SettingsResponse, requestSettings } from "api/npm"; | ||||
| import { Table } from "components"; | ||||
| import { SuspenseLoader } from "components"; | ||||
| import { intl } from "locale"; | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { useInterval } from "rooks"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| @@ -59,7 +58,10 @@ function Settings() { | ||||
| 					<div className="card-status-top bg-cyan" /> | ||||
| 					<div className="card-header"> | ||||
| 						<h3 className="card-title"> | ||||
| 							<FormattedMessage id="settings.title" defaultMessage="Settings" /> | ||||
| 							{intl.formatMessage({ | ||||
| 								id: "settings.title", | ||||
| 								defaultMessage: "Settings", | ||||
| 							})} | ||||
| 						</h3> | ||||
| 					</div> | ||||
| 					<Table | ||||
|   | ||||
| @@ -13,7 +13,6 @@ function Setup() { | ||||
| 	const { refreshHealth } = useHealthState(); | ||||
| 	const { login } = useAuthState(); | ||||
| 	const [loading, setLoading] = useState(false); | ||||
| 	const [renderCount, setRenderCount] = useState(0); | ||||
| 	const [errorMessage, setErrorMessage] = useState(""); | ||||
|  | ||||
| 	const [formData, setFormData] = useState({ | ||||
| @@ -93,11 +92,7 @@ function Setup() { | ||||
| 								</h2> | ||||
| 							</div> | ||||
| 							<div className="col col-md-2"> | ||||
| 								<LocalePicker | ||||
| 									onChange={() => { | ||||
| 										setRenderCount(renderCount + 1); | ||||
| 									}} | ||||
| 								/> | ||||
| 								<LocalePicker /> | ||||
| 							</div> | ||||
| 						</div> | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React from "react"; | ||||
|  | ||||
| import { FormattedMessage } from "react-intl"; | ||||
| import { intl } from "locale"; | ||||
| import styled from "styled-components"; | ||||
|  | ||||
| const Root = styled.div` | ||||
| @@ -15,7 +15,10 @@ function Users() { | ||||
| 				<div className="card-status-top bg-cyan" /> | ||||
| 				<div className="card-header"> | ||||
| 					<h3 className="card-title"> | ||||
| 						<FormattedMessage id="users.title" defaultMessage="Users" /> | ||||
| 						{intl.formatMessage({ | ||||
| 							id: "users.title", | ||||
| 							defaultMessage: "Users", | ||||
| 						})} | ||||
| 					</h3> | ||||
| 				</div> | ||||
| 			</div> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user