chore: add code

This commit is contained in:
JesusPerez 2022-01-10 10:40:05 +00:00
parent dd6d1aa7bd
commit 88bd388d75
74 changed files with 8239 additions and 0 deletions

485
locales/en.yaml Normal file
View File

@ -0,0 +1,485 @@
edit: Edit
reset: Reset
onedit: Edit On
changeview: View
showview: Show View
menufixed: Menu fixed
fixmen: Fix Menu
from: From
link: Link
since: Since
date: Date
location: In
position: Role
name: Name
code: Code
license: License
for: For
role: Role
tasks: Tasks
description: Description
tools: Tools
purpose: Purpose
code_on: Code on
deployed_to: Deployed to
features: Features
builtwith: Built with
hidepanel: Hide panel
showpanel: Show panel
id: Id
org: Org.
title: Title
cert: Cert.
capture: Capture
site: Site
total: Total
items: Items
in: in
of: of
fixmenu: Fix Menu
phone: Phone
address: Address
save: Save
close: Close
DataNeedSaved: Data changes not saved !
click_link_to_save: Click on link to save
download_json_data: 'Download JSON data '
cancel: Cancel
signup: Sign Up
select: Select
selectOption: Select options
selectModel: Select Model
send: Send
local: Local
path: Path
route: Web Route
param: Parameter
type: Type
Route: Routue
Template: Template
Lang: Lang
Name: Name
saveload:
editMode: Edit mode
saveLocalFile: Save to local file
saveData: Save data
dataSaved: Data saved
sendData: Send data
saveOptions: Save options
dataLoaded: Data loaded
autherror: Auth error
loaderror: Load error
auth:
invalidcredentials: Invalid credentials
loginForm: ''
newaccount: new account
password: password
register: Register
signin: Sign in
subHeading: Login info
subTitle: ''
welcome: welcome !
button:
about: About
back: Back
cancel: Cancel
close-panel: Close panel
delete: Delete
go: Go
home: Home
new: Nuevo
reset: Reset
save: Save
toggle_dark: Toggle dark mode
toggle_langs: Change language
connection:
error: Error conexión
dashboard: Dashboard
form:
delete_done: Delete done
delete_error: Delete Error
save_done: Data saved
save_error: Data save Error
intro:
desc: Opinionated Vite Starter Template
dynamic-route: Demo of dynamic route
hi: Hi, {name}!
whats-your-name: What's your name?
login:
end_session: End session
form: Credentials
invalidcredentials: Incorrect data
newuser: New user
nodata: Data incomplete
password: Password
register: Register
forgot: Forgot ?
checkemail: Check email
reset: Reset
see_you_soon: See You soon !
signin: Signin
subHeading: Access
subtitle: LibreCloud online
user_email: User / email
username: User name
email: Email
welcome: welcome
message:
loading: Loadding
not-found: Not found
notes: Notes
notifications:
content: content
notifications: Notifications
search:
search: Search
payload:
exp: Expiration
id: id
uuid: uuid
data: Data
orig_iat: Issued
tracking:
when: When
what: What
where: Where
context: Context
ctx: Ctx.
data: Datos
auth: Authorization
payload: Payload Auth.
list: Data Dates
menu:
lang: Lang
Config: Configuration
Tracking: Tracking
Users: Users
sec_langs: Langs
sec_webserver: WebServer
sec_logs: Logs
sec_jwt: JWT
sec_auth: Auth
sec_perms: Permissions
sec_routes: Routes
sec_templates: Templates
sec_mail: Mail
sec_assets: WebAssets
sec_models: DataModels
sec_git: GIT
sec_redis: Redis
sec_others: Others
sec_usersData: Accounts
sec_modelsData: Models
sec_invitations: Invitations
sec_authzModel: Authz.Model
sec_authzPolicy: Authz.Policy
sec_tracking_list: Logs
sec_tracking_data: Data
users:
usersData: Users Data
msg_usersData: Data for <b>YAML</b> file collecting basic user data.<br> <u>Password</u> is encrypted by server command, use prefix <b>passwdEnc</b> (passwdEncmypasswd) to request encryption.
modelsData: Users Models
msg_modelsData: In is a <b>YAML</b> file collecting data for specific keys (<b>uuid</b>) to load CV models and permissions.
invitations: New Users Invitations
msg_invitations: Use <b>Uses num.</b> with negative number for no limit uses.<br>Use <b>Expire</b> (ISO date format). Shortcut Atajo <b>?+num(d,h,m)</b>, by example <b>?+2h</b> for expire in two hours or use <b>?+20m</b> for 20 minutes, then adjust result.<br>Set a <b>Role</b> for new User.
authzModel: Authorization Models
msg_authzModel: It is a <a class='link' href='https://casbin.org/' target='_blank'>Cabin RBAC with resources model<a/> to controll resources access.
authzPolicy: Authorization Policies
msg_authzPolicy: It is a <a class='link' href='https://casbin.org/' target='_blank'>Cabin RBAC with resources policy model<a/> to controll resources access.
sec_modelsData:
model: Modelo
user: Usuario
data: Datos
active: Activo
sec_usersData:
description: Description
email: Email
passwd: Password
data: Data
web: Web
username: Username
sec_invitations:
email: Email
createdby: Created by
expire: Expire
howmany: Uses num.
role: Role
description: Description
data: Data
active: Active
cv:
'on': active
'off': unactive
profile: Profile
experience: Experience
work_experience: Work Experience
projects: Projects
teaching: Teaching
talks: Talks
education: Education
expand_skills: Expand Skills
others: Others
contact: Contact
personal: Personal
certifications: Certifications
languages: Languages
infrastructures: Infrastructures
skills_tools: Skills & Tools
skills: Skills
onweb: On the Web
mission: Mission
mission_how: Missión detail
lang:
spanish: Spanish
english: English
french: French
fluent: Fluent
native: Native
conversation: Conversation
vars:
sec_logs: Web Service Logs
sec_webserver: Webserver Settings
sec_jwt: JSON Web Tokens
sec_auth: Access Authorizations
sec_perms: Permissions
sec_routes: Webserver Routes
sec_templates: Webserver Templates
sec_mail: Mail Services
sec_assets: Webserver Assets
sec_models: CV Models
sec_langs: Languages
sec_git: Git
sec_redis: Redis
sec_others: Others
sec_logs:
logOut: Main Log filepath
requestOut: Web Request Log DirPath
requestStore: Store Web Request
trackingOut: Tracking actions DirPath
trackingStore: Store Tracking actions
debugLevel: Debugging Level
sec_webserver:
host: Host
port: Port
protocol: Protocol
keyPem: Primary key Certificate
certPem: Public key Certificate
allowOrigins: Allowed URL origins (CORS)
sec_jwt:
useJWT: Use JSON Web Tokens
jwtRealm: JWT Realm
jwtKey: JWT Key
jwtTimeout: JWT Timeout
jwtMaxRefresh: JWT Refresh tiemout
signingAlgorithm: Signing Algorithm for firma JWT
jwtKeyPem: Private Certificate Path to sign JWT
jwtCertPem: Public Certificate Path to sign JWT
sec_auth:
authSep: Authorizations separator
passwdEnc: Password Encription prefix
invitationsPath: Path to users invitations
recoveryTime: Expire Id Recovery access time
useAuthz: Use Casbin Authorizations
authzModel: Casbin Authorizations Path
authzPolicy: Casbin Policies Path
adminRole: Administrador Role
usersStore: Users store
sec_perms:
pubUser: Public User
usersPath: Users filepath
usersModelsPath: Users Models filepath
identityKey: Identity Key
usersStore: Store Users
sec_routes:
rootAuthGroup: Authentication Group
routes: Webserver Routes
path: Path
param: Param
sec_templates:
templatesRoot: Main Template filepath
templatesExt: Templates files extension
templatesIncludes: Templates Include FilePath
templatesLayouts: Templates Layout filepath
templatesFiles: Templates Paths List
path: Path
route: Route
sec_mail:
mailHost: Server
mailPort: Port
mailFrom: From Email
mailUser: Mail user
mailPswd: User passwrord
mailCertPath: Cert. Path
mailCertDom: Cert. Domain
tplsMailPath: Mail Templates Path
tplsMail: Mail Templates List
path: Path
route: Route
sec_assets:
assetsPath: Web Assets Path
assetsURL: URL Web Assets
sec_models:
useDist: Use Distribution (JSON)
genDist: Generate Distribution (JSON)
genExcludeList: Exclude Names List
dataDistPath: Distribution Path (JSON)
dataPath: Data Path (YAML files)
dataModelsRoot: Models-List Root-Path
dataCorePath: Data Core Path
dataDflt: Default Model Name
dataItems: Data Items List
dataStore: Store Data
sec_langs:
langs: Languages List
mainLang: Main Language
sec_git:
useRepo: Use GIT Repository
useRepoOnReq: Use Repository in web requests
quietGit: Not show additional GIT messages
backgGit: Use background mode for GIT
repoPath: Local path for Repository
repoName: Repository Name
repoCommit: Commits GIT message
sec_redis:
redisHost: Redis Host
redisPort: Redis Port
redisDB: Redis DataBase
redisPswd: Redis DB Password
sec_others:
openBrowser: Open web browser at web server start
info:
logs: >-
<div class='title'>Web Service Logs where server runs.</div>
<div class='note'>Paths with read/write permissions.</div>
webserver: >-
<div class='title'>Web Server Configuration</div>
jwt: >-
<div class='title'>JSON Web Tokens (JWT) Configuration</div>
auth: >-
<div class='title'>Access Authorizations for web requests</div>
perms: >-
<div class='title'>Access Permissions to resources (users, groups, policies)</div>
routes: >-
<div class='title'>Web Server Requests Routes</div>
templates: >-
<div class='title'>Web Server Templates</div>
mail: >-
<div class='title'>Mail Services Configuration</div>
assets: >-
<div class='title'>Web Server Assets</div>
<div class='note'>Static files</div>
models: >-
<div class='title'>Data Models. JSON and YAML formats</div>
langs: >-
<div class='title'>Languages </div>
git: >-
<div class='title'>GIT as Data Source</div>
redis: >-
<div class='title'>Redis</div>
others: >-
<div class='title'>Others</div>
infosec_logs:
logOut: Main Logs relative or absolute Path
requestOut: Web Requests daily Logs files, relative or absolute Directory Path
requestStore: Web Requests Store (fs/redis)
trackingOut: Tracking daily Actions files, absolute Directory Path
trackingStore: Tracking Actions Store (fs/redis)
debugLevel: Debuggin Level (0,1,2)
infosec_webserver:
host: Web server Host name or domain
port: Web server listen Port (80, 443, etc)
protocol: Web server Protocol (http | https)
keyPem: Digital Certificate private key Path
certPem: Digital Certificate public key Path
allowOrigins: URLs Allowed Access (origins) List (CORS)
infosec_jwt:
useJWT: Use JSON Web Token for access identification
jwtRealm: JWT Realm
jwtKey: JWT Key
jwtTimeout: Token timeout in minutes
jwtMaxRefresh: Max refresh token time before expiration
signingAlgorithm: Signing Algorithm for JWT (RS256 or '')
jwtKeyPem: Private Certificate Path to sing JWT
jwtCertPem: Public Certificate Path to sing JWT
infosec_auth:
authSep: Separation character for user and credentials in web requests
passwdEnc: Password Encription prefix for server
invitationsPath: Path to new user invitations
recoveryTime: Recovery access id expire in minutes
useAuthz: Use Casbin Authorizations for web requests
authzModel: Casbin authioization model path
authzPolicy: Authorization Policies Path
adminRole: Administration name role for Casbin authorizations
usersStore: Users store (fs/redis)
infosec_perms:
pubUser: Name to identify public users (usually 'none')
usersPath: Users path file with access credentials encrypted
usersModelsPath: Models associated with users path
identityKey: Field name to identify user ('id')
infosec_routes:
rootAuthGroup: Web route group name for routes with authorization required
routes: >-
Rotues List, use 'path' (url relative) and 'param' field name.
New routes has to be defined/managed in server.
infosec_templates:
templatesRoot: Web server Templates relative or absolute Path
templatesExt: Template files Extensión (default .tmpl)
templatesIncludes: File path to include in Templates (Headers, Footer, etc.)
templatesLayouts: Templates Layout file path
templatesFiles: >-
Templates List with 'path' (relative or absolute), 'route' relative URL
info_mail:
mailHost: Mail service Server, domain or IP
mailPort: Mail service Port (SSL)
mailFrom: Email (from) who send
mailUser: User for Mail service access
mailPswd: Password form User Mail service access
mailCertPath: Certificate Path (DKIM) for emails
mailCertDom: Certificate Domain DKIM
tplsMailPath: Templates Email Path
tplsMail: >-
Mail Templates List with 'path' (relative or absolute), 'type' (format 'text' 'html')
infosec_assets:
assetsPath: Static resources Path for web server
assetsURL: Static resources relative URL for web server
infosec_models:
useDist: Use data models in distributiuon mode (one file in JSON format)
genDist: >-
Genearte Data models for distribution before web server start or when saving data
genExcludeList: >-
Names List to exclude for distribution, will keep items structure in several YAML files.
dataDistPath: Path for Data model distribution with modelos de datos para distribución (JSON)
dataPath: Data directory path, contains folders with items in YAML
dataModelsRoot: File Name (YAML) which contains the models available to the user
dataCorePath: Main Data file name or 'Core'
dataDflt: Main/default data models filename
dataItems: >-
Items List with each Data Model.
Contains a 'showinfo' folder with authorizations (view/edit) for each item and his attributes or fields.
Includes a 'langs' directory with availables languages folders and their translated items.
dataStore: Data Store (fs/redis)
infosec_langs:
langs: Languages List in two lowercase letters code
mainLang: Main/defaul Language
infosec_git:
useRepo: Use GIT Repository as Data Source
useRepoOnReq: Use GIT with each web request (pull/push)
quietGit: Do not show additional messages for GIT tasks
backgGit: Run GIT taks in background
repoPath: Local Directory Path to use GIT (contains .git)
repoName: Repository/branch name for pull/push (default origin)
repoCommit: Commit text for automatic changes with GIT
infosec_redis:
redisHost: Redis Host
redisPort: Redis Port
redisDB: Redis DataBase
redisPswd: Redis DB Password
infosec_others:
openBrowser: URL to open with web browser when web server starts
useTracking: Use Tracking actions (add appropriate routes)

482
locales/es.yaml Normal file
View File

@ -0,0 +1,482 @@
edit: Editar
reset: Reset
onedit: Activar Edición
changeview: Vista
showview: Mostrar Vista
menufixed: Menú fijo
fixmenu: Fijar Menú
from: De
link: Enlace
since: Desde
date: Fecha
location: En
position: Rol
name: Nombre
code: Código
license: Licencia
for: Para
tasks: Tareas
role: Rol
description: Descripción
tools: Software
purpose: Propósito
code_on: Código en
deployed_to: Desplegado en
features: Funciones
builtwith: Creado con
hidepanel: Ocultar panel
showpanel: Mostrar panel
id: Id
org: Org.
title: Título
cert: Cert.
capture: Captura
site: Sitio
total: Total
items: Items
in: en
of: de
phone: Teléfono
address: Dirección
save: Guardar
close: Cerrar
DataNeedSaved: Cambios no guardados
click_link_to_save: Click en enlace de arriba para guardar
download_json_data: Descargar datos en JSON
cancel: Cancelar
signup: Registrarse
select: Seleccionar
selectOption: Selección de opciones
selectModel: Seleccionar modelo
send: Enviar
local: Local
path: Ruta Fichero
route: Ruta Web
param: Parámetro
type: Tipo
Route: Ruta
Template: Plantilla
Lang: Idioma
Name: Nombre
saveload:
editMode: Modo edición
saveLocalFile: Guardar como fichero local
saveData: Guardar datos
dataSaved: Datos guardados
sendData: Enviar datos
saveOptions: Opciones para guardar
dataLoaded: Datos cargados
autherror: Error autenticación
loaderror: Error de carga
auth:
invalidcredentials: Credenciale no válidas
loginForm: ''
newaccount: nueva cuenta
password: password
register: Registrarse
signin: Inicial sessión
subHeading: Login info
subTitle: ''
welcome: ¡ Bienvenido/a !
button:
about: Acerca de
back: Atrás
cancel: Cancelar
close-panel: Cerarr panel
delete: Borrar
go: Ir
home: Inicio
new: Nuevo
reset: Reset
save: Guardar
toggle_dark: Alternar modo oscuro
toggle_langs: Cambiar idiomas
connection:
error: Error conexión
dashboard: Panel de control
form:
delete_done: Borrado realizado
delete_error: Error al borrar datos
save_done: Datos guardados
save_error: Error al guardar datos
intro:
desc: Plantilla de Inicio de Vite Dogmática
dynamic-route: Demo de ruta dinámica
hi: ¡Hola, {name}!
whats-your-name: ¿Cómo te llamas?
login:
end_session: Sesión finalizada
form: Credenciales
invalidcredentials: Datos incorrectos
newuser: Nuevo usuario
nodata: Datos incompetos
password: Password
register: Registrarse
forgot: ¿ Olvidado ?
checkemail: Comprobar email
reset: Reset
see_you_soon: ¡ Nos vemos pronto !
signin: Iniciar sesión
subHeading: Acceso
subtitle: LibreCloud online
user_email: Usuario / email
username: Usuario
email: Email
welcome: Bienvenido/a
message:
loading: Cargando
not-found: No se ha encontrado
notes: Notas
notifications:
content: contenido
notifications: Notificiaciones
search:
search: Buscar
payload:
exp: Expiración
id: id
uuid: uuid
data: Datos
orig_iat: Emitida
tracking:
when: Cuando
what: Qué
where: Dónde
context: Contexto
ctx: Ctx.
data: Datos
auth: Autorización
payload: Info/Autorización
list: Fechas Datos
menu:
lang: Idioma
Config: Configuración
Tracking: Seguimiento
Users: Usuarios
sec_langs: Idiomas
sec_webserver: ServidorWeb
sec_logs: Logs
sec_jwt: JWT
sec_auth: Autorizaciones
sec_perms: Permisos
sec_routes: Rutas
sec_templates: Plantillas
sec_mail: Correo
sec_assets: RecurosWeb
sec_models: ModelosDatos
sec_git: GIT
sec_redis: Redis
sec_others: Otros
sec_usersData: Cuentas
sec_modelsData: Modelos
sec_invitations: Invitaciones
sec_authzModel: Aut.Modelos
sec_authzPolicy: Aut.Polítcas
sec_tracking_list: Logs
sec_tracking_data: Datos
users:
usersData: Datos Usuarios
msg_usersData: Es un fichero en <b>YAML</b> quer reune los datos básicos de cada usuarios.<br>El <u>password</u> está encriptado en el servidor, usar el prefijo <b>passwdEnc</b> (passwdEncmiclave) para pedir que se encripte.
modelsData: Modelos de Usuarios
msg_models: Es un fichero en <b>YAML</b> que reune las claves específcas (<b>uuid</b>) los datos para la carga de modelos CV y permisos de uso.
invitations: Invitaciones para nuevos usuarios
msg_invitations: Usar <b>Num.Usos</b> con número negativo para uso no limitado.<br>Usar <b>Expira</b> (fecha en formato ISO). Atajo <b>?+num(d,h,m)</b>, por ejemplo <b>?+2h</b> para que expire en dos horas o usar <b>?+20m</b> para 20 minutos, luego ajustar el resultado.<br>Asignar un <b>Rol</b> al nuevo usuario
authzModel: Modelos Autorización
msg_authzModel: Es un <a class='link' href='https://casbin.org/' target='_blank'>modelo Casbin en RBAC <a/> para controlar el acceso a los recursos.
authzPolicy: Políticas Autorización
msg_authzPolicy: Es un <a class='link' href='https://casbin.org/' target='_blank'>modelo Casbin en RBAC de polítcas de recursos<a/> para controlar el acceso a los recursos.
sec_modelsData:
model: Modelo
user: Usuario
data: Datos
active: Activo
sec_usersData:
description: Descripción
email: Email
passwd: Password
data: Datos
web: Web
username: Usuario
sec_invitations:
email: Email
createdby: Creada por
expire: Expira
howmany: Num.Usos
role: Rol
description: Descripción
data: Datos
active: Activo
cv:
'on': activo
'off': desactivado
profile: Perfil
work_experience: Experiencia
projects: Proyectos
teaching: Enseñanza
talks: Charlas
education: Educación
expand_skills: Ampliando habilidades
others: Otros
contact: Contacto
personal: Personal
certifications: Certificaciones
languages: Idiomas
infrastructures: Infrastructuras
skills_tools: Habilidades y Herramientas
skills: Habilidades
onweb: En la Web
mission: Misión
mission_how: Detalle misión
lang:
spanish: Español
english: Inglés
french: Francés
fluent: Fluído
native: Nativo
conversation: Conversación
vars:
sec_logs: Logs Servicio Web
sec_webserver: Configuración Servidor Web
sec_jwt: JSON Web Tokens
sec_auth: Autorizaciones de acceso
sec_perms: Permisos
sec_routes: Rutas Servidor Web
sec_templates: Plantillas Servidor Web
sec_mail: Configuración Servicios de Correo
sec_assets: Recusos estáticos Servidor Web
sec_models: Modelos CV
sec_langs: Idiomas
sec_git: Git
sec_redis: Redis
sec_others: Otros
sec_logs:
logOut: Ruta fichero log principal
requestOut: Ruta directorio logs peticiones web
requestStore: Almacenamiento peticiones web
trackingOut: Ruta directorio seguimiento acciones (tracking)
trackingStore: Almacenamiento seguimiento
debugLevel: Nivel depuración
sec_webserver:
host: Host
port: Puerto
protocol: Protocolo
keyPem: Certificado clave privada
certPem: Certificado clave pública
allowOrigins: Lista orígenes permitidos (CORS)
sec_jwt:
useJWT: Usar JSON Web Tokens
jwtRealm: Ámbito JWT
jwtKey: Clave JWT
jwtTimeout: Expiración JWT
jwtMaxRefresh: Tiempo para refrescar JWT
signingAlgorithm: Algoritmo de firma JWT
jwtKeyPem: Ruta clave privada firma JWT
jwtCertPem: Ruta clave pública firma JWT
sec_auth:
authSep: Separador autorizaciones
passwdEnc: Prefijo Encriptación password
invitationsPath: Ruta fichero invitaciones usuarios
recoveryTime: Tiempo expiración recuperación
useAuthz: usar Autorizaciones Casbin
authzModel: Ruta fichero de Autorizaciones Casbin
authzPolicy: Ruta fichero de Políticas Casbin
adminRole: Rol administrador
usersStore: Almacenamiento usuarios
sec_perms:
pubUser: Usuario público
usersPath: Ruta fichero de Usuarios
usersModelsPath: Ruta fichero de Modelos de usuarios
identityKey: Clave de identidad
sec_routes:
rootAuthGroup: Grupo de Autenticación
routes: Rutas Webserver
path: Ruta fichero
param: Parámetro
sec_templates:
templatesRoot: Ruta fichero Plantilla Principal
templatesExt: Extensión ficheros plantilla
templatesIncludes: Ruta fichero a Incluir en plantillas
templatesLayouts: Ruta fichero plantilla base o maqueta
templatesFiles: Lista de Rutas de Plantillas
path: Ruta fichero
route: Ruta web
sec_mail:
mailHost: Servidor
mailPort: Puerto
mailFrom: Email de envío
mailUser: Usuario de mail
mailPswd: Password del usuario
tplsMailPath: Ruta Plantillas de Correo
tplsMail: Lista Plantillas de Correo
path: Ruta fichero
route: Ruta web
sec_assets:
assetsPath: Ruta fichero recursos web
assetsURL: URL recursos web
sec_models:
useDist: Usar Distribución (JSON)
genDist: Generar Distribución (JSON)
genExcludeList: Lista nombres a excluir
dataDistPath: Ruta fichero de Distribución (JSON)
dataPath: Ruta Ficherod e Datos (YAML files)
dataModelsRoot: Ruta fichero Principal Lista de Modelos
dataCorePath: Ruta fichero de Datos Core (principales)
dataDflt: Nombre Modelo por defecto
dataItems: Lista Items de Datos
dataStore: Almacenamiento datos
sec_langs:
langs: Lista de Idiomas
mainLang: Idioma Principal
sec_git:
useRepo: Usar Repositorio GIT
useRepoOnReq: Usar Repositorio en peticiones web (requests)
quietGit: No mostrar mensajes GIT adicionales
backgGit: Usar modo 'background' para GIT
repoPath: Directorio Local del Repositorio
repoName: Nombre del Repositorio
repoCommit: Mensaje de GIT para los 'commits'
sec_redis:
redisHost: Host Redis
redisPort: Puerto Redis
redisDB: BaseDatos Redis
redisPswd: Password BD Redis
sec_others:
openBrowser: Abrir un navegador al iniciar servidor web
info:
logs: >-
<div class='title'>Logs del Servicio Web en el servidor donde se ejecuta.</div>
<div class='note'>Ruta con permisos de lectura/escritura.</div>
webserver: >-
<div class='title'>Configuración del Servidor Web</div>
jwt: >-
<div class='title'>Configuración JSON Web Tokens (JWT)</div>
auth: >-
<div class='title'>Autorizaciones de acceso para las peticiones web</div>
perms: >-
<div class='title'>Permisos de acceso a los recursos (usuarios, grupos, políticas)</div>
routes: >-
<div class='title'>Rutas para las peticiones del Servidor Web</div>
templates: >-
<div class='title'>Plantillas para el Servidor Web</div>
mail: >-
<div class='title'>Configuración Servicios de Correo</div>
assets: >-
<div class='title'>Recursos para el Servidor Web</div>
<div class='note'>Ficheros estáticos</div>
models: >-
<div class='title'>Modelos de Datos. En formatos JSON y YAML</div>
langs: >-
<div class='title'>Idiomas </div>
git: >-
<div class='title'>GIT como Fuente de Datos</div>
redis: >-
<div class='title'>Redis</div>
others: >-
<div class='title'>Otros</div>
infosec_logs:
logOut: Ruta relativa o absoluta del fichero de logs principal
requestOut: Ruta relativa o absoluta del directorio de ficheros diarios de logs de peticiones web (requests)
requestStore: Almacenamiento peticiones web (fs/redis)
trackingOut: Ruta relativa o absoluta del directorio de ficheros diarios de seguimiento de acciones (tracking)
trackingStore: Almacenamiento sequimeinto (fs/redis)
debugLevel: Nivel de depuración (0,1,2)
infosec_webserver:
host: Nombre del 'host' o dominio para el servidor web
port: Puerto de escucha del servidor web (80, 443, etc)
protocol: Protocol del servidor web (http | https)
keyPem: Ruta fichero de la clave privada del certificado digital
certPem: Ruta fichero de la clave pública del certificado digital
allowOrigins: Lista de URLs con acceso permitido (CORS)
infosec_jwt:
useJWT: Usar JSON Web Token para gestionar/identificar accesos
jwtRealm: Nombre del ámbito o espaciio para la configuración del JWT
jwtKey: Clave para la configuración de JWT
jwtTimeout: Tiempo que tardan en expirar los Token en minutos
jwtMaxRefresh: Tiempo máximo para solicitar un Token antes de que expire
signingAlgorithm: Algoritmo de firma JWT (RS256 o '')
jwtKeyPem: Ruta de la clave privada del certificado digital para firmar JWT
jwtCertPem: Ruta de la clave pública del certificado digital para firmar JWT
infosec_auth:
authSep: Carácter de separación del usuario y su credencial en las peticiones web
passwdEnc: Prefijo Encriptación password en el servidor
invitationsPath: Ruta fichero de invitaciones a usuarios nuevos
recoveryTime: Tiempo expiración ID de recuperacción de acceso en minutos
useAuthz: Usar Autorizaciones Casbin para las peticiones web
authzModel: Ruta fichero del modelo de autorizaciones (Casbin)
authzPolicy: Ruta fichero de las políticas de autorizaciones
adminRole: Nombre del rol de administración para autorizaciones Casbin
usersStore: Almacenamiento usuarios (fs/redis)
infosec_perms:
pubUser: Nombre para identificar al usuario público (normalmente 'none')
usersPath: Ruta fichero de usuarios con credenciales de acceso encriptadas
usersModelsPath: Ruta fichero de los modelos asociados a usuarios
identityKey: Nombre del campo que identifica al usuario ('id')
infosec_routes:
rootAuthGroup: Nombre del grupo de Rutas web que requieren autorización
routes: >-
Lista de rutas, especificando 'path' (url relativa) y 'param' nombre campo o parámetro.
Las rutas nuevas han de estar definidas/gestionadas en el servidor.
infosec_templates:
templatesRoot: Ruta fichero relativa o absoluta de las plantillas para el servidor web
templatesExt: Extensión de los ficheros de plantillas (por defecto .tmpl)
templatesIncludes: Ruta de fichero a incluir en las plantillas (Cabecera, Pie, etc.)
templatesLayouts: Ruta del fichero base o maqueta para las plantillas
templatesFiles: >-
Lista de plantillas con 'path' (relativo o absoluto), 'route' ruta URL relativa
info_mail:
mailHost: Servidor, dominio o IP del servicio de Correo
mailPort: Puerto del servicio de Correo (SSL)
mailFrom: Email que envía los correos
mailUser: Usuario de acceso a los servicios de Correo
mailPswd: Password del usuario de acceso a los servicios de Correo
mailCertPath: Ruta certificado DKIM para emails
mailCertDom: Dominio certificado DKIM
tplsMailPath: Ruta Plantillas de Correo
tplsMail: >-
Lista Plantillas de Correo con 'path' (relativo o absoluto), 'tipo' (formato 'text' 'html')
infosec_assets:
assetsPath: Ruta de los ficheros de recursos estáticos para el servidor web
assetsURL: URL relativa de los recursos estáticos para el servidor web
infosec_models:
useDist: Usar los modelos de datos en modo distribución (un fichero en formato JSON)
genDist: >-
Generar los modelos de datos para distribución antes de arrancar el servidor web
o al salvar datos
genExcludeList: >-
Lista de nombres a excluir de la distribución que mantendrán la estructura de items en varios ficheros YAML.
dataDistPath: Ruta del directorio que contiene los modelos de datos para distribución (JSON)
dataPath: Ruta del directorio de datos, contiene carpetas con items en YAML
dataModelsRoot: Nombre del fichero (YAML) que contiene los modelos disponibles para el usuario
dataCorePath: Nombre del fichero de datos principal o 'Core'
dataDflt: Nombre del modelo de datos principal/por defecto
dataItems: >-
Lista de items de cada modelo de datos.
Contiene una carpeta 'showinfo' con autorizacions (ver/editar) de cada item y sus atributos o campos.
Incluye un directorio 'langs' con carpetas de los idiomas disponibles y los items que tiene traducidos.
dataStore: Almacenamiento datos (fs/redis)
infosec_langs:
langs: Lista de idiomas en código de dos letras minúscula
mainLang: Idioma pricipal y por defecto
infosec_git:
useRepo: Usar Repositorio GIT como Fuente de datos
useRepoOnReq: Usar GIT en cada petición web (pull/push)
quietGit: No mostrar mensajes adicionales de las tareas GIT
backgGit: Realizar tareas GIT en segundo plano o background
repoPath: Ruta local del directorio para usar GIT (contiene .git)
repoName: Nombre del repositorio/rama para pull/push (por defecto origin)
repoCommit: Texto para los 'commits' automáticos sobre cambios con GIT
infosec_others:
openBrowser: URL para abrir en un navegador al iniciar el servidor web
useTracking: Usa seguimiento de acciones o tracking (incluir rutas)
infosec_redis:
redisHost: Host Redis
redisPort: Puerto Redis
redisDB: BaseDatos Redis
redisPswd: Password BD Redis

