2018-10-27 00:16:23 +11:00
|
|
|
import Vue from 'vue'
|
|
|
|
import VueRouter from 'vue-router'
|
2018-12-04 05:36:59 +11:00
|
|
|
import routes from './routes'
|
2018-10-27 00:16:23 +11:00
|
|
|
import App from '../App.vue'
|
2019-04-02 06:41:34 +11:00
|
|
|
import { windowWidth } from '../services/window_utils/window_utils'
|
2019-05-23 02:13:41 +10:00
|
|
|
import { getOrCreateApp, getClientToken } from '../services/new_api/oauth.js'
|
|
|
|
import backendInteractorService from '../services/backend_interactor_service/backend_interactor_service.js'
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 21:57:30 +11:00
|
|
|
const getStatusnetConfig = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/api/statusnet/config.json')
|
2019-03-13 22:41:39 +11:00
|
|
|
if (res.ok) {
|
|
|
|
const data = await res.json()
|
2019-04-03 01:26:14 +11:00
|
|
|
const { name, closed: registrationClosed, textlimit, uploadlimit, server, vapidPublicKey, safeDMMentionsEnabled } = data.site
|
2018-10-27 00:16:23 +11:00
|
|
|
|
|
|
|
store.dispatch('setInstanceOption', { name: 'name', value: name })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'registrationOpen', value: (registrationClosed === '0') })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'textlimit', value: parseInt(textlimit) })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'server', value: server })
|
2019-04-03 01:26:14 +11:00
|
|
|
store.dispatch('setInstanceOption', { name: 'safeDM', value: safeDMMentionsEnabled !== '0' })
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2018-12-19 05:26:14 +11:00
|
|
|
// TODO: default values for this stuff, added if to not make it break on
|
|
|
|
// my dev config out of the box.
|
|
|
|
if (uploadlimit) {
|
|
|
|
store.dispatch('setInstanceOption', { name: 'uploadlimit', value: parseInt(uploadlimit.uploadlimit) })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'avatarlimit', value: parseInt(uploadlimit.avatarlimit) })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'backgroundlimit', value: parseInt(uploadlimit.backgroundlimit) })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'bannerlimit', value: parseInt(uploadlimit.bannerlimit) })
|
|
|
|
}
|
|
|
|
|
2018-12-11 02:36:25 +11:00
|
|
|
if (vapidPublicKey) {
|
|
|
|
store.dispatch('setInstanceOption', { name: 'vapidPublicKey', value: vapidPublicKey })
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 22:41:39 +11:00
|
|
|
return data.site.pleromafe
|
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
2019-03-13 21:57:30 +11:00
|
|
|
} catch (error) {
|
|
|
|
console.error('Could not load statusnet config, potentially fatal')
|
|
|
|
console.error(error)
|
|
|
|
}
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 21:57:30 +11:00
|
|
|
const getStaticConfig = async () => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/static/config.json')
|
2019-03-13 22:41:39 +11:00
|
|
|
if (res.ok) {
|
|
|
|
return res.json()
|
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
2019-03-13 21:57:30 +11:00
|
|
|
} catch (error) {
|
|
|
|
console.warn('Failed to load static/config.json, continuing without it.')
|
|
|
|
console.warn(error)
|
|
|
|
return {}
|
|
|
|
}
|
|
|
|
}
|
2019-01-25 05:03:13 +11:00
|
|
|
|
2019-03-13 21:57:30 +11:00
|
|
|
const setSettings = async ({ apiConfig, staticConfig, store }) => {
|
|
|
|
const overrides = window.___pleromafe_dev_overrides || {}
|
|
|
|
const env = window.___pleromafe_mode.NODE_ENV
|
|
|
|
|
|
|
|
// This takes static config and overrides properties that are present in apiConfig
|
|
|
|
let config = {}
|
|
|
|
if (overrides.staticConfigPreference && env === 'development') {
|
|
|
|
console.warn('OVERRIDING API CONFIG WITH STATIC CONFIG')
|
|
|
|
config = Object.assign({}, apiConfig, staticConfig)
|
|
|
|
} else {
|
|
|
|
config = Object.assign({}, staticConfig, apiConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
const copyInstanceOption = (name) => {
|
|
|
|
store.dispatch('setInstanceOption', { name, value: config[name] })
|
|
|
|
}
|
|
|
|
|
|
|
|
copyInstanceOption('nsfwCensorImage')
|
|
|
|
copyInstanceOption('background')
|
|
|
|
copyInstanceOption('hidePostStats')
|
|
|
|
copyInstanceOption('hideUserStats')
|
|
|
|
copyInstanceOption('hideFilteredStatuses')
|
|
|
|
copyInstanceOption('logo')
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 21:57:30 +11:00
|
|
|
store.dispatch('setInstanceOption', {
|
|
|
|
name: 'logoMask',
|
|
|
|
value: typeof config.logoMask === 'undefined'
|
|
|
|
? true
|
|
|
|
: config.logoMask
|
|
|
|
})
|
|
|
|
|
|
|
|
store.dispatch('setInstanceOption', {
|
|
|
|
name: 'logoMargin',
|
|
|
|
value: typeof config.logoMargin === 'undefined'
|
|
|
|
? 0
|
|
|
|
: config.logoMargin
|
|
|
|
})
|
2019-06-13 06:16:55 +10:00
|
|
|
store.commit('authFlow/setInitialStrategy', config.loginMethod)
|
2019-03-13 21:57:30 +11:00
|
|
|
|
|
|
|
copyInstanceOption('redirectRootNoLogin')
|
|
|
|
copyInstanceOption('redirectRootLogin')
|
|
|
|
copyInstanceOption('showInstanceSpecificPanel')
|
2019-03-30 21:31:50 +11:00
|
|
|
copyInstanceOption('minimalScopesMode')
|
2019-03-13 21:57:30 +11:00
|
|
|
copyInstanceOption('formattingOptionsEnabled')
|
2019-03-03 00:07:14 +11:00
|
|
|
copyInstanceOption('hideMutedPosts')
|
2019-03-13 21:57:30 +11:00
|
|
|
copyInstanceOption('collapseMessageWithSubject')
|
|
|
|
copyInstanceOption('scopeCopy')
|
|
|
|
copyInstanceOption('subjectLineBehavior')
|
|
|
|
copyInstanceOption('postContentType')
|
|
|
|
copyInstanceOption('alwaysShowSubjectInput')
|
|
|
|
copyInstanceOption('noAttachmentLinks')
|
|
|
|
copyInstanceOption('showFeaturesPanel')
|
|
|
|
|
|
|
|
if ((config.chatDisabled)) {
|
|
|
|
store.dispatch('disableChat')
|
|
|
|
} else {
|
|
|
|
store.dispatch('initializeSocket')
|
|
|
|
}
|
|
|
|
|
|
|
|
return store.dispatch('setTheme', config['theme'])
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 22:10:57 +11:00
|
|
|
const getTOS = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/static/terms-of-service.html')
|
|
|
|
if (res.ok) {
|
|
|
|
const html = await res.text()
|
2018-10-27 00:16:23 +11:00
|
|
|
store.dispatch('setInstanceOption', { name: 'tos', value: html })
|
2019-03-13 22:10:57 +11:00
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.warn("Can't load TOS")
|
|
|
|
console.warn(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const getInstancePanel = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/instance/panel.html')
|
|
|
|
if (res.ok) {
|
|
|
|
const html = await res.text()
|
|
|
|
store.dispatch('setInstanceOption', { name: 'instanceSpecificPanelContent', value: html })
|
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.warn("Can't load instance panel")
|
2019-03-13 22:26:40 +11:00
|
|
|
console.warn(e)
|
|
|
|
}
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 22:26:40 +11:00
|
|
|
const getStaticEmoji = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/static/emoji.json')
|
|
|
|
if (res.ok) {
|
|
|
|
const values = await res.json()
|
2018-10-27 00:16:23 +11:00
|
|
|
const emoji = Object.keys(values).map((key) => {
|
|
|
|
return { shortcode: key, image_url: false, 'utf': values[key] }
|
|
|
|
})
|
|
|
|
store.dispatch('setInstanceOption', { name: 'emoji', value: emoji })
|
2019-03-13 22:26:40 +11:00
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.warn("Can't load static emoji")
|
|
|
|
console.warn(e)
|
|
|
|
}
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-03-13 22:26:40 +11:00
|
|
|
// This is also used to indicate if we have a 'pleroma backend' or not.
|
|
|
|
// Somewhat weird, should probably be somewhere else.
|
|
|
|
const getCustomEmoji = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/api/pleroma/emoji.json')
|
|
|
|
if (res.ok) {
|
2019-04-10 05:54:14 +10:00
|
|
|
const result = await res.json()
|
|
|
|
const values = Array.isArray(result) ? Object.assign({}, ...result) : result
|
2019-03-13 22:26:40 +11:00
|
|
|
const emoji = Object.keys(values).map((key) => {
|
2019-04-10 05:54:14 +10:00
|
|
|
return { shortcode: key, image_url: values[key].image_url || values[key] }
|
2019-03-13 22:26:40 +11:00
|
|
|
})
|
|
|
|
store.dispatch('setInstanceOption', { name: 'customEmoji', value: emoji })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'pleromaBackend', value: true })
|
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
store.dispatch('setInstanceOption', { name: 'pleromaBackend', value: false })
|
|
|
|
console.warn("Can't load custom emojis, maybe not a Pleroma instance?")
|
|
|
|
console.warn(e)
|
|
|
|
}
|
|
|
|
}
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2019-05-23 02:13:41 +10:00
|
|
|
const getAppSecret = async ({ store }) => {
|
|
|
|
const { state, commit } = store
|
|
|
|
const { oauth, instance } = state
|
|
|
|
return getOrCreateApp({ ...oauth, instance: instance.server, commit })
|
|
|
|
.then((app) => getClientToken({ ...app, instance: instance.server }))
|
|
|
|
.then((token) => {
|
|
|
|
commit('setClientToken', token.access_token)
|
|
|
|
commit('setBackendInteractor', backendInteractorService(store.getters.getToken()))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-03-13 22:26:40 +11:00
|
|
|
const getNodeInfo = async ({ store }) => {
|
|
|
|
try {
|
|
|
|
const res = await window.fetch('/nodeinfo/2.0.json')
|
|
|
|
if (res.ok) {
|
|
|
|
const data = await res.json()
|
2018-10-27 00:16:23 +11:00
|
|
|
const metadata = data.metadata
|
2018-10-27 01:13:05 +11:00
|
|
|
|
|
|
|
const features = metadata.features
|
|
|
|
store.dispatch('setInstanceOption', { name: 'mediaProxyAvailable', value: features.includes('media_proxy') })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'chatAvailable', value: features.includes('chat') })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'gopherAvailable', value: features.includes('gopher') })
|
2018-10-27 00:16:23 +11:00
|
|
|
|
2018-12-27 00:50:48 +11:00
|
|
|
store.dispatch('setInstanceOption', { name: 'restrictedNicknames', value: metadata.restrictedNicknames })
|
2019-03-15 08:35:45 +11:00
|
|
|
store.dispatch('setInstanceOption', { name: 'postFormats', value: metadata.postFormats })
|
2018-12-27 00:50:48 +11:00
|
|
|
|
2018-10-27 00:16:23 +11:00
|
|
|
const suggestions = metadata.suggestions
|
|
|
|
store.dispatch('setInstanceOption', { name: 'suggestionsEnabled', value: suggestions.enabled })
|
|
|
|
store.dispatch('setInstanceOption', { name: 'suggestionsWeb', value: suggestions.web })
|
2019-03-11 10:58:12 +11:00
|
|
|
|
|
|
|
const software = data.software
|
|
|
|
store.dispatch('setInstanceOption', { name: 'backendVersion', value: software.version })
|
|
|
|
|
|
|
|
const frontendVersion = window.___pleromafe_commit_hash
|
|
|
|
store.dispatch('setInstanceOption', { name: 'frontendVersion', value: frontendVersion })
|
2019-02-19 01:49:32 +11:00
|
|
|
store.dispatch('setInstanceOption', { name: 'tagPolicyAvailable', value: metadata.federation.mrf_policies.includes('TagPolicy') })
|
2019-03-13 22:26:40 +11:00
|
|
|
} else {
|
|
|
|
throw (res)
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.warn('Could not load nodeinfo')
|
|
|
|
console.warn(e)
|
2019-03-13 22:10:57 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-02 04:46:30 +11:00
|
|
|
const setConfig = async ({ store }) => {
|
|
|
|
// apiConfig, staticConfig
|
|
|
|
const configInfos = await Promise.all([getStatusnetConfig({ store }), getStaticConfig()])
|
|
|
|
const apiConfig = configInfos[0]
|
|
|
|
const staticConfig = configInfos[1]
|
|
|
|
|
2019-05-23 02:13:41 +10:00
|
|
|
await setSettings({ store, apiConfig, staticConfig }).then(getAppSecret({ store }))
|
2019-04-02 04:46:30 +11:00
|
|
|
}
|
|
|
|
|
2019-04-02 06:32:13 +11:00
|
|
|
const checkOAuthToken = async ({ store }) => {
|
|
|
|
return new Promise(async (resolve, reject) => {
|
2019-06-13 07:39:51 +10:00
|
|
|
if (store.state.oauth.userToken) {
|
2019-04-02 06:32:13 +11:00
|
|
|
try {
|
2019-06-13 07:39:51 +10:00
|
|
|
await store.dispatch('loginUser', store.state.oauth.userToken)
|
2019-04-02 06:32:13 +11:00
|
|
|
} catch (e) {
|
|
|
|
console.log(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resolve()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-03-13 21:57:30 +11:00
|
|
|
const afterStoreSetup = async ({ store, i18n }) => {
|
2019-03-13 23:29:34 +11:00
|
|
|
if (store.state.config.customTheme) {
|
|
|
|
// This is a hack to deal with async loading of config.json and themes
|
|
|
|
// See: style_setter.js, setPreset()
|
|
|
|
window.themeLoaded = true
|
|
|
|
store.dispatch('setOption', {
|
|
|
|
name: 'customTheme',
|
|
|
|
value: store.state.config.customTheme
|
2018-10-27 00:16:23 +11:00
|
|
|
})
|
2019-03-13 23:29:34 +11:00
|
|
|
}
|
|
|
|
|
2019-04-02 06:41:34 +11:00
|
|
|
const width = windowWidth()
|
2019-03-24 07:21:57 +11:00
|
|
|
store.dispatch('setMobileLayout', width <= 800)
|
|
|
|
|
2019-04-02 04:46:30 +11:00
|
|
|
// Now we can try getting the server settings and logging in
|
2019-04-02 06:32:13 +11:00
|
|
|
await Promise.all([
|
|
|
|
checkOAuthToken({ store }),
|
|
|
|
setConfig({ store }),
|
|
|
|
getTOS({ store }),
|
|
|
|
getInstancePanel({ store }),
|
|
|
|
getStaticEmoji({ store }),
|
|
|
|
getCustomEmoji({ store }),
|
|
|
|
getNodeInfo({ store })
|
|
|
|
])
|
2019-03-13 21:57:30 +11:00
|
|
|
|
|
|
|
const router = new VueRouter({
|
|
|
|
mode: 'history',
|
|
|
|
routes: routes(store),
|
|
|
|
scrollBehavior: (to, _from, savedPosition) => {
|
|
|
|
if (to.matched.some(m => m.meta.dontScroll)) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return savedPosition || { x: 0, y: 0 }
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2019-03-13 22:10:57 +11:00
|
|
|
/* eslint-disable no-new */
|
|
|
|
return new Vue({
|
|
|
|
router,
|
|
|
|
store,
|
|
|
|
i18n,
|
|
|
|
el: '#app',
|
|
|
|
render: h => h(App)
|
|
|
|
})
|
2018-10-27 00:16:23 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
export default afterStoreSetup
|