pleroma-fe/src/components/user_settings/user_settings.js

361 lines
12 KiB
JavaScript
Raw Normal View History

import unescape from 'lodash/unescape'
import get from 'lodash/get'
import map from 'lodash/map'
import reject from 'lodash/reject'
import TabSwitcher from '../tab_switcher/tab_switcher.js'
2019-02-07 19:05:59 +11:00
import ImageCropper from '../image_cropper/image_cropper.vue'
import StyleSwitcher from '../style_switcher/style_switcher.vue'
import ScopeSelector from '../scope_selector/scope_selector.vue'
import fileSizeFormatService from '../../services/file_size_format/file_size_format.js'
2019-02-14 06:55:02 +11:00
import BlockCard from '../block_card/block_card.vue'
2019-02-14 14:04:28 +11:00
import MuteCard from '../mute_card/mute_card.vue'
import SelectableList from '../selectable_list/selectable_list.vue'
import EmojiInput from '../emoji-input/emoji-input.vue'
2019-04-03 07:10:03 +11:00
import Autosuggest from '../autosuggest/autosuggest.vue'
import withSubscription from '../../hocs/with_subscription/with_subscription'
import userSearchApi from '../../services/new_api/user_search.js'
const BlockList = withSubscription({
fetch: (props, $store) => $store.dispatch('fetchBlocks'),
select: (props, $store) => get($store.state.users.currentUser, 'blockIds', []),
childPropName: 'items'
})(SelectableList)
const MuteList = withSubscription({
fetch: (props, $store) => $store.dispatch('fetchMutes'),
select: (props, $store) => get($store.state.users.currentUser, 'muteIds', []),
childPropName: 'items'
})(SelectableList)
2019-02-14 14:04:28 +11:00
const UserSettings = {
data () {
return {
newName: this.$store.state.users.currentUser.name,
2019-02-05 07:03:01 +11:00
newBio: unescape(this.$store.state.users.currentUser.description),
newLocked: this.$store.state.users.currentUser.locked,
newNoRichText: this.$store.state.users.currentUser.no_rich_text,
newDefaultScope: this.$store.state.users.currentUser.default_scope,
hideFollows: this.$store.state.users.currentUser.hide_follows,
hideFollowers: this.$store.state.users.currentUser.hide_followers,
showRole: this.$store.state.users.currentUser.show_role,
role: this.$store.state.users.currentUser.role,
2017-12-24 01:44:22 +11:00
followList: null,
followImportError: false,
followsImported: false,
2018-05-14 09:47:08 +10:00
enableFollowsExport: true,
2019-02-09 13:59:33 +11:00
pickAvatarBtnVisible: true,
2018-12-13 19:25:03 +11:00
bannerUploading: false,
backgroundUploading: false,
followListUploading: false,
bannerPreview: null,
backgroundPreview: null,
bannerUploadError: null,
backgroundUploadError: null,
deletingAccount: false,
deleteAccountConfirmPasswordInput: '',
2018-05-22 08:01:09 +10:00
deleteAccountError: false,
changePasswordInputs: [ '', '', '' ],
changedPassword: false,
2018-08-19 09:23:03 +10:00
changePasswordError: false,
activeTab: 'profile'
}
},
created () {
this.$store.dispatch('fetchTokens')
},
components: {
StyleSwitcher,
ScopeSelector,
2019-02-07 19:05:59 +11:00
TabSwitcher,
ImageCropper,
BlockList,
MuteList,
2019-04-02 21:12:31 +11:00
EmojiInput,
2019-04-03 07:10:03 +11:00
Autosuggest,
2019-04-03 07:14:45 +11:00
BlockCard,
MuteCard
},
computed: {
user () {
return this.$store.state.users.currentUser
2017-12-24 01:44:22 +11:00
},
pleromaBackend () {
2018-09-10 04:21:23 +10:00
return this.$store.state.instance.pleromaBackend
},
2019-03-30 21:41:42 +11:00
minimalScopesMode () {
return this.$store.state.instance.minimalScopesMode
},
vis () {
return {
public: { selected: this.newDefaultScope === 'public' },
unlisted: { selected: this.newDefaultScope === 'unlisted' },
private: { selected: this.newDefaultScope === 'private' },
direct: { selected: this.newDefaultScope === 'direct' }
}
},
currentSaveStateNotice () {
return this.$store.state.interface.settings.currentSaveStateNotice
},
oauthTokens () {
return this.$store.state.oauthTokens.tokens.map(oauthToken => {
return {
id: oauthToken.id,
appName: oauthToken.app_name,
validUntil: new Date(oauthToken.valid_until).toLocaleDateString()
}
})
2018-06-27 23:28:07 +10:00
}
},
methods: {
updateProfile () {
2018-12-15 11:30:39 +11:00
const name = this.newName
const description = this.newBio
const locked = this.newLocked
// Backend notation.
2018-06-27 23:28:07 +10:00
/* eslint-disable camelcase */
const default_scope = this.newDefaultScope
const no_rich_text = this.newNoRichText
const hide_follows = this.hideFollows
const hide_followers = this.hideFollowers
const show_role = this.showRole
2018-06-27 23:28:07 +10:00
/* eslint-enable camelcase */
this.$store.state.api.backendInteractor
.updateProfile({
params: {
name,
description,
locked,
// Backend notation.
/* eslint-disable camelcase */
default_scope,
no_rich_text,
hide_follows,
hide_followers,
show_role
/* eslint-enable camelcase */
}}).then((user) => {
if (!user.error) {
this.$store.commit('addNewUsers', [user])
this.$store.commit('setCurrentUser', user)
}
})
},
changeVis (visibility) {
this.newDefaultScope = visibility
},
uploadFile (slot, e) {
const file = e.target.files[0]
if (!file) { return }
2018-12-14 01:51:29 +11:00
if (file.size > this.$store.state.instance[slot + 'limit']) {
const filesize = fileSizeFormatService.fileSizeFormat(file.size)
2018-12-14 01:51:29 +11:00
const allowedsize = fileSizeFormatService.fileSizeFormat(this.$store.state.instance[slot + 'limit'])
2018-12-14 01:44:37 +11:00
this[slot + 'UploadError'] = this.$t('upload.error.base') + ' ' + this.$t('upload.error.file_too_big', {filesize: filesize.num, filesizeunit: filesize.unit, allowedsize: allowedsize.num, allowedsizeunit: allowedsize.unit})
return
}
// eslint-disable-next-line no-undef
const reader = new FileReader()
reader.onload = ({target}) => {
const img = target.result
2018-12-14 01:44:37 +11:00
this[slot + 'Preview'] = img
}
reader.readAsDataURL(file)
},
submitAvatar (cropper, file) {
2019-03-19 12:19:42 +11:00
let img
if (cropper) {
img = cropper.getCroppedCanvas().toDataURL(file.type)
} else {
img = file
}
2019-02-09 13:59:33 +11:00
return this.$store.state.api.backendInteractor.updateAvatar({ params: { img } }).then((user) => {
if (!user.error) {
this.$store.commit('addNewUsers', [user])
this.$store.commit('setCurrentUser', user)
2018-12-13 03:31:16 +11:00
} else {
2019-02-09 14:17:53 +11:00
throw new Error(this.$t('upload.error.base') + user.error)
}
})
},
clearUploadError (slot) {
2018-12-14 01:44:37 +11:00
this[slot + 'UploadError'] = null
},
submitBanner () {
2018-12-13 19:25:03 +11:00
if (!this.bannerPreview) { return }
2018-12-13 19:25:03 +11:00
let banner = this.bannerPreview
2017-08-16 07:46:55 +10:00
// eslint-disable-next-line no-undef
let imginfo = new Image()
2017-08-16 07:46:55 +10:00
/* eslint-disable camelcase */
let offset_top, offset_left, width, height
imginfo.src = banner
width = imginfo.width
height = imginfo.height
offset_top = 0
offset_left = 0
2018-12-13 19:25:03 +11:00
this.bannerUploading = true
this.$store.state.api.backendInteractor.updateBanner({params: {banner, offset_top, offset_left, width, height}}).then((data) => {
if (!data.error) {
let clone = JSON.parse(JSON.stringify(this.$store.state.users.currentUser))
clone.cover_photo = data.url
this.$store.commit('addNewUsers', [clone])
this.$store.commit('setCurrentUser', clone)
2018-12-13 19:25:03 +11:00
this.bannerPreview = null
2018-12-13 03:31:16 +11:00
} else {
2018-12-13 19:25:03 +11:00
this.bannerUploadError = this.$t('upload.error.base') + data.error
}
2018-12-13 19:25:03 +11:00
this.bannerUploading = false
})
2017-08-16 07:46:55 +10:00
/* eslint-enable camelcase */
},
submitBg () {
2018-12-13 19:25:03 +11:00
if (!this.backgroundPreview) { return }
let img = this.backgroundPreview
2017-08-16 07:46:55 +10:00
// eslint-disable-next-line no-undef
let imginfo = new Image()
let cropX, cropY, cropW, cropH
imginfo.src = img
cropX = 0
cropY = 0
cropW = imginfo.width
cropH = imginfo.width
2018-12-13 19:25:03 +11:00
this.backgroundUploading = true
this.$store.state.api.backendInteractor.updateBg({params: {img, cropX, cropY, cropW, cropH}}).then((data) => {
if (!data.error) {
let clone = JSON.parse(JSON.stringify(this.$store.state.users.currentUser))
clone.background_image = data.url
this.$store.commit('addNewUsers', [clone])
this.$store.commit('setCurrentUser', clone)
2018-12-13 19:25:03 +11:00
this.backgroundPreview = null
2018-12-13 03:31:16 +11:00
} else {
2018-12-13 19:25:03 +11:00
this.backgroundUploadError = this.$t('upload.error.base') + data.error
}
2018-12-13 19:25:03 +11:00
this.backgroundUploading = false
})
2017-12-24 01:44:22 +11:00
},
importFollows () {
2018-12-13 19:25:03 +11:00
this.followListUploading = true
2017-12-24 01:44:22 +11:00
const followList = this.followList
this.$store.state.api.backendInteractor.followImport({params: followList})
.then((status) => {
if (status) {
this.followsImported = true
} else {
this.followImportError = true
}
2018-12-13 19:25:03 +11:00
this.followListUploading = false
2017-12-24 01:44:22 +11:00
})
},
2018-05-14 09:47:08 +10:00
/* This function takes an Array of Users
* and outputs a file with all the addresses for the user to download
*/
2018-05-17 08:51:52 +10:00
exportPeople (users, filename) {
2018-05-14 09:47:08 +10:00
// Get all the friends addresses
2018-05-17 08:51:52 +10:00
var UserAddresses = users.map(function (user) {
// check is it's a local user
if (user && user.is_local) {
// append the instance address
// eslint-disable-next-line no-undef
user.screen_name += '@' + location.hostname
}
return user.screen_name
}).join('\n')
2018-05-14 09:47:08 +10:00
// Make the user download the file
2018-05-17 08:51:52 +10:00
var fileToDownload = document.createElement('a')
fileToDownload.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(UserAddresses))
fileToDownload.setAttribute('download', filename)
fileToDownload.style.display = 'none'
document.body.appendChild(fileToDownload)
fileToDownload.click()
document.body.removeChild(fileToDownload)
2018-05-14 09:47:08 +10:00
},
2018-05-17 08:51:52 +10:00
exportFollows () {
this.enableFollowsExport = false
2018-05-14 09:47:08 +10:00
this.$store.state.api.backendInteractor
.exportFriends({
id: this.$store.state.users.currentUser.id
2019-02-07 03:17:23 +11:00
})
2018-05-18 20:34:36 +10:00
.then((friendList) => {
2018-05-17 08:51:52 +10:00
this.exportPeople(friendList, 'friends.csv')
setTimeout(() => { this.enableFollowsExport = true }, 2000)
2018-05-18 20:34:36 +10:00
})
2018-05-14 09:47:08 +10:00
},
2017-12-24 01:44:22 +11:00
followListChange () {
// eslint-disable-next-line no-undef
let formData = new FormData()
formData.append('list', this.$refs.followlist.files[0])
this.followList = formData
},
dismissImported () {
this.followsImported = false
this.followImportError = false
},
confirmDelete () {
this.deletingAccount = true
},
deleteAccount () {
this.$store.state.api.backendInteractor.deleteAccount({password: this.deleteAccountConfirmPasswordInput})
.then((res) => {
if (res.status === 'success') {
this.$store.dispatch('logout')
this.$router.push({name: 'root'})
} else {
this.deleteAccountError = res.error
}
})
2018-05-22 08:01:09 +10:00
},
changePassword () {
const params = {
password: this.changePasswordInputs[0],
newPassword: this.changePasswordInputs[1],
newPasswordConfirmation: this.changePasswordInputs[2]
}
this.$store.state.api.backendInteractor.changePassword(params)
.then((res) => {
if (res.status === 'success') {
this.changedPassword = true
this.changePasswordError = false
this.logout()
2018-05-22 08:01:09 +10:00
} else {
this.changedPassword = false
this.changePasswordError = res.error
}
})
2018-08-19 09:23:03 +10:00
},
activateTab (tabName) {
this.activeTab = tabName
2018-12-01 00:30:55 +11:00
},
logout () {
2018-12-01 00:30:55 +11:00
this.$store.dispatch('logout')
this.$router.replace('/')
},
revokeToken (id) {
2019-02-21 10:51:28 +11:00
if (window.confirm(`${this.$i18n.t('settings.revoke_token')}?`)) {
this.$store.dispatch('revokeToken', id)
}
},
filterUnblockedUsers (userIds) {
return reject(userIds, (userId) => {
const user = this.$store.getters.findUser(userId)
return !user || user.statusnet_blocking || user.id === this.$store.state.users.currentUser.id
})
},
2019-04-03 07:14:45 +11:00
filterUnMutedUsers (userIds) {
return reject(userIds, (userId) => {
const user = this.$store.getters.findUser(userId)
return !user || user.muted || user.id === this.$store.state.users.currentUser.id
})
},
queryUserIds (query) {
return userSearchApi.search({query, store: this.$store})
.then((users) => {
this.$store.dispatch('addNewUsers', users)
return map(users, 'id')
})
}
}
}
export default UserSettings