21
src/App.vue Normal file
View File

@ -0,0 +1,21 @@
<template>
<component :is="layout" class="gray-700 dark:gray-200 dark:bg-gray-800 dark:text-warm-gray-100 text-warm-gray-800 dark:bg-warm-gray-800">
<transition name="slide">
<router-view />
</transition>
</component>
</template>
<script setup lang="ts">
import { useHead } from '@vueuse/head'
useHead({
title: 'CV genadmin',
meta: [
{ name: 'description', content: 'CV GenAdmin' },
],
})
const { currentRoute } = useRouter()
const appLayout = 'AppLayout'
const layout = computed(() => {
return `${currentRoute.value.meta.layout || appLayout}`
})
</script>

20
src/components/Footer.vue Normal file
View File

@ -0,0 +1,20 @@
<script setup lang="ts">
import { isDark, toggleDark } from '~/composables'
</script>
<template>
<nav text-xl mt-6 inline-flex gap-2 dark:bg-cool-gray-800 dark:text-white >
<button class="icon-btn !outline-none" @click="toggleDark()">
<div v-if="isDark" i-carbon-moon />
<div v-else i-carbon-sun />
</button>
<a
icon-btn
i-carbon-logo-github
rel="noreferrer"
href="https://github.com/antfu/vitesse-lite"
target="_blank"
title="GitHub"
/>
</nav>
</template>

View File

@ -0,0 +1,175 @@
<template>
<div>
<CheckInput
v-if="typ === InputValueOption.bool"
:typ="typ"
:inputvalue="(dataValue as boolean)"
:varname="varname"
:validate="validate"
:title="title"
:edititem="edititem"
:showitem="showitem"
:messages="messages"
@input-change="onInputChange"
/>
<TextInput
v-if="typ !== InputValueOption.bool"
:typ="typ"
:inputvalue="(dataValue as any)"
:varname="varname"
:idx="-1"
:rows="rows"
:mode="mode"
:vardef="vardef"
:validate="validate"
:title="title"
:edititem="edititem"
:showitem="showitem"
:messages="messages"
@input-change="onInputChange"
/>
<div v-if="typ === InputValueOption.arrayString">
<div v-for="(val,elmindex) in dataValue as any" :key="elmindex" class="flex flex-1">
<TextInput
:typ="InputValueOption.string"
:inputvalue="val"
:varname="varname"
:idx="elmindex"
:vardef="vardef"
:validate="validate"
:title="title"
:edititem="edititem"
:showitem="showitem"
:messages="messages"
@input-change="onInputChange"
/>
<button
v-if="edititem"
class="no-print text-sm icon-btn mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onDeleteItem(elmindex,val)"
>
<div class="noprint flex"> <div i-carbon-trash-can /> </div>
</button>
</div>
<h4 v-if="edititem" class="w-full flex flex-1 mt-5 px-1 pb-1 border-t-1 border-b-1 border-dotted border-gray-300 dark:border-gray-600 bg-gray-800">
<label v-if="labelitms && labelitms.length > 0" :for="`id-${section}`"> <span class="text-xs mr-2 text-gray-600 dark:text-gray-500">{{t(labelitms,labelitms)}}</span> </label>
<input
type="text"
class="appearance-none bg-transparent w-full text-gray-700 dark:text-gray-200 py-1 leading-tight focus:outline-none"
:class="{'border-b border-indigo-500': edititem && showitem}"
v-model="inputAdd"
:disabled="!edititem"
:id="`id-${section}`"
/>
<button
v-if="edititem"
class="no-print text-sm float-right icon-btn mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onAddItem()"
>
<div class="noprint flex">
<div i-carbon-add-filled />
</div>
</button>
</h4>
</div>
</div>
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
import { PropType } from 'vue'
// import { isDefined } from '@vueuse/core'
import TextInput from './forms/TextInput.vue'
import CheckInput from './forms/CheckInput.vue'
import { InputValueOption,InputValidateOption,InputValueModeOption} from '~/typs/inputs'
import { generate_value } from '~/hooks/utils'
const emit = defineEmits(['inputChange','inputAdd','inputDelete'])
const t = useI18n().t
const props = defineProps({
typ: {
type: String as PropType<InputValueOption>,
default: InputValueOption.string,
required: true,
},
section: {
type: String,
default: '',
required: false,
},
dataValue: {
type: [String, Number, Boolean, Array], // as PropType<String[]>],
default: '',
required: true,
},
validate: {
type: String as PropType<InputValidateOption>,
default: InputValidateOption.none,
required: true,
},
varname: {
type: String,
default: '',
required: true,
},
vardef: {
type: Object,
default: {},
required: false,
},
title: {
type: String,
default: '',
required: true,
},
labelitms: {
type: String,
default: '',
required: false,
},
rows: {
type: Number,
default: 3,
required: false,
},
mode: {
type: String as PropType<InputValueModeOption>,
default: InputValueModeOption.none,
required: false,
},
edititem: {
type: Boolean,
default: true,
required: true,
},
showitem: {
type: Boolean,
default: true,
required: true,
},
messages: {
type: Boolean,
default: true,
required: false,
},
})
const inputAdd = ref('')
const onInputChange = (data: {val: any,idx: number}) => {
emit('inputChange',{ sec: props.section, key: props.varname, idx: data.idx, val: data.val})
}
const onDeleteItem= (elmindex: number, val: string) => {
const data = props.dataValue as string[]
const newData = data.filter((_: any,idx: number) => idx !== elmindex)
emit('inputChange',{ sec: props.section, key: props.varname, val: newData})
}
const onAddItem= () => {
const newData = props.dataValue as string[]
if (inputAdd.value.startsWith('?') && props.vardef[props.varname] && props.vardef[props.varname].gen) {
inputAdd.value = generate_value(props.vardef[props.varname].gen,inputAdd.value)
}
newData.push(inputAdd.value)
emit('inputChange',{ sec: props.section, key: props.varname, val: newData})
inputAdd.value=''
}
onBeforeMount(async() => {
})
</script>

View File

@ -0,0 +1,92 @@
<template>
<button class="icon-btn mx-2" @click="toggleShowLocales">
<carbon-language class="inline-block text-gray-900 dark:text-white" />
</button>
<div v-if="showLocalesSelector" :class="`select !normal max-w-xs ${selWidth} float-right`">
<select v-model="currentLocale" class="dark:text-white dark:bg-gray-800 !outline-none">
<option
v-for="(itm: string) in listLocales"
:key="itm"
:selected="itm === currentLocale"
:value="itm"
>
{{ localeLabel(itm) }}
</option>
</select>
</div>
</template>
<script setup lang="ts">
// <vue:window on:keydown={cleanOverlay} />
import {
computed,
PropType,
} from 'vue'
import { useI18n } from 'vue-i18n'
import {LocalesLabelModes} from '~/typs'
const props = defineProps({
labelMode: {
type: String as PropType<LocalesLabelModes>,
//default: LocalesLabelModes.translation,
required: false,
},
includeCurrent: {
type: Boolean,
default: false,
required: false,
},
})
const i18n = useI18n()
const showLocalesSelector = ref(false)
const emit = defineEmits(['onLocale'])
const currentLocale = computed({
get: (): string => i18n.locale.value,
set: (val: string) => {
if (i18n.availableLocales.includes(val)) {
// console.log(`change to ${val} from ${i18n.locale.value}`)
i18n.locale.value = val
emit('onLocale', val)
showLocalesSelector.value = false
}
},
})
const locales = computed(() => i18n.availableLocales)
const listLocales = computed(() =>
props.includeCurrent ? locales.value : locales.value.filter(lcl => lcl !== currentLocale.value))
const toggleShowLocales = () => {
showLocalesSelector.value = !showLocalesSelector.value
// change to some real logic
// i18n.locale.value = i18n.availableLocales[(i18n.availableLocales.indexOf(i18n.locale.value) + 1) % i18n.availableLocales.length]
}
const localeLabel = (value: string): string => {
let label = ''
switch (props.labelMode) {
case LocalesLabelModes.auto:
case LocalesLabelModes.translation:
label = i18n.t(value, value)
break
case LocalesLabelModes.value:
label = value
break
}
return label
}
const selWidth = computed(() => {
let width = ''
switch (props.labelMode) {
case LocalesLabelModes.auto:
width = 'w-auto'
break
case LocalesLabelModes.translation:
width = 'w-28'
break
case LocalesLabelModes.value:
width = 'w-24'
break
}
return width
})
</script>

View File

@ -0,0 +1,41 @@
<template>
<div class="flex">
<div class="box">
<div class="text-center space-y-2">
<div class="space-y-0.5">
<p class="text-lg text-black dark:text-white font-semibold mb-2">
<slot name="header"></slot>
</p>
<p class="text-gray-500 dark:text-gray-300 font-medium pb-3">
<slot name="content"></slot>
</p>
</div>
<slot name="button"></slot>
</div>
</div>
</div>
</template>
<script setup lang="ts">
/*
import { PropType } from 'vue'
import { OtherType, AuthInfoType, ShowInfoType } from '~/typs/cv'
import useState from '~/hooks/useState'
const props = defineProps({
data: {
type: Array as PropType<OtherType[]>,
default: [],
required: true,
},
showinfo: {
type: Object as PropType<ShowInfoType>,
default: {},
required: true,
},
authinfo: {
type: Object as PropType<AuthInfoType>,
default: { editable: false, viewchange: false },
required: true,
},
})
*/
</script>

View File

@ -0,0 +1,191 @@
<template>
<MessageBox v-if="openMessageBox && messageType !== MessageBoxType.NotSet" class="z-99 absolute top-10 left-10 openbox">
<template v-slot:header>
<h2 class="text-indigo-700 dark:text-indigo-300">
<span v-if="messageType === MessageBoxType.Select">{{ t('selectModel', 'Select Model') }}</span>
<span v-if="messageType === MessageBoxType.Save"> {{ t('DataNeedSaved', 'Data changes not saved') }}</span>
</h2>
</template>
<template v-slot:content>
<div class="task" v-if="messageType === MessageBoxType.Save && data_url_encoded !== MessageBoxType.NotSet">
<a
:href="`data:${data_url_encoded}`"
:download="`${inputValue}${inputValue.includes('json') ? '' : '.json'}`"
>{{ t('download_json_data', 'download JSON data') }}</a>
</div>
<div
v-if="messageType === MessageBoxType.Save && data_url_encoded !== MessageBoxType.NotSet"
class="mt-2 text-xs text-800 dark:text-indigo-100"
>{{ t('click_link_to_save', 'Click on link to save') }} {{inputValue}}</div>
<form v-if="messageType === MessageBoxType.OneInput" @submit.prevent="onMessageOkBtn" class="w-full max-w-sm">
<div class="flex items-center border-b border-indigo-500 py-2">
<input
class="appearance-none bg-transparent border-none w-full text-gray-700 dark:text-gray-200 mr-3 py-1 px-2 leading-tight focus:outline-none"
:type="inputType"
v-model="oneinputValue"
:placeholder="input_placeholder"
aria-label="Full name"
/>
<div v-if="inpType === inputType" i-carbon-view @click="onInputType"/>
<div v-else i-carbon-view-off @click="onInputType"/>
</div>
</form>
<form v-if="messageType === MessageBoxType.Save && show_input" @submit.prevent="onMessageOkBtn" class="w-full max-w-sm">
<div class="flex items-center border-b border-indigo-500 py-2">
<input
class="appearance-none bg-transparent border-none w-full text-gray-700 dark:text-gray-200 mr-3 py-1 px-2 leading-tight focus:outline-none"
type="text"
v-model="inputValue"
:placeholder="input_placeholder"
aria-label="Full name"
/>
</div>
<div class="flex items-center mt-2 py-2">
<span class="text-gray-700 dark:text-gray-500 text-sm">{{t('saveload.saveOptions','Save options')}}:</span>
<div class="mt-0 flex">
<div v-if="sendurl !== ''" class="mx-2">
<label class="inline-flex items-center">
<input type="radio" class="form-radio" name="radio" v-model="inputSaveMode" :checked="inputSaveMode === 'send'" />
<span class="ml-2 text-sm">{{t('send','Send')}}</span>
</label>
</div>
<div>
<label class="inline-flex items-center">
<input type="radio" class="form-radio" name="radio" v-model="inputSaveMode" :checked="inputSaveMode !== 'send'"/>
<span class="ml-2 text-sm">{{t('local','Local')}}</span>
</label>
</div>
</div>
<button v-for="(btn:any,idx:number) in input_btns" :key="idx"
class="flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
:class="{'dark:bg-red-400 bg-red-700': btn.typ === 'cancel'}"
type="button"
@click="onInputBtn(btn)"
>{{ t(btn.title) }}</button>
</div>
</form>
<div v-if="messageType === 'select' && Object.keys(select_ops).length > 0" class="inline-block relative w-64">
<select
class="block appearance-none w-full bg-white dark:bg-gray-600 border border-gray-400 hover:border-gray-500 px-4 py-2 pr-8 rounded shadow leading-tight focus:outline-none focus:shadow-outline"
v-model="selectValue"
>
<option value=""></option>
<option v-for="(op: any) in select_ops" :key="op.id" :value="op.id">{{op.title}}</option>
</select>
<div
class="pointer-events-none absolute inset-y-0 right-0 flex items-center px-2 dark:text-indigo-200 text-indigo-700 dark:bg-gray-600 border-r-1 border-t-1 border-b-1 border-l-0 border-gray-400 "
>
<svg class="fill-current h-4 w-4" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20">
<path d="M9.293 12.95l.707.707L15.657 8l-1.414-1.414L10 10.828 5.757 6.586 4.343 8z" />
</svg>
</div>
</div>
</template>
<template v-slot:button>
<div class="flex items-center gap-5">
<button
v-if="messageType === MessageBoxType.Select || (messageType === MessageBoxType.Save && data_url_encoded === '')"
class="btn-msg flex-grow"
@click="onMessageOkBtn"
>
<span v-if="messageType === MessageBoxType.Select">{{ t('select', 'Select') }}</span>
<span v-if="messageType === MessageBoxType.Save && data_url_encoded === ''" >{{ t('save', 'Save') }}</span>
</button>
<button v-if="messageType === MessageBoxType.OneInput" :disabled="oneinputValue === ''" class="btn-msg flex-grow" @click="onMessageOkBtn">{{ t('save', 'Save') }}</button>
<button v-else class="btn-msg flex-grow" @click="onMessageCloseBtn">{{ t('close', 'Close') }}</button>
</div>
</template>
</MessageBox>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { useI18n } from 'vue-i18n'
import useState from '~/hooks/useState'
import { MessageBoxType, InputBtnsType, ModelType} from '~/typs/cv'
import MessageBox from '~/components/MessageBox.vue'
const props = defineProps({
messageType: {
type: String as PropType<MessageBoxType>,
default: MessageBoxType.NotSet,
required: true,
},
openMessageBox: {
type: Boolean,
default: false,
required: true,
},
show_input: {
type: Boolean,
default: false,
required: false,
},
input_btns: {
type: Array as PropType<InputBtnsType[]>,
default: [],
required: false,
},
input_placeholder: {
type: String,
default: '',
required: false,
},
select_ops: {
type: Object as PropType<ModelType>,
default: {},
required: true,
},
data_url_encoded: {
type: String,
default: '',
required: false,
},
inpType: {
type: String,
default: 'text',
required: false,
},
})
const emit = defineEmits(['onMessageBox','onInput','onLoadModel'])
const router = useRouter()
const inputType = ref(props.inpType)
const sendurl = router.currentRoute.value.meta.sendurl as string || '/'
const inputValue = useState().current_modelid
const oneinputValue = ref('')
const inputSaveMode = ref(sendurl !== '' ? 'send': 'local')
const selectValue = ref(useState().current_model.value.id)
const t = useI18n().t
const onInputBtn = (btn: InputBtnsType) => {
emit('onInput', btn )
}
const onMessageOkBtn = () => {
switch(props.messageType) {
case MessageBoxType.Save:
if (inputSaveMode.value === 'send') {
emit('onMessageBox', { src: 'sendata', val: inputValue.value })
} else {
emit('onMessageBox', { src: 'savedata', val: inputValue.value })
}
break
case MessageBoxType.Select:
if (useState().current_model.value.id !== selectValue.value) {
emit('onLoadModel', { id: selectValue.value })
}
emit('onMessageBox', { src: 'done' })
break
case MessageBoxType.OneInput:
if (oneinputValue.value !== '') {
emit('onMessageBox', { src: 'oneinput', val: oneinputValue.value })
}
// emit('onMessageBox', { src: 'done' })
break
}
}
const onInputType = () => {
inputType.value = inputType.value === props.inpType ? 'text' : props.inpType
}
const onMessageCloseBtn = () => {
emit('onMessageBox', { src: 'done' })
}
</script>

166
src/components/Modal.vue Normal file
View File

@ -0,0 +1,166 @@
<template>
<transition id="modal" name="modal">
<div class="modal-mask">
<div
class="modal-wrapper"
>
<div
class="modal-container bg-gray-500 dark:bg-gray-600 shadow-inner rounded border border-gray-500 dark:border-gray-400"
:style="`${cssStyle}`"
>
<div class="relative">
<div
class="absolute top-0 right-2 h-8 w-8 p-2"
>
<button
class="rounded-md text-gray-300 hover:text-white
focus:outline-none focus:ring-2 focus:ring-white
dark:focus:ring-black dark:bg-cool-gray-600 dark:text-white"
@click="OnCloseButton"
>
<span class="sr-only">Close panel</span>
<!-- Heroicon name: outline/x -->
<svg
class="h-6 w-6 dark:text-white dark:hover:text-gray-400"
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
stroke="currentColor"
aria-hidden="true"
>
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
</svg>
</button>
</div>
</div>
<div class="modal-header p-4">
<slot name="header">
</slot>
</div>
<div class="modal-body pl-4">
<slot name="body">
</slot>
</div>
<div class="modal-footer p-4">
<slot name="footer">
<button
class="modal-default-button rounded border p-1 text-gray-400 dark:text-gray-200 hover:text-black focus:outline-none focus:ring-2 focus:ring-black dark:hover:text-white dark:focus:ring-white"
@click="OnCloseButton"
>
OK
</button>
</slot>
</div>
</div>
</div>
</div>
</transition>
</template>
<script setup lang="ts">
import { onMounted, onUnmounted, ref, computed } from 'vue'
import { useI18n } from 'vue-i18n'
import useState from '~/hooks/useState'
const { t } = useI18n()
const props = defineProps({
cssStyle: {
type: String,
required: false,
default: () => '',
},
})
const emit = defineEmits(['onCloseModal'])
const OnCloseButton = () => {
useState().showModal.value = false
emit('onCloseModal')
}
const OnModalMask = (ev: any) => {
}
const OnKeydownEsc = (event: any) => {
switch (event.key) {
case 'Escape':
if (document.getElementById('modal') && useState().showModal.value) {
event.preventDefault()
event.stopImmediatePropagation()
event.stopPropagation()
useState().showModal.value = false
emit('onCloseModal')
}
break
}
}
onMounted(async() => {
document.addEventListener('keydown', OnKeydownEsc)
})
onUnmounted(async() => {
document.removeEventListener('keydown', OnKeydownEsc)
})
</script>
<style scoped>
.modal-mask {
position: fixed;
z-index: 9998;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 0.75);
display: table;
transition: opacity 0.3s ease;
}
.modal-wrapper {
display: table-cell;
vertical-align: middle;
}
.modal-container {
margin: 0px auto;
/* padding: 20px 30px; */
/* border-radius: 2px; */
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.33);
transition: all 0.3s ease;
font-family: Helvetica, Arial, sans-serif;
}
.modal-header h3 {
margin-top: 0;
/* color: #42b983; */
}
.modal-body {
margin: 20px 0;
}
.modal-default-button {
display: block;
margin-top: 1rem;
}
/*
* The following styles are auto-applied to elements with
* transition="modal" when their visibility is toggled
* by Vue.js.
*
* You can easily play with the modal transition by editing
* these styles.
*/
.modal-enter {
opacity: 0;
}
.modal-leave-active {
opacity: 0;
}
.modal-enter .modal-container,
.modal-leave-active .modal-container {
-webkit-transform: scale(1.1);
transform: scale(1.1);
}
</style>

307
src/components/NavMenu.vue Normal file
View File

@ -0,0 +1,307 @@
<template>
<nav
class="noprint flex flex-row gap-2 mx-1 mt-1 border-b-1 border-gray-800 dark:border-gray-500 bg-indigo-100 dark:bg-gray-800 dark:text-white"
:class="{ 'fixed z-50 w-full lg:w-screen-xl -top-2 pt-2 border-1': position === NavPosition.header && fixMenu, 'z-1 opacity-20': openMessageBox }"
>
<button
type="button"
class="flex-none h-6 mt-1 lg:hidden px-2 text-gray-300 hover:text-white focus:outline-none focus:text-white"
:class="{ 'transition transform-180': isOpen }"
@click.prevent="isOpen = !isOpen"
>
<svg class="h-6 w-6 fill-current" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path
v-show="isOpen"
fill-rule="evenodd"
clip-rule="evenodd"
d="M18.278 16.864a1 1 0 0 1-1.414 1.414l-4.829-4.828-4.828 4.828a1 1 0 0 1-1.414-1.414l4.828-4.829-4.828-4.828a1 1 0 0 1 1.414-1.414l4.829 4.828 4.828-4.828a1 1 0 1 1 1.414 1.414l-4.828 4.829 4.828 4.828z"
/>
<path
v-show="!isOpen"
fill-rule="evenodd"
d="M4 5h16a1 1 0 0 1 0 2H4a1 1 0 1 1 0-2zm0 6h16a1 1 0 0 1 0 2H4a1 1 0 0 1 0-2zm0 6h16a1 1 0 0 1 0 2H4a1 1 0 0 1 0-2z"
/>
</svg>
</button>
<div
class="flex-grow mt-2 text-indigo-800 dark:text-indigo-500 inline-flex gap-2 dark:bg-cool-gray-800 dark:text-white"
>
<ul
:class="isOpen ? '-mt-3 flex flex-col gap-5 bg-gray-300 pr-3 pb-5 dark:bg-gray-700 w-full' : 'hidden lg:flex lg:flex-row'"
>
<li
:class="isOpen ? 'inline-block pt-5' : ''"
class="nav-item"
style="margin-top: 0px"
@click="onHome()"
>
<carbon-home />
</li>
<li v-if="needSave" class="nav-item mt-2 lg:-mt-1" @click.prevent="onNeedSaveBtn">
<div class="flex rounded-full px-2 py-1 dark:bg-gray-600 bg-gray-300">
<div i-carbon-save />
<span class="ml-1 mt-0.5 text-xs">{{ t('save', 'Save') }}</span>
</div>
</li>
<li
v-if="currModelid && select_ops && Object.keys(select_ops).length > 0 && (authinfo.viewchange || authinfo.editable)"
class="nav-item mt-2 lg:-mt-1"
@click.prevent="onSelectBtn"
>
<div class="flex rounded-full px-2 py-1 dark:bg-gray-600 bg-gray-300">
<div i-carbon-list />
<span class="ml-1 -mt-0.2 text-xs"> {{currModelid}} </span>
</div>
</li>
<li
v-if="position === NavPosition.header && data_sections && data_sections.length > 0"
v-for="(sec) in data_sections"
:key="sec"
:id="`nav-${sec}`"
class="nav-item"
:class="{'hidden': dataAuth && !showinfo[`${sec}_itms`]}"
@click="onNavMenu(`${prefix}-${sec}`)"
>{{ t(`menu.${sec}`, sec) }}</li>
<li
v-if="position === NavPosition.header && showinfo.skills && !show_infopanel"
id="nav-skills"
class="nav-item"
@click="onNavMenu('cv-skills')"
>{{ t('cv.skills', 'Skills') }}</li>
</ul>
</div>
<div
class="flex-none text-indigo-800 dark:text-indigo-500 inline-flex gap-2 dark:bg-cool-gray-800 dark:text-white"
:class="{ 'p-0': fixMenu }"
>
<div v-if="canChange || canWrite || isAdmin" class="hidden lg:block border-gray-400 dark:border-gray-600 border-l"/>
<div v-if="currModelid" class="hidden flex-grow-0">
<div class="nav-item hover:mt-1 -mt-2 lg:-mt-1 lg:mt-1 flex rounded-full p-1.2 mt-1 border-1 border-gray-500"
@click.prevent="onSelectBtn">
<div i-carbon-list />
<span class="ml-1 text-xs"> {{currModelid}} </span>
</div>
</div>
<div v-if="useEdit || dataAuth && (canWrite || isAdmin)" class="flex-grow-0 icon-btn">
<div
class="flex icon-btn nav-item hover:mt-1 my-1.5 px-0.8 pb-0.8 rounded-sm"
:class="{ 'bg-indigo-700 text-gray-100 hover:text-gray-300 dark:bg-indigo-300 dark:text-gray-700 dark:hover:text-gray-600': authinfo.editable && needSave }"
@click="onEdit()"
>
<carbon-edit />
<span v-if="authinfo.editable" class="ml-2 mt-1 text-xs">{{ t('edit', 'Edit') }}</span>
</div>
</div>
<div v-if="dataAuth && (canChange || isAdmin)" class="flex-grow-0 icon-btn">
<div class="flex icon-btn nav-item hover:mt-1 my-1.5" @click="onViewChange()">
<carbon-erase />
<span
v-if="authinfo.viewchange"
class="ml-2 mt-1 text-xs"
>{{ t('changeview', 'Change View') }}</span>
</div>
</div>
<div class="hidden lg:block mx-1 border-gray-400 dark:border-gray-600 border-l"/>
<div v-if="useInfoPanel" class="flex-grow-0 icon-btn ml-3">
<button class="flex flex-row my-2" @click="onInfoPanel">
<svg width="1.2em" height="1.2em" class="panel-visibility-toggle">
<path
v-if="!show_infopanel"
class="eye-open"
d="M8 2.36365C4.36364 2.36365 1.25818 4.62547 0 7.81819C1.25818 11.0109 4.36364 13.2727 8 13.2727C11.6364 13.2727 14.7418 11.0109 16 7.81819C14.7418 4.62547 11.6364 2.36365 8 2.36365ZM8 11.4546C5.99273 11.4546 4.36364 9.82547 4.36364 7.81819C4.36364 5.81092 5.99273 4.18183 8 4.18183C10.0073 4.18183 11.6364 5.81092 11.6364 7.81819C11.6364 9.82547 10.0073 11.4546 8 11.4546ZM8 5.63637C6.79273 5.63637 5.81818 6.61092 5.81818 7.81819C5.81818 9.02547 6.79273 10 8 10C9.20727 10 10.1818 9.02547 10.1818 7.81819C10.1818 6.61092 9.20727 5.63637 8 5.63637Z"
/>
<path
v-else
class="eye-closed"
fill-rule="evenodd"
clip-rule="evenodd"
d="M14.8222 1.85355C15.0175 1.65829 15.0175 1.34171 14.8222 1.14645C14.627 0.951184 14.3104 0.951184 14.1151 1.14645L12.005 3.25653C10.8901 2.482 9.56509 1.92505 8.06 1.92505C3 1.92505 0 7.92505 0 7.92505C0 7.92505 1.16157 10.2482 3.25823 12.0033L1.19366 14.0679C0.998396 14.2632 0.998396 14.5798 1.19366 14.775C1.38892 14.9703 1.7055 14.9703 1.90076 14.775L14.8222 1.85355ZM4.85879 10.4028L6.29159 8.96998C6.10643 8.66645 6 8.3089 6 7.92505C6 6.81505 6.89 5.92505 8 5.92505C8.38385 5.92505 8.7414 6.03148 9.04493 6.21664L10.4777 4.78384C9.79783 4.24654 8.93821 3.92505 8 3.92505C5.8 3.92505 4 5.72505 4 7.92505C4 8.86326 4.32149 9.72288 4.85879 10.4028ZM11.8644 6.88906L13.8567 4.8968C15.2406 6.40616 16 7.92505 16 7.92505C16 7.92505 13 13.925 8.06 13.925C7.09599 13.925 6.20675 13.7073 5.39878 13.3547L6.96401 11.7895C7.29473 11.8779 7.64207 11.925 8 11.925C10.22 11.925 12 10.145 12 7.92505C12 7.56712 11.9529 7.21978 11.8644 6.88906ZM9.33847 9.41501L9.48996 9.26352C9.44222 9.31669 9.39164 9.36726 9.33847 9.41501Z"
/>
</svg>
<span class="nav-item ml-2">Panel</span>
</button>
</div>
<div class="flex-grow-0 icon-btn">
<div class="icon-btn nav-item hover:mt-1 my-1.5"
:class="{'border-1 p-1 rounded-full mt-0.3 border-gray-900 dark:border-gray-100': fixMenu}" @click="onFixMenu()">
<div v-if="fixMenu" i-carbon-pin/>
<div v-else i-carbon-pin-filled />
</div>
</div>
<div class="hidden lg:block mx-1 border-gray-400 dark:border-gray-600 border-l"/>
<div class="flex-grow-0 icon-btn">
<button class="nav-item hover:mt-1 my-1.5 !outline-none" @click="toggleDark()">
<div v-if="isDark" i-carbon-sun />
<div v-else i-carbon-moon />
</button>
</div>
<div v-if="useLogin" class="nav-item hover:mt-1 my-1.5 !outline-none">
<router-link v-if="is_logged_user()" to="Logout">
<div i-carbon-logout />
</router-link>
<router-link v-else to="Login">
<div i-carbon-login />
</router-link>
</div>
<div class="ml-2 flex-grow-0">
<MenuLocales :label-mode="localesLabelMode" :include-current="includeCurrLocale" @onLocale="onLocale" />
</div>
</div>
</nav>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n'
import { NavPosition,ShowInfoType,AuthInfoType } from '~/typs/cv'
import { isDark, toggleDark } from '~/composables'
import MenuLocales from '@/MenuLocales.vue'
import {LocalesLabelModes} from '~/typs'
import {is_logged_user} from '~/hooks/utilsAuth'
import useState from '~/hooks/useState'
const props =
defineProps({
position: {
type: String as PropType<NavPosition>,
default: '',
required: true,
},
fixMenu: {
type: Boolean,
default: false,
required: true,
},
show_infopanel: {
type: Boolean,
default: false,
required: true,
},
dataAuth: {
type: Boolean,
default: true,
required: false,
},
useInfoPanel: {
type: Boolean,
default: true,
required: false,
},
useLogin: {
type: Boolean,
default: true,
required: false,
},
showinfo: {
type: Object as PropType<ShowInfoType>,
default: {},
required: true,
},
authinfo: {
type: Object as PropType<AuthInfoType>,
default: { editable: false, viewchange: false, show: true },
required: true,
},
openMessageBox: {
type: Boolean,
default: false,
required: true,
},
needSave: {
type: Boolean,
default: false,
required: true,
},
useEdit: {
type: Boolean,
default: false,
required: false,
},
prefix: {
type: String,
default: 'cv',
required: true,
},
})
const emit = defineEmits(['onNavMenu'])
const localesLabelMode = ref(LocalesLabelModes.auto)
const includeCurrLocale = true
const t = useI18n().t
const router = useRouter()
const currModelid = useState().current_modelid
const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
const isOpen = ref(false)
const onInfoPanel = () => {
emit('onNavMenu', { src: 'infopanel' })
}
const onFixMenu = () => {
emit('onNavMenu', { src: 'fixmenu' })
}
const onLocale = (target: string) => {
emit('onNavMenu',{src: 'locale', target})
}
const isAdmin = computed(() => {
return routeKy !== '' && useState().showinfo.value.ky === routeKy ? useState().showinfo.value.admin : false
})
const canWrite = computed(() => {
const val = routeKy !== '' && useState().showinfo.value.ky === routeKy ? useState().showinfo.value.write : false
//auth_info.value.editable = val
return val
})
const canChange = computed(() => {
const val = routeKy !== '' && useState().showinfo.value.ky === routeKy ? useState().showinfo.value.change : false
//auth_info.value.viewchange = val
return val
})
const select_ops = useState().selectOps
const data_sections = useState().dataSections
const goToItem = (item: string) => {
const dom_id = document.getElementById(item)
if (dom_id) {
dom_id.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
setTimeout(() => window.scrollBy(0, isOpen.value ? -140 : -40), 4000)
isOpen.value = false
}
}
const onNavMenu = (item: string) => {
let dom_id = document.getElementById(item)
if (dom_id) {
goToItem(item)
} else {
const itm = `${item.replace(props.prefix, '')}_itms`
if (typeof useState().showinfo.value[itm] !== 'undefined') {
useState().showinfo.value[itm] = useState().showinfo.value[itm] = true
setTimeout(() => goToItem(item), 1000)
} else {
const rtitm = item.replace(`${props.prefix}-`,'')
const rt = router.getRoutes().filter(rt => rt.name === rtitm)
if (rt[0]) {
router.push(rt[0].path)
}
}
}
}
const onHome = () => {
router.push('/home')
}
const goTopPage = () => {
const dom_body = document.getElementsByTagName('body')[0]
if (dom_body) {
dom_body.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
}
}
const onEdit = () => {
emit('onNavMenu', { src: 'editable' })
}
const onViewChange = () => {
emit('onNavMenu', { src: 'viewchange' })
}
const onSelectBtn = () => {
goTopPage()
emit('onNavMenu', { src: 'select' })
}
const onNeedSaveBtn = () => {
goTopPage()
emit('onNavMenu', { src: 'save' })
}
</script>

