2025-03-22 10:19:15 +00:00

667 lines
15 KiB
JavaScript

import Localization from '@/mixins/Localization'
import { Form } from '@/util/FormValidation'
import { setupAxios } from '@/bootstrap/axios'
import { setupCodeMirror } from '@/bootstrap/codemirror'
import { setupInertia } from '@/bootstrap/inertia'
import { setupNumbro } from '@/bootstrap/numbro'
import url from '@/util/url'
import { createApp, h } from 'vue'
import { hideProgress, revealProgress } from '@inertiajs/core'
import { createInertiaApp, Head, Link, router } from '@inertiajs/vue3'
import { registerViews } from './components'
import { registerFields } from './fields'
import Mousetrap from 'mousetrap'
import { createNovaStore } from './store'
import resourceStore from './store/resources'
import NProgress from 'nprogress'
import FloatingVue from 'floating-vue'
import camelCase from 'lodash/camelCase'
import fromPairs from 'lodash/fromPairs'
import isString from 'lodash/isString'
import omit from 'lodash/omit'
import upperFirst from 'lodash/upperFirst'
import Toasted from 'toastedjs'
import Emitter from 'tiny-emitter'
import Layout from '@/layouts/AppLayout'
import { Settings } from 'luxon'
import { ColorTranslator } from 'colortranslator'
const { parseColor } = require('tailwindcss/lib/util/color')
setupCodeMirror()
const emitter = new Emitter()
/**
* @typedef {import('vuex').Store} VueStore
* @typedef {import('vue').App} VueApp
* @typedef {import('vue').Component} VueComponent
* @typedef {import('vue').DefineComponent} DefineComponent
* @typedef {import('axios').AxiosInstance} AxiosInstance
* @typedef {import('axios').AxiosRequestConfig} AxiosRequestConfig
* @typedef {Object<string, any>} AppConfig
* @typedef {import('./util/FormValidation').Form} Form
* @typedef {(app: VueApp, store: VueStore) => void} BootingCallback
*/
export default class Nova {
/**
* @param {AppConfig} config
*/
constructor(config) {
/**
* @protected
* @type {Array<BootingCallback>}
*/
this.bootingCallbacks = []
/** @readonly */
this.appConfig = config
/**
* @private
* @type {boolean}
*/
this.useShortcuts = true
/**
* @protected
* @type {{[key: string]: VueComponent|DefineComponent}}
*/
this.pages = {
'Nova.Attach': require('@/pages/Attach').default,
'Nova.ConfirmPassword': require('@/pages/ConfirmPassword').default,
'Nova.Create': require('@/pages/Create').default,
'Nova.Dashboard': require('@/pages/Dashboard').default,
'Nova.Detail': require('@/pages/Detail').default,
'Nova.EmailVerification': require('@/pages/EmailVerification').default,
'Nova.UserSecurity': require('@/pages/UserSecurity').default,
'Nova.Error': require('@/pages/AppError').default,
'Nova.Error403': require('@/pages/Error403').default,
'Nova.Error404': require('@/pages/Error404').default,
'Nova.ForgotPassword': require('@/pages/ForgotPassword').default,
'Nova.Index': require('@/pages/Index').default,
'Nova.Lens': require('@/pages/Lens').default,
'Nova.Login': require('@/pages/Login').default,
'Nova.Replicate': require('@/pages/Replicate').default,
'Nova.ResetPassword': require('@/pages/ResetPassword').default,
'Nova.TwoFactorChallenge': require('@/pages/TwoFactorChallenge').default,
'Nova.Update': require('@/pages/Update').default,
'Nova.UpdateAttached': require('@/pages/UpdateAttached').default,
}
/** @protected */
this.$toasted = new Toasted({
theme: 'nova',
position: config.rtlEnabled ? 'bottom-left' : 'bottom-right',
duration: 6000,
})
/** @public */
this.$progress = NProgress
/** @public */
this.$router = router
if (config.debug === true) {
/** @readonly */
this.$testing = {
timezone: timezone => {
Settings.defaultZoneName = timezone
},
}
}
/** @private */
this.__started = false
/** @private */
this.__booted = false
/** @private */
this.__liftOff = false
}
/**
* Register a callback to be called before Nova starts. This is used to bootstrap
* addons, tools, custom fields, or anything else Nova needs
*
* @param {BootingCallback} callback
*/
booting(callback) {
this.bootingCallbacks.push(callback)
}
/**
* Execute all of the booting callbacks.
*/
boot() {
if (!this.__started || !this.__liftOff || this.__booted) {
return
}
this.debug('engage thrusters')
/** @type {VueStore} */
this.store = createNovaStore()
this.bootingCallbacks.forEach(callback => callback(this.app, this.store))
this.bootingCallbacks = []
this.registerStoreModules()
this.app.mixin(Localization)
setupInertia(this, this.store)
this.app.mixin({
methods: {
$url: (path, parameters) => this.url(path, parameters),
},
})
this.component('Link', Link)
this.component('InertiaLink', Link)
this.component('Head', Head)
registerViews(this)
registerFields(this)
this.app.mount(this.mountTo)
let mousetrapDefaultStopCallback = Mousetrap.prototype.stopCallback
Mousetrap.prototype.stopCallback = (e, element, combo) => {
if (!this.useShortcuts) {
return true
}
return mousetrapDefaultStopCallback.call(this, e, element, combo)
}
Mousetrap.init()
this.applyTheme()
this.log('All systems go...')
this.__booted = true
}
/**
* @param {BootingCallback} callback
*/
booted(callback) {
callback(this.app, this.store)
}
async countdown() {
this.log('Initiating Nova countdown...')
const appName = this.config('appName')
await createInertiaApp({
title: title => (!title ? appName : `${title} - ${appName}`),
progress: false,
resolve: name => {
const page =
this.pages[name] != null
? this.pages[name]
: require('@/pages/Error404').default
page.layout = page.layout || Layout
return page
},
setup: ({ el, App, props, plugin }) => {
this.debug('engine start')
/** @protected */
this.mountTo = el
/**
* @protected
* @type VueApp
*/
this.app = createApp({ render: () => h(App, props) })
this.app.use(plugin)
this.app.use(FloatingVue, {
preventOverflow: true,
flip: true,
themes: {
Nova: {
$extend: 'tooltip',
triggers: ['click'],
autoHide: true,
placement: 'bottom',
html: true,
},
},
})
},
}).then(() => {
this.__started = true
this.debug('engine ready')
this.boot()
})
}
/**
* Start the Nova app by calling each of the tool's callbacks and then creating
* the underlying Vue instance.
*/
liftOff() {
this.log('We have lift off!')
let currentTheme = null
new MutationObserver(() => {
const element = document.documentElement.classList
const theme = element.contains('dark') ? 'dark' : 'light'
if (theme !== currentTheme) {
this.$emit('nova-theme-switched', {
theme,
element,
})
currentTheme = theme
}
}).observe(document.documentElement, {
attributes: true,
attributeOldValue: true,
attributeFilter: ['class'],
})
if (this.config('notificationCenterEnabled')) {
/** @private */
this.notificationPollingInterval = setInterval(() => {
if (document.hasFocus()) {
this.$emit('refresh-notifications')
}
}, this.config('notificationPollingInterval'))
}
this.__liftOff = true
this.boot()
}
/**
* Return configuration value from a key.
*
* @param {string} key
* @returns {any}
*/
config(key) {
return this.appConfig[key]
}
/**
* Return a form object configured with Nova's preconfigured axios instance.
*
* @param {{[key: string]: any}} data
* @returns {Form}
*/
form(data) {
return new Form(data, {
http: this.request(),
})
}
/**
* Return an axios instance configured to make requests to Nova's API
* and handle certain response codes.
*
* @param {AxiosRequestConfig|null} [options=null]
* @returns {AxiosInstance}
*/
request(options = null) {
/** @type AxiosInstance */
let axios = setupAxios()
if (options != null) {
return axios(options)
}
return axios
}
/**
* Get the URL from base Nova prefix.
*
* @param {string} path
* @param {any} parameters
* @returns {string}
*/
url(path, parameters) {
if (path === '/') {
path = this.config('initialPath')
}
return url(this.config('base'), path, parameters)
}
/**
* @returns {boolean}
*/
hasSecurityFeatures() {
const features = this.config('fortifyFeatures')
return (
features.includes('update-passwords') ||
features.includes('two-factor-authentication')
)
}
/**
* Register a listener on Nova's built-in event bus
*
* @param {string} name
* @param {Function} callback
* @param {any} ctx
*/
$on(...args) {
emitter.on(...args)
}
/**
* Register a one-time listener on the event bus
*
* @param {string} name
* @param {Function} callback
* @param {any} ctx
*/
$once(...args) {
emitter.once(...args)
}
/**
* Unregister an listener on the event bus
*
* @param {string} name
* @param {Function} callback
*/
$off(...args) {
emitter.off(...args)
}
/**
* Emit an event on the event bus
*
* @param {string} name
*/
$emit(...args) {
emitter.emit(...args)
}
/**
* Determine if Nova is missing the requested resource with the given uri key
*
* @param {string} uriKey
* @returns {boolean}
*/
missingResource(uriKey) {
return this.config('resources').find(r => r.uriKey === uriKey) == null
}
/**
* Register a keyboard shortcut.
*
* @param {string} keys
* @param {Function} callback
*/
addShortcut(keys, callback) {
Mousetrap.bind(keys, callback)
}
/**
* Unbind a keyboard shortcut.
*
* @param {string} keys
*/
disableShortcut(keys) {
Mousetrap.unbind(keys)
}
/**
* Pause all keyboard shortcuts.
*/
pauseShortcuts() {
this.useShortcuts = false
}
/**
* Resume all keyboard shortcuts.
*/
resumeShortcuts() {
this.useShortcuts = true
}
/**
* Register the built-in Vuex modules for each resource
*/
registerStoreModules() {
this.app.use(this.store)
this.config('resources').forEach(resource => {
this.store.registerModule(resource.uriKey, resourceStore)
})
}
/**
* Register Inertia component.
*
* @param {string} name
* @param {VueComponent|DefineComponent} component
*/
inertia(name, component) {
this.pages[name] = component
}
/**
* Register a custom Vue component.
*
* @param {string} name
* @param {VueComponent|DefineComponent} component
*/
component(name, component) {
if (this.app._context.components[name] == null) {
this.app.component(name, component)
}
}
/**
* Check if custom Vue component exists.
*
* @param {string} name
* @returns {boolean}
*/
hasComponent(name) {
return Boolean(
this.app._context.components[upperFirst(camelCase(name))] != null
)
}
/**
* Show an error message to the user.
*
* @param {string} message
*/
info(message) {
this.$toasted.show(message, { type: 'info' })
}
/**
* Show an error message to the user.
*
* @param {string} message
*/
error(message) {
this.$toasted.show(message, { type: 'error' })
}
/**
* Show a success message to the user.
*
* @param {string} message
*/
success(message) {
this.$toasted.show(message, { type: 'success' })
}
/**
* Show a warning message to the user.
*
* @param {string} message
*/
warning(message) {
this.$toasted.show(message, { type: 'warning' })
}
/**
* Format a number using numbro.js for consistent number formatting.
*
* @param {number} number
* @param {Object|string} format
* @returns {string}
*/
formatNumber(number, format) {
const numbro = setupNumbro(
document.querySelector('meta[name="locale"]').content
)
const num = numbro(number)
if (format !== undefined) {
return num.format(format)
}
return num.format()
}
/**
* Log a message to the console with the NOVA prefix
*
* @param {string} message
* @param {string} [type=log]
*/
log(message, type = 'log') {
console[type](`[NOVA]`, message)
}
/**
* Log a message to the console for debugging purpose
*
* @param {any} message
* @param {string} [type=log]
*/
debug(message, type = 'log') {
const debugEnabled =
process.env.NODE_ENV === true || (this.config('debug') ?? false)
if (debugEnabled === true) {
if (type === 'error') {
console.error(message)
} else {
this.log(message, type)
}
}
}
/**
* Redirect to login path.
*/
redirectToLogin() {
const url =
!this.config('withAuthentication') && this.config('customLoginPath')
? this.config('customLoginPath')
: this.url('/login')
this.visit({
remote: true,
url,
})
}
/**
* Visit page using Inertia visit or window.location for remote.
*
* @param {{url: string, remote: boolean} | string} path
* @param {any} [options={}]
*/
visit(path, options = {}) {
options = options
const openInNewTab = options?.openInNewTab || null
if (isString(path)) {
router.visit(this.url(path), omit(options, ['openInNewTab']))
return
}
if (isString(path.url) && path.hasOwnProperty('remote')) {
if (path.remote === true) {
if (openInNewTab === true) {
window.open(path.url, '_blank')
} else {
window.location = path.url
}
return
}
router.visit(path.url, omit(options, ['openInNewTab']))
}
}
applyTheme() {
const brandColors = this.config('brandColors')
if (Object.keys(brandColors).length > 0) {
const style = document.createElement('style')
// Handle converting any non-RGB user strings into valid RGB strings.
// This allows the user to specify any color in HSL, RGB, and RGBA
// format, and we'll convert it to the proper format for them.
let css = Object.keys(brandColors).reduce((carry, v) => {
let colorValue = brandColors[v]
let validColor = parseColor(colorValue)
if (validColor) {
let parsedColor = parseColor(
ColorTranslator.toRGBA(convertColor(validColor))
)
let rgbaString = `${parsedColor.color.join(' ')} / ${
parsedColor.alpha
}`
return carry + `\n --colors-primary-${v}: ${rgbaString};`
}
return carry + `\n --colors-primary-${v}: ${colorValue};`
}, '')
style.innerHTML = `:root {${css}\n}`
document.head.append(style)
}
}
}
function convertColor(parsedColor) {
let color = fromPairs(
Array.from(parsedColor.mode).map((v, i) => {
return [v, parsedColor.color[i]]
})
)
if (parsedColor.alpha !== undefined) {
color.a = parsedColor.alpha
}
return color
}