[P_UI] Added django settings for p_ui (#5343)

* Added django settings for pui

* Fix: server version is not loaded on initial load

* Moved server version out of server selector icon

* Use polling only for WSL

* Added comment and extracted to constant instead of function

* Default show server selector to false if not in dev mode

* Refactored hostList settings

* Move json serialization into global scope

* Show server selector in netlify builds

* Use demo server for netlify

* Renamed netilfy mode to dev or demo mode

* Translate for netlify

* Dont use translation in main as the are not working there
This commit is contained in:
Lukas 2023-08-10 12:57:33 +02:00 committed by GitHub
parent 86ca0b27a4
commit 89795f632c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 95 additions and 21 deletions

View File

@ -979,6 +979,10 @@ CUSTOM_LOGO = get_custom_file('INVENTREE_CUSTOM_LOGO', 'customize.logo', 'custom
CUSTOM_SPLASH = get_custom_file('INVENTREE_CUSTOM_SPLASH', 'customize.splash', 'custom splash') CUSTOM_SPLASH = get_custom_file('INVENTREE_CUSTOM_SPLASH', 'customize.splash', 'custom splash')
CUSTOMIZE = get_setting('INVENTREE_CUSTOMIZE', 'customize', {}) CUSTOMIZE = get_setting('INVENTREE_CUSTOMIZE', 'customize', {})
# Frontend settings
PUI_SETTINGS = get_setting("INVENTREE_PUI_SETTINGS", "pui_settings", {})
if DEBUG: if DEBUG:
logger.info("InvenTree running with DEBUG enabled") logger.info("InvenTree running with DEBUG enabled")

View File

@ -240,6 +240,15 @@ remote_login_header: HTTP_REMOTE_USER
# hide_admin_link: true # hide_admin_link: true
# hide_password_reset: true # hide_password_reset: true
# Platform UI options
# pui_settings:
# server_list:
# my_server1:
# host: https://demo.inventree.org/api/
# name: InvenTree Demo
# default_server: my_server1
# show_server_selector: false
# Custom flags # Custom flags
# InvenTree uses django-flags; read more in their docs at https://cfpb.github.io/django-flags/conditions/ # InvenTree uses django-flags; read more in their docs at https://cfpb.github.io/django-flags/conditions/
# Use environment variable INVENTREE_FLAGS or the settings below # Use environment variable INVENTREE_FLAGS or the settings below

View File

@ -12,6 +12,7 @@
<body> <body>
<div id="root"></div> <div id="root"></div>
{% spa_settings %}
{% spa_bundle %} {% spa_bundle %}
</body> </body>

View File

@ -7,9 +7,11 @@ from django import template
from django.conf import settings from django.conf import settings
from django.utils.safestring import mark_safe from django.utils.safestring import mark_safe
logger = getLogger("gwaesser_backend") logger = getLogger("InvenTree")
register = template.Library() register = template.Library()
PUI_SETTINGS = json.dumps(settings.PUI_SETTINGS)
@register.simple_tag @register.simple_tag
def spa_bundle(): def spa_bundle():
@ -36,3 +38,9 @@ def spa_bundle():
f"""<link rel="stylesheet" href="{settings.STATIC_URL}web/{css_index['file']}" /> f"""<link rel="stylesheet" href="{settings.STATIC_URL}web/{css_index['file']}" />
<script type="module" src="{settings.STATIC_URL}web/{index['file']}"></script>{imports_files}""" <script type="module" src="{settings.STATIC_URL}web/{index['file']}"></script>{imports_files}"""
) )
@register.simple_tag
def spa_settings():
"""Render settings for spa."""
return mark_safe(f"""<script>window.INVENTREE_SETTINGS={PUI_SETTINGS}</script>""")

View File

@ -2,9 +2,12 @@
# https://www.netlify.com/docs/netlify-toml-reference/ # https://www.netlify.com/docs/netlify-toml-reference/
[build] [build]
command = "yarn run build --outDir dist" command = "yarn run extract && yarn run compile && yarn run build --outDir dist"
publish = "dist" publish = "dist"
[build.environment]
VITE_DEMO = "true"
# Send requests to subpath # Send requests to subpath
[[redirects]] [[redirects]]

View File