76
src/components/Navbar.vue Normal file
View File

@ -0,0 +1,76 @@
<template>
<nav class="bg-white border-gray-200 px-2">
<div class="container mx-auto flex flex-wrap items-center justify-between">
<a href="#" class="flex">
<svg class="h-10 mr-3" width="51" height="70" viewBox="0 0 51 70" fill="none" xmlns="http://www.w3.org/2000/svg"><g clip-path="url(#clip0)"><path d="M1 53H27.9022C40.6587 53 51 42.7025 51 30H24.0978C11.3412 30 1 40.2975 1 53Z" fill="#76A9FA"></path><path d="M-0.876544 32.1644L-0.876544 66.411C11.9849 66.411 22.4111 55.9847 22.4111 43.1233L22.4111 8.87674C10.1196 8.98051 0.518714 19.5571 -0.876544 32.1644Z" fill="#A4CAFE"></path><path d="M50 5H23.0978C10.3413 5 0 15.2975 0 28H26.9022C39.6588 28 50 17.7025 50 5Z" fill="#1C64F2"></path></g><defs><clipPath id="clip0"><rect width="51" height="70" fill="white"></rect></clipPath></defs></svg>
<span class="self-center text-lg font-semibold whitespace-nowrap">FlowBite</span>
</a>
<div class="flex md:order-2">
<div class="relative mr-3 md:mr-0 hidden md:block">
<div class="absolute inset-y-0 left-0 pl-3 flex items-center pointer-events-none">
<svg class="w-5 h-5 text-gray-500" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M8 4a4 4 0 100 8 4 4 0 000-8zM2 8a6 6 0 1110.89 3.476l4.817 4.817a1 1 0 01-1.414 1.414l-4.816-4.816A6 6 0 012 8z" clip-rule="evenodd"></path></svg>
</div>
<input type="text" id="email-adress-icon" class="bg-gray-50 border border-gray-300 text-gray-900 sm:text-sm rounded-lg focus:ring-blue-500 focus:border-blue-500 block w-full pl-10 p-2" placeholder="Search...">
</div>
<button @click.prevent="onUserMenu" type="button" class="mr-3 md:mr-0 bg-gray-800 flex text-sm rounded-full focus:ring-4 focus:ring-gray-300" id="user-menu-button" aria-expanded="false" data-dropdown-toggle="dropdown">
<span class="sr-only">Open user menu</span>
<img class="h-8 w-8 rounded-full" src="/images/people/profile-picture-3.jpg" alt="user photo">
</button>
<!-- Dropdown menu -->
<div :class="{hidden: hide_user_menu}" class="absolute bg-white text-base z-50 list-none divide-y divide-gray-100 rounded shadow top-10 right-8" id="dropdown">
<div class="px-4 py-3">
<span class="block text-sm">Bonnie Green</span>
<span class="block text-sm font-medium text-gray-900 truncate">name@flowbite.com</span>
</div>
<ul class="py-1" aria-labelledby="dropdown">
<li>
<a href="#" class="text-sm hover:bg-gray-100 text-gray-700 block px-4 py-2">Dashboard</a>
</li>
<li>
<a href="#" class="text-sm hover:bg-gray-100 text-gray-700 block px-4 py-2">Settings</a>
</li>
<li>
<a href="#" class="text-sm hover:bg-gray-100 text-gray-700 block px-4 py-2">Earnings</a>
</li>
<li>
<a href="#" class="text-sm hover:bg-gray-100 text-gray-700 block px-4 py-2">Sign out</a>
</li>
</ul>
</div>
<button data-collapse-toggle="mobile-menu-2" type="button" class="md:hidden text-gray-400 hover:text-gray-900 focus:outline-none focus:ring-2 focus:ring-blue-300 rounded-lg inline-flex items-center justify-center" aria-controls="mobile-menu-2" aria-expanded="false">
<span class="sr-only">Open main menu</span>
<svg class="w-6 h-6" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 10a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 15a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1z" clip-rule="evenodd"></path></svg>
<svg class="hidden w-6 h-6" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" clip-rule="evenodd"></path></svg>
</button>
</div>
<div class="hidden md:flex justify-between items-center w-full md:w-auto md:order-1" id="mobile-menu-2">
<ul class="flex-col md:flex-row flex md:space-x-8 mt-4 md:mt-0 md:text-sm md:font-medium">
<li>
<a href="#" class="bg-blue-700 md:bg-transparent text-white block pl-3 pr-4 py-2 md:text-blue-700 md:p-0 rounded" aria-current="page">Home</a>
</li>
<li>
<a href="#" class="text-gray-700 hover:bg-gray-50 border-b border-gray-100 md:hover:bg-transparent md:border-0 block pl-3 pr-4 py-2 md:hover:text-blue-700 md:p-0">About</a>
</li>
<li>
<a href="#" class="text-gray-700 hover:bg-gray-50 border-b border-gray-100 md:hover:bg-transparent md:border-0 block pl-3 pr-4 py-2 md:hover:text-blue-700 md:p-0">Services</a>
</li>
<li>
<a href="#" class="text-gray-700 hover:bg-gray-50 border-b border-gray-100 md:hover:bg-transparent md:border-0 block pl-3 pr-4 py-2 md:hover:text-blue-700 md:p-0">Pricing</a>
</li>
<li>
<a href="#" class="text-gray-700 hover:bg-gray-50 border-b border-gray-100 md:hover:bg-transparent md:border-0 block pl-3 pr-4 py-2 md:hover:text-blue-700 md:p-0">Contact</a>
</li>
</ul>
</div>
</div>
</nav>
</template>
<script setup lang="ts" >
// import '@themesberg/flowbite'
// import { isDark, toggleDark } from '~/composables'
const hide_user_menu=ref(true)
const onUserMenu = () => hide_user_menu.value = !hide_user_menu.value
</script>

9
src/components/README.md Normal file
View File