@ -19,9 +19,11 @@ export function AuthFormOptions({
<Group> <Group>
<ColorToggle /> <ColorToggle />
<LanguageToggle /> <LanguageToggle />
<Tooltip label={hostname}> {window.INVENTREE_SETTINGS.show_server_selector && (
<IconServer onClick={toggleHostEdit} /> <Tooltip label={hostname}>
</Tooltip> <IconServer onClick={toggleHostEdit} />
</Tooltip>
)}
<Text c={'dimmed'}> <Text c={'dimmed'}>
{server.version} | {server.apiVersion} {server.version} | {server.apiVersion}
</Text> </Text>

View File

@ -2,18 +2,5 @@ import { t } from '@lingui/macro';
import { HostList } from '../states/states'; import { HostList } from '../states/states';
export const defaultHostList: HostList = { export const defaultHostList: HostList = window.INVENTREE_SETTINGS.server_list;
'mantine-u56l5jt85': { export const defaultHostKey = window.INVENTREE_SETTINGS.default_server;
host: 'https://demo.inventree.org/api/',
name: t`InvenTree Demo`
},
'mantine-g8t1zrj50': {
host: 'https://sample.app.invenhost.com/api/',
name: 'InvenHost: Sample'
},
'mantine-cqj63coxn': {
host: 'http://localhost:8000/api/',
name: t`Local Server`
}
};
export const defaultHostKey = 'mantine-cqj63coxn';

View File

@ -4,6 +4,44 @@ import 'react-grid-layout/css/styles.css';
import 'react-resizable/css/styles.css'; import 'react-resizable/css/styles.css';
import App from './App'; import App from './App';
import { HostList } from './states/states';
// define settings
declare global {
interface Window {
INVENTREE_SETTINGS: {
server_list: HostList;
default_server: string;
show_server_selector: boolean;
};
}
}
export const IS_DEV = import.meta.env.DEV;
export const IS_DEMO = import.meta.env.VITE_DEMO === 'true';
export const IS_DEV_OR_DEMO = IS_DEV || IS_DEMO;
window.INVENTREE_SETTINGS = {
server_list: {
'mantine-cqj63coxn': {
host: `${window.location.origin}/api/`,
name: 'Current Server'
},
...(IS_DEV_OR_DEMO
? {
'mantine-u56l5jt85': {
host: 'https://demo.inventree.org/api/',
name: 'InvenTree Demo'
}
}
: {})
},
default_server: IS_DEMO ? 'mantine-u56l5jt85' : 'mantine-cqj63coxn', // use demo server for demo mode
show_server_selector: IS_DEV_OR_DEMO,
// merge in settings that are already set via django's spa_view or for development
...((window.INVENTREE_SETTINGS || {}) as any)
};
ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render( ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render(
<React.StrictMode> <React.StrictMode>

View File

@ -3,6 +3,7 @@ import { Center, Container } from '@mantine/core';
import { useToggle } from '@mantine/hooks'; import { useToggle } from '@mantine/hooks';
import { useEffect } from 'react'; import { useEffect } from 'react';
import { setApiDefaults } from '../../App';
import { AuthFormOptions } from '../../components/forms/AuthFormOptions'; import { AuthFormOptions } from '../../components/forms/AuthFormOptions';
import { AuthenticationForm } from '../../components/forms/AuthenticationForm'; import { AuthenticationForm } from '../../components/forms/AuthenticationForm';
import { InstanceOptions } from '../../components/forms/InstanceOptions'; import { InstanceOptions } from '../../components/forms/InstanceOptions';
@ -27,6 +28,7 @@ export default function Login() {
// Data manipulation functions // Data manipulation functions
function ChangeHost(newHost: string): void { function ChangeHost(newHost: string): void {
setHost(hostList[newHost].host, newHost); setHost(hostList[newHost].host, newHost);
setApiDefaults();
fetchServerApiState(); fetchServerApiState();
} }

View File

@ -1 +1,9 @@
/// <reference types="vite/client" /> /// <reference types="vite/client" />
interface ImportMetaEnv {
readonly VITE_DEMO: string;
}
interface ImportMeta {
readonly env: ImportMetaEnv;
}

View File

@ -1,6 +1,9 @@
import react from '@vitejs/plugin-react'; import react from '@vitejs/plugin-react';
import { platform } from 'node:os';
import { defineConfig, splitVendorChunkPlugin } from 'vite'; import { defineConfig, splitVendorChunkPlugin } from 'vite';
const IS_IN_WSL = platform().includes('WSL');
// https://vitejs.dev/config/ // https://vitejs.dev/config/
export default defineConfig({ export default defineConfig({
plugins: [ plugins: [
@ -16,8 +19,17 @@ export default defineConfig({
outDir: '../../InvenTree/web/static/web' outDir: '../../InvenTree/web/static/web'
}, },
server: { server: {
proxy: {
'/api': {
target: 'http://localhost:8000',
changeOrigin: true,
secure: true
}
},
watch: { watch: {
usePolling: true // use polling only for WSL as the file system doesn't trigger notifications for Linux apps
// ref: https://github.com/vitejs/vite/issues/1153#issuecomment-785467271
usePolling: IS_IN_WSL
} }
} }
}); });