@ -0,0 +1,9 @@
## Components
Components in this dir will be auto-registered and on-demand, powered by [`unplugin-vue-components`](https://github.com/antfu/unplugin-vue-components).
### Icons
You can use icons from almost any icon sets by the power of [Iconify](https://iconify.design/).
It will only bundle the icons you use. Check out [`unplugin-icons`](https://github.com/antfu/unplugin-icons) for more details.

View File

@ -0,0 +1,212 @@
<template>
<div>
<bubble-menu
class="bubble-menu"
:tippy-options="{ duration: 100 }"
:editor="editor"
v-if="editor"
>
<button
@click="editor.chain().focus().toggleBold().run()"
:class="{ 'is-active': editor.isActive('bold') }"
>Bold</button>
<button
@click="editor.chain().focus().toggleItalic().run()"
:class="{ 'is-active': editor.isActive('italic') }"
>Italic</button>
<button
@click="editor.chain().focus().toggleStrike().run()"
:class="{ 'is-active': editor.isActive('strike') }"
>Strike</button>
</bubble-menu>
<floating-menu
class="floating-menu"
:tippy-options="{ duration: 100 }"
:editor="editor"
v-if="editor"
>
<button
@click="editor.chain().focus().toggleHeading({ level: 1 }).run()"
:class="{ 'is-active': editor.isActive('heading', { level: 1 }) }"
>H1</button>
<button
@click="editor.chain().focus().toggleHeading({ level: 2 }).run()"
:class="{ 'is-active': editor.isActive('heading', { level: 2 }) }"
>H2</button>
<button
@click="editor.chain().focus().toggleBulletList().run()"
:class="{ 'is-active': editor.isActive('bulletList') }"
>Bullet List</button>
</floating-menu>
<editor-content :editor="editor" />
</div>
</template>
<script setup lang="ts">
import {
Editor, EditorContent,
FloatingMenu,
BubbleMenu
} from '@tiptap/vue-3'
import { PropType } from 'vue'
import Link from '@tiptap/extension-link'
import StarterKit from '@tiptap/starter-kit'
import TextStyle from '@tiptap/extension-text-style'
import { HtmlAttrsType } from '~/typs/cv'
// import { auth_data } from '~/hooks/utils'
import useState from '~/hooks/useState'
const props = defineProps({
data: {
type: String,
default: '',
required: true,
},
src: {
type: String,
default: '',
required: true,
},
field: {
type: [String, Number],
default: '',
required: true,
},
idx: {
type: Number,
default: -1,
required: true,
},
editable: {
type: Boolean,
default: false,
required: true,
},
htmlattrs: {
type: Object as PropType<HtmlAttrsType>,
default: useState().htmlAttrs,
required: false,
},
})
const emit = defineEmits(['onEditorBlur'])
const editor = new Editor({
extensions: [
StarterKit.configure({
// history: false,
bold: {
HTMLAttributes: {
class: props.htmlattrs.bold ? props.htmlattrs.bold : useState().htmlAttrs.bold,
},
},
listItem: {
HTMLAttributes: {
class: props.htmlattrs.list ? props.htmlattrs.list : useState().htmlAttrs.list,
},
},
}),
TextStyle.configure({
HTMLAttributes: {
class: props.htmlattrs.text ? props.htmlattrs.text : useState().htmlAttrs.text,
},
}),
Link.configure({
openOnClick: false,
HTMLAttributes: {
class: props.htmlattrs.link ? props.htmlattrs.link : useState().htmlAttrs.link,
},
}),
],
content: props.data,
editorProps: {
attributes: {
class: 'focus:p-2 focus:dark:bg-gray-700 focus:bg-gray-100 focus:border-gray-500 focus:border-1 prose prose-sm sm:prose lg:prose-lg xl:prose-2xl mx-auto focus:outline-none',
},
transformPastedText(text) {
return text.toUpperCase()
}
},
// '<p>Im running Tiptap with Vue.js. 🎉</p>',
onBeforeCreate({ editor }) {
// Before the view is created.
},
onCreate({ editor }) {
// The editor is ready.
//editor.commands.setContent(data)
// console.log('debugger create '+editor.state)
},
onUpdate({ editor }) {
// The content has changed.
},
onBlur({ editor, event }) {
const data = editor.getHTML()
emit('onEditorBlur', { src: props.src, field: props.field, idx: props.idx, data, ev: event })
},
injectCSS: true,
editable: props.editable,
})
onBeforeMount(async () => {
// const d = useState()
// console.log('Editor: ' + props.data)
})
onBeforeUnmount(async () => {
editor.destroy()
// console.log('Editor: destroy')
})
</script>
<style scoped>
/* Basic editor styles */
/* .ProseMirror {
> * + * {
margin-top: 0.75em;
}
ul,
ol {
padding: 0 1rem;
}
blockquote {
padding-left: 1rem;
border-left: 2px solid rgba(#0D0D0D, 0.1);
}
} */
.bubble-menu {
display: flex;
background-color: #0D0D0D;
padding: 0.2rem;
border-radius: 0.5rem;
}
.bubble-menu button {
border: none;
background: none;
color: #FFF;
font-size: 0.85rem;
font-weight: 500;
padding: 0 0.2rem;
opacity: 0.6;
}
.bubble-menu button:hover,
.bubble-menu button.is-active {
opacity: 1;
}
.floating-menu {
display: flex;
background-color: #0D0D0D10;
padding: 0.2rem;
border-radius: 0.5rem;
}
.dark .floating-menu { background-color: #0D0D0D;}
.floating-menu button {
border: none;
background: none;
font-size: 0.85rem;
font-weight: 500;
padding: 0 0.2rem;
border-left: #7c7c7c 1px solid;
opacity: 0.6;
}
.floating-menu button:hover,
.floating-menu button.is-active {
opacity: 1;
}
</style>

View File

@ -0,0 +1,61 @@
<template>
<div class="mt-3">
<input class="checkbox"
type="checkbox"
:disabled="!edititem"
:checked="inputvalue"
@change="onCheckbox($event)"
/>
<label class="form-check-label inline-block text-gray-800 dark:text-gray-200" :for="varname">
<span class="text-base">{{t(title,varname)}}</span>
</label>
</div>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { useI18n } from 'vue-i18n'
import { InputValueOption,InputValidateOption } from '~/typs/inputs'
const emit = defineEmits(['inputChange'])
const t = useI18n().t
defineProps({
typ: {
type: String as PropType<InputValueOption>,
default: InputValueOption.string,
required: true,
},
inputvalue: {
type: Boolean,
required: true,
},
varname: {
type: String,
required: true,
},
validate: {
type: String as PropType<InputValidateOption>,
default: InputValidateOption.none,
required: true,
},
title: {
type: String,
default: '',
required: true,
},
edititem: {
type: Boolean,
required: true,
},
showitem: {
type: Boolean,
required: true,
},
messages: {
type: Boolean,
default: true,
required: false,
},
})
const onCheckbox = (ev: any) => {
emit('inputChange',{val: ev.target.checked})
}
</script>

View File

@ -0,0 +1,182 @@
<template>
<div v-if="dataValue && dataVars" class="pb-2 w-full" >
<div v-for="(key: string) in Object.keys(dataValue)" :key="key">
<h4 class="w-full mt-5 px-1 pb-1 border-t-1 border-b-1 border-dotted border-gray-300 dark:border-gray-600 bg-gray-200 dark:bg-gray-700">
<span
class="ml-1 mt-5 text-normal font-medium text-indigo-800 dark:text-indigo-400"
:class="{'text-gray-300 dark:text-gray-600': !showitem}"
>
{{ key }}
</span>
<button
v-if="edititem"
class="no-print mt-2 text-sm float-right icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onDeleteItem(key)"
>
<div class="noprint flex">
<div i-carbon-trash-can />
</div>
</button>
<button
v-if="vardef.email && dataValue[key] && dataValue[key].active"
class="flex no-print text-sm float-right icon-btn mt-2 mx-2 !outline-none text-gray-500 dark:text-gray-400"
@click="onEmailItem(key)"
>
<div class="noprint flex">
<div i-carbon-email />
</div>
</button>
</h4>
<div class="w-full lg:w-6/12 sm:max-w-sm">
<div v-for="(varky: string) in Object.keys(dataVars)" :key="varky" class="">
<label :for="`${varky}-${key}`">
<span class="text-xs text-gray-600 dark:text-gray-500">
{{t(`${section}.${varky}`,varky)}}
</span>
</label>
<InputValue
:typ="dataVars[varky].typ"
:class="{'border-b border-indigo-500': edititem && showitem}"
:dataValue="dataValue[key][varky]"
:section="key"
:varname="varky"
:vardef="dataVars"
:title="`${section}.${varky}`"
:edititem="edititem"
:showitem="showitem"
:validate="dataVars[varky].valid ? dataVars[varky].valid : InputValidateOption.none"
:messages="true"
@input-change="onInputChange"
/>
</div>
</div>
</div>
<h4 v-if="edititem" class="w-full flex flex-1 mt-5 px-1 pb-1 border-t-1 border-b-1 border-dotted border-gray-300 dark:border-gray-600 bg-gray-100 dark:bg-gray-700">
<label v-if="labelitms && labelitms.length > 0" :for="`id-${varname}`"> <span class="text-xs mr-2 text-gray-600 dark:text-gray-500">{{t(labelitms,labelitms)}}</span> </label>
<input
type="text"
class="appearance-none bg-transparent w-full text-gray-700 dark:text-gray-200 py-1 leading-tight focus:outline-none"
:class="{'border-b border-indigo-500': edititem && showitem}"
v-model="inputAdd"
:disabled="!edititem"
:id="`id-${varname}`"
@keyup.enter.exact="onAddItem"
/>
<button
v-if="edititem"
class="no-print mt-2 text-sm float-right icon-btn !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onAddItem()"
>
<div class="noprint flex">
<div i-carbon-add-filled />
</div>
</button>
</h4>
</div>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { useI18n } from 'vue-i18n'
import InputValue from '@/InputValue.vue'
import { InputValueOption, InputValidateOption,VarsDataType} from '~/typs/inputs'
import { generate_value,email_info } from '~/hooks/utils'
const emit = defineEmits(['inputChange','inputModelAdd','inputModelDelete'])
const t = useI18n().t
const props = defineProps({
typ: {
type: String as PropType<InputValueOption>,
default: InputValueOption.string,
required: true,
},
section: {
type: String,
required: false,
},
dataValue: {
type: Object,
required: true,
},
dataVars: {
type: Object as PropType<VarsDataType>,
required: true,
},
validate: {
type: String as PropType<InputValidateOption>,
default: InputValidateOption.none,
required: true,
},
varname: {
type: String,
default: '',
required: true,
},
vardef: {
type: Object,
default: {},
required: false,
},
title: {
type: String,
default: '',
required: true,
},
labelitms: {
type: String,
default: '',
required: false,
},
edititem: {
type: Boolean,
default: true,
required: true,
},
showitem: {
type: Boolean,
default: true,
required: true,
},
messages: {
type: Boolean,
default: true,
required: false,
},
})
const inputAdd = ref('')
const onInputChange = (data: {sec: string, key: string, val: any,idx: number}) => {
const newData = props.dataValue
if (newData[data.sec])
newData[data.sec][data.key]=data.val
emit('inputChange',{ sec: props.section, key: props.varname, idx: data.idx, val: newData})
}
const onDeleteItem= (id: string) => {
const newData = Object.keys(props.dataValue).filter((key) => key !== id).
reduce((cur, key) => { return Object.assign(cur, { [key]: props.dataValue[key] })}, {});
emit('inputChange',{ sec: props.section, key: props.varname, idx: -1, val: newData})
}
const onAddItem= () => {
const newData = props.dataValue
const newItem : any= {}
Object.keys(props.dataVars).forEach(itm => {
// switch(props.dataVars[itm].typ) {
// case InputValueOption.string:
// break
// }
newItem[itm] = props.dataVars[itm].dflt
})
if (inputAdd.value.startsWith('?') && props.vardef.gen)
inputAdd.value = generate_value(props.vardef.gen,inputAdd.value)
newData[inputAdd.value] = newItem
emit('inputChange',{ sec: props.section, key: props.varname, idx: -1, val: newData})
inputAdd.value = ''
}
const onEmailItem= (id: string) => {
const targetData: any = Object.keys(props.dataValue).filter((key) => key === id).
reduce((cur, key) => { return Object.assign(cur, { [key]: props.dataValue[key] })}, {});
Object.keys(targetData).forEach((ky: string) => {
email_info(props.varname,props.vardef,ky,targetData[ky],t(props.title,props.varname))
})
}
</script>

View File

@ -0,0 +1,144 @@
<template>
<div>
<input
v-if="typ === InputValueOption.string"
class="appearance-none bg-transparent border-none w-full text-gray-700 dark:text-gray-200 py-1 leading-tight focus:outline-none"
:class="{'text-gray-300 dark:text-gray-600': !showitem}"
type="text"
:disabled="!edititem || !showitem"
:value="inputvalue"
:placeholder="t(title,varname)"
:aria-label="t(title,varname)"
@change="onInputChange($event)"
/>
<div v-if="typ === InputValueOption.password" class="flex">
<input
class="appearance-none bg-transparent border-none w-full text-gray-700 dark:text-gray-200 py-1 leading-tight focus:outline-none"
:class="{'text-gray-300 dark:text-gray-600': !showitem}"
:type="show_input ? 'text' : 'password'"
:disabled="!edititem || !showitem"
:value="inputvalue"
:placeholder="t(title,varname)"
:aria-label="t(title,varname)"
@change="onInputChange($event)"
/>
<button
class="no-print text-sm float-right icon-btn ml-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="show_input = !show_input"
>
<div v-if="show_input" i-carbon-view-off />
<div v-else class="noprint flex">
<div i-carbon-view />
</div>
</button>
</div>
<input
v-if="typ === InputValueOption.number"
class="appearance-none bg-transparent border-none w-full text-gray-700 dark:text-gray-200 py-1 leading-tight focus:outline-none"
type="number"
:disabled="!edititem || !showitem"
:value="inputvalue"
:placeholder="t(title,varname)"
:aria-label="t(title,varname)"
@change="onInputChange($event)"
/>
<textarea
v-if="typ === InputValueOption.content && showitem"
class="form-control block w-full px-3 py-1.5 text-base font-normal rounded-md resized transition ease-in-out m-0
text-gray-800 dark:text-gray-200
bg-white dark:bg-gray-800 bg-clip-padding
border border-solid border-gray-300 dark:border-gray-500
focus:text-gray-700 dark:focus:text-gray-300 focus:bg-gray-100 dark:focus:bg-gray-900
focus:border-gray-700 dark:focus:border-gray-300 focus:outline-none
"
:id="`input-${varname}`"
:rows="rows"
:value="inputvalue"
:disabled="!edititem || !showitem"
:placeholder="t(title,varname)"
@change="onInputChange($event)"
></textarea>
</div>
</template>
<script setup lang="ts">
import { PropType } from 'vue'
import { useI18n } from 'vue-i18n'
const { t } = useI18n()
import { MessageType} from '~/typs'
import { InputValueOption,InputValidateOption,InputValueModeOption} from '~/typs/inputs'
import { show_message,generate_value} from '~/hooks/utils'
import { input_validate } from '~/hooks/validation'
const emit = defineEmits(['inputChange'])
const props = defineProps({
typ: {
type: String as PropType<InputValueOption>,
default: InputValueOption.string,
required: true,
},
inputvalue: {
type: [String,Number,Array],
required: true,
},
varname: {
type: String,
default: '',
required: true,
},
vardef: {
type: Object,
default: {},
required: false,
},
idx: {
type: Number,
default: -1,
required: true,
},
rows: {
type: Number,
default: 3,
required: false,
},
validate: {
type: String as PropType<InputValidateOption>,
default: InputValidateOption.none,
required: true,
},
title: {
type: String,
default: '',
required: true,
},
mode: {
type: String as PropType<InputValueModeOption>,
default: InputValueModeOption.none,
required: false,
},
edititem: {
type: Boolean,
default: true,
required: true,
},
showitem: {
type: Boolean,
default: true,
required: true,
},
messages: {
type: Boolean,
default: true,
required: false,
},
})
const show_input = ref(false)
const onInputChange = (ev: any) => {
let val = ev.target.value
if (val.startsWith('?') && props.vardef[props.varname] && props.vardef[props.varname].gen)
val = generate_value(props.vardef[props.varname].gen,val)
if (input_validate(props.typ,props.validate,val)) {
emit('inputChange', {val, idx: props.idx})
} else if (props.messages) {
show_message(MessageType.Error, `${t(props.title,props.varname)} not valid`,5000)
}
}
</script>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M12 6V4m0 2a2 2 0 100 4m0-4a2 2 0 110 4m-6 8a2 2 0 100-4m0 4a2 2 0 110-4m0 4v2m0-6V4m6 6v10m6-2a2 2 0 100-4m0 4a2 2 0 110-4m0 4v2m0-6V4"
/>
</svg>
</template>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M15 17h5l-1.405-1.405A2.032 2.032 0 0118 14.158V11a6.002 6.002 0 00-4-5.659V5a2 2 0 10-4 0v.341C7.67 6.165 6 8.388 6 11v3.159c0 .538-.214 1.055-.595 1.436L4 17h5m6 0v1a3 3 0 11-6 0v-1m6 0H9"
/>
</svg>
</template>

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 5l7 7-7 7M5 5l7 7-7 7" />
</svg>
</template>

View File

@ -0,0 +1,5 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
</svg>
</template>

View File

@ -0,0 +1,19 @@
<template>
<svg
class=""
fill="#fff"
xmlns="http://www.w3.org/2000/svg"
width="76.15"
height="51.53"
viewBox="0 0 76.15 51.53"
>
<path
d="M171.78,71.73h-45.4a14.26,14.26,0,0,1-2-28.38.66.66,0,0,0,.56-.71,14.31,14.31,0,0,1-.07-1.51,18.32,18.32,0,0,1,36.43-2.73A.65.65,0,0,0,162,39h.33a14.31,14.31,0,0,1,14.2,13.11.63.63,0,0,0,.24.43,3.43,3.43,0,0,1,.56.07,4.4,4.4,0,0,1,1.1-1c-.21-.06-.43-.11-.65-.15a15.63,15.63,0,0,0-15.23-13.8,19.61,19.61,0,0,0-38.92,3.48q0,.54,0,1A15.56,15.56,0,0,0,126.38,73h45.4a.65.65,0,0,0,0-1.3Z"
transform="translate(-110.83 -21.5)"
/>
<path d="M130.56,39.18a.66.66,0,0,1-.65-.65,12.07,12.07,0,0,1,12.05-12,.65.65,0,1,1,0,1.3,10.76,10.76,0,0,0-10.75,10.75A.66.66,0,0,1,130.56,39.18Z" transform="translate(-110.83 -21.5)" /><path
d="M177.75,51.32c0-.15-.05-.29-.08-.45a5.42,5.42,0,0,1-1.22.68c0,.12,0,.24.05.36a.64.64,0,0,0,.56.57,9.67,9.67,0,0,1-1.44,19.27H162.55a5.24,5.24,0,0,1-.1,1.25h13.17a10.91,10.91,0,0,0,2.13-21.68Z"
transform="translate(-110.83 -21.5)"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M3 12l2-2m0 0l7-7 7 7M5 10v10a1 1 0 001 1h3m10-11l2 2m-2-2v10a1 1 0 01-1 1h-3m-6 0a1 1 0 001-1v-4a1 1 0 011-1h2a1 1 0 011 1v4a1 1 0 001 1m-6 0h6"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M11 16l-4-4m0 0l4-4m-4 4h14m-5 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h7a3 3 0 013 3v1"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M18 9v3m0 0v3m0-3h3m-3 0h-3m-2-5a4 4 0 11-8 0 4 4 0 018 0zM3 20a6 6 0 0112 0v1H3v-1z"
/>
</svg>
</template>

View File

@ -0,0 +1,10 @@
<template>
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"
/>
</svg>
</template>

1
src/composables/index.ts Normal file
View File

@ -0,0 +1 @@
export * from './use'

4
src/composables/use.ts Normal file
View File

@ -0,0 +1,4 @@
export const isDark = useDark()
export const toggleDark = useToggle(isDark)
// import { useMediaQuery } from '@vueuse/core'
export const isLargeScreen = useMediaQuery('(min-width: 1024px)')

172
src/hooks/config.ts Normal file
View File

@ -0,0 +1,172 @@
// import store from '~/store'
import useState from '~/hooks/useState'
import { show_message, fetch_json } from '~/hooks/utils'
import { MessageType} from '~/typs'
import 'toastify-js/src/toastify.css'
import YAML from 'yaml'
export const set_config = (res: any) => {
if (res && res.URLS && res.URLS.root) {
Object.keys(res.URLS).forEach(it => {
if (it === 'root' || res.URLS[it].includes('yaml') || res.URLS[it].includes('json') )
useState().CONFURLS.value[it] = res.URLS[it]
else
useState().CONFURLS.value[it] = res.URLS.root.includes(':') ? `${res.URLS.root}${res.URLS[it]}` : res.URLS[it]
})
}
if (res && res.ASSETS_PATH)
useState().ASSETS_PATH.value = res.ASSETS_PATH
if (res && res.DATA_PATH)
useState().DATA_PATH.value = res.DATA_PATH
if (res && res.MODEL_ID)
useState().MODEL_ID.value = res.MODEL_ID
if (res && res.URLKEY)
useState().URLKEY.value = res.URLKEY
if (res && res.APPNAME)
useState().APPNAME.value = res.APPNAME
if (res && res.AUTHKEY)
useState().AUTHKEY.value = res.AUTHKEY
if (res && res.AUTH_SEPCHAR)
useState().AUTH_SEPCHAR.value = res.AUTH_SEPCHAR
if (res && res.UUIDONE)
useState().UUIDNONE.value = res.UUIDNONE
if (res && res.TKNLIMIT)
useState().TKNLIMIT.value = res.TKNLIMIT
if (res && res.REFRESHTIME)
useState().REFRESHTIME.value = res.REFRESHTIME
if (res && res.ALLOW_REGISTER)
useState().ALLOW_REGISTER.value = res.ALLOW_REGISTER
if (res && res.RESET_PASSWORD)
useState().RESET_PASSWORD.value = res.RESET_PASSWORD
if (res && res.isDevelmode)
useState().isDevelmode.value = res.isDevelmode
if (res && res.htmlAttrs)
useState().htmlAttrs.value = res.htmlAttrs
if (res && res.dataSections)
useState().dataSections.value = res.dataSections
if (res && res.cv_model)
useState().cv_model.value = res.cv_model
}
export const get_urlpath = async (url: string, mode?: string): Promise<[any, string]> => {
if (url.length == 0)
return [{},'No URL defined']
let text = ''
let res: any = {}
let error = null
try {
const response = await self.fetch(url, {
method: 'GET',
})
if (url.includes('.json') || mode === 'json') {
res = await response.json()
} else {
text = await response.text()
}
if (response.ok) {
if (url.includes('.yaml') || mode === 'yaml') {
res = YAML.parse(text)
} else if (!url.includes('.json') && mode !== 'json') {
res = text
}
} else {
error = res.message
}
}
catch (err: any) {
error = err.message
}
return [res,error]
}
export const load_config = async (): Promise<[any, string]> => {
const url = window.CONFIG_LOCATION ? window.CONFIG_LOCATION : '/config.yaml'
// const url = window.CONFIG_LOCATION ? window.CONFIG_LOCATION : 'config.json'
const [res,error] = await get_urlpath(url)
if (error) {
show_message(MessageType.Error, `'Config Load' -> ${error}`, 5000)
useState().connection.value.state = 'connection.error'
return [{},error]
}
return [res, error]
}
export const load_serv_config = async (): Promise<[any,string]> => {
const url = useState().CONFURLS.value.srvconfig || ''
if (url.length == 0)
return [{},'No SRV URL defined']
const [res,error] = await get_urlpath(url)
if (error) {
show_message(MessageType.Error, `'Serv Config Load' -> ${error}`, 5000)
useState().connection.value.state = 'connection.error'
return [{},error]
}
return [res,error]
}
export const get_serv_config = async (): Promise<[any,string]> => {
const url = useState().CONFURLS.value.srvconfig || ''
if (url.length == 0)
return [{},'No SRV URL defined']
const [res,errmsg] = await fetch_json(url, 2000,true)
// fetch_json(url, 2000,to.meta.withauth,(res,errmsg) => {
if (errmsg.length > 0) {
return [{},errmsg]
}
return [res,errmsg]
}
export const load_vars_config = async (): Promise<[any,string]> => {
const url = useState().CONFURLS.value.varsconfig || ''
if (url.length == 0)
return [{},'No URL defined']
let text = ''
let res: any = {}
let error = null
try {
const response = await self.fetch(url, {
method: 'GET',
})
if (url.includes('.json')) {
res = await response.json()
} else {
text = await response.text()
}
if (response.ok) {
if (url.includes('.yaml')) {
res = YAML.parse(text)
} else if (!url.includes('.json')) {
res = text
}
} else {
error = res.message
}
}
catch (err: any) {
error = err.message
}
if (error) {
show_message(MessageType.Error, `'Vars Config Load' -> ${error}`, 5000)
useState().connection.value.state = 'connection.error'
return [{}, error]
}
return [res,error]
}
export const has_config = async(): Promise<boolean> => {
let url = useState().CONFURLS.value.root || ''
if (url.length === 0) {
const [res,err] = await load_config()
if (err && err.length > 0) {
return false
}
set_config(res)
url = useState().CONFURLS.value.root || ''
if (url.length === 0)
return false
}
return true
}
export default {
set_config,
load_config,
get_urlpath,
load_serv_config,
get_serv_config,
load_vars_config,
has_config,
}

112
src/hooks/loads.ts Normal file
View File

@ -0,0 +1,112 @@
import useState from '~/hooks/useState'
import { fetch_json } from '~/hooks/utils'
import { load_config } from '~/hooks/config'
export const store_info = (key:string,res: any) => {
useState().cvdata.value = {}
useState().datalang.value = {}
let models = []
if (res.models) {
models = res.models
}
Object.keys(res.data).forEach(ky => {
if (ky === 'main') {
useState().cvdata.value = res.data.main
if (res.data.main.models) {
res.data.main.models.forEach((mdl: any) => models.push(mdl))
}
if (res.data.main.showinfo) {
const showItms = res.data.main.showinfo.filter((it: any) => it.ky === key)
if (showItms[0]) {
useState().showinfo.value = showItms[0]
} else {
const showPub = res.data.main.showinfo.filter((it: any) => it.ky === '')
if (showPub[0])
useState().showinfo.value = showPub[0]
}
}
} else {
useState().datalang.value[ky] = {}
Object.keys(res.data[ky]).forEach(sec => {
if (res.data[ky][sec]) {
useState().datalang.value[ky][sec] = res.data[ky][sec]
}
})
}
})
useState().models.value = models
models.forEach((mdl: any) => useState().selectOps.value[mdl.id] = mdl)
}
export const load_data_model = async(id: string, rootpath: string, ky: string, with_auth: boolean, cllbck: any) => {
if (id && useState().selectOps.value[id]) {
const url = rootpath.includes('http') ? `${rootpath}/${id}` : useState().selectOps.value[id].path.replace('~',rootpath)
const [res,err] = await fetch_json(url, 2000, with_auth)
if (err.length === 0 && res) {
store_info(ky,res)
useState().current_model.value = useState().selectOps.value[id]
useState().current_modelid.value = id
if (cllbck) {
cllbck()
}
}
}
}
export const load_data_info = async (to: any, _from: any, next: any) => {
let url = useState().CONFURLS.value.root || ''
if (url.length > 0) {
const [_res,err] = await load_config()
if (err && err.length > 0) {
next()
return
}
}
url = useState().CONFURLS.value.data || ''
if (url.length == 0) {
next()
return
}
url = `${url}/${useState().MODEL_ID.value}`
const [res,errmsg] = await fetch_json(url, 2000,to.meta.withauth)
// fetch_json(url, 2000,to.meta.withauth,(res,errmsg) => {
if (errmsg.length > 0) {
next()
return
}
if (res && res.data) {
const ky = to.params.ky || to.query.k || ''
store_info(ky,res)
if (useState().selectOps.value[useState().MODEL_ID.value]) {
useState().current_model.value = useState().selectOps.value[useState().MODEL_ID.value]
useState().current_modelid.value = useState().MODEL_ID.value
}
} else {
//next('404')
next()
return
}
next()
}
export const load_currentRoute= async(rt: any): Promise<boolean> => {
let url = useState().CONFURLS.value.data || ''
if (url.length == 0) {
return false
}
url = `${url}/${useState().MODEL_ID.value}`
const [res,errmsg] = await fetch_json(url, 2000,rt.meta.withauth)
if (errmsg.length === 0 && res && res) {
const ky = rt.params.ky || rt.query.k || ''
store_info(ky,res)
if (useState().models.value[rt.meta.model]) {
useState().current_model.value = useState().models.value[rt.meta.model]
}
useState().current_modelid.value = rt.meta.model
return true
}
return false
}
export default {
store_info,
load_data_model,
load_data_info,
load_currentRoute,
}

75
src/hooks/tools.ts Normal file
View File

@ -0,0 +1,75 @@
import { send_data, show_message } from '~/hooks/utils'
import useState from '~/hooks/useState'
import { MessageType} from '~/typs'
import 'toastify-js/src/toastify.css'
import { DateTime } from 'luxon'
export const format_date = (ndate: number):string => {
let value = ''
try {
const d = new Date(ndate);
value = d.toLocaleString();
} catch(e) {
console.log(`error date to string ${ndate}`)
}
return value
}
export const utf8_to_b64 = (str: string):string => {
return window.btoa(unescape(encodeURIComponent( str )));
}
export const b64_to_utf8 = (str: string):string => {
return decodeURIComponent(escape(window.atob( str )));
}
export const generate_uuid = ():string => {
let dt = new Date().getTime()
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
const r = (dt + Math.random()*16)%16 | 0;
dt = Math.floor(dt/16);
return (c=='x' ? r :(r&0x3|0x8)).toString(16);
})
}
export const generate_value = (gen: string, val: string):string => {
switch (gen) {
case 'uuid':
return generate_uuid()
case 'NowSet':
const now = DateTime.now().toUTC()
if (val === '?') {
return now.toISO()
} else if (val.endsWith('d')) {
const num = parseInt(val.replace('?', '').replace('d', '')) || 0
return now.plus({days: num}).toISO()
} else if (val.endsWith('h')) {
const num = parseInt(val.replace('?', '').replace('h', '')) || 0
return now.plus({hours: num}).toISO()
} else if (val.endsWith('m')) {
const num = parseInt(val.replace('?', '').replace('m', '')) || 0
return now.plus({minutes: num}).toISO()
}
default:
return val ? val : ''
}
}
export const email_info = async(src: string, defs: any, key: string, data: any, msg: string) => {
switch (src) {
case 'invitations':
const url = useState().CONFURLS.value.sendinvitation || ''
if (url.length > 0) {
const [res, err] = await send_data(url, { id: key, data }, true, true, () => {
show_message(MessageType.Success,`${msg} sent`, 2000)
})
if (err && err.length > 0 && res && res.status && res.status !== 'ok') {
show_message(MessageType.Error, `Error: ${err}`, 2000)
}
}
break
}
}
export default {
format_date,
utf8_to_b64,
b64_to_utf8,
generate_uuid,
generate_value,
}

48
src/hooks/tracking.ts Normal file
View File

@ -0,0 +1,48 @@
import useState from '~/hooks/useState'
import { MessageType, TrackActionType } from '~/typs'
import { show_message,send_data} from '~/hooks/utils'
export const track_action = async(e: any, src?: {ref: string,text?: string}, act = 'click') => {
const url = useState().CONFURLS.value.tracking || ''
if (url.length == 0) {
return
}
let href=''
let text=''
if (src && src.ref && src.ref.length > 0 ) {
href=src.ref
text=src.text ? src.text : src.ref
} else if (e && e.target) {
switch(e.target.tagName) {
case 'IMG':
const imgTarget = e.target.parentNode.tagName === 'A' ? e.target.parentNode : null
if (imgTarget && imgTarget.tagName === 'A') {
href = imgTarget.href ? imgTarget.href : ''
text = e.target.alt ? e.target.alt : e.target.src.slice(-1)[0]
}
break
case 'A':
const linkTarget = e.target
if (linkTarget && linkTarget.tagName === 'A') {
href = linkTarget.href ? linkTarget.href : ''
text = linkTarget.innerHTML ? linkTarget.innerHTML : ''
}
break
}
}
const trackAction: TrackActionType = {
when: Date.now().toString(),
where: `${useState().APPNAME.value}>${text}: ${href}`,
what: act,
context: navigator.userAgent,
data: useState().userID.value,
}
const [_res,err] = await send_data(url, trackAction, true, true)
if (err && err.length > 0 ) {
show_message(MessageType.Error, `Error: ${err}`,2000)
}
}
export default {
track_action,
}

97
src/hooks/useComponent.ts Normal file
View File

@ -0,0 +1,97 @@
import { ref } from 'vue'
// import WysiwygEditor from '@/WysiwygEditor.vue'
// // import CodeEditor from '@/CodeEditor.vue'
// import GridSettings from '~/views/GridSettings.vue'
// import GridView from '~/views/GridView.vue'
// import TableView from '~/views/TableView.vue'
// import ListView from '~/views/ListView.vue'
// import FormView from '~/views/FormView.vue'
// import TaFormView from '/app_modules/bm/ta/views/ta_form.vue'
// import TaTableView from '/app_modules/bm/ta/views/ta_table.vue'
// import TaListView from '/app_modules/bm/ta/views/ta_list.vue'
export enum DynComponent {
// GridSettings,
// GridJs,
// TableView,
// ListView,
// FormView,
// WysiwygEditor,
// CodeEditor,
}
const asideComponent = ref({})
const settingsComponent = ref({})
const fullSliderComponent = ref({})
const formViewComponent = ref({})
const dataViewComponent = ref({})
const topPaneComponent = ref({})
const bottomPaneComponent = ref({})
const moduleComponent = ref({})
const getModuleComponent = (key: string, target: string): any => {
// switch (key) {
// case 'ta':
// switch (target) {
// case 'form':
// return TaFormView
// break
// case 'table':
// return TaTableView
// break
// case 'list':
// return TaListView
// break
// }
// break
// }
}
const getComponent = (cmpnt: DynComponent): any => {
// switch (cmpnt) {
// case DynComponent.WysiwygEditor:
// return WysiwygEditor
// break
// // case DynComponent.CodeEditor:
// // return CodeEditor
// // break
// case DynComponent.GridSettings:
// return GridSettings
// break
// case DynComponent.GridJs:
// return GridView
// break
// case DynComponent.TableView:
// return TableView
// break
// case DynComponent.ListView:
// return ListView
// break
// case DynComponent.FormView:
// return FormView
// break
// }
}
export default function useComponent() {
return {
getModuleComponent,
asideComponent,
settingsComponent,
fullSliderComponent,
dataViewComponent,
formViewComponent,
getComponent,
topPaneComponent,
bottomPaneComponent,
moduleComponent,
}
}

196
src/hooks/useState.ts Normal file
View File

@ -0,0 +1,196 @@
import { ref } from 'vue'
import type { SideMenuItemType, UiPanelsType } from '~/typs/cmpnts'
import { DataLangsType, ShowInfoType, HtmlAttrsType, ModelType, ModelSelType, CVDataType } from '~/typs/cv'
import { VarsConfigType,ConfUrlsType } from '~/typs/config'
import { VarsUserType} from '~/typs/users'
const reqError = ref({ defs: '', lang: '', data: '', gql: '', api: '' })
const currentMapKey = ref('')
const isSidebarOpen = ref(false)
const toggleSidebar = () => {
isSidebarOpen.value = !isSidebarOpen.value
}
const showModal = ref(false)
const isAsidePanelOpen = ref(false)
const sideSettingButton = ref(false)
const isSettingsPanelOpen = ref(false)
const isSearchPanelOpen = ref(false)
const usideSettingButton = ref(false)
const isNotificationsPanelOpen = ref(false)
const useSettings = ref(false)
const useSearch = ref(false)
const search = ref('')
const isDevelmode = ref(false)
const bcPath = ref('')
const navTitle = ref({
text: '',
textclick: null as null | Function,
title: '',
cmpnt: '',
ops: [] as any[],
btntype: '',
cllbck: null as null | Function,
})
const dfltNavTitle = () => {
navTitle.value = {
text: '',
textclick: null as null | Function,
title: '',
cmpnt: '',
ops: [] as any[],
btntype: '',
cllbck: null as null | Function,
}
}
const bookCllbck = ref()
const bookSelec = (data: string) => {
if (data === '#') {
bcPath.value = ''
dfltNavTitle()
}
else {
if (bookCllbck.value)
bookCllbck.value(data)
// else ...
}
}
const sidebarMenuItems = ref([] as SideMenuItemType[])
const checkin = ref(false)
const connection = ref({
state: '',
})
const side_menu_click = ref()
const app_home_click = ref()
const backdrop_blur = ref(14)
const back_opacity = ref(60)
const panels = ref({} as UiPanelsType)
const show_profile = ref(false)
const cvdata = ref({} as CVDataType | any)
const datalang = ref({} as DataLangsType | any)
const showinfo = ref({} as ShowInfoType)
const authinfo = ref({ editable: false, viewchange: false, show: true })
const dataSections = ref([] as string[])
const htmlAttrs = ref({
bold: '', //'itm-title',
list: 'list-circle ml-5',
link: 'link',
text: '',
} as HtmlAttrsType)
const cv_model = ref({} as ModelType)
const models = ref({} as ModelType)
const selectOps = ref({} as ModelSelType)
const current_model = ref({} as ModelType)
const current_modelid = ref("")
const APPNAME = ref('')
const AUTHKEY = ref('auth')
const UUIDNONE = ref('none')
const TKNLIMIT = ref(20)
const REFRESHTIME = ref(5)
const ASSETS_PATH = ref("/assets")
const DATA_PATH = ref("/assets/data")
// const DATA_URL = ref("${DATA_PATH}/dist/info.json")
const MODEL_ID = ref("cv")
const URLKEY = ref("?k: ") // location.search.replace('?k: ','')
const AUTH_SEPCHAR = ref(";")
const PASSWD_ENC = ref("enc;")
const ALLOW_REGISTER = ref(false)
const RESET_PASSWORD = ref(false)
const CONFURLS = ref({} as ConfUrlsType)
const timeoutAuth = ref(0)
const allowView = ref(false)
const userID = ref('')
const sourceRoute = ref('')
const varsConfig = ref({} as VarsConfigType)
const varsUser = ref({} as VarsUserType)
export default function useState() {
return {
reqError,
currentMapKey,
isSidebarOpen,
toggleSidebar,
isAsidePanelOpen,
isSettingsPanelOpen,
isSearchPanelOpen,
isNotificationsPanelOpen,
useSettings,
usideSettingButton,
sideSettingButton,
bcPath,
bookSelec,
bookCllbck,
navTitle,
dfltNavTitle,
checkin,
connection,
sidebarMenuItems,
showModal,
side_menu_click,
app_home_click,
useSearch,
search,
back_opacity,
backdrop_blur,
panels,
show_profile,
cvdata,
datalang,
showinfo,
authinfo,
dataSections,
htmlAttrs,
selectOps,
models,
cv_model,
current_model,
current_modelid,
timeoutAuth,
isDevelmode,
allowView,
userID,
sourceRoute,
varsConfig,
varsUser,
// APP CONFIG VARS
APPNAME,
AUTHKEY,
UUIDNONE,
TKNLIMIT,
REFRESHTIME,
ASSETS_PATH,
DATA_PATH,
MODEL_ID,
URLKEY,
AUTH_SEPCHAR,
PASSWD_ENC,
ALLOW_REGISTER,
RESET_PASSWORD,
CONFURLS,
}
}

178
src/hooks/utils.ts Normal file
View File

@ -0,0 +1,178 @@
import Toastify from 'toastify-js'
import useState from '~/hooks/useState'
import { checkAuth} from '~/hooks/utilsAuth'
import { MessageType} from '~/typs'
import 'toastify-js/src/toastify.css'
export const show_message = (typ: MessageType, msg: string, timeout?: number, cllbck?: any): void => {
let cssClass = 'msg-box'
switch (typ) {
case MessageType.Show:
cssClass += ' msg-show'
break
case MessageType.Success:
cssClass += ' msg-ok'
break
case MessageType.Error:
cssClass += ' msg-error'
break
case MessageType.Warning:
cssClass += ' msg-warn'
break
case MessageType.Info:
cssClass += ' msg-info'
break
}
// https://github.com/apvarun/toastify-js
Toastify({
text: msg,
duration: timeout || 3000,
//destination: '',
//selector: '',
// newWindow: true,
// close: true,
className: cssClass,
offset: {
x: 10, // horizontal axis - can be a number or a string indicating unity. eg: '2em'
y: 50, // vertical axis - can be a number or a string indicating unity. eg: '2em
},
gravity: 'top', // `top` or `bottom`
position: 'left', // `left`, `center` or `right`
// backgroundColor: 'linear-gradient(to right, #00b09b, #96c93d)',
stopOnFocus: true, // Prevents dismissing of toast on hover
callback: cllbck
// onClick() {}, // Callback after click
}).showToast()
}
export const translate = (store: any, map: string, ky: string, ctx: string, dflt?: string): string => {
const val = dflt || ky
const lang: any = {}
if (lang && lang.value)
return lang.value[ctx] && lang.value[ctx][ky] ? lang.value[ctx][ky] : val
else if (lang && lang[ctx])
return lang[ctx][ky] ? lang[ctx][ky] : val
else
return val
}
export const fetch_text = async(path: RequestInfo): Promise<any> => {
try {
const response = await fetch(path)
return !response.ok ? response.text() : new Error('No items found')
}
catch (err) {
return err
}
}
const request_headers = async (with_auth: boolean): Promise<[any, string]> => {
const headers: any = {} // { 'Content-Type': 'application/json' }
headers['Accept']='application/json'
let error= ''
if (with_auth) {
const token = await checkAuth()
if (token && token.length > 0) {
headers['Authorization'] =`Bearer ${token}`
} else {
return [null,`error no auth found`]
}
}
return [headers, error]
}
export const fetch_json = async(path: RequestInfo, timeout = 2000, with_auth = false, cllbck?: any): Promise<any> => {
useState().connection.value.state = ''
let res = null
let error = ''
const showError = (err: string) => {
show_message(MessageType.Error, `'Data Load' -> ${err}`,5000)
useState().connection.value.state = 'connection.error'
}
const [headers,errorHeader] = await request_headers(with_auth)
if (errorHeader.length > 0) {
return [null,errorHeader]
}
try {
const response = await self.fetch(path, {
method: 'GET',
headers,
//mode: 'cors',
})
res = await response.json()
if (!response.ok && res.message.length > 0)
error=res.message
}
catch (err: any) {
error = err
}
if (error.length > 0)
showError(error)
return [res,error]
}
export const send_data = async(url: string, formData: any, with_auth = true, show = true, cllbck?: any): Promise<any> => {
const [headers,errorHeader] = await request_headers(with_auth)
if (errorHeader.length > 0) {
return [null,errorHeader]
}
headers['Content-Type'] = 'application/json'
let formDataJsonString = ''
try {
formDataJsonString = JSON.stringify(formData)
}
catch (e) {
console.log(e)
return [null,e]
}
try {
const response = await fetch(url, {
method: 'POST',
headers,
body: formDataJsonString,
})
if (!response.ok && response.status !== 200) {
const errorMessage = await response.json()
console.log(errorMessage)
if (show)
show_message(MessageType.Error, `Send data -> ${errorMessage.error ? errorMessage.error : ''}`)
return [null,errorMessage]
// throw new Error(errorMessage)
} else if (response.ok && response.status === 200) {
const res = await response.json()
if (cllbck) {
cllbck(res)
}
return [res,null]
}
}
catch (e) {
if (show)
show_message(MessageType.Error, `Send data -> ${e}`)
useState().connection.value.state = 'connection.error'
console.log(e)
return [null,e]
}
}
export const run_task = (val: number, task: Function) => {
const now = new Date().getTime()
const timePassed = now % val
const run_at = val - timePassed
setTimeout(task, run_at)
}
export const parse_str_json_data = (src: string, dflt: object|any):object|any => {
let data = dflt
try {
data = JSON.parse(src)
}
catch (e) {
data = dflt
}
return data
}
export default {
fetch_text,
fetch_json,
send_data,
show_message,
translate,
run_task,
parse_str_json_data,
}

270
src/hooks/utilsAuth.ts Normal file
View File

@ -0,0 +1,270 @@
import useState from '~/hooks/useState'
import { MessageType} from '~/typs'
import { show_message } from './utils.js'
export const check_credentials = async(url: string, data: any): Promise<any> => {
let dataJsonString = ''
try {
dataJsonString = JSON.stringify(data)
}
catch (e) {
console.log(e)
return
}
try {
const response = await fetch(url, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: dataJsonString,
})
if (!response.ok) {
const errorMessage = await response.text()
// throw new Error(errorMessage)
console.log(errorMessage)
return
}
if (response.ok && response.status === 200)
return response.json()
}
catch (e) {
useState().connection.value.state = 'connection.error'
console.log(e)
}
}
export const parseJwt = (token: string|null): any => {
let res = {}
if (token && token.length === 0 || token === "?") {
return res
}
if (token) {
try {
const base64Url = token.split('.')[1];
const base64 = base64Url.replace(/-/g, '+').replace(/_/g, '/');
const jsonPayload = decodeURIComponent(atob(base64).split('').map((c) =>
'%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2)
).join(''));
res = JSON.parse(jsonPayload);
} catch(e) {
console.log(e)
localStorage.removeItem(useState().AUTHKEY.value)
}
}
return res
}
export const authExpired = (): boolean => {
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
if (payload.exp) {
const now = Date.now() / 1000;
return payload.exp - now < 0
} else {
return true
}
}
export const onTimeoutAuth = () => {
if (useState().timeoutAuth.value > 0) {
if (useState().isDevelmode.value)
console.log(`timeout already set to: ${useState().timeoutAuth.value}`)
return
}
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
const now = Date.now() / 1000;
if (payload.exp) {
let timeout = Math.round(payload.exp - now)
const limit = useState().TKNLIMIT.value * 60
timeout = timeout < limit ? timeout - (useState().REFRESHTIME.value * 60) : limit - (useState().REFRESHTIME.value * 60)
timeout = timeout < 0 ? 100 : timeout
useState().timeoutAuth.value = timeout
if (useState().isDevelmode.value)
console.log(`AUTH timeout(${limit}): ${timeout} [${Math.round(payload.exp - now)}]`)
setTimeout(() => {
const r = async() => {
useState().timeoutAuth.value=0
await refreshAuth()
}
r()
}, (timeout*1000))
}
}
export const refreshAuth = async(val = 0) => {
const url = useState().CONFURLS.value.refreshauth || ''
if (url.length == 0) {
console.log('URL not found for refreshToken')
return
}
let res: any = '', error: any = ''
const headers: any = {} // { 'Content-Type': 'application/json' }
headers['Accept']='application/json'
const token = localStorage.getItem(useState().AUTHKEY.value)
if (token && token.length > 0) {
headers['Authorization'] =`Bearer ${token}`
}
try {
const response = await self.fetch(url, {
method: 'GET',
headers,
//mode: 'cors',
})
res = await response.json()
if (!response.ok && res.message && res.message.length > 0)
error=res.message
}
catch (err: any) {
error = err.message
}
if (res && res.error && res.error.length > 0) {
error = res.error
}
if (error && error.length > 0) {
show_message(MessageType.Error, `'Auth Refresh' -> ${error}`,5000)
useState().connection.value.state = 'connection.error'
setTimeout(() => {
location.href = '/logout'
}, 1000)
return
}
if (res && res[useState().AUTHKEY.value]) {
localStorage.setItem(useState().AUTHKEY.value,res[useState().AUTHKEY.value])
console.log(`New auth: ${localStorage.getItem(useState().AUTHKEY.value)}`)
onTimeoutAuth()
}
}
export const getAuth = async(urlpath = '',val = 0): Promise<[any,string]> => {
const urlkey = location.search.replace(useState().URLKEY.value,'')
const uuid = urlkey.length > 0 ? urlkey : useState().UUIDNONE.value
const url = urlpath.length > 0 ? urlpath : (useState().CONFURLS.value.auth ? useState().CONFURLS.value.auth.replace('URLKEY',uuid) : '')
if (url.length === 0) {
console.log('URL not found to get Auth')
return [null,'']
}
let res: any = ''
let errmsg: any = ''
try {
const response = await self.fetch(url, {
method: 'GET',
//mode: 'cors',
})
res = await response.json()
if (!response.ok && res.message.length > 0)
errmsg=res.message
}
catch (e: any) {
errmsg = e.message
}
if (errmsg && errmsg.length > 0) {
show_message(MessageType.Error, `'Auth Data Load' -> ${errmsg}`,5000)
useState().connection.value.state = 'connection.error'
}
if (res && res.auth) {
if (res.auth === "?") {
useState().userID.value = "?"
} else {
useState().userID.value = res.user ? res.user : ''
localStorage.setItem(useState().AUTHKEY.value,res.auth)
onTimeoutAuth()
}
}
return [res,errmsg]
}
export const checkUserAuth = async(val: string): Promise<boolean> => {
if (val.length === 0) {
return false
}
const urlkey = location.search.replace(useState().URLKEY.value,'')
const uuid = urlkey.length > 0 ? urlkey : useState().UUIDNONE.value
const urlpath = useState().CONFURLS.value.auth ? useState().CONFURLS.value.auth.replace('URLKEY',uuid) : ''
if (urlpath.length === 0) {
console.log('URL not found to check Auth')
return false
}
const psw = btoa(val)
const url = `${urlpath}${useState().AUTH_SEPCHAR.value}${psw}`
const [res,errmsg] = await getAuth(url,0)
if (errmsg.length === 0 && res && res.user) {
useState().allowView.value=true
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
if (payload && payload.id && payload.id === res.user) {
return true
}
}
return false
}
export const checkPerms = async(auth: string|null): Promise<boolean> => {
if (auth && auth.length === 0 || auth === '?')
return false
if (useState().allowView.value)
return true
const payload = parseJwt(auth)
if (payload && payload.uuid && payload.uuid !== useState().UUIDNONE.value) {
useState().allowView.value=false
return true
}
useState().allowView.value=true
return true
}
export const checkAuth = async(urlpath = '',val = 0): Promise<string|null> => {
if (authExpired()) {
const [res,errmsg] = await getAuth(urlpath,val)
if (errmsg.length === 0 && res && res.auth) {
return await checkPerms(res.auth) ? res.auth : ''
}
return ''
}
if (location.pathname === '/' && useState().URLKEY.value.length > 0) {
const urlkey = location.search.replace(useState().URLKEY.value,'')
const uuid = urlkey.length > 0 ? urlkey : useState().UUIDNONE.value
const data = authData()
if (data.payload && data.payload.uuid && data.payload.uuid !== uuid) {
const url = useState().CONFURLS.value.auth ? useState().CONFURLS.value.auth.replace('URLKEY',uuid) : ''
if (url.length === 0) {
console.log('URL not found to check Perms')
return ''
}
const [res,_err] = await getAuth(url,val)
if (res.auth) {
return await checkPerms(res.auth) ? res.auth : ''
}
return ''
} else {
onTimeoutAuth()
return await checkPerms(data.auth) ? data.auth : ''
}
} else {
const data = authData()
if (data.payload && data.payload.id)
useState().userID.value = data.payload.id
else
return ''
}
onTimeoutAuth()
const res: string|null = localStorage.getItem(useState().AUTHKEY.value)
return await checkPerms(res) ? res : ''
}
export const authData = ():any => {
const auth = localStorage.getItem(useState().AUTHKEY.value) || ''
const uidefs = {}
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
return {auth, payload, uidefs}
}
export const is_logged_user = (): boolean => {
const urlkey = location.search.replace(useState().URLKEY.value, '')
if (urlkey.length > 0) {
if (authExpired()) {
return false
}
return true
}
return false
}
export default {
check_credentials,
parseJwt,
authExpired,
checkAuth,
checkUserAuth,
refreshAuth,
getAuth,
authData,
is_logged_user,
}

13
src/hooks/validation.ts Normal file
View File

@ -0,0 +1,13 @@
import { InputValueOption,InputValidateOption} from '~/typs/inputs'
export const input_validate = (typ: InputValueOption, validator: InputValidateOption, val: any): boolean => {
switch (validator) {
case InputValidateOption.none:
// if (typ === InputValueOption.number) {
// }
break
case InputValidateOption.mail:
return val.includes('@')
}
return true
}

11
src/layouts/Default.vue Normal file
View File

@ -0,0 +1,11 @@
<template>
<main class="px-4 py-10 text-center text-gray-700 dark:text-gray-200">
<transition name="fade">
<router-view />
</transition>
<Footer />
<div class="mt-5 mx-auto text-center opacity-25 text-sm">
[Default Layout]
</div>
</main>
</template>

View File

@ -0,0 +1,8 @@
<template>
<div class="antialiased text-gray-900 bg-white">
<slot />
</div>
<Footer />
</template>
<script setup></script>

25
src/logics/dark.ts Normal file
View File

@ -0,0 +1,25 @@
import { watch, computed } from 'vue'
import { usePreferredDark, useToggle } from '@vueuse/core'
import { colorSchema } from './store'
const preferredDark = usePreferredDark()
export const isDark = computed({
get() {
return colorSchema.value === 'auto' ? preferredDark.value : colorSchema.value === 'dark'
},
set(v: boolean) {
if (v === preferredDark.value)
colorSchema.value = 'auto'
else
colorSchema.value = v ? 'dark' : 'light'
},
})
export const toggleDark = useToggle(isDark)
watch(
isDark,
v => typeof document !== 'undefined' && document.documentElement.classList.toggle('dark', v),
{ immediate: true },
)

1
src/logics/index.ts Normal file
View File

@ -0,0 +1 @@
export * from './dark'

4
src/logics/store.ts Normal file
View File

@ -0,0 +1,4 @@
import { Ref } from 'vue'
import { useStorage } from '@vueuse/core'
export const colorSchema = useStorage('vueuse-color-scheme', 'auto') as Ref<'auto' | 'dark' | 'light'>

35
src/main.ts Normal file
View File

@ -0,0 +1,35 @@
import { createApp } from 'vue'
import App from './App.vue'
import '@unocss/reset/tailwind.css'
import './styles/main.css'
import 'uno.css'
import { createI18n } from 'vue-i18n'
import { createHead, useHead } from '@vueuse/head'
import { messages } from './modules/i18n'
import VueHighlightJS from 'vue3-highlightjs'
import 'highlight.js/styles/solarized-light.css'
const app = createApp(App)
app.use(VueHighlightJS)
import routes from './router'
//import AppLayout from '~/layouts/AppLayout.vue'
import SimpleLayout from '~/layouts/SimpleLayout.vue'
const i18n = createI18n({
locale: navigator.language === 'es' ? 'es' : 'en',
legacy: false,
fallbackLocale: ['en'],
fallbackWarn: false,
missing: (locale, key, instance) => {
console.warn(`detect '${key}' key missing in '${locale}'`)
},
messages,
})
const head = createHead()
app.use(routes)
app.use(i18n)
app.use(head)
app.component('SimpleLayout', SimpleLayout)
app.mount('#app')

11
src/modules/README.md Normal file
View File

@ -0,0 +1,11 @@
## Modules
A custom user module system. Place a `.ts` file with the following template, it will be installed automatically.
```ts
import { UserModule } from '~/types'
export const install: UserModule = ({ app, router, isClient }) => {
// do something
}
```

25
src/modules/i18n.ts Normal file
View File

@ -0,0 +1,25 @@
import { createI18n } from 'vue-i18n'
// import { UserModule } from '~/types'
// import i18n resources
// https://vitejs.dev/guide/features.html#glob-import
export const messages = Object.fromEntries(
Object.entries(
import.meta.globEager('../../locales/*.y(a)?ml'))
.map(([key, value]) => {
const yaml = key.endsWith('.yaml')
return [key.slice(14, yaml ? -5 : -4), value.default]
}),
)
/*
export const install: UserModule = ({ app }) => {
const i18n = createI18n({
legacy: false,
locale: 'en',
messages,
})
app.use(i18n)
}
*/

9
src/modules/nprogress.ts Normal file
View File

@ -0,0 +1,9 @@
import NProgress from 'nprogress'
import { UserModule } from '~/types'
export const install: UserModule = ({ isClient, router }) => {
if (isClient) {
router.beforeEach(() => { NProgress.start() })
router.afterEach(() => { NProgress.done() })
}
}

12
src/modules/sw.ts Normal file
View File

@ -0,0 +1,12 @@
import { UserModule } from '~/types'
export const install: UserModule = ({ isClient, router }) => {
if (isClient) {
router.isReady().then(async() => {
if (isClient) {
const { registerSW } = await import('virtual:pwa-register')
registerSW({ immediate: true })
}
})
}
}

21
src/pages/About.md Normal file
View File

@ -0,0 +1,21 @@
---
title: About
---
<div class="text-center">
<!-- You can use Vue components inside markdown -->
<carbon-dicom-overlay class="text-4xl mb-6 m-auto" />
<h3>About</h3>
</div>
[Vitesse](https://github.com/antfu/vitesse) is an opinionated [Vite](https://github.com/vitejs/vite) starter template made by [@antfu](https://github.com/antfu) for mocking apps swiftly. With **file-based routing**, **components auto importing**, **markdown support**, I18n, PWA and uses **Tailwind** v2 for UI.
<pre v-highlightjs><code class="javascript">
// syntax highlighting example
function vitesse() {
const foo = 'bar'
console.log(foo)
}
</code></pre>
heck out the [GitHub repo](https://github.com/antfu/vitesse) for more details.

20
src/pages/README.md Normal file
View File

@ -0,0 +1,20 @@
## File-based Routing
Routes will be auto-generated for Vue files in this dir with the same file structure.
Check out [`vite-plugin-pages`](https://github.com/hannoeru/vite-plugin-pages) for more details.
### Path Aliasing
`~/` is aliased to `./src/` folder.
For example, instead of having
```ts
import { isDark } from '../../../../composables'
```
now, you can use
```ts
import { isDark } from '~/composables'
```

5
src/pages/[...all].vue Executable file
View File

@ -0,0 +1,5 @@
<template>
<div>
Not Found
</div>
</template>

50
src/pages/base.vue Normal file
View File

@ -0,0 +1,50 @@
<script setup lang="ts">
const name = ref('')
const router = useRouter()
const go = () => {
if (name.value)
router.push(`/hi/${encodeURIComponent(name.value)}`)
}
</script>
<template>
<div>
<div i-carbon-campsite text-4xl inline-block />
<p>
<a rel="noreferrer" href="https://github.com/antfu/vitesse-lite" target="_blank">
Vitesse Lite
</a>
</p>
<p>
<em text-sm op75>Opinionated Vite Starter Template</em>
</p>
<div py-4 />
<input
id="input"
v-model="name"
placeholder="What's your name?"
type="text"
autocomplete="false"
p="x-4 y-2"
w="250px"
text="center"
bg="transparent"
border="~ rounded gray-200 dark:gray-700"
outline="none active:none"
@keydown.enter="go"
>
<div>
<button
class="m-3 text-sm btn"
:disabled="!name"
@click="go"
>
Go
</button>
</div>
</div>
</template>

25
src/pages/hi/[name].vue Normal file
View File

@ -0,0 +1,25 @@
<script setup lang="ts">
const props = defineProps<{ name: string }>()
const router = useRouter()
</script>
<template>
<div>
<div i-carbon-pedestrian text-4xl inline-block />
<p>
Hi, {{ props.name }}
</p>
<p text-sm op50>
<em>Dynamic route!</em>
</p>
<div>
<button
class="btn m-3 text-sm mt-8"
@click="router.back()"
>
Back
</button>
</div>
</div>
</template>

160
src/router.ts Normal file
View File

@ -0,0 +1,160 @@
// eslint-disable-next-line no-unused-vars
import { createRouter, createWebHistory, RouteRecordRaw } from 'vue-router'
// import { defineAsyncComponent } from 'vue'
import generatedRoutes from 'virtual:generated-pages'
import useState from '~/hooks/useState'
import {has_config} from '~/hooks/config'
import {authExpired} from '~/hooks/utilsAuth'
// // const Login = defineAsyncComponent(() => import('./views/Login.vue'))
import Home from '~/views/Home.vue'
import Config from '~/views/Config.vue'
import Tracking from '~/views/Tracking.vue'
import Users from '~/views/Users.vue'
//import About from '~/pages/About.md'
import Register from '~/views/Register.vue'
import Login from '~/views/Login.vue'
import Logout from '~/views/Logout.vue'
// https://medium.com/swlh/adding-meta-fields-and-transitions-to-vue-router-routes-f5cb78a7ed97
const check_config = async(to: any, _from: any, next: any) => {
if (await has_config()) {
next()
} else {
next('/noconfig')
}
}
const check_auth = async(to: any, _from: any, next: any) => {
if (await has_config()) {
if (authExpired()) {
useState().sourceRoute.value = to.path
next('/login')
} else {
useState().sourceRoute.value = ""
next()
}
} else {
next('/login')
}
}
const routes: RouteRecordRaw[] = [
{
path: '/config',
name: 'Config',
component: Config,
meta: {
layout: 'AppLayout',
ctx: 'config',
withauth: true,
},
beforeEnter: check_auth,
},
{
path: '/tracking',
name: 'Tracking',
component: Tracking,
meta: {
requireAuth: true,
layout: 'AppLayout',
withauth: true,
},
beforeEnter: check_auth,
},
{
path: '/users',
name: 'Users',
component: Users,
meta: {
requireAuth: true,
layout: 'AppLayout',
withauth: true,
},
beforeEnter: check_auth,
},
{
path: '/',
name: 'Home',
component: Home,
meta: {
ctx: 'home',
withauth: true,
},
beforeEnter: check_auth,
},
{
path: '/login',
name: 'Login',
component: Login,
meta: {
requireAuth: false,
//layout: 'empty',
layout: 'AppLayout',
use_logo: true,
ctx: 'login',
withauth: false,
},
beforeEnter: check_config,
},
{
path: '/register/:id',
name: 'Register',
component: Register,
meta: {
requireAuth: false,
//layout: 'empty',
layout: 'AppLayout',
use_logo: true,
ctx: 'register',
withauth: false,
},
beforeEnter: check_config,
},
{
path: '/recoveryuser/:id',
name: 'Recovery',
component: Register,
meta: {
requireAuth: false,
//layout: 'empty',
layout: 'AppLayout',
use_logo: true,
ctx: 'register',
withauth: false,
},
beforeEnter: check_config,
},
{
path: '/logout',
name: 'Logout',
component: Logout,
meta: {
layout: 'AppLayout',
use_logo: true,
requireAuth: false,
ctx: 'logout',
withauth: false,
}
},
{
path: '/noconfig',
name: 'NoConfig',
component: () => import('./views/404.vue'),
meta: {
}
},
{
path: '/:catchAll(.*)',
name: '404',
meta: { layout: 'Page404' },
component: () => import('./views/404.vue'),
},
...generatedRoutes,
]
const router: any = createRouter({
history: createWebHistory(),
routes,
})
export default router

62
src/shims.d.ts vendored Normal file
View File

@ -0,0 +1,62 @@
/* eslint-disable import/no-duplicates */
declare const ASSETS_PATH: string
declare const DATA_PATH: string
declare interface Window {
// extend the window
ROOT_LOCATION: string
CONFIG_LOCATION: string
VARSCONFIG_LOCATION: string
SERVCONFIG_LOCATION: string
}
// with vite-plugin-md, markdowns can be treat as Vue components
declare module '*.md' {
import { ComponentOptions } from 'vue'
const component: ComponentOptions
export default component
}
/*
declare module '*.vue' {
import Vue from 'vue'
export default Vue
}
*/
declare const _APP_VERSION: string
/*
* Keep states in the global scope to be reusable across Vue instances.
*
* @see {@link /createGlobalState}
* @param stateFactory A factory function to create the state
*/
/*
declare function createGlobalState<T extends object>(
stateFactory: () => T
): () => T
*/
interface EventTarget {
value: EventTarget|null
name: string| null
/**
* Appends an event listener for events whose type attribute value is type. The callback argument sets the callback that will be invoked when the event is dispatched.
*
* The options argument sets listener-specific options. For compatibility this can be a boolean, in which case the method behaves exactly as if the value was specified as options's capture.
*
* When set to true, options's capture prevents callback from being invoked when the event's eventPhase attribute value is BUBBLING_PHASE. When false (or not present), callback will not be invoked when event's eventPhase attribute value is CAPTURING_PHASE. Either way, callback will be invoked if event's eventPhase attribute value is AT_TARGET.
*
* When set to true, options's passive indicates that the callback will not cancel the event by invoking preventDefault(). This is used to enable performance optimizations described in § 2.8 Observing event listeners.
*
* When set to true, options's once indicates that the callback will only be invoked once after which the event listener will be removed.
*
* The event listener is appended to target's event listener list and is not appended if it has the same type, callback, and capture.
*/
addEventListener(type: string, listener: EventListenerOrEventListenerObject | null, options?: boolean | AddEventListenerOptions): void
/**
* Dispatches a synthetic event event to target and returns true if either event's cancelable attribute value is false or its preventDefault() method was not invoked, and false otherwise.
*/
dispatchEvent(event: Event): boolean
/**
* Removes the event listener in target's event listener list with the same type, callback, and options.
*/
removeEventListener(type: string, callback: EventListenerOrEventListenerObject | null, options?: EventListenerOptions | boolean): void
}

193
src/styles/main.css Executable file
View File

@ -0,0 +1,193 @@
html,
body,
#app {
height: 100%;
margin: 0;
padding: 0;
}
html.dark {
background: #27272a; /*bg-gray-800*/
color: #f0f0f0;
currentColor: #f0f0f0;
}
.right-item ul li {
margin-left: 1em;
}
.features li b {
color: #6366F1; /* indigo-600 */
font-weight: 300;
}
.dark .features li b {
color: #A5B4FC; /* indigo-400 */
}
li a, link,
.task a,h3 a,.link a, .link {
color: #4F46E5; /*indigo-600*/
text-decoration: underline;
}
.dark li a,.dark link,
.dark .task a,.dark h3 a,.dark .link a,.dark .link {
color: #A5B4FC; /* indigo-400 */
}
.project-purpose, .project-name {
width: 100%;
padding: 4px 11px;
border-radius: 20px;
margin-bottom: 8px;
border: 1px solid #A5B4FC; /* indigo-400 */
background-color: #F3F4F6; /* gray-100 */
color: #6366F1; /* indigo-600 */
font-size: 1.02em;
font-weight: 600;
}
.dark .project-purpose, .dark .project-name {
background-color: #374151; /* gray-500 */
border: 1px solid #9CA3AF; /* gray-400 */
color: #A5B4FC; /* indigo-400 */
}
.dark svg {
fill: #f0f0f0;
}
.openbox {
border-radius: 10px;
border: 1px solid #374151; /* gray-500 */
}
.dark .openbox {
border: 1px solid #F3F4F66; /* gray-100 */
}
.msg-box {
border-radius: 40px !important;
border: 1px solid #374151; /* indigo-500 */
}
.dark .msg-box {
border: 1px solid #F3F4F6; /* gray-100 */
}
.msg-ok {
background: linear-gradient(135deg, #83D475, #2EB62C) !important;
}
.msg-error {
background: linear-gradient(135deg, #D90708, #9E1444) !important;
}
.msg-warn {
background: linear-gradient(135deg, #FDA766, #FD7F2C) !important;
}
.msg-show {
background: linear-gradient(135deg, #4949FF, #0000FF) !important;
}
.msg-info {
background: linear-gradient(135deg, #9E9E9E, #696969) !important;
}
.hljs {
background: none !important;
}
.markdown-body {
margin: 2em;
}
.markdown-body p {
margin-top: 1em;;
font-size: 1em;
}
.markdown-body pre {
padding: 2em;
background: rgba(243,244,246);
}
.dark .markdown-body pre {
background: rgba(75, 85, 99,var(--tw-bg-opacity))
}
.markdown-body a {
color: rgba(37, 99, 235);
}
.no-list { list-style-type: none; margin-top: 1em;}
/*
.btn {
@apply px-4 py-1 rounded inline-block
bg-teal-600 text-white cursor-pointer
hover:bg-teal-700
disabled:cursor-default disabled:bg-gray-600 disabled:opacity-50;
}
.icon-btn {
@apply inline-block cursor-pointer select-none
opacity-75 transition duration-200 ease-in-out
hover:opacity-100 hover:text-teal-600;
font-size: 0.9em;
}
/*
.btn {
@apply px-4 py-1 rounded inline-block
bg-teal-600 text-white cursor-pointer
hover:bg-teal-700
disabled:cursor-default disabled:bg-gray-600 disabled:opacity-50;
}
.icon-btn {
@apply inline-block cursor-pointer select-none
opacity-75 transition duration-200 ease-in-out
hover:opacity-100 hover:text-teal-600;
font-size: 0.9em;
}
*/
progress {
margin-top: 0.5rem;
width: 100%;
height: 10px;
margin-bottom: 0.4rem;
}
progress::-webkit-progress-bar {
border-radius: 10px;
background-color: #d1d5db;
box-shadow: 0 2px 6px #555;
}
progress::-webkit-progress-value {
border-radius: 10px 0 0 10px;
background-image: linear-gradient(36deg, #d1fae5, #818cf8);
}
.prose>:first-child {
margin-top: 0 !important;
}
.prose h2 {
font-size: 1.5em;
font-weight: 500;
line-height: 1.3333333;
/* margin-bottom: 1em; */
/* margin-top: 2em; */
text-transform: uppercase;
}
#cv-wrapper {
box-shadow: rgba(0, 0, 0, 0.16) 0px 10px 36px 0px, rgba(0, 0, 0, 0.06) 0px 0px 0px 1px;
/* box-shadow: rgba(50, 50, 93, 0.25) 0px 50px 100px -20px, rgba(0, 0, 0, 0.3) 0px 30px 60px -30px, rgba(10, 37, 64, 0.35) 0px -2px 6px 0px inset; */
/* https://getcssscan.com/css-box-shadow-examples */
}
div strong, p strong, .itm-title {
color: #6366F1; /* indigo-600 */
font-weight: 600;
}
.dark div strong, .dark p strong, .dark .itm-title {
color: #A5B4FC; /* indigo-400 */
font-weight: 600;
}
img.project {
width: 10em;
max-height: 5em;
}
@media print {
* {
line-height: 1em !important;
}
.main-container { max-width: fit-content !important;}
.noprint {
visibility: hidden;
display: none;
margin-top: 1em;
}
#cv-wrapper { margin-top: 0 !important;}
.page-break { display: block; break-before: auto; }
}
/* @media all {
.page-break { display: none; }
} */

86
src/typs/clouds/index.ts Normal file
View File

@ -0,0 +1,86 @@
export const enum LanguageType {
en = 'en',
es = 'es',
None = 'None',
}
export interface StatusItemType {
title: string
content: string
lang: LanguageType
datetime: string
isOpen: boolean
}
export interface StatusItemDataType {
[key: string]: any
title: string
content: string
lang: LanguageType
datetime: string
}
export const enum ReqType {
tcp = 'tcp',
https = 'https',
NotSet = 'NotSet',
}
export const enum CriticalType {
yes = 'yes',
cloud = 'cloud',
group = 'group',
ifresized = 'ifresized',
no = 'no',
}
export interface SrvcType {
name: string
path: string
req: ReqType
target: string
liveness: string
critical: CriticalType
}
export interface SrvcInfoType {
[key: string]: any
name: string
info: string
srvc: SrvcType
}
export interface CloudGroupItemType {
[key: string]: any
hostname: string
tsksrvcs: SrvcType[]
appsrvcs: SrvcType[]
}
export interface CloudGroupServcType {
[key: string]: any
hostname: string
name: string
tsksrvcs: SrvcInfoElemType[]
appsrvcs: SrvcInfoElemType[]
}
export interface CloudDataCheck {
[key: string]: any
name: string
apps: Map<string, Map<string, CloudGroupServcType>>
cloud: Map<string, Map<string, CloudGroupServcType>>
infos: StatusItemType[]
}
export interface CloudOptionType {
name: string
option: number
}
export interface CloudGroupDataType {
[key: string]: CloudGroupItemType[] | any
}
export interface ResCloudDataCheck {
[key: string]: any
name: string
cloud: CloudGroupDataType
apps: CloudGroupDataType
// cloud: CloudGroupSrvcType
// statusentries: StatusItemDataType[]
}
export interface ResCloudDataCheckDefs {
[key: string]: any
check: ResCloudDataCheck[]
defs: any
}

59
src/typs/cmpnts/index.ts Normal file
View File

@ -0,0 +1,59 @@
export const enum Profile {
Premium,
Basic,
Pro,
NotSet,
}
export enum Auth {
Allow,
Denied,
NotSet,
}
export enum Permission {
Read,
Write,
ReadWrite,
NotSet,
}
export interface DataContext {
profile: Profile
auth: Auth
perms: Permission
}
export interface AccordionItemType {
title?: string
status: boolean
h: number
}
export interface SideMenuItemType {
title: string
ctx?: string
mode?: string
name?: string
icon_on?: string
img?: string
name_to?: string
show_to: boolean
path?: string
click?: string
type: NavItemType
pfx?: string
href?: string
label?: string
vif?: string
active?: boolean
cllbck?: Function | null
}
export interface UiPanelsType {
[key: string]: any
id: string
style: string
}

196
src/typs/config.ts Normal file
View File

@ -0,0 +1,196 @@
import { VarDefType,VarBoolDefType,VarNumDefType } from '~/typs/inputs'
export interface ReqRouteType {
path: string
param: string
}
export interface WebRouteType {
[key: string]: ReqRouteType
}
export interface TemplateFileType {
path: string
rooute: string
}
export interface TemplateType {
[key: string]: TemplateFileType
}
export interface TplMailType{
path: string
type: string
}
export interface TplsMailType {
[key: string]: TplMailType
}
export interface VarsConfigType {
[key: string]: any
sec_logs: {
logOut: VarDefType
requestOut: VarDefType
trackOut: VarDefType
debugLevel: VarDefType
}
sec_webserver: {
host: VarDefType
port: VarNumDefType
protocol: VarDefType
keyPem: VarDefType
certPem: VarDefType
allowOrigins: {
typ: string
dflt: string[]
}
}
sec_jwt: {
useJWT: VarBoolDefType
jwtRealm: VarDefType
jwtKey: VarDefType
jwtTimeout: VarNumDefType
jwtMaxRefresh: VarNumDefType
}
sec_auth: {
authSep: VarDefType
useAuthz: VarBoolDefType
authzModel: VarDefType
authzPolicy: VarDefType
adminRole: VarDefType
}
sec_perms: {
pubUser: VarDefType
usersPath: VarDefType
usersModelsPath: VarDefType
identityKey: VarDefType
}
sec_routes: {
rootAuthGroup: VarDefType
routes: {
typ: string
dflt: WebRouteType[]
}
}
sec_templates: {
templatesRoot: VarDefType
templatesExt: VarDefType
templatesIncludes: VarDefType
templatesLayouts: VarDefType
templatesFiles: {
typ: string
dflt: TemplateType
}
}
sec_assets: {
assetsPath: VarDefType
assetsURL: VarDefType
}
sec_models: {
useDist: VarBoolDefType
genDist: VarBoolDefType
genExcludeList: {
typ: string
dflt: string[]
}
dataDistPath: VarDefType
dataPath: VarDefType
dataModelsRoot: VarDefType
dataCorePath: VarDefType
dataDflt: VarDefType
dataItems: {
typ: string
dflt: string[]
}
}
sec_langs: {
langs: {
typ: string
dflt: string[]
}
mainLang: VarDefType
}
sec_git: {
useRepo: VarBoolDefType
useRepoOnReq: VarBoolDefType
quietGit: VarBoolDefType
backgGit: VarBoolDefType
repoPath: VarDefType
repoName:VarDefType
repoCommit:VarDefType
}
sec_others: {
openBrowser: VarBoolDefType
}
}
export interface ConfUrlsType {
[key: string]: string
}
export interface ServConfigType {
[key: string]: any
logOut: string
requestOut: string
requestStore: string
trackOut: string
trackStore: string
debugLevel: number
host: string
port: number
protocol: string
keyPem: string
certPem: string
allowOrigins: string[]
useJWT: boolean
jwtRealm: string
jwtKey: string
jwtTimeout: number
jwtMaxRefresh: number
signingAlgorithm: string
jwtKeyPem: string
jwtCertPem: string
authSep: string
useAuthz: boolean
authzModel: string
authzPolicy: string
adminRole: string
pubUser: string
usersPath: string
usersModelsPath: string
identityKey: string
usersStore: string
rootAuthGroup: string
routes: WebRouteType
templatesRoot: string
templatesExt: string
templatesIncludes: string
templatesLayouts: string
templatesFiles: TemplateType
mailHost: string
mailPort: number
mailFrom: string
mailUser: string
mailPswd: string
tplsMailPath: string
tplsMail: TplsMailType
assetsPath: string
assetsURL: string
useDist: boolean
genDist: boolean
genExcludeList: string[]
dataDistPath: string
dataPath: string
dataModelsRoot: string
dataCorePath: string
dataDflt: string
dataItems: string[]
dataStore: string
langs: string[]
mainLang: string
useRepo: boolean
useRepoOnReq: boolean
quietGit: boolean
backgGit: boolean
repoPath: boolean
repoName: string
repoCommit: string
openBrowser: boolean
redisHost: string
redisPort: number
redisDB: string
redisPswd: string
}

320
src/typs/cv.ts Normal file
View File

@ -0,0 +1,320 @@
export interface AuthInfoType {
editable: boolean
viewchange: boolean
show: boolean
}
export interface SkillsType {
id: string
auth: AuthInfoType
title: string
max: number
value: number
}
export interface CertificationType {
id: string
auth: AuthInfoType
title: string
author: string
date: string
link: string
href: string
certid: string
}
export interface SitesType {
id: string
auth: AuthInfoType
title: string
sub: string
link: string
type: string
alt: string
img: string
}
export interface LangsType {
id: string
title: string
mode: string
}
export interface WorkExperienceType {
[key: string]: any
auth: AuthInfoType
date: string
where: string
wheredef: string
location: string
position: string
description: string
tools: string[]
tasks: string[]
}
export interface TalksType {
[key: string]: any
auth: AuthInfoType
date: string
title: string
org: string
location: string
description: string[]
}
export interface TeachingType {
[key: string]: any
auth: AuthInfoType
date: string
title: string
org: string
location: string
description: string[]
}
export interface EducationType {
[key: string]: any
auth: AuthInfoType
date: string
title: string
org: string
location: string
cert: string
description: string[]
tools: string[]
}
export interface ProjectType {
[key: string]: any
auth: AuthInfoType
date: string
name: string
title: string
img: string
site: string
code: string
purpose: string
for: string
position: string
license: string
demo: string
capture: string
description: string
features: string[]
builtwith: string[]
}
export interface ProfileType {
[key: string]: unknown;
auth: AuthInfoType
desc: string
}
export interface OtherType {
[key: string]: unknown;
auth: AuthInfoType
desc: string
}
export interface MissionHowType {
[key: string]: unknown;
auth: AuthInfoType
desc: string
}
export interface DataCoreType {
[key: string]: any
name: string
fullname: string
title1: string
title2: string
imgalt: string
imgsrc: string
email: string
phone: string
address: string
postalcode: string
state: string
city: string
country: string
birthdate: string
status: string
mission: string
mission_how: MissionHowType[]
profile: ProfileType[]
certifications: CertificationType[]
skills: SkillsType[]
infra: SkillsType[]
sites: SitesType[]
langs: LangsType[]
}
export interface DataLangType {
[key: string]: any
imgalt: string
title1: string
title2: string
country: string
birthdate: string
status: string
mission: string
mission_how: MissionHowType[]
profile: ProfileType[]
certifications: CertificationType[]
work_experiences: WorkExperienceType[]
projects: ProjectType[]
education: EducationType[]
talks: TalksType[]
teaching: TeachingType[]
others: OtherType[]
}
export interface DataLangsType {
[key: string]: DataLangType
}
export interface ShowTalksType {
[key: string]: unknown;
auth: AuthInfoType
date: boolean
title: boolean
org: boolean
location: boolean
description: boolean
}
export interface ShowTeachingType {
[key: string]: unknown;
auth: AuthInfoType
date: boolean
title: boolean
org: boolean
location: boolean
description: boolean
}
export interface ShowWorkExperienceType {
[key: string]: unknown;
auth: AuthInfoType
date: boolean
where: boolean
wheredef: boolean
location: boolean
position: boolean
description: boolean
tools: boolean
tasks: boolean
}
export interface ShowEducationType {
[key: string]: unknown;
auth: AuthInfoType
date: boolean
title: boolean
org: boolean
location: boolean
cert: boolean
description: boolean
tools: boolean
}
export interface ShowProjectType {
[key: string]: unknown;
auth: AuthInfoType
name: boolean
img: boolean
title: boolean
purpose: boolean
site: boolean
code: boolean
date: boolean
for: boolean
position: boolean
license: boolean
demo: boolean
capture: boolean
description: boolean
features: boolean
builtwith: boolean
}
export interface ShowInfoType {
[key: string]: any
id: string
ky: string
auth: AuthInfoType
write: boolean
change: boolean
admin: boolean
// save: boolean
fullname: boolean
personal: boolean
title: boolean
image: boolean
mission: boolean
mission_how: boolean
phone: boolean
address: boolean
status: boolean
birthdate: boolean
sites: boolean
skills: boolean
skills_itms: boolean
infra: boolean
certs: boolean
langs: boolean
profile: boolean
work_experience_itms: boolean
work_experience: ShowWorkExperienceType
project_itms: boolean
projects: ShowProjectType
education_itms: boolean
education: ShowEducationType
talks_itms: boolean
talks: ShowTalksType
teaching_itms: boolean
teaching: ShowTeachingType
others_itms: boolean
others: boolean
}
export interface HtmlAttrsType {
[key: string]: any
bold: string
list: string
text: string
link: string
}
export interface OptionsSelectorType {
title: string
val: string
}
export interface InputBtnsType {
id: string
title: string
typ: string
show: boolean
}
export enum MessageBoxType {
Save = 'save',
Select = 'select',
Input = 'input',
OneInput = 'oneinput',
NotSet = '',
}
export enum NavPosition {
header = 'header',
footer = 'footer',
}
export interface ModelType {
[key: string]: any
id: string
title: string
path: string
}
export interface ModelSelType {
[key: string]: ModelType
}
export interface CVDataType {
models: ModelType[]
showinfo: ShowInfoType[]
core: DataCoreType[]
work_experience: WorkExperienceType[]
projects: ProjectType[]
education: EducationType[]
talks: TalksType[]
teaching: TeachingType[]
others: OtherType[]
}
export interface CVLangDataType {
[key: string]: CVDataType
}
export interface CVModelDataType {
[key: string]: CVLangDataType
}
export interface CVPostDataType {
u: string
data: CVModelDataType
}

70
src/typs/index.ts Normal file
View File

@ -0,0 +1,70 @@
export enum LocalesLabelModes {
auto = 'auto',
value = 'val',
translation = 'trans',
}
export enum MessageType {
Show,
Success,
Error,
Warning,
Info,
}
export interface MenuItemType {
id: string
title: string
active: boolean
link: string
}
export interface DataSections {
[key: string]: string
}
export enum NavItemType {
router_link = 'router_link',
app_link = 'app_link',
a_blank = 'a_blank',
a_link = 'a_link',
cloud_link = 'cloud_link',
module_label = 'module_label',
separator = 'separator',
}
export interface TrackActionShowType {
[key: string]: any
when: boolean
where: boolean
what: boolean
context: boolean
data: boolean
auth: boolean
payload: boolean
}
export interface AuthPayloadType {
[key: string]: any
exp: number|string
id: string
uuid: string
orig_iat: number|string
data: string
}
export interface TrackActionType {
[key: string]: any
when: string|number
where: string
what: string
context: string
data: string
auth: string
payload?: AuthPayloadType
idx: number
}
export interface InvitationDataType {
[key: string]: any
email: string
role: string
data: string
}

53
src/typs/inputs.ts Normal file
View File

@ -0,0 +1,53 @@
export enum InputValueOption {
string = 'String',
password = 'Password',
number = 'Number',
bool = 'Bool',
arrayString = 'Array:String',
object = 'Object',
content = 'Content',
// templates = 'Templates',
// webroutes = 'WebRoutes',
// mailtemplates = 'MailTemplates',
// user = 'User',
}
export enum InputValueModeOption {
text = 'Text',
html = 'HTML',
yaml = 'YAML',
json = 'JSON',
toml = 'TOML',
none = '',
}
export enum InputValidateOption {
email = 'Email',
none = '',
}
export interface SecStateType {
view: boolean
enabled: boolean
help: boolean
}
export interface SecViewType {
[key: string]: SecStateType
}
export interface VarsDataType {
[key: string]: VarDefType
}
export interface VarDefType {
[key: string]: any
typ: InputValueOption
valid?: InputValidateOption
dflt: string
labelitms?: string
vars?: VarsDataType
}
export interface VarNumDefType {
typ: string
dflt: number
}
export interface VarBoolDefType {
typ: string
dflt: boolean
}

58
src/typs/users.ts Normal file
View File

@ -0,0 +1,58 @@
import { VarDefType,VarBoolDefType,VarNumDefType } from '~/typs/inputs'
export interface UserAccountType {
[key: string]: any
password: string
email: string
description: string
data: string
web: boolean
dflt: true
}
export interface UsersAccountsType {
accounts: UserAccountType[]
}
export interface ModelUserType {
[key: string]: any
model: string
user: string
data: string
active: boolean
}
export interface ModelsUsersType {
[key: string]: any
}
export interface UsersInvitationsType {
[key: string]: any
email: string
createdby: string
expire: string
howmany: number
role: string
description: string
data: string
active: boolean
}
export interface UsersDataType {
[key: string]: string
usersData: string
modelsData: string
authzModel: string
authzPolicy: string
}
export interface UsersDataViewType {
[key: string]: boolean
usersData: boolean
modelsData: boolean
authzModel: boolean
authzPolicy: boolean
}
export interface VarsUserType {
[key: string]: any
sec_data: {
logOut: VarDefType
password: VarDefType
email: VarDefType
debugLevel: VarDefType
}
}

28
src/views/404.vue Normal file
View File

@ -0,0 +1,28 @@
<script setup lang="ts">
import { useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n'
const router = useRouter()
const { t } = useI18n()
</script>
<template>
<main class="px-4 py-10 text-center text-teal-700 dark:text-gray-200">
<div>
<p class="text-4xl">
<carbon-warning class="inline-block" />
</p>
</div>
<transition name="fade">
<router-view />
</transition>
<div>
<button
class="btn m-3 text-sm mt-8"
@click="router.back()"
>
{{ t('button.back') }}
</button>
</div>
</main>
</template>

494
src/views/Config.vue Normal file
View File

@ -0,0 +1,494 @@
<template>
<div
v-if="openMessageBox"
class="fixed inset-0 bg-gray-600 overflow-y-auto h-full w-full z-80 dark:bg-gray-900 opacity-97"
>
<message-box-view
:messageType="message_type"
:openMessageBox="openMessageBox"
:show_input="show_input"
:input_placeholder="input_placeholder"
:input_btns="input_btns"
:select_ops="select_ops"
:data_url_encoded="data_url_encoded"
:inpType="oneInputType"
@onInput="onInput"
@onMessageBox="onMessageBox"
@onLoadModel="onLoadModel"
/>
</div>
<div v-if="show_content && servConfig" class="font-sans antialiased">
<div class="container mx-auto max-w-screen-2xl main-container">
<nav-menu
class="bg-light-300 dark:bg-gray-900"
:position="NavPosition.header"
:openMessageBox="openMessageBox"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
:prefix="prefix"
@onNavMenu="onNavMenu"
/>
<main
:id="`${prefix}-wrapper`"
class="rounded flex flex-col sm:flex-row-reverse shadow-2xl"
:class="{ 'mt-11': fixMenu }"
>
<div
v-if="show_infopanel"
id="sidebar"
class="rounded-r w-full lg:w-80 sm:max-w-sm p-8 border-l-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-500 dark:via-indigo-400 dark:to-indigo-800"
>
</div>
<div class="content w-full p-5">
<div class="prose flex-1 p-4 border-1 rounded-md border-gray-300 dark:border-gray-600 lg:w-4/12 sm:w-full sm:max-w-sm shadow-xl bg-light-300 dark:bg-gray-900">
<div v-for="(sec,idxsec) in data_sections" :key="idxsec"
:id="`${prefix}-${sec}`"
class="prose mb-2"
:class="{ 'page-break': idxsec > 0 }"
>
<h2 v-if="show_sections[sec]" class="section-headline my-4 px-5 pb-2"
:class="{'bg-gray-200 dark:bg-gray-700 border-1 rounded-tl-xl rounded-r-3xl border-indigo-500': show_sections[sec].enabled && show_sections[sec].view}"
>
<span v-if="show_sections[sec].enabled && show_sections[sec].view" class="text-base">{{t(`vars.${sec}`,sec)}}</span>
<button
class="no-print text-sm float-right icon-btn mt-3 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onSection(sec)"
>
<div v-if="show_sections[sec].view && show_sections[sec].enabled" i-carbon-view-off />
<div v-else class="noprint flex -mt-4">
<div class="mr-2 text-xs" :class="{'line-through text-indigo-600 dark:text-indigo-300': !show_sections[sec].enabled}">{{t(`vars.${sec}`,sec)}}</div>
<div i-carbon-view />
</div>
</button>
<button
v-if="show_sections[sec].view && show_sections[sec].enabled"
class="no-print text-sm float-right icon-btn mt-3 mr-1 !outline-none text-gray-500 dark:text-gray-400"
@click="onHelp(sec)"
>
<div i-carbon-help/>
</button>
<button
v-if="show_sections[sec].view && show_sections[sec].enabled"
class="no-print text-sm float-right icon-btn mt-3 mr-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onHome()"
>
<div i-carbon-home />
</button>
</h2>
<div v-if="show_sections[sec] && show_sections[sec].enabled && show_sections[sec].view && show_sections[sec].help"
class="helpbox" v-html="t(`info.${sec.replace('sec_','')}`,'')"
/>
<div v-if="show_sections[sec] && show_sections[sec].enabled && show_sections[sec].view"
v-for="(itm,idxitm) in Object.keys(varsConfig[sec])"
:key="idxitm"
>
<div
v-if="servConfig[itm]"
class="flex flex-col items-start py-2"
:class="{'border-b border-indigo-500': edititems && show_items[`${sec}.${itm}`].enabled && simpleInputype(varsConfig[sec][itm].typ)}"
>
<h4 class="sectionitem w-full" :class="{'-mt-2 -mb-7': varsConfig[sec][itm].typ === 'Bool'}">
<span v-if="varsConfig[sec][itm].typ !== 'Bool'"
class="text-base"
:class="{'text-gray-300 dark:text-gray-600': !show_items[`${sec}.${itm}`].enabled}"
>
{{t(`${sec}.${itm}`,itm)}}
</span>
<button
v-if="!auth_info.viewchange"
class="no-print text-sm float-right icon-btn mt-1 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onHelp(sec,itm)"
>
<div class="noprint flex">
<div i-carbon-help />
</div>
</button>
</h4>
<div v-if="show_items[`${sec}.${itm}`].help" class="helpbox" v-html="t(`info${sec}.${itm}`,'')"/>
<DataInput
v-if="varsConfig[sec][itm].typ === InputValueOption.object"
:dataVars="varsConfig[sec][itm].vars"
:section="sec"
:typ="varsConfig[sec][itm].typ"
:dataValue="servConfig[itm]"
:varname="itm"
:title="`${sec}.${itm}`"
:labelitms="varsConfig[sec][itm].labelitms? varsConfig[sec][itm].labelitms : ''"
:edititem="edititems"
:showitem="show_items[`${sec}.${itm}`].enabled"
:validate="varsConfig[sec][itm].valid ? varsConfig[sec][itm].valid : ''"
:messages="true"
@input-change="onInputChange"
/>
<InputValue
v-if="varsConfig[sec][itm].typ !== InputValueOption.object"
:section="sec"
:typ="varsConfig[sec][itm].typ"
:vardef="varsConfig[sec][itm]"
:dataValue="servConfig[itm]"
:varname="itm"
:title="`${sec}.${itm}`"
:labelitms="varsConfig[sec][itm].labelitms? varsConfig[sec][itm].labelitms : ''"
:edititem="edititems"
:showitem="show_items[`${sec}.${itm}`].enabled"
:validate="varsConfig[sec][itm].valid ? varsConfig[sec][itm].valid : ''"
:messages="true"
@input-change="onInputChange"
/>
</div>
</div>
</div>
<div class="py-4 flex items-center gap-5 border-t-1 border-gray-500">
<button
v-if="edititems"
class="btn-msg flex-grow"
@click="onSaveBtn"
>
<span>{{ t('save', 'Save') }}</span>
</button>
<button v-if="!edititems" class="btn-msg flex-grow" @click="onEditBtn">{{ t('edit', 'Edit') }}</button>
<button v-else class="btn-msg flex-grow" @click="onCloseBtn">{{ t('close', 'Close') }}</button>
</div>
</div>
</div>
<hr class="hr-sep" />
</main>
<div
v-if="show_content"
class="mr-auto w-full lg:w-1/2 text-center text-sm mx-2 text-gray-600 border-gray-300 border-1 border-b-0 rounded-t-xl"
>
<nav-menu
class="shadow-xl bg-light-300 dark:bg-gray-900"
:position="NavPosition.footer"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
prefix="sec"
:openMessageBox="openMessageBox"
@onNavMenu="onNavMenu"
/>
</div>
</div>
</div>
<div v-else class="text-center m-auto">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
<h2 class="mt-8 text-2xl text-gray-700 dark:text-gray-400">{{ t('message.loading', 'Loading') }}...</h2>
<h4>{{useState().connection.value.state}}</h4>
</div>
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
import MessageBoxView from '~/components/MessageBoxView.vue'
import NavMenu from '@/NavMenu.vue'
import DataInput from '@/forms/DataInput.vue'
import InputValue from '@/InputValue.vue'
import { isDark } from '~/composables'
import useState from '~/hooks/useState'
import { parseJwt,checkUserAuth } from '~/hooks/utilsAuth'
import { load_data_model,load_currentRoute } from '~/hooks/loads'
import { track_action } from '~/hooks/tracking'
import { load_vars_config,get_serv_config} from '~/hooks/config'
import { MessageType} from '~/typs'
import { MessageBoxType, NavPosition, InputBtnsType, ShowInfoType, ModelType} from '~/typs/cv'
import { InputValueOption,SecViewType} from '~/typs/inputs'
import { ServConfigType} from '~/typs/config'
import { show_message,send_data} from '~/hooks/utils'
const prefix = 'cfg'
const varsConfig = useState().varsConfig
const servConfig = ref({} as ServConfigType)
const show_sections = ref({} as SecViewType)
const show_items = ref({} as SecViewType)
const edititems = ref(false)
const assetsPath = useState().ASSETS_PATH
const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
const message_type = ref(MessageBoxType.NotSet)
const fixMenu = ref(false)
const show_content = ref(true)
const show_infopanel = ref(false)
const openMessageBox = ref(false)
const needSave = ref(false)
const show_input = ref(false)
const data_url_encoded = ref('')
const input_btns = ref([] as InputBtnsType[])
const input_placeholder = ref('')
const oneInputType = ref('')
const onSection = (sec: string) => {
if (!show_sections.value[sec].enabled) {
show_sections.value[sec].view = true
show_sections.value[sec].enabled = true
return
} else {
show_sections.value[sec].view = !show_sections.value[sec].view
}
}
const onHelp = (sec: string, itm?: string) => {
if (itm)
show_items.value[`${sec}.${itm}`].help = !show_items.value[`${sec}.${itm}`].help
else
show_sections.value[sec].help = !show_sections.value[sec].help
}
const onInputChange = (data: {sec: string, key: string, idx:number, val: any}) => {
if (data.key && typeof servConfig.value[data.key] !== 'undefined') {
servConfig.value[data.key] = data.val
if (data.sec)
check_item_section(data.sec,data.key)
}
}
const simpleInputype = (typ: string): boolean => {
switch(typ) {
case 'String':
case 'Number':
case 'Array:String':
return true
default:
return false
}
}
const check_item_section = (sec: string, itm: string) => {
// show_items.value[`${sec}.${itm}`].enabled = servConfig.value[itm]
switch(itm) {
case 'useJWT':
case 'useRepo':
show_sections.value[sec].enabled = servConfig.value[itm]
break
case 'useAuthz':
show_items.value[`${sec}.authzModel`].enabled = servConfig.value[itm]
show_items.value[`${sec}.authzPolicy`].enabled = servConfig.value[itm]
break
}
}
const onInput = (btn: InputBtnsType) => {
switch (btn.id) {
case 'ok':
break
case 'cancel':
break
}
show_input.value = false
}
const select_ops = ref({} as ModelType)
const show_info = ref({} as ShowInfoType)
const auth_info = useState().authinfo
const useEdit = ref(true)
const data_sections = useState().dataSections
const refresh = () => {
show_content.value = false
setTimeout(() => show_content.value = true, 1)
}
const onSaveBtn = () => {
// edititems.value = false
// needSave.value = false
onNavMenu({ src: 'save', target: '' })
}
const onCloseBtn = () => {
edititems.value = false
needSave.value = false
}
const onEditBtn = () => {
edititems.value = true
needSave.value = true
auth_info.value.editable = true
useEdit.value = true
}
const onNavMenu = (item: { src: string, target: string }) => {
switch (item.src) {
case 'editable':
if (edititems.value) {
onCloseBtn()
} else {
onEditBtn()
}
break
case 'locale':
track_action(null, { ref: 'locale',text: i18n.locale.value}, 'locale')
refresh()
break
case 'infopanel':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
show_infopanel.value = !show_infopanel.value
break
case 'fixmenu':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
fixMenu.value = !fixMenu.value
break
case 'endinput':
show_input.value = false
break
case 'select':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
message_type.value = MessageBoxType.Select
openMessageBox.value = true
break
case 'save':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
input_placeholder.value = `${t('name', 'Name')} o 'config'`
show_input.value = true
message_type.value = MessageBoxType.Save
data_url_encoded.value = ''
openMessageBox.value = true
break
case 'editable':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.editable = !useState().authinfo.value.editable
const state = useState().authinfo.value.editable ? 'on' : 'off'
const msgTyp = state === 'on' ? MessageType.Warning : MessageType.Info
show_message(msgTyp, `${t('saveload.editMode','Edit Mode')} ${t('cv.'+state,'')}`)
break
case 'viewchange':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.viewchange = !useState().authinfo.value.viewchange
break
case 'goto':
if (item.target) {
const dom_id = document.getElementById(item.target)
if (dom_id) {
track_action(null, { ref: 'gotp',text: item.target}, 'goto')
dom_id.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
setTimeout(() => window.scrollBy(0, -40), 4000)
}
}
break
}
}
const saveData = async(mode: string,val: string) => {
const url = useState().CONFURLS.value.sendsrvconfig || ''
message_type.value = MessageBoxType.Save
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
const data: any = {
id: payload.id ? payload.id : '',
val: val.length > 0 ? val : 'config',
config: servConfig.value,
}
if (mode == 'send' && url.length > 0) {
onMessageBox({src: 'done', val: ''})
const [res,err] = await send_data(url, data, true, true)
if (err && err.length > 0 ) {
show_message(MessageType.Error, `Error: ${err}`,2000)
} else if (res && res.status && res.status === 'ok') {
track_action(null, { ref: 'saveData',text: `${url} -> ${val}`})
show_message(MessageType.Success, `${t('saveload.dataSaved','Data saved')}`,2000,() => {
edititems.value = false
needSave.value = false
})
}
} else {
track_action(null, { ref: 'saveData',text: `local_json -> ${val}`})
show_message(MessageType.Warning, `${t('saveload.saveLocalFile','Save Data to Local File')}`,2000,() => {
data_url_encoded.value = `text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(data))}`
})
}
}
const onMessageBox = (item: { src: string, val: string }) => {
switch (item.src) {
case 'savedata':
show_input.value = false
saveData('local', item.val)
break
case 'sendata':
show_input.value = false
saveData('send', item.val)
break
case 'oneinput':
const r = async() => {
if (await checkUserAuth(item.val)) {
oneInputType.value='text'
openMessageBox.value = false
if (!await load_currentRoute(router.currentRoute.value)) {
show_message(MessageType.Error, `${t('saveload.loaderror','Load error')}`,2000)
}
} else {
show_message(MessageType.Error, `${t('saveload.autherror','Auth error')}`,2000)
}
}
r()
break
case 'endinput':
show_input.value = false
break
case 'done':
openMessageBox.value = false
break
case 'open':
openMessageBox.value = true
break
}
}
const onHome = () => {
const dom_body = document.getElementsByTagName('body')[0]
if (dom_body) {
dom_body.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
}
}
const onLoadModel = (model: { id: string}) => {
const url = useState().CONFURLS.value.data || ''
if ( url.length > 0) {
return
}
if (useState().selectOps.value[model.id]) {
load_data_model(model.id, url, routeKy as string,true,() => {
show_content.value = false
show_message(MessageType.Success, `${t('saveload.dataLoaded','Data loaded')}`,2000,() => {
show_content.value = true
})
})
}
}
onBeforeMount(async() => {
if (Object.keys(useState().varsConfig.value).length === 0) {
const [res, err] = await load_vars_config()
if (err && err.length > 0) {
show_content.value = false
} else {
useState().varsConfig.value = res
}
}
useState().dataSections.value = Object.keys(useState().varsConfig.value)
useState().dataSections.value.forEach(sec => {
show_sections.value[sec] = { view: true, enabled: true, help: false}
Object.keys(useState().varsConfig.value[sec]).forEach(itm => {
show_items.value[`${sec}.${itm}`] = { view: true, enabled: true, help: false}
})
})
useState().current_modelid.value = ''
const [res,errmsg] = await get_serv_config()
if (errmsg && errmsg.length > 0) {
show_content.value = false
} else {
servConfig.value = res
}
track_action(null, { ref: 'config',text: 'server config'}, 'mount')
})
</script>

302
src/views/Home.vue Normal file
View File

@ -0,0 +1,302 @@
<template>
<div
v-if="openMessageBox"
class="fixed inset-0 bg-gray-600 overflow-y-auto h-full w-full z-80 dark:bg-gray-900 opacity-97"
>
<message-box-view
:messageType="message_type"
:openMessageBox="openMessageBox"
:show_input="show_input"
:input_placeholder="input_placeholder"
:input_btns="input_btns"
:select_ops="select_ops"
:data_url_encoded="data_url_encoded"
:inpType="oneInputType"
@onInput="onInput"
@onMessageBox="onMessageBox"
@onLoadModel="onLoadModel"
/>
</div>
<div v-if="show_content" class="font-sans antialiased h-screen">
<div class="container mx-auto max-w-screen-2xl main-container">
<nav-menu
class="bg-light-300 dark:bg-gray-900"
:position="NavPosition.header"
:openMessageBox="openMessageBox"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:prefix="prefix"
@onNavMenu="onNavMenu"
/>
<main
id="cv-wrapper"
class="rounded flex flex-col sm:flex-row-reverse shadow-2xl"
:class="{ 'mt-11': fixMenu }"
>
<div
v-if="show_infopanel"
id="sidebar"
class="rounded-r w-full lg:w-80 sm:max-w-sm p-8 border-l-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-500 dark:via-indigo-400 dark:to-indigo-800"
>
</div>
<div class="content w-full p-5">
<div class="prose flex lg:items-center lg:justify-center p-4 border-1 rounded-md border-gray-300 dark:border-gray-600 lg:w-full sm:w-full sm:max-w-sm shadow-xl bg-light-300 dark:bg-gray-900">
<img v-if="isDark"
class="m-auto m-5"
width="500"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="500"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
</div>
<div class="prose my-4">
CVgen admin ...
</div>
</div>
<hr class="hr-sep" />
</main>
</div>
</div>
<div v-else class="text-center m-auto">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
<h2 class="mt-8 text-2xl text-gray-700 dark:text-gray-400">{{ t('message.loading', 'Loading') }}...</h2>
<h3> </h3>
<h4>{{useState().connection.value.state}}</h4>
</div>
<div
v-if="show_content"
class="flex-1 lg:mx-11 sm:mx-0 -mt-10 text-center text-sm text-gray-600 border-gray-300 border-1 border-b-0 rounded-t-lg"
>
<nav-menu
class="shadow-xl bg-light-300 dark:bg-gray-900"
:position="NavPosition.footer"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
prefix="sec"
:openMessageBox="openMessageBox"
@onNavMenu="onNavMenu"
/>
</div>
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
import MessageBoxView from '~/components/MessageBoxView.vue'
import NavMenu from '@/NavMenu.vue'
import { isDark } from '~/composables'
// import Modal from '@/Modal.vue'
import useState from '~/hooks/useState'
import { parseJwt,checkUserAuth } from '~/hooks/utilsAuth'
import { send_data,show_message} from '~/hooks/utils'
import { MessageType} from '~/typs'
import { MessageBoxType, NavPosition, InputBtnsType, ModelType, ShowInfoType} from '~/typs/cv'
import { load_currentRoute } from '~/hooks/loads'
import { track_action } from '~/hooks/tracking'
const prefix = 'home'
const assetsPath = useState().ASSETS_PATH
// const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
const message_type = ref(MessageBoxType.NotSet)
const fixMenu = ref(false)
const show_content = ref(true)
const show_infopanel = ref(false)
const openMessageBox = ref(false)
const needSave = ref(false)
const show_input = ref(false)
const data_url_encoded = ref('')
const input_btns = ref([] as InputBtnsType[])
const input_placeholder = ref('')
const oneInputType = ref('')
const onInput = (btn: InputBtnsType) => {
switch (btn.id) {
case 'ok':
break
case 'cancel':
break
}
show_input.value = false
}
const select_ops = ref({} as ModelType)
const show_info = ref({} as ShowInfoType)
const auth_info = useState().authinfo
const refresh = () => {
show_content.value = false
setTimeout(() => show_content.value = true, 1)
}
const onNavMenu = (item: { src: string, target: string }) => {
switch (item.src) {
case 'locale':
track_action(null, { ref: 'locale',text: i18n.locale.value})
refresh()
break
case 'infopanel':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
show_infopanel.value = !show_infopanel.value
break
case 'fixmenu':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
fixMenu.value = !fixMenu.value
break
case 'endinput':
show_input.value = false
break
case 'select':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
message_type.value = MessageBoxType.Select
openMessageBox.value = true
break
case 'save':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
input_placeholder.value = `${t('name', 'Name')} o 'data'`
show_input.value = true
message_type.value = MessageBoxType.Save
data_url_encoded.value = ''
openMessageBox.value = true
break
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.editable = !useState().authinfo.value.editable
const state = useState().authinfo.value.editable ? 'on' : 'off'
const msgTyp = state === 'on' ? MessageType.Warning : MessageType.Info
show_message(msgTyp, `${t('saveload.editMode','Edit Mode')} ${t('cv.'+state,'')}`)
break
case 'viewchange':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.viewchange = !useState().authinfo.value.viewchange
break
case 'goto':
const dom_id = document.getElementById(item.target)
if (dom_id) {
track_action(null, { ref: 'gotp',text: item.target})
dom_id.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
setTimeout(() => window.scrollBy(0, -40), 4000)
}
break
}
}
const saveData = async(mode: string,val: string) => {
const showinfo: ShowInfoType[] = useState().cvdata.value.showinfo
message_type.value = MessageBoxType.Save
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
const cv: any = {
u: payload.id ? payload.id : '',
data: {}
}
cv.data[val]={}
Object.keys(useState().datalang.value).forEach(lng =>
cv.data[val][lng] = useState().datalang.value[lng]
)
cv.data[val].main = useState().cvdata.value
showinfo.forEach((it,idx) => {
if (it.ky === useState().showinfo.value.ky) {
cv.data[val].main[idx] = useState().showinfo.value
}
})
const url = useState().CONFURLS.value.send || ''
if (mode == 'send' && url !== '') {
onMessageBox({src: 'done', val: ''})
const res = await send_data(url, cv, true, true, () => {
track_action(null, { ref: 'saveData',text: `${url} -> ${val}`})
show_message(MessageType.Success, `${t('saveload.dataSaved','Data saved')}`,2000,() => {
})
})
console.log(res)
} else {
track_action(null, { ref: 'saveData',text: `local_json -> ${val}`})
show_message(MessageType.Warning, `${t('saveload.saveData','Save data')}`,2000,() => {
data_url_encoded.value = `text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(cv))}`
})
}
}
const onMessageBox = (item: { src: string, val: string }) => {
switch (item.src) {
case 'savedata':
show_input.value = false
saveData('local', item.val)
break
case 'sendata':
show_input.value = false
saveData('send', item.val)
break
case 'oneinput':
const r = async() => {
if (await checkUserAuth(item.val)) {
oneInputType.value='text'
openMessageBox.value = false
if (!await load_currentRoute(router.currentRoute.value)) {
show_message(MessageType.Error, `${t('saveload.loaderror','Load error')}`,2000)
}
} else {
show_message(MessageType.Error, `${t('saveload.autherror','Auth error')}`,2000)
}
}
r()
break
case 'endinput':
show_input.value = false
break
case 'done':
openMessageBox.value = false
break
case 'open':
openMessageBox.value = true
break
}
}
// const onHome = () => {
// const dom_body = document.getElementsByTagName('body')[0]
// if (dom_body) {
// dom_body.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
// }
// }
const onLoadModel = (model: { id: string}) => {
// if (useState().selectOps.value[model.id]) {
// load_data_model(model.id, useState().DATA_URL.value, routeKy as string,true,() => {
// show_content.value = false
// show_message(MessageType.Success, `${t('saveload.dataLoaded','Data loaded')}`,2000,() => {
// show_content.value = true
// })
// })
// }
}
onBeforeMount(async() => {
useState().dataSections.value = ['Config','Tracking','Users']
// if (!useState().allowView.value && useState().userID.value === "?") {
// oneInputType.value='password'
// message_type.value=MessageBoxType.OneInput
// openMessageBox.value=true
// show_input.value=false
// input_placeholder.value='Enter password'
// }
})
</script>

237
src/views/Login.vue Normal file
View File

@ -0,0 +1,237 @@
<template>
<div v-if="isCheckin">
<div class="spinner" style="margin: 100px auto; width: 80px; height: 80px;">
<div class="dot1" />
<div class="dot2" />
</div>
<div class="flex justify-center text-2xl">
{{ t('message.loading') }} ...
</div>
</div>
<div v-else>
<section class="page-w bg-gray-400 dark:bg-gray-600 min-h-screen flex justify-center flex-wrap ">
<figure class="my-auto w-[calc(100%+1rem)]">
<div class="max-w-sm mx-auto overflow-hidden shadow-xl dark:bg-gray-900 bg-gray-100 rounded-3xl p-2">
<div class="flex flex-row justify-center">
<div class="mt-1.4 ml-2">
<button class="!outline-none text-gray-800 dark:text-gray-200" @click="toggleDark()">
<div v-if="isDark" i-carbon-sun />
<div v-else i-carbon-moon />
</button>
</div>
<div class="order-last flex-shrink-0">
<MenuLocales :label-mode="localesLabelMode" :include-current="includeCurrLocale" />
</div>
<div class="p-1 ml-8 flex flex-shrink-0 justify-center flex-grow bg-transparent">
<span v-if="use_logo" class="text-lg rounded-full">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
</span>
<h2 v-else class="text-center mb-1 text-lg heading ~info">
{{ t('login.welcome') }}
</h2>
</div>
</div>
<div v-if="use_logo" class="-mt-2 flex flex-row justify-center flex-none">
<span class="mb-1 text-lg ">
</span>
</div>
<div class="p-2">
<p class="text-center mb-3 text-base support ">
<span class="text-gray-400">{{ t('login.subtitle') }}</span>
</p>
<div class="container">
<h5 class="hidden text-center subheading">
<span class="text-gray-500">{{ t('login.subHeading') }}</span>
</h5>
<div class="hidden messages global mb-5">
<span class="text-gray-500">{{ t('login.form','') }}</span>
</div>
<div id="login-password" class="mt-8">
<form action="">
<fieldset class="mt-10 relative">
<input
v-model="user"
name="identifier"
id="identifier"
type="text"
class="input-fld peer h-10 w-full border-b border-gray-500 py-2 focus:placeholder- focus:outline-none focus:border-indigo-500"
autocomplete="username"
@focus="errorMessage = ''"
>
<label for="identifier"
class="input-lbl absolute left-0 -top-3.5 transition-all peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-400 peer-placeholder-shown:top-2 peer-focus:-top-3.5 peer-focus:text-gray-600 peer-focus:text-sm text-xs text-gray-600 dark:text-gray-500"
>{{ t('login.user_email') }}</label>
<div class="messages " />
</fieldset>
<fieldset v-if="!isRecovery" class="mt-10 relative">
<input
v-model="password"
name="password"
class="input-fld peer h-10 w-full border-b border-gray-500 py-2 focus:placeholder- focus:outline-none focus:border-indigo-500"
:type="passwordVisible ? 'text' : 'password'"
autocomplete="current-password"
@focus="errorMessage = ''"
>
<label for="password"
class="input-lbl absolute left-0 -top-3.5 transition-all peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-400 peer-placeholder-shown:top-2 peer-focus:-top-3.5 peer-focus:text-gray-600 peer-focus:text-sm text-xs text-gray-600 dark:text-gray-500"
>
{{ t('login.password') }}
</label>
<svg class="password-visibility-toggle cursor-pointer absolute top-4 -right-70" @click="togglePasswordVisible">
<path v-if="!passwordVisible" class="eye-open" d="M8 2.36365C4.36364 2.36365 1.25818 4.62547 0 7.81819C1.25818 11.0109 4.36364 13.2727 8 13.2727C11.6364 13.2727 14.7418 11.0109 16 7.81819C14.7418 4.62547 11.6364 2.36365 8 2.36365ZM8 11.4546C5.99273 11.4546 4.36364 9.82547 4.36364 7.81819C4.36364 5.81092 5.99273 4.18183 8 4.18183C10.0073 4.18183 11.6364 5.81092 11.6364 7.81819C11.6364 9.82547 10.0073 11.4546 8 11.4546ZM8 5.63637C6.79273 5.63637 5.81818 6.61092 5.81818 7.81819C5.81818 9.02547 6.79273 10 8 10C9.20727 10 10.1818 9.02547 10.1818 7.81819C10.1818 6.61092 9.20727 5.63637 8 5.63637Z" />
<path v-else class="eye-closed" fill-rule="evenodd" clip-rule="evenodd" d="M14.8222 1.85355C15.0175 1.65829 15.0175 1.34171 14.8222 1.14645C14.627 0.951184 14.3104 0.951184 14.1151 1.14645L12.005 3.25653C10.8901 2.482 9.56509 1.92505 8.06 1.92505C3 1.92505 0 7.92505 0 7.92505C0 7.92505 1.16157 10.2482 3.25823 12.0033L1.19366 14.0679C0.998396 14.2632 0.998396 14.5798 1.19366 14.775C1.38892 14.9703 1.7055 14.9703 1.90076 14.775L14.8222 1.85355ZM4.85879 10.4028L6.29159 8.96998C6.10643 8.66645 6 8.3089 6 7.92505C6 6.81505 6.89 5.92505 8 5.92505C8.38385 5.92505 8.7414 6.03148 9.04493 6.21664L10.4777 4.78384C9.79783 4.24654 8.93821 3.92505 8 3.92505C5.8 3.92505 4 5.72505 4 7.92505C4 8.86326 4.32149 9.72288 4.85879 10.4028ZM11.8644 6.88906L13.8567 4.8968C15.2406 6.40616 16 7.92505 16 7.92505C16 7.92505 13 13.925 8.06 13.925C7.09599 13.925 6.20675 13.7073 5.39878 13.3547L6.96401 11.7895C7.29473 11.8779 7.64207 11.925 8 11.925C10.22 11.925 12 10.145 12 7.92505C12 7.56712 11.9529 7.21978 11.8644 6.88906ZM9.33847 9.41501L9.48996 9.26352C9.44222 9.31669 9.39164 9.36726 9.33847 9.41501Z" />
</svg>
<div class="float-right mt-2">
<a v-if="allow_reset" class="flex border border-dotted border-gray-700 dark:border-gray-500 rounded-lg p-2 hover:bg-gray-300 dark:hover:bg-gray-700" href="#"
@click.prevent="forgot">
<div class="text-xs text-gray-400 dark:text-gray-500">{{ t('login.forgot') }} </div>
</a>
</div>
</fieldset>
<div v-if="connection.state !== ''" class="messages text-red-400 ">
{{ t(connection.state) }}
</div>
<div v-if="errorMessage !== ''" class="messages text-red-400 ">
{{ t(errorMessage) }}
</div>
<input name="csrf_token" type="hidden" value="">
<div class="mt-5 flex justify-center">
<button v-if="isRecovery" type="submit" @click.prevent="recovery"
class="text-lg flex-grow flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
>
{{ t('login.reset') }} {{ t('login.password') }}
</button>
<button v-else type="submit" @click.prevent="signIn"
class="text-lg flex-grow flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
>
{{ t('login.signin') }}
</button>
</div>
</form>
</div>
<div v-if="message.length > 0" class="messages text-green-600 dark:text-green-300 mt-3 ml-2">
{{ t(message) }}
</div>
<div v-if="allow_register && allow_reset" class="alternative-actions border-b border-gray-700 dark:border-gray-500 flex gap-4 py-4 section info justify-center ">
<!-- <a href="auth/registration">Recover password</a> -->
<a v-if="allow_register" class="border border-dotted border-gray-700 dark:border-gray-500 rounded-lg p-2 flex flex-col items-center hover:bg-gray-300 dark:hover:bg-gray-700" href="#" @click.prevent="register">
<div>{{ t('login.register') }} </div>
<div class="text-xs border-t border-indigo-500 dark:border-indigo-300 pt-1">{{ t('login.newuser') }}</div>
</a>
<a v-if="allow_reset" class="border border-dotted border-gray-700 dark:border-gray-500 rounded-lg p-2 flex flex-col items-center hover:bg-gray-300 dark:hover:bg-gray-700" href="#" @click.prevent="recovery">
<div>{{ t('login.reset') }} </div>
<div class="text-xs border-t border-indigo-500 dark:border-indigo-300 pt-1"> {{ t('login.password') }}</div>
</a>
</div>
</div>
</div>
</div>
<section style="display: none" class="section p-4">
</section>
</figure>
</section>
</div>
</template>
<script setup lang="ts">
import { ref, } from 'vue'
import { useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n'
import MenuLocales from '@/MenuLocales.vue'
import {LocalesLabelModes} from '~/typs'
import { isDark, toggleDark } from '~/composables'
import useState from '~/hooks/useState'
import { fetch_json,send_data } from '~/hooks/utils'
import { utf8_to_b64 } from '~/hooks/tools'
import { onTimeoutAuth } from '~/hooks/utilsAuth'
const assetsPath = useState().ASSETS_PATH.value
const router = useRouter()
const use_logo = router.currentRoute.value.meta.use_logo
const allow_register = useState().ALLOW_REGISTER
const allow_reset = useState().RESET_PASSWORD
const t = useI18n().t
const includeCurrLocale = true
const { connection } = useState()
connection.value.state = ''
const localesLabelMode = ref(LocalesLabelModes.auto)
const errorMessage = ref('')
const isCheckin = ref(false)
const isRecovery = ref(false)
const user = ref('')
const password = ref('')
const passwordVisible = ref(false)
const message = ref('')
const togglePasswordVisible = () => passwordVisible.value = !passwordVisible.value
const forgot = () => {
isRecovery.value= true
}
const signIn = async() => {
if (user.value.length === 0 || password.value.length === 0) {
errorMessage.value = 'login.nodata'
return
}
console.log('signIn')
let success = false
const url = useState().CONFURLS.value.login || ''
if (url.length == 0 ) {
errorMessage.value = 'login.no_url_found'
return
}
errorMessage.value = ''
connection.value.state = ''
const formData = { username: user.value, password: utf8_to_b64(password.value)}
const [responseData,_err] = await send_data(url, formData, false, false)
if (responseData && responseData.token && responseData.token.length > 0) {
localStorage.setItem(useState().AUTHKEY.value,responseData.token)
onTimeoutAuth()
// if (responseData.defs)
// store.dispatch(AppDefsAction.addDefs, { key: map_key, defs: responseData.defs })
success = true
useState().checkin.value = false
useState().userID.value = user.value
}
if (success) {
const srcRoute = useState().sourceRoute.value.length > 0 ? useState().sourceRoute.value : '/'
router.push(srcRoute)
} else {
useState().userID.value = ''
errorMessage.value = 'login.invalidcredentials'
}
}
const register = () => {
// TODO register
console.log('register')
}
const recovery = async() => {
if (user.value.length === 0) {
errorMessage.value = 'login.nodata'
return
}
// let success = false
const url = useState().CONFURLS.value.recovery || ''
if (url.length == 0 ) {
errorMessage.value = 'login.no_url_found'
return
}
errorMessage.value = ''
connection.value.state = ''
const urlPath = `${url}/${user.value}`
const [_,err] = await fetch_json(urlPath, 2000, false)
if (err) {
errorMessage.value = 'login.invalidcredentials'
} else {
message.value = 'login.checkemail'
}
}
</script>

111
src/views/Logout.vue Normal file
View File

@ -0,0 +1,111 @@
<template>
<div v-if="isCheckin">
<div class="spinner" style="margin: 100px auto; width: 80px; height: 80px;">
<div class="dot1" />
<div class="dot2" />
</div>
<div class="flex justify-center text-2xl">
{{ t('message.loading') }} ...
</div>
</div>
<div v-else>
<section class="page-w bg-gray-400 dark:bg-gray-600 min-h-screen flex justify-center flex-wrap ">
<figure class="my-auto login w-[calc(100%+1rem)]">
<div class="max-w-sm mx-auto overflow-hidden shadow-xl dark:bg-gray-900 bg-gray-100 rounded-3xl p-2">
<div class="flex flex-row justify-center">
<div class="mt-1.4 ml-2">
<button class="!outline-none text-gray-800 dark:text-gray-200" @click="toggleDark()">
<div v-if="isDark" i-carbon-sun />
<div v-else i-carbon-moon />
</button>
</div>
<div class="order-last flex-shrink-0">
<MenuLocales :label-mode="localesLabelMode" :include-current="includeCurrLocale" />
</div>
<div class="p-1 ml-8 flex flex-shrink-0 justify-center flex-grow bg-transparent">
<span v-if="use_logo" class="text-lg rounded-full">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
</span>
<h2 v-else class="text-center mb-1 text-lg heading ~info">
{{ t('login.welcome') }}
</h2>
</div>
</div>
<div v-if="use_logo" class="-mt-2 flex flex-row justify-center flex-none">
<span class="mb-1 text-lg ">
</span>
</div>
<div class="p-2">
<p class="text-center mb-3 text-base support ~info dark:text-gray-100">
{{ t('login.subtitle') }}
</p>
<div class="container">
<h5 class="text-center subheading mb-4 dark:text-gray-200">
{{ t('login.end_session') }}
</h5>
<div class="messages global">
</div>
<div class="alternative-actions border-t border-b flex gap-4 py-2 section justify-center bg-gray-300 dark:bg-gray-600">
<!-- <a href="auth/registration">Recover password</a> -->
<h3 class=" ~critical text-xl p-4 dark:text-gray-100">
{{ t('login.see_you_soon','') }}
</h3>
</div>
<div class="mt-8 flex justify-center">
<button type="submit"
class="text-lg flex-grow flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
@click.prevent="signIn">
{{ t('login.signin') }}
</button>
</div>
</div>
</div>
</div>
</figure>
</section>
</div>
</template>
<script setup lang="ts">
import {
ref,
onMounted,
} from 'vue'
import { useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n'
// import { useStore } from 'vuex'
// import { AppDefsAction } from '~/store/types'
import MenuLocales from '@/MenuLocales.vue'
import {LocalesLabelModes} from '~/typs'
import { isDark, toggleDark } from '~/composables'
import useState from '~/hooks/useState'
const assetsPath = useState().ASSETS_PATH.value
const router = useRouter()
const use_logo = router.currentRoute.value.meta.use_logo
const t = useI18n().t
// const store = useStore()
// const lang = computed(() => store.state.app_lang.lang)
const includeCurrLocale = true // computed(() => appDefs.value.includeCurrLocale || false)
const localesLabelMode = ref(LocalesLabelModes.auto)
const isCheckin = ref(false)
const signIn = async() => {
router.push('/login')
}
onMounted(() => {
// TODO call backend for logout
localStorage.removeItem(useState().AUTHKEY.value)
})
</script>

318
src/views/Register.vue Normal file
View File

@ -0,0 +1,318 @@
<template>
<div v-if="isCheckin">
<div class="spinner" style="margin: 100px auto; width: 80px; height: 80px;">
<div class="dot1" />
<div class="dot2" />
</div>
<div class="flex justify-center text-2xl">
{{ t('message.loading') }} ...
</div>
</div>
<div v-else>
<section v-if="show_content && (invitationData.role || request_id)" class="page-w bg-gray-400 dark:bg-gray-600 min-h-screen flex justify-center flex-wrap ">
<figure class="my-auto w-[calc(100%+1rem)]">
<div class="max-w-sm mx-auto overflow-hidden shadow-xl dark:bg-gray-900 bg-gray-100 rounded-3xl p-2">
<div class="flex flex-row justify-center">
<div class="mt-1.4 ml-2">
<button class="!outline-none text-gray-800 dark:text-gray-200" @click="toggleDark()">
<div v-if="isDark" i-carbon-sun />
<div v-else i-carbon-moon />
</button>
</div>
<div class="order-last flex-shrink-0">
<MenuLocales :label-mode="localesLabelMode" :include-current="includeCurrLocale" />
</div>
<div class="p-1 ml-8 flex flex-shrink-0 justify-center flex-grow bg-transparent">
<span v-if="use_logo" class="text-lg rounded-full">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
</span>
<h2 v-else class="text-center mb-1 text-lg heading ~info">
{{ t('login.welcome') }}
</h2>
</div>
</div>
<div v-if="use_logo" class="-mt-2 flex flex-row justify-center flex-none">
<span class="mb-1 text-lg ">
</span>
</div>
<div class="p-2">
<p class="text-center mb-3 text-base support ">
<span class="text-gray-400">{{ t('login.subtitle') }}</span>
</p>
<div class="container">
<h5 class="hidden text-center subheading">
<span class="text-gray-500">{{ t('login.subHeading') }}</span>
</h5>
<div class="hidden messages global mb-5">
<span class="text-gray-500">{{ t('login.form','') }}</span>
</div>
<div id="login-password" class="mt-8">
<form action="">
<fieldset v-if="is_register" class="mt-10 relative">
<input
v-model="user"
name="identifier"
id="identifier"
type="text"
class="input-fld peer h-10 w-full border-b border-gray-500 py-2 focus:placeholder- focus:outline-none focus:border-indigo-500"
autocomplete="username"
@focus="errorMessage = ''"
>
<label for="identifier"
class="input-lbl absolute left-0 -top-3.5 transition-all peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-400 peer-placeholder-shown:top-2 peer-focus:-top-3.5 peer-focus:text-gray-600 peer-focus:text-sm text-xs text-gray-600 dark:text-gray-500 text-xs text-gray-600 dark:text-gray-500"
>{{ t('login.username') }}</label>
<div class="messages " />
</fieldset>
<fieldset v-if="is_register" class="mt-10 relative">
<input
v-model="email"
name="email"
id="email"
type="text"
class="input-fld peer h-10 w-full border-b border-gray-500 py-2 focus:placeholder- focus:outline-none focus:border-indigo-500"
autocomplete="email"
@focus="errorMessage = ''"
>
<label for="identifier"
class="input-lbl absolute left-0 -top-3.5 transition-all peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-400 peer-placeholder-shown:top-2 peer-focus:-top-3.5 peer-focus:text-gray-600 peer-focus:text-sm text-xs text-gray-600 dark:text-gray-500"
>{{ t('login.email') }}</label>
<div class="messages " />
</fieldset>
<fieldset class="mt-10 relative">
<input
v-model="password"
name="password"
class="input-fld peer h-10 w-full border-b border-gray-500 py-2 focus:placeholder- focus:outline-none focus:border-indigo-500"
:type="passwordVisible ? 'text' : 'password'"
autocomplete="current-password"
@focus="errorMessage = ''"
>
<label for="password"
class="input-lbl absolute left-0 -top-3.5 transition-all peer-placeholder-shown:text-base peer-placeholder-shown:text-gray-400 peer-placeholder-shown:top-2 peer-focus:-top-3.5 peer-focus:text-gray-600 peer-focus:text-sm text-xs text-gray-600 dark:text-gray-500"
>
{{ t('login.password') }}
</label>
<div v-if="connection.state !== ''" class="messages text-red-400 ">
{{ t(connection.state) }}
</div>
<div v-if="errorMessage !== ''" class="messages text-red-400 ">
{{ t(errorMessage) }}
</div>
<svg class="password-visibility-toggle absolute top-4 -right-70 cursor-pointer" @click="togglePasswordVisible">
<path v-if="!passwordVisible" class="eye-open" d="M8 2.36365C4.36364 2.36365 1.25818 4.62547 0 7.81819C1.25818 11.0109 4.36364 13.2727 8 13.2727C11.6364 13.2727 14.7418 11.0109 16 7.81819C14.7418 4.62547 11.6364 2.36365 8 2.36365ZM8 11.4546C5.99273 11.4546 4.36364 9.82547 4.36364 7.81819C4.36364 5.81092 5.99273 4.18183 8 4.18183C10.0073 4.18183 11.6364 5.81092 11.6364 7.81819C11.6364 9.82547 10.0073 11.4546 8 11.4546ZM8 5.63637C6.79273 5.63637 5.81818 6.61092 5.81818 7.81819C5.81818 9.02547 6.79273 10 8 10C9.20727 10 10.1818 9.02547 10.1818 7.81819C10.1818 6.61092 9.20727 5.63637 8 5.63637Z" />
<path v-else class="eye-closed" fill-rule="evenodd" clip-rule="evenodd" d="M14.8222 1.85355C15.0175 1.65829 15.0175 1.34171 14.8222 1.14645C14.627 0.951184 14.3104 0.951184 14.1151 1.14645L12.005 3.25653C10.8901 2.482 9.56509 1.92505 8.06 1.92505C3 1.92505 0 7.92505 0 7.92505C0 7.92505 1.16157 10.2482 3.25823 12.0033L1.19366 14.0679C0.998396 14.2632 0.998396 14.5798 1.19366 14.775C1.38892 14.9703 1.7055 14.9703 1.90076 14.775L14.8222 1.85355ZM4.85879 10.4028L6.29159 8.96998C6.10643 8.66645 6 8.3089 6 7.92505C6 6.81505 6.89 5.92505 8 5.92505C8.38385 5.92505 8.7414 6.03148 9.04493 6.21664L10.4777 4.78384C9.79783 4.24654 8.93821 3.92505 8 3.92505C5.8 3.92505 4 5.72505 4 7.92505C4 8.86326 4.32149 9.72288 4.85879 10.4028ZM11.8644 6.88906L13.8567 4.8968C15.2406 6.40616 16 7.92505 16 7.92505C16 7.92505 13 13.925 8.06 13.925C7.09599 13.925 6.20675 13.7073 5.39878 13.3547L6.96401 11.7895C7.29473 11.8779 7.64207 11.925 8 11.925C10.22 11.925 12 10.145 12 7.92505C12 7.56712 11.9529 7.21978 11.8644 6.88906ZM9.33847 9.41501L9.48996 9.26352C9.44222 9.31669 9.39164 9.36726 9.33847 9.41501Z" />
</svg>
</fieldset>
<input name="csrf_token" type="hidden" value="">
<div class="mt-10 flex justify-center">
<button v-if="is_register" type="submit" @click.prevent="onRegister"
class="text-lg flex-grow flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
>
{{ t('login.register') }}
</button>
<button v-else type="submit" @click.prevent="onSend"
class="text-lg flex-grow flex-shrink-0 bg-indigo-500 hover:bg-indigo-700 border-indigo-500 hover:border-indigo-700 text-sm border-4 text-white py-1 px-2 rounded"
>
{{ t('send') }}
</button>
</div>
</form>
</div>
</div>
</div>
</div>
<section style="display: none" class="section p-4">
</section>
</figure>
</section>
<div v-else class="text-center m-auto">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
@click="router.push(goRoute)"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
@click="router.push(goRoute)"
/>
<h2 class="mt-8 text-2xl text-gray-700 dark:text-gray-400">{{ t('message.loading', 'Loading') }}...</h2>
<h4>{{useState().connection.value.state}}</h4>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, } from 'vue'
import { useRouter } from 'vue-router'
import { useI18n } from 'vue-i18n'
import MenuLocales from '@/MenuLocales.vue'
import {LocalesLabelModes,InvitationDataType} from '~/typs'
import { isDark, toggleDark } from '~/composables'
import useState from '~/hooks/useState'
import { onTimeoutAuth } from '~/hooks/utilsAuth'
import { MessageType } from '~/typs'
import { show_message,send_data } from '~/hooks/utils'
import { b64_to_utf8,utf8_to_b64 } from '~/hooks/tools'
const assetsPath = useState().ASSETS_PATH.value
const router = useRouter()
const goRoute = "/"
const use_logo = router.currentRoute.value.meta.use_logo
const t = useI18n().t
const includeCurrLocale = true
const is_register = computed(() => {
switch(router.currentRoute.value.name) {
case 'Register':
return true
default:
return false
}
})
const show_content = ref(true)
const invitationData = ref({} as InvitationDataType)
const { connection } = useState()
connection.value.state = ''
const localesLabelMode = ref(LocalesLabelModes.auto)
const errorMessage = ref('')
const isCheckin = ref(false)
const user = ref('')
const email = ref('')
const password = ref('')
const passwordVisible = ref(false)
const togglePasswordVisible = () => passwordVisible.value = !passwordVisible.value
const request_id = ref('')
const onSend = async() => {
if (request_id.value.length === 0 || password.value.length === 0 ) {
errorMessage.value = 'login.nodata'
return
}
const url = useState().CONFURLS.value.sendrecovery || ''
if (url.length == 0 ) {
errorMessage.value = 'login.no_url_found'
return
}
errorMessage.value = ''
connection.value.state = ''
const formData = {
id: request_id.value,
val: utf8_to_b64(password.value),
}
const [responseData,_err] = await send_data(url, formData, false, false)
if (responseData && responseData.status && responseData.status === 'ok') {
const srcRoute = useState().sourceRoute.value.length > 0 ? useState().sourceRoute.value : '/'
router.push(srcRoute)
} else {
useState().userID.value = ''
errorMessage.value = 'login.invalidcredentials'
}
}
const onRegister = async() => {
if (user.value.length === 0 || password.value.length === 0 || email.value.length === 0 ) {
errorMessage.value = 'login.nodata'
return
}
let success = false
const url = useState().CONFURLS.value.newuser || ''
if (url.length == 0 ) {
errorMessage.value = 'login.no_url_found'
return
}
errorMessage.value = ''
connection.value.state = ''
const formData = {
username: user.value,
email: email.value,
password: utf8_to_b64(password.value),
role: invitationData.value.role,
data: invitationData.value.data,
}
const [responseData,_err] = await send_data(url, formData, true, false)
if (responseData && responseData.status && responseData.status === 'ok') {
// if (responseData.defs)
// store.dispatch(AppDefsAction.addDefs, { key: map_key, defs: responseData.defs })
success = true
useState().checkin.value = false
useState().userID.value = user.value
}
if (success) {
localStorage.removeItem(useState().AUTHKEY.value)
const srcRoute = useState().sourceRoute.value.length > 0 ? useState().sourceRoute.value : '/'
router.push(srcRoute)
} else {
useState().userID.value = ''
errorMessage.value = 'login.invalidcredentials'
}
}
const load_invitation_data = async (id: string): Promise<[any,string]> => {
let url = useState().CONFURLS.value.invitation || ''
if (url.length == 0)
return [{},'No URL defined']
url = `${url}/${id}`
let res: any = {}
let error = null
try {
const response = await self.fetch(url, {
method: 'GET',
})
if (response.ok) {
res = await response.json()
if (res.data) {
error = null
} else {
error = res.message
}
} else {
error = response.statusText
}
}
catch (err: any) {
error = err.message
}
if (error) {
show_message(MessageType.Error, `'Load invitation info' -> ${error}`, 5000)
useState().connection.value.state = 'connection.error'
return [{}, error]
}
return [res,error]
}
onBeforeMount(async() => {
const id = router.currentRoute.value.params.id || router.currentRoute.value.query.id || ''
request_id.value = id as string
show_content.value = false
if (is_register.value) {
const [res,error] = await load_invitation_data(id as string)
if (error && error.length > 0) {
return
}
try {
invitationData.value = JSON.parse(b64_to_utf8(res.data))
} catch (err: any) {
console.log(err)
}
if (res && res.auth) {
localStorage.setItem(useState().AUTHKEY.value,res.auth)
onTimeoutAuth()
}
if (invitationData.value.email) {
email.value = invitationData.value.email
show_content.value = true
}
} else if (request_id.value.length > 0) {
show_content.value = true
}
})
</script>

678
src/views/Tracking.vue Normal file
View File

@ -0,0 +1,678 @@
<template>
<div
v-if="openMessageBox"
class="fixed inset-0 bg-gray-600 overflow-y-auto h-full w-full z-80 dark:bg-gray-900 opacity-97"
>
<message-box-view
:messageType="message_type"
:openMessageBox="openMessageBox"
:show_input="show_input"
:input_placeholder="input_placeholder"
:input_btns="input_btns"
:select_ops="select_ops"
:data_url_encoded="data_url_encoded"
:inpType="oneInputType"
@onInput="onInput"
@onMessageBox="onMessageBox"
@onLoadModel="onLoadModel"
/>
</div>
<div v-if="show_content" class="font-sans antialiased">
<div class="container mx-auto max-w-screen-2xl main-container">
<nav-menu
class="bg-light-300 dark:bg-gray-900"
:position="NavPosition.header"
:openMessageBox="openMessageBox"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="true"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
:prefix="prefix"
@onNavMenu="onNavMenu"
/>
<div v-if="trackingList.length > 0" class="font-sans antialiased pb-3 border-b-1 border-gray-500 ">
<h2 class="section-headline my-4 px-5 pb-2"
:class="{'bg-gray-200 dark:bg-gray-700 border-1 rounded-tl-xl rounded-r-3xl border-indigo-500': show_payload}"
:id="`${prefix}-sec_${prefix}_list`"
>
<span v-if="show_list" class="text-sm">{{t('tracking.list','List')}}</span>
<button
class="no-print text-sm float-right icon-btn mt-2 -mr-3 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onShowListBtn"
>
<div v-if="show_list" i-carbon-view-off />
<div v-else class="noprint flex -mt-3">
<div class="text-xs" :class="{'text-indigo-600 dark:text-indigo-300 mr-1': !show_payload}">{{t('tracking.list','List')}}</div>
<div i-carbon-view />
</div>
</button>
</h2>
<ul v-if="show_list" class="flex mx-2 pb-2">
<li v-for="(item,index) in trackingList" >
<button
@click="onClickItemList(item,index)"
class="btn-msg flex ml-2"
>
<div i-carbon-list /> <div class="ml-2 text-sm"> {{item}} </div>
</button>
</li>
</ul>
</div>
<main
v-if="trackingData"
:id="`${prefix}-sec_${prefix}_data`"
class="rounded flex flex-col sm:flex-row-reverse shadow-2xl"
:class="{ 'mt-11': fixMenu }"
>
<div
v-if="show_infopanel"
id="sidebar"
class="rounded-r w-full lg:w-60 sm:max-w-sm p-8 border-l-1 border-b-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-700 dark:via-indigo-600 dark:to-indigo-800"
>
<div v-for="ky in dataFields">
<label
v-if="( ky === 'context' && showField.context ) || (ky !== 'payload' && ky !== 'context')"
:for="`filter-${ky}`">
<span class="text-xs text-gray-800 dark:text-gray-400">
{{t(`tracking.${ky}`,ky)}}
</span>
</label>
<input
v-if="( ky === 'context' && showField.context ) || (ky !== 'payload' && ky !== 'context')"
type="text"
class="appearance-none bg-transparent w-full text-gray-900 dark:text-gray-200 py-2 leading-tight focus:outline-none"
:class="{'border-b border-indigo-500': editsearch}"
v-model="filterData[ky]"
:disabled="!editsearch"
:id="`filter-${ky}`"
/>
</div>
<h2 class="section-headline my-4 px-5 pb-2"
:class="{'bg-gray-200 dark:bg-gray-700 border-1 rounded-tl-xl rounded-r-3xl border-indigo-500': show_payload}"
>
<span v-if="show_payload" class="text-sm">{{t('tracking.payload','payload')}}</span>
<button
class="no-print text-sm float-right icon-btn mt-2 -mr-3 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onPayloadBtn"
>
<div v-if="show_payload" i-carbon-view-off />
<div v-else class="noprint flex -mt-3">
<div class="text-xs" :class="{'text-indigo-600 dark:text-indigo-300 mr-1': !show_payload}">{{t('tracking.payload','payload')}}</div>
<div i-carbon-view />
</div>
</button>
</h2>
<div v-if="show_payload && showField.payload" v-for="ky in payloadFields">
<label :for="`filter-${ky}`">
<span class="text-xs text-gray-800 dark:text-gray-400">
{{t(`payload.${ky}`,ky)}}
</span>
</label>
<input
type="text"
class="appearance-none bg-transparent w-full text-gray-900 dark:text-gray-200 py-2 leading-tight focus:outline-none"
:class="{'border-b border-indigo-500': editsearch}"
v-model="filterData.payload[ky]"
:disabled="!editsearch"
:id="`filter-${ky}`"
/>
</div>
<div class="py-4 flex items-center gap-5 border-t-1 border-gray-500">
<button
v-if="editsearch"
class="btn-msg flex-grow"
@click="onSaveSearchBtn"
>
<span>{{ t('save', 'Save') }}</span>
</button>
<button v-else class="btn-msg flex-grow" @click="onResetSearchBtn">{{ t('reset', 'Reset') }}</button>
<button v-if="!editsearch" class="btn-msg flex-grow" @click="onEditSearchBtn">{{ t('edit', 'Edit') }}</button>
<button v-else class="btn-msg flex-grow" @click="onCloseSearchBtn">{{ t('close', 'Close') }}</button>
</div>
</div>
<div class="content w-full p-5">
<div class="prose flex-1 p-4 border-1 rounded-md border-gray-300 dark:border-gray-600 lg:w-11/12 sm:w-full sm:max-w-sm shadow-xl bg-light-300 dark:bg-gray-900">
<div class="relative float-right -top-2 right-2">
<span class=" text-gray-500 dark:text-gray-500 text-xs pr-1"> {{t('total','Total')}}: </span>
<span class=" text-indigo-600 dark:text-indigo-400 text-sm pr-1"> {{trackingDataFilter.length}} </span>
<span class=" text-gray-500 dark:text-gray-500 text-xs pr-1"> {{t('of','of')}} </span>
<span class=" text-gray-600 dark:text-gray-400 text-sm pr-1"> {{trackingData.length}} </span>
</div>
<div class="flex flex-col container max-w-fit mt-5 mx-auto w-full lg:items-center lg:justify-center bg-white dark:bg-gray-800 rounded-lg shadow">
<ul class="flex flex-col divide-y divide-gray-500 w-full">
<li class="flex flex-row py-2 lg:py-0 text-sm lg:text-base">
<div class="flex-row select-none flex lg:flex-1 lg:items-center">
<div class="flex flex-row w-15 lg:h-10 lg:justify-center lg:items-center text-gray-600 dark:text-gray-400 ml-2 lg:ml-3 cursor-pointer font-normal lg:text-sm" :class="{'m-0': orderBy === 'what'}">
<div v-if="orderBy === 'what' && sortMode !== 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400" i-carbon-arrow-up />
<div @click="onField('what')" :class="{'border-b border-indigo-500': orderBy === 'what'}">{{t('tracking.what','what')}} </div>
<div v-if="orderBy === 'what' && sortMode === 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400" i-carbon-arrow-down/>
</div>
<div class="flex flex-row w-15 lg:w-30 sm:max-w-sm lg:h-10 lg:justify-center lg:items-center ml-2 text-gray-600 dark:text-gray-400 lg:text-xs cursor-pointer" :class="{'m-0': orderBy === 'data'}">
<div v-if="orderBy === 'data' && sortMode !== 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400" i-carbon-arrow-up />
<div @click="onField('data')" :class="{'border-b border-indigo-500': orderBy === 'data'}">{{t('tracking.data','data')}} </div>
<div v-if="orderBy === 'data' && sortMode === 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400" i-carbon-arrow-down/>
</div>
<div class="flex-1 pl-1">
<div class="flex flex-row">
<div class="flex flex-row w-20 text-gray-600 dark:text-gray-400 cursor-pointer font-normal lg:text-sm">
<div v-if="orderBy === 'where' && sortMode !== 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400 mt-1" i-carbon-arrow-up />
<div @click="onField('where')" :class="{'border-b border-indigo-500': orderBy === 'where'}">{{t('tracking.where','where')}} </div>
<div v-if="orderBy === 'where' && sortMode === 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400 mt-1" i-carbon-arrow-down/>
</div>
<button
class="no-print text-sm icon-btn mt-2 mr-3 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onToggleView('context')"
>
<div v-if="showField.context" class="flex -mt-1" >
<div class="hidden lg:block ml-5 mr-2 mt-0.8 text-gray-500 dark:text-gray-500 text-[0.8em]"> {{t('tracking.context','context')}} </div>
<div class="lg:hidden ml-5 mt-1 text-gray-500 dark:text-gray-500 text-[0.8em]"> {{t('tracking.ctx','ctx')}} </div>
<div i-carbon-view-off />
</div>
<div v-else class="noprint flex -mt-1">
<div class="mr-2 text-xs" :class="{'line-through text-indigo-600 dark:text-indigo-300': !showField.context}">{{t('tracking.context','context')}}</div>
<div i-carbon-view />
</div>
</button>
</div>
</div>
<div class="flex-1 pl-1 mr-5 ">
<button
class="no-print text-sm icon-btn mt-2 -mr-3 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onToggleView('payload')"
>
<div v-if="showField.payload" class="flex -mt-1">
<div class="font-normal text-gray-600 dark:text-gray-400 lg:text-sm mr-1 -mt-0.8">{{t('tracking.auth','auth')}}</div>
<div i-carbon-view-off />
</div>
<div v-else class="noprint flex -mt-3">
<div class="font-normal text-gray-600 dark:text-gray-400 lg:text-sm" :class="{'line-through text-indigo-600 dark:text-indigo-300 mr-1': !showField.payload}">{{t('tracking.auth','auth')}}</div>
<div i-carbon-view />
</div>
</button>
</div>
<div class="flex flex-row lg:justify-center lg:mr-10 cursor-pointer font-normal lg:text-sm">
<div v-if="orderBy === 'when' && sortMode !== 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400 mt-1" i-carbon-arrow-up />
<div @click="onField('when')" class="text-gray-600 dark:text-gray-400" :class="{'border-b border-indigo-500': orderBy === 'when'}">
{{t('tracking.when','when')}}
</div>
<div v-if="orderBy === 'when' && sortMode === 'descending'" @click="onToggelSort" class="text-indigo-600 dark:text-indigo-400 mt-1" i-carbon-arrow-down/>
</div>
</div>
</li>
<li v-for="(item,index) in trackingDataFilter" class="flex flex-col lg:flex-row">
<div v-if="item.when" class="flex-col select-none cursor-pointer hover:bg-gray-200 dark:hover:bg-gray-900 flex lg:flex-row lg:flex-1 lg:items-center p-4">
<div class="flex flex-row lg:flex-col lg:w-15 sm:max-w-xs lg:h-10 lg:justify-center lg:items-center pb-2 lg:pb-0 pl-1 text-gray-900 dark:text-gray-100 font-normal text-sm">
<div class="lg:hidden text-sm w-20 text-gray-600 dark:text-gray-400 mt-0.8"> {{t('tracking.what','what')}}: </div>
<div class="flex-grow-0">{{item.what}}</div>
</div>
<div class="flex flex-row lg:flex-col lg:w-30 sm:max-w-sm lg:h-10 lg:justify-center lg:items-center pl-1">
<div class="lg:hidden text-xs w-20 text-gray-600 dark:text-gray-400 mt-0.8"> {{t('tracking.data','data')}}: </div>
<div class=" text-gray-900 dark:text-gray-100 text-xs"> {{item.data}} </div>
</div>
<div class="flex flex-row lg:flex-1 pl-1">
<div class="lg:hidden text-sm w-20 text-gray-600 dark:text-gray-400 mt-0.8"> {{t('tracking.where','where')}}: </div>
<div class="flex-1 flex-col">
<div class="font-normal text-sm text-gray-600 dark:text-gray-400">{{item.where}}</div>
<div v-if="showField.context" v-for="(line,pos) in item.context.split(')')" :key="pos" class="text-gray-500 dark:text-gray-500 text-[0.8em] mt-2">
{{`${line})`}}
</div>
</div>
</div>
<div v-if="showField.payload" class="flex-1 pl-1 pb-2 lg:pb-0">
<div class="lg:hidden text-xs w-20 text-gray-600 dark:text-gray-400 mt-0.8"> {{t('tracking.auth','auth')}}: </div>
<table v-if="item.payload" class="table-auto lg:mt-2">
<tbody>
<tr v-for="key,idx in payloadFields" :key="idx" class="text-gray-500 dark:text-gray-500 text-[0.8em] mt-2">
<td v-if="item.payload[key]" class="text-xs text-gray-500 dark:text-gray-400">{{t(`payload.${key}`,key)}}</td>
<td v-if="item.payload[key]" class="pl-5 text-gray-600 dark:text-gray-300">{{`${auth_get_value(key,item.payload)}`}}</td>
</tr>
</tbody>
</table>
</div>
<div class="flex flex-row lg:justify-center">
<div class="lg:hidden pl-1 text-xs w-20 text-gray-600 dark:text-gray-400 mt-0.8"> {{t('tracking.when','when')}}: </div>
<div class="pl-1 text-gray-600 dark:text-gray-200 text-xs">{{`${format_date(parseInt(item.when))}`}}</div>
<button v-if="edititems"
@click="onClickItem(item,index)"
class="absolute right-15 lg:relative lg:right-0 w-10 text-right flex justify-end"
>
<svg width="20" fill="currentColor" height="20" class="hover:text-gray-800 dark:hover:text-white dark:text-gray-200 text-gray-500" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg">
<path d="M1363 877l-742 742q-19 19-45 19t-45-19l-166-166q-19-19-19-45t19-45l531-531-531-531q-19-19-19-45t19-45l166-166q19-19 45-19t45 19l742 742q19 19 19 45t-19 45z"></path>
</svg>
</button>
</div>
</div>
</li>
</ul>
</div>
</div>
</div>
</main>
<div
v-if="show_content"
class="mr-auto w-full lg:w-1/2 text-center text-sm mx-2 text-gray-600 border-gray-300 border-1 border-b-0 rounded-t-xl"
>
<nav-menu
class="shadow-xl bg-light-300 dark:bg-gray-900"
:position="NavPosition.footer"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
prefix="sec"
:openMessageBox="openMessageBox"
@onNavMenu="onNavMenu"
/>
</div>
</div>
</div>
<div v-else class="text-center m-auto">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
<h2 class="mt-8 text-2xl text-gray-700 dark:text-gray-400">{{ t('message.loading', 'Loading') }}...</h2>
<h4>{{useState().connection.value.state}}</h4>
</div>
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
import MessageBoxView from '~/components/MessageBoxView.vue'
import NavMenu from '@/NavMenu.vue'
import { isDark} from '~/composables'
import useState from '~/hooks/useState'
import { parseJwt,checkUserAuth } from '~/hooks/utilsAuth'
import { load_data_model,load_currentRoute} from '~/hooks/loads'
import { fetch_json } from '~/hooks/utils'
import { track_action } from '~/hooks/tracking'
import { AuthPayloadType,TrackActionShowType, MessageType, TrackActionType} from '~/typs'
import { MessageBoxType, NavPosition, InputBtnsType, ShowInfoType, ModelType} from '~/typs/cv'
import { show_message,send_data} from '~/hooks/utils'
import { format_date} from '~/hooks/tools'
const trackingList = ref([] as string[])
const trackingData = ref([] as TrackActionType[])
const filterData = ref({} as TrackActionType)
const dataFields = ref(['what','data','where','context','payload','when'])
const payloadFields = ref(['id','exp','uuid','data','orig_iat'])
const showField = ref({what: true,data: true,where: true,context: true,payload: true,when: true} as TrackActionShowType)
const show_payload = ref(false)
const orderBy = ref('')
const sortMode = ref('')
const show_list = ref(true)
const filterPayload = (item: AuthPayloadType):boolean => {
let include = true
for (const key in item) {
if (!include)
break
const keyval = filterData.value.payload && filterData.value.payload[key] ? filterData.value.payload[key] : ''
if (keyval.length == 0)
continue
let val = ''
switch(key) {
case 'orig_iat':
case 'exp':
val = typeof item[key] === 'string' ? format_date(parseInt(item[key] as string)*1000) : format_date((item[key] as number)*1000)
break
default:
val = item[key]
}
if (val.length > 0 && !val.includes(keyval)) {
include = false
}
}
return include
}
const filterItem = (item: TrackActionType, index: number):boolean => {
let include = true
for (const key in item) {
if (!include)
break
if (key !== 'payload' && filterData.value[key].length == 0)
continue
let val = ''
switch(key) {
case 'auth':
break
case 'payload':
include = filterPayload(item.payload as AuthPayloadType)
// const auth = parseJwt(item.auth)
break
case 'when':
val = format_date(parseInt(item.when as string))
if (!val.includes(filterData.value[key] as string)) {
include = false
}
break
default:
val = item[key]
}
if (val.length > 0 && !val.includes(filterData.value[key])) {
include = false
}
}
return include
}
const trackingDataFilter = computed(() => {
if (orderBy.value.length > 0) {
if (sortMode.value === 'descending')
return trackingData.value.filter(filterItem).sort((a,b) => a[orderBy.value].localeCompare(b[orderBy.value])).reverse()
else
return trackingData.value.filter(filterItem).sort((a,b) => a[orderBy.value].localeCompare(b[orderBy.value]))
} else {
return trackingData.value.filter(filterItem)
}
})
const prefix = 'tracking'
const edititems = ref(false)
const editsearch = ref(false)
const assetsPath = useState().ASSETS_PATH
const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
const message_type = ref(MessageBoxType.NotSet)
const fixMenu = ref(false)
const show_content = ref(true)
const show_infopanel = ref(false)
const openMessageBox = ref(false)
const needSave = ref(false)
const show_input = ref(false)
const data_url_encoded = ref('')
const input_btns = ref([] as InputBtnsType[])
const input_placeholder = ref('')
const oneInputType = ref('')
const onInput = (btn: InputBtnsType) => {
switch (btn.id) {
case 'ok':
break
case 'cancel':
break
}
show_input.value = false
}
const select_ops = ref({} as ModelType)
const show_info = ref({} as ShowInfoType)
const auth_info = useState().authinfo
const useEdit = ref(true)
const onShowListBtn = () => {
show_list.value = !show_list.value
}
const resetFilter = () => {
filterData.value = {
when: '',
what: '',
where: '',
context: '',
data: '',
auth: '',
payload: {
id: '',
data: '',
uuid: '',
exp: '',
orig_iat: '',
},
idx: -1,
}
}
const refresh = () => {
show_content.value = false
setTimeout(() => show_content.value = true, 1)
}
const onCloseBtn = () => {
edititems.value = false
needSave.value = false
}
const onEditBtn = () => {
edititems.value = true
needSave.value = true
auth_info.value.editable = true
useEdit.value = true
}
const onSaveSearchBtn = () => {
// TODO save search
}
const onCloseSearchBtn = () => {
editsearch.value = false
}
const onResetSearchBtn = () => {
editsearch.value = true
resetFilter()
}
const onEditSearchBtn = () => {
editsearch.value = true
}
const onPayloadBtn = () => {
show_payload.value = !show_payload.value
if (show_payload.value) {
showField.value.payload = true
}
}
const onNavMenu = (item: { src: string, target: string }) => {
switch (item.src) {
case 'editable':
if (edititems.value) {
onCloseBtn()
} else {
onEditBtn()
}
break
case 'locale':
track_action(null, { ref: 'locale',text: i18n.locale.value}, 'locale')
refresh()
break
case 'infopanel':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
show_infopanel.value = !show_infopanel.value
break
case 'fixmenu':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
fixMenu.value = !fixMenu.value
break
case 'endinput':
show_input.value = false
break
case 'select':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
message_type.value = MessageBoxType.Select
openMessageBox.value = true
break
case 'save':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
input_placeholder.value = `${t('name', 'Name')} o 'config'`
show_input.value = true
message_type.value = MessageBoxType.Save
data_url_encoded.value = ''
openMessageBox.value = true
break
case 'editable':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.editable = !useState().authinfo.value.editable
const state = useState().authinfo.value.editable ? 'on' : 'off'
const msgTyp = state === 'on' ? MessageType.Warning : MessageType.Info
show_message(msgTyp, `${t('saveload.editMode','Edit Mode')} ${t('cv.'+state,'')}`)
break
case 'viewchange':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.viewchange = !useState().authinfo.value.viewchange
break
case 'goto':
if (item.target) {
const dom_id = document.getElementById(item.target)
if (dom_id) {
track_action(null, { ref: 'gotp',text: item.target}, 'goto')
dom_id.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
setTimeout(() => window.scrollBy(0, -40), 4000)
}
}
break
}
}
const saveData = async(mode: string,val: string) => {
const url = useState().CONFURLS.value.sendsrvconfig || ''
message_type.value = MessageBoxType.Save
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
const data: any = {
id: payload.id ? payload.id : '',
val: val.length > 0 ? val : 'config',
}
if (mode == 'send' && url.length > 0) {
onMessageBox({src: 'done', val: ''})
const [res,err] = await send_data(url, data, true, true)
if (err && err.length > 0 ) {
show_message(MessageType.Error, `Error: ${err}`,2000)
} else if (res && res.status && res.status === 'ok') {
track_action(null, { ref: 'saveData',text: `${url} -> ${val}`})
show_message(MessageType.Success, `${t('saveload.dataSaved','Data saved')}`,2000,() => {
edititems.value = false
needSave.value = false
})
}
} else {
track_action(null, { ref: 'saveData',text: `local_json -> ${val}`})
show_message(MessageType.Warning, `${t('saveload.saveLocalFile','Save Data to Local File')}`,2000,() => {
data_url_encoded.value = `text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(data))}`
})
}
}
const onMessageBox = (item: { src: string, val: string }) => {
switch (item.src) {
case 'savedata':
show_input.value = false
saveData('local', item.val)
break
case 'sendata':
show_input.value = false
saveData('send', item.val)
break
case 'oneinput':
const r = async() => {
if (await checkUserAuth(item.val)) {
oneInputType.value='text'
openMessageBox.value = false
if (!await load_currentRoute(router.currentRoute.value)) {
show_message(MessageType.Error, `${t('saveload.loaderror','Load error')}`,2000)
}
} else {
show_message(MessageType.Error, `${t('saveload.autherror','Auth error')}`,2000)
}
}
r()
break
case 'endinput':
show_input.value = false
break
case 'done':
openMessageBox.value = false
break
case 'open':
openMessageBox.value = true
break
}
}
const onLoadModel = (model: { id: string}) => {
const url = useState().CONFURLS.value.data || ''
if ( url.length > 0) {
return
}
if (useState().selectOps.value[model.id]) {
load_data_model(model.id, url, routeKy as string,true,() => {
show_content.value = false
show_message(MessageType.Success, `${t('saveload.dataLoaded','Data loaded')}`,2000,() => {
show_content.value = true
})
})
}
}
const onField = (field: string) => {
orderBy.value = orderBy.value === field ? '': field
}
const onToggleView = (field: string) => {
showField.value[field] = !showField.value[field]
}
const onToggelSort = () => {
sortMode.value = sortMode.value === '' ? 'descending' : ''
}
const onClickItem = (item: TrackActionType, pos: number) => {
// if (item && item.auth && item.auth.length > 0) {
// const payload = parseJwt(item.auth)
// }
}
const auth_get_value = (ky: string, itm: AuthPayloadType):string => {
let value = ''
switch(ky) {
case 'exp':
case 'orig_iat':
value = format_date(parseInt(`${itm[ky]}`)*1000)
break
default:
value=itm[ky] ? itm[ky] : ''
}
return value
}
const get_tracking_data = async (target: string): Promise<[any,string]> => {
const url = useState().CONFURLS.value.tracking || ''
if (url.length == 0)
return [{},'No SRV URL defined']
const urlpath = `${url}/${target}`
const [res,errmsg] = await fetch_json(urlpath, 2000,true)
if (errmsg.length > 0) {
return [{},errmsg]
}
return [res,errmsg]
}
const load_tracking_data = async (target: string): Promise<[any,string]> => {
const [res,errmsg] = await get_tracking_data(target)
if (errmsg && errmsg.length > 0) {
show_content.value = false
} else {
trackingData.value = res.map((itm: TrackActionType,idx: number) => {
itm.payload = parseJwt(itm.auth)
itm.idx = idx
return itm
})
}
resetFilter()
return [res,errmsg]
// track_action(null, { ref: 'tracking',text: 'tracking data'}, 'load')
}
const onClickItemList = async (item: string, index: Number) => {
await load_tracking_data(item)
}
onBeforeMount(async() => {
useState().dataSections.value = [`sec_${prefix}_list`,`sec_${prefix}_data`]
const url = useState().CONFURLS.value.trackinglist || ''
if (url.length == 0)
return [{},'No SRV URL defined']
const [res,errmsg] = await fetch_json(url, 2000,true)
if (errmsg.length > 0) {
show_content.value = false
} else {
show_list.value = true
show_content.value = true
trackingList.value = res
}
})
</script>

596
src/views/Users.vue Normal file
View File

@ -0,0 +1,596 @@
<template>
<div
v-if="openMessageBox"
class="fixed inset-0 bg-gray-600 overflow-y-auto h-full w-full z-80 dark:bg-gray-900 opacity-97"
>
<message-box-view
:messageType="message_type"
:openMessageBox="openMessageBox"
:show_input="show_input"
:input_placeholder="input_placeholder"
:input_btns="input_btns"
:select_ops="select_ops"
:data_url_encoded="data_url_encoded"
:inpType="oneInputType"
@onInput="onInput"
@onMessageBox="onMessageBox"
@onLoadModel="onLoadModel"
/>
</div>
<div v-if="show_content && usersData" class="font-sans antialiased">
<div class="container mx-auto max-w-screen-2xl main-container">
<nav-menu
class="bg-light-300 dark:bg-gray-900"
:position="NavPosition.header"
:openMessageBox="openMessageBox"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
:prefix="prefix"
@onNavMenu="onNavMenu"
/>
<main
:id="`${prefix}-wrapper`"
class="rounded flex flex-col sm:flex-row-reverse shadow-2xl"
:class="{ 'mt-11': fixMenu }"
>
<div
v-if="show_infopanel"
id="sidebar"
class="rounded-r w-full lg:w-60 sm:max-w-sm p-8 border-l-1 border-indigo-200 bg-gradient-to-b from-indigo-300 via-indigo-200 to-indigo-100 dark:bg-gray-600 dark:from-indigo-700 dark:via-indigo-600 dark:to-indigo-800"
>
</div>
<div class="content w-full p-5">
<div v-for="(sec,idx) in dataSections" :key="idx" class="flex justify-center border-gray-300 dark:border-gray-500 border-b-1 py-2 my-1">
<div :id="`${prefix}-${sec}`" class="mb-3 w-full">
<h2 class="section-headline noprint" >
<span v-if="show_data[sec]"> {{t(`users.${sec.replace('sec_','')}`,sec)}} </span>
<button
v-if="show_data[sec]"
class="no-print text-sm float-right icon-btn mt-2 ml-2 !outline-none text-gray-500 dark:text-gray-400"
@click="onHelp(sec)"
>
<div i-carbon-help/>
</button>
<button
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400"
@click.prevent="onDataView(sec)"
>
<div v-if="show_data[sec]" i-carbon-view-off />
<div v-else class="noprint flex">
<div class="-mt-0.5 mr-2 line-through">{{t(`menu.${sec}`,sec)}}</div>
<div i-carbon-view />
</div>
</button>
<button
v-if="show_data[sec]"
class="no-print text-sm float-right icon-btn mt-2 !outline-none text-gray-500 dark:text-gray-400 mr-2"
@click="onSaveUser(sec)"
>
<div i-carbon-checkmark-outline/>
</button>
</h2>
<div v-if="varsUser[sec] && show_data[sec]" v-for="(item) in Object.keys(varsUser[sec])">
<label
:for="`input-${sec}-${item}`" class="form-label inline-block mb-2 text-gray-700 dark:text-gray-500">
{{t(`menu.${sec}`,item)}}
</label>
<div v-if="varsUser[sec][item].vars">
<DataInput
:dataVars="varsUser[sec][item].vars"
:section="sec"
:typ="varsUser[sec][item].typ"
:dataValue="getDataFromItem(item) as Object"
:varname="item"
:vardef="varsUser[sec][item]"
:title="`${sec}`"
:labelitms="varsUser[sec][item].labelitms? varsUser[sec][item].labelitms : ''"
:edititem="edititems"
:showitem="show_data[sec]"
:validate="varsUser[sec][item].valid ? varsUser[sec][item].valid : ''"
:messages="true"
@input-change="onInputChange"
/>
<div v-if="varsUser[sec][item].from">
<button
v-if="show_data[sec]"
class="flex no-print text-sm float-right icon-btn mt-2 ml-2 !outline-none text-gray-500 dark:text-gray-400"
@click="onViewInput(sec)"
>
<div class="mr-2 mb-2">{{t(`menu.${sec}`,sec)}}</div>
<carbon-edit />
</button>
<InputValue
v-if="varsUser[sec][item].from && show_inputText[sec]"
:section="sec"
:typ="varsUser[sec][item].from.typ"
:dataValue="usersData[item] as string"
:varname="item"
:vardef="varsUser[sec][item]"
:title="`${prefix}.${item}`"
:rows="varsUser[sec][item].from.rows"
:labelitms="varsUser[sec][item].from.labelitms? varsUser[sec][item].from.labelitms : ''"
:edititem="edititems"
:showitem="show_data[sec]"
:validate="varsUser[sec][item].from.valid ? varsUser[sec][item].from.valid : ''"
:messages="true"
@input-change="onInputValueChange"
/>
</div>
</div>
<div v-if=" varsUser[sec][item].typ !== InputValueOption.object">
<InputValue
:section="sec"
:typ="varsUser[sec][item].typ"
:dataValue="getDataFromItem(item) as string"
:varname="item"
:vardef="varsUser[sec][item]"
:title="`${prefix}.${item}`"
:rows="varsUser[sec][item].rows"
:labelitms="varsUser[sec][item].labelitms? varsUser[sec][item].labelitms : ''"
:edititem="edititems"
:showitem="show_data[sec]"
:validate="varsUser[sec][item].valid ? varsUser[sec][item].valid : ''"
:messages="true"
@input-change="onInputChange"
/>
</div>
<div v-if="show_help[sec]" class="my-2 text-gray-600 dark:text-gray-500" :class="{'mt-8 border-gray-200 dark:border-gray-600 border-t-1 pt-2': !show_inputText[sec]}">
<span v-html="t(`users.msg_${sec.replace('sec_','')}`,'').replaceAll('passwdEnc',useState().PASSWD_ENC.value)" />
</div>
</div>
</div>
</div>
</div>
</main>
<div
v-if="show_content"
class="mr-auto w-full lg:w-1/2 text-center text-sm mx-2 text-gray-600 border-gray-300 border-1 border-b-0 rounded-t-xl"
>
<nav-menu
class="shadow-xl bg-light-300 dark:bg-gray-900"
:position="NavPosition.footer"
:fixMenu="fixMenu"
:show_infopanel="show_infopanel"
:dataAuth="false"
:useInfoPanel="false"
:useLogin="true"
:showinfo="show_info"
:authinfo="auth_info"
:needSave="needSave"
:useEdit="true"
prefix="sec"
:openMessageBox="openMessageBox"
@onNavMenu="onNavMenu"
/>
</div>
</div>
</div>
<div v-else class="text-center m-auto">
<img v-if="isDark"
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_b.svg`"
/>
<img v-else
class="m-auto mt-5"
width="250"
:src="`${assetsPath}/images/cvgen_w.svg`"
/>
<h2 class="mt-8 text-2xl text-gray-700 dark:text-gray-400">{{ t('message.loading', 'Loading') }}...</h2>
<h4>{{useState().connection.value.state}}</h4>
</div>
</template>
<script setup lang="ts">
import { useI18n } from 'vue-i18n'
const i18n = useI18n()
const { t } = useI18n()
const router = useRouter()
import YAML from 'yaml'
import DataInput from '@/forms/DataInput.vue'
import InputValue from '@/InputValue.vue'
import MessageBoxView from '~/components/MessageBoxView.vue'
import NavMenu from '@/NavMenu.vue'
import { isDark} from '~/composables'
import useState from '~/hooks/useState'
import { parseJwt,checkUserAuth } from '~/hooks/utilsAuth'
import { load_data_model,load_currentRoute} from '~/hooks/loads'
import { fetch_json } from '~/hooks/utils'
import { track_action } from '~/hooks/tracking'
import { MessageType } from '~/typs'
import { UsersDataType, UsersDataViewType, UsersInvitationsType } from '~/typs/users'
import { MessageBoxType, NavPosition, InputBtnsType, ShowInfoType, ModelType } from '~/typs/cv'
import { InputValueOption } from '~/typs/inputs'
import { UsersAccountsType, ModelsUsersType } from '~/typs/users'
import { show_message,send_data } from '~/hooks/utils'
import { b64_to_utf8,utf8_to_b64 } from '~/hooks/tools'
const dataSections = useState().dataSections
const dataKeys = ['usersData','modelsData','invitations','authzModel','authzPolicy']
const usersData= ref({} as UsersDataType)
const varsUser = useState().varsUser
const show_data = ref({} as UsersDataViewType)
const show_help = ref({} as UsersDataViewType)
const usersAccounts = ref({} as UsersAccountsType)
const modelsUsers = ref({} as ModelsUsersType)
const invitations = ref({} as UsersInvitationsType)
const prefix = 'users'
const edititems = ref(false)
const show_inputText = ref({} as UsersDataViewType)
const assetsPath = useState().ASSETS_PATH
const routeKy = router.currentRoute.value.params.ky || router.currentRoute.value.query.k || ''
const message_type = ref(MessageBoxType.NotSet)
const fixMenu = ref(false)
const show_content = ref(true)
const show_infopanel = ref(false)
const openMessageBox = ref(false)
const needSave = ref(false)
const show_input = ref(false)
const data_url_encoded = ref('')
const input_btns = ref([] as InputBtnsType[])
const input_placeholder = ref('')
const oneInputType = ref('')
const onInputChange = (data: {sec: string, key: string, idx:number, val: any}) => {
if (data.key && data.val) {
switch (data.key) {
case 'usersData':
usersAccounts.value = data.val
usersData.value.usersData = YAML.stringify(data.val)
break
case 'modelsData':
modelsUsers.value= data.val
usersData.value.modelsData = YAML.stringify(data.val)
break
case 'invitations':
invitations.value= data.val
usersData.value.invitations = YAML.stringify(data.val)
break
case 'authzModel':
usersData.value.authzModel = data.val
break
case 'authzPolicy':
usersData.value.authzPolicy = data.val
break
}
}
}
const onInputValueChange = (data: {sec: string, key: string, idx:number, val: string}) => {
if (data.key && data.val && data.val.length ) {
switch (data.key) {
case 'usersData':
usersData.value.usersData = data.val
usersAccounts.value = YAML.parse(data.val)
break
case 'modelsData':
usersData.value.modelsData = data.val
modelsUsers.value = YAML.parse(data.val)
break
case 'invitations':
usersData.value.invitations = data.val
invitations.value = YAML.parse(data.val)
break
case 'authzModel':
usersData.value.authzModel = data.val
break
case 'authzPolicy':
usersData.value.authzPolicy = data.val
break
}
}
}
const onInput = (btn: InputBtnsType) => {
switch (btn.id) {
case 'ok':
break
case 'cancel':
break
}
show_input.value = false
}
const getDataFromItem= (item: string):any => {
switch (item) {
case 'usersData':
return usersAccounts.value
case 'modelsData':
return modelsUsers.value
case 'invitations':
return invitations.value
case 'authzModel':
return usersData.value.authzModel
case 'authzPolicy':
return usersData.value.authzPolicy
default:
return {}
}
}
const select_ops = ref({} as ModelType)
const show_info = ref({} as ShowInfoType)
const auth_info = useState().authinfo
const useEdit = ref(true)
const refresh = () => {
show_content.value = false
setTimeout(() => show_content.value = true, 1)
}
const onCloseBtn = () => {
edititems.value = false
needSave.value = false
}
const onEditBtn = () => {
edititems.value = true
needSave.value = true
auth_info.value.editable = true
useEdit.value = true
}
const onNavMenu = (item: { src: string, target: string }) => {
switch (item.src) {
case 'editable':
if (edititems.value) {
onCloseBtn()
} else {
onEditBtn()
}
break
case 'locale':
track_action(null, { ref: 'locale',text: i18n.locale.value}, 'locale')
refresh()
break
case 'infopanel':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
show_infopanel.value = !show_infopanel.value
break
case 'fixmenu':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
fixMenu.value = !fixMenu.value
break
case 'endinput':
show_input.value = false
break
case 'select':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
message_type.value = MessageBoxType.Select
openMessageBox.value = true
break
case 'save':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
input_placeholder.value = `${t('name', 'Name')} o 'config'`
show_input.value = true
message_type.value = MessageBoxType.Save
data_url_encoded.value = ''
openMessageBox.value = true
break
case 'editable':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.editable = !useState().authinfo.value.editable
const state = useState().authinfo.value.editable ? 'on' : 'off'
const msgTyp = state === 'on' ? MessageType.Warning : MessageType.Info
show_message(msgTyp, `${t('saveload.editMode','Edit Mode')} ${t('cv.'+state,'')}`)
break
case 'viewchange':
track_action(null, { ref: 'onNavMenu',text: 'item.src'})
useState().authinfo.value.viewchange = !useState().authinfo.value.viewchange
break
case 'goto':
if (item.target) {
const dom_id = document.getElementById(item.target)
if (dom_id) {
track_action(null, { ref: 'gotp',text: item.target}, 'goto')
dom_id.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
setTimeout(() => window.scrollBy(0, -40), 4000)
}
}
break
}
}
const saveData = async(mode: string,val: string, newData?: UsersDataType ) => {
const url = useState().CONFURLS.value.sendusers || ''
message_type.value = MessageBoxType.Save
const payload = parseJwt(localStorage.getItem(useState().AUTHKEY.value))
const postdata: any = {
id: payload.id ? payload.id : '',
val: val.length > 0 ? val : 'usersData',
data: newData ? newData : {...usersData.value},
}
Object.keys(postdata.data).forEach(ky => {
if (postdata.data[ky]) {
postdata.data[ky]= utf8_to_b64(postdata.data[ky])
}
})
if (mode == 'send' && url.length > 0) {
onMessageBox({src: 'done', val: ''})
const [res,err] = await send_data(url, postdata, true, true)
if (err && err.length > 0 ) {
show_message(MessageType.Error, `Error: ${err}`,2000)
} else if (res && res.status && res.status === 'ok') {
track_action(null, { ref: 'saveData',text: `${url} -> ${val}`})
show_message(MessageType.Success, `${t('saveload.dataSaved','Data saved')}`,2000,() => {
edititems.value = false
needSave.value = false
})
}
} else {
track_action(null, { ref: 'saveData',text: `local_json -> ${val}`})
show_message(MessageType.Warning, `${t('saveload.saveLocalFile','Save Data tLocal File')}`,2000,() => {
data_url_encoded.value = `text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(postdata))}`
})
}
}
const onMessageBox = (item: { src: string, val: string }) => {
switch (item.src) {
case 'savedata':
show_input.value = false
saveData('local', item.val)
break
case 'sendata':
show_input.value = false
saveData('send', item.val)
break
case 'oneinput':
const r = async() => {
if (await checkUserAuth(item.val)) {
oneInputType.value='text'
openMessageBox.value = false
if (!await load_currentRoute(router.currentRoute.value)) {
show_message(MessageType.Error, `${t('saveload.loaderror','Load error')}`,2000)
}
} else {
show_message(MessageType.Error, `${t('saveload.autherror','Auth error')}`,2000)
}
}
r()
break
case 'endinput':
show_input.value = false
break
case 'done':
openMessageBox.value = false
break
case 'open':
openMessageBox.value = true
break
}
}
const onLoadModel = (model: { id: string}) => {
const url = useState().CONFURLS.value.data || ''
if ( url.length > 0) {
return
}
if (useState().selectOps.value[model.id]) {
load_data_model(model.id, url, routeKy as string,true,() => {
show_content.value = false
show_message(MessageType.Success, `${t('saveload.dataLoaded','Data loaded')}`,2000,() => {
show_content.value = true
})
})
}
}
const onDataView = (ky: string) => {
show_data.value[ky] = ! show_data.value[ky]
}
const onHelp = (ky: string) => {
show_help.value[ky] = ! show_help.value[ky]
}
const onViewInput = (ky: string) => {
show_inputText.value[ky] = ! show_inputText.value[ky]
}
const onSaveUser = (ky: string) => {
const key = ky.replace('sec_','')
let data = {} as UsersDataType
if (usersData.value[key]) {
data[key] = usersData.value[key]
} else {
data = usersData.value
}
saveData('send',ky ,data)
}
const load_vars_user = async (): Promise<[any,string]> => {
const url = useState().CONFURLS.value.varsuser || ''
if (url.length == 0)
return [{},'No URL defined']
let text = ''
let res: any = {}
let error = null
try {
const response = await self.fetch(url, {
method: 'GET',
})
if (url.includes('.json')) {
res = await response.json()
} else {
text = await response.text()
}
if (response.ok) {
if (url.includes('.yaml')) {
res = YAML.parse(text)
} else if (!url.includes('.json')) {
res = text
}
} else {
error = res.message
}
}
catch (err: any) {
error = err.message
}
if (error) {
show_message(MessageType.Error, `'Vars User Load' -> ${error}`, 5000)
useState().connection.value.state = 'connection.error'
return [{}, error]
}
return [res,error]
}
const get_users_data = async (): Promise<[any,string]> => {
const url = useState().CONFURLS.value.users || ''
if (url.length == 0)
return [{},'No SRV URL defined']
const [res,errmsg] = await fetch_json(url, 2000,true)
if (errmsg.length > 0) {
return [{},errmsg]
}
return [res,errmsg]
}
onBeforeMount(async() => {
//if (Object.keys(useState().varsUser.value).length === 0) {
const [resvars, err] = await load_vars_user()
if (err && err.length > 0) {
show_content.value = false
} else {
varsUser.value = resvars
useState().dataSections.value = []
Object.keys(resvars).forEach(ky => {
useState().dataSections.value.push(ky)
show_data.value[ky] = true
show_inputText.value[ky] = false
show_help.value[ky] = false
})
}
//}
const [res,errmsg] = await get_users_data()
if (errmsg && errmsg.length > 0) {
debugger
show_content.value = false
} else {
usersData.value = {} as UsersDataType
usersAccounts.value = {} as UsersAccountsType
invitations.value = {} as UsersInvitationsType
modelsUsers.value = {} as ModelsUsersType
if (res.data) {
dataKeys.forEach(ky => {
if (res.data[ky]) {
usersData.value[ky]= b64_to_utf8(res.data[ky])
try {
switch(ky) {
case 'usersData':
usersAccounts.value = YAML.parse(usersData.value[ky])
break
case 'modelsData':
modelsUsers.value = YAML.parse(usersData.value[ky])
break
case 'invitations':
invitations.value = YAML.parse(usersData.value[ky])
break
}
} catch(e) {
show_message(MessageType.Error, `'Error parse data' -> ${e}`, 5000)
useState().connection.value.state = 'data.error'
}
} else {
show_data.value[ky] = false
}
})
}
}
track_action(null, { ref: 'users',text: 'users data'}, 'mount')
})
</script>