mirror of
https://github.com/Eugeny/tabby
synced 2024-12-14 07:12:50 +00:00
wip ref(core): Profile avoid direct config interraction outside of profiles.services
This commit is contained in:
parent
ef040ee342
commit
48d4b8e8f8
2 changed files with 142 additions and 63 deletions
|
@ -37,6 +37,117 @@ export class ProfilesService {
|
||||||
@Inject(ProfileProvider) private profileProviders: ProfileProvider<Profile>[],
|
@Inject(ProfileProvider) private profileProviders: ProfileProvider<Profile>[],
|
||||||
) { }
|
) { }
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Methods used to interract with ProfileProvider
|
||||||
|
*/
|
||||||
|
|
||||||
|
getProviders (): ProfileProvider<Profile>[] {
|
||||||
|
return [...this.profileProviders]
|
||||||
|
}
|
||||||
|
|
||||||
|
providerForProfile <T extends Profile> (profile: PartialProfile<T>): ProfileProvider<T>|null {
|
||||||
|
const provider = this.profileProviders.find(x => x.id === profile.type) ?? null
|
||||||
|
return provider as unknown as ProfileProvider<T>|null
|
||||||
|
}
|
||||||
|
|
||||||
|
getDescription <P extends Profile> (profile: PartialProfile<P>): string|null {
|
||||||
|
profile = this.getConfigProxyForProfile(profile)
|
||||||
|
return this.providerForProfile(profile)?.getDescription(profile) ?? null
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Methods used to interract with Profile
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return ConfigProxy for a given Profile
|
||||||
|
* arg: skipUserDefaults -> do not merge global provider defaults in ConfigProxy
|
||||||
|
*/
|
||||||
|
getConfigProxyForProfile <T extends Profile> (profile: PartialProfile<T>, skipUserDefaults = false): T {
|
||||||
|
const defaults = this.getProfileDefaults(profile, skipUserDefaults).reduce(configMerge, {})
|
||||||
|
return new ConfigProxy(profile, defaults) as unknown as T
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return an Array of Profiles
|
||||||
|
* arg: includeBuiltin (default: true) -> include BuiltinProfiles
|
||||||
|
* arg: clone (default: false) -> return deepclone Array
|
||||||
|
*/
|
||||||
|
async getProfiles (includeBuiltin = true, clone = false): Promise<PartialProfile<Profile>[]> {
|
||||||
|
let list = this.config.store.profiles ?? []
|
||||||
|
if (includeBuiltin) {
|
||||||
|
const lists = await Promise.all(this.config.enabledServices(this.profileProviders).map(x => x.getBuiltinProfiles()))
|
||||||
|
list = [
|
||||||
|
...this.config.store.profiles ?? [],
|
||||||
|
...lists.reduce((a, b) => a.concat(b), []),
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
const sortKey = p => `${this.resolveProfileGroupName(p.group ?? '')} / ${p.name}`
|
||||||
|
list.sort((a, b) => sortKey(a).localeCompare(sortKey(b)))
|
||||||
|
list.sort((a, b) => (a.isBuiltin ? 1 : 0) - (b.isBuiltin ? 1 : 0))
|
||||||
|
return clone ? deepClone(list) : list
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write a Profile in config
|
||||||
|
* arg: saveConfig (default: true) -> invoke after the Profile was updated
|
||||||
|
*/
|
||||||
|
async writeProfile (profile: PartialProfile<Profile>, saveConfig = true): Promise<void> {
|
||||||
|
this.deleteProfile(profile, false)
|
||||||
|
|
||||||
|
this.config.store.profiles.push(profile)
|
||||||
|
if (saveConfig) {
|
||||||
|
return this.config.save()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a Profile from config
|
||||||
|
* arg: saveConfig (default: true) -> invoke after the Profile was deleted
|
||||||
|
*/
|
||||||
|
async deleteProfile (profile: PartialProfile<Profile>, saveConfig = true): Promise<void> {
|
||||||
|
this.providerForProfile(profile)?.deleteProfile(this.getConfigProxyForProfile(profile))
|
||||||
|
this.config.store.profiles = this.config.store.profiles.filter(p => p.id !== profile.id)
|
||||||
|
|
||||||
|
const profileHotkeyName = ProfilesService.getProfileHotkeyName(profile)
|
||||||
|
if (this.config.store.hotkeys.profile.hasOwnProperty(profileHotkeyName)) {
|
||||||
|
const profileHotkeys = deepClone(this.config.store.hotkeys.profile)
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
||||||
|
delete profileHotkeys[profileHotkeyName]
|
||||||
|
this.config.store.hotkeys.profile = profileHotkeys
|
||||||
|
}
|
||||||
|
|
||||||
|
if (saveConfig) {
|
||||||
|
return this.config.save()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete all Profiles from config using option filter
|
||||||
|
* arg: options { group: string } -> options used to filter which profile have to be deleted
|
||||||
|
* arg: saveConfig (default: true) -> invoke after the Profile was deleted
|
||||||
|
*/
|
||||||
|
async deleteBulkProfiles (options: { group: string }, saveConfig = true): Promise<void> {
|
||||||
|
for (const profile of this.config.store.profiles.filter(p => p.group === options.group)) {
|
||||||
|
this.providerForProfile(profile)?.deleteProfile(this.getConfigProxyForProfile(profile))
|
||||||
|
|
||||||
|
const profileHotkeyName = ProfilesService.getProfileHotkeyName(profile)
|
||||||
|
if (this.config.store.hotkeys.profile.hasOwnProperty(profileHotkeyName)) {
|
||||||
|
const profileHotkeys = deepClone(this.config.store.hotkeys.profile)
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
||||||
|
delete profileHotkeys[profileHotkeyName]
|
||||||
|
this.config.store.hotkeys.profile = profileHotkeys
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.config.store.profiles = this.config.store.profiles.filter(p => p.group !== options.group)
|
||||||
|
|
||||||
|
if (saveConfig) {
|
||||||
|
return this.config.save()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async openNewTabForProfile <P extends Profile> (profile: PartialProfile<P>): Promise<BaseTabComponent|null> {
|
async openNewTabForProfile <P extends Profile> (profile: PartialProfile<P>): Promise<BaseTabComponent|null> {
|
||||||
const params = await this.newTabParametersForProfile(profile)
|
const params = await this.newTabParametersForProfile(profile)
|
||||||
if (params) {
|
if (params) {
|
||||||
|
@ -64,32 +175,27 @@ export class ProfilesService {
|
||||||
return params
|
return params
|
||||||
}
|
}
|
||||||
|
|
||||||
getProviders (): ProfileProvider<Profile>[] {
|
async launchProfile (profile: PartialProfile<Profile>): Promise<void> {
|
||||||
return [...this.profileProviders]
|
await this.openNewTabForProfile(profile)
|
||||||
|
|
||||||
|
let recentProfiles: PartialProfile<Profile>[] = JSON.parse(window.localStorage['recentProfiles'] ?? '[]')
|
||||||
|
if (this.config.store.terminal.showRecentProfiles > 0) {
|
||||||
|
recentProfiles = recentProfiles.filter(x => x.group !== profile.group || x.name !== profile.name)
|
||||||
|
recentProfiles.unshift(profile)
|
||||||
|
recentProfiles = recentProfiles.slice(0, this.config.store.terminal.showRecentProfiles)
|
||||||
|
} else {
|
||||||
|
recentProfiles = []
|
||||||
|
}
|
||||||
|
window.localStorage['recentProfiles'] = JSON.stringify(recentProfiles)
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProfiles (): Promise<PartialProfile<Profile>[]> {
|
static getProfileHotkeyName (profile: PartialProfile<Profile>): string {
|
||||||
const lists = await Promise.all(this.config.enabledServices(this.profileProviders).map(x => x.getBuiltinProfiles()))
|
return (profile.id ?? profile.name).replace(/\./g, '-')
|
||||||
let list = lists.reduce((a, b) => a.concat(b), [])
|
|
||||||
list = [
|
|
||||||
...this.config.store.profiles ?? [],
|
|
||||||
...list,
|
|
||||||
]
|
|
||||||
const sortKey = p => `${this.resolveProfileGroupName(p.group ?? '')} / ${p.name}`
|
|
||||||
list.sort((a, b) => sortKey(a).localeCompare(sortKey(b)))
|
|
||||||
list.sort((a, b) => (a.isBuiltin ? 1 : 0) - (b.isBuiltin ? 1 : 0))
|
|
||||||
return list
|
|
||||||
}
|
}
|
||||||
|
|
||||||
providerForProfile <T extends Profile> (profile: PartialProfile<T>): ProfileProvider<T>|null {
|
/*
|
||||||
const provider = this.profileProviders.find(x => x.id === profile.type) ?? null
|
* Methods used to interract with Profile Selector
|
||||||
return provider as unknown as ProfileProvider<T>|null
|
*/
|
||||||
}
|
|
||||||
|
|
||||||
getDescription <P extends Profile> (profile: PartialProfile<P>): string|null {
|
|
||||||
profile = this.getConfigProxyForProfile(profile)
|
|
||||||
return this.providerForProfile(profile)?.getDescription(profile) ?? null
|
|
||||||
}
|
|
||||||
|
|
||||||
selectorOptionForProfile <P extends Profile, T> (profile: PartialProfile<P>): SelectorOption<T> {
|
selectorOptionForProfile <P extends Profile, T> (profile: PartialProfile<P>): SelectorOption<T> {
|
||||||
const fullProfile = this.getConfigProxyForProfile(profile)
|
const fullProfile = this.getConfigProxyForProfile(profile)
|
||||||
|
@ -103,12 +209,6 @@ export class ProfilesService {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
getRecentProfiles (): PartialProfile<Profile>[] {
|
|
||||||
let recentProfiles: PartialProfile<Profile>[] = JSON.parse(window.localStorage['recentProfiles'] ?? '[]')
|
|
||||||
recentProfiles = recentProfiles.slice(0, this.config.store.terminal.showRecentProfiles)
|
|
||||||
return recentProfiles
|
|
||||||
}
|
|
||||||
|
|
||||||
showProfileSelector (): Promise<PartialProfile<Profile>|null> {
|
showProfileSelector (): Promise<PartialProfile<Profile>|null> {
|
||||||
if (this.selector.active) {
|
if (this.selector.active) {
|
||||||
return Promise.resolve(null)
|
return Promise.resolve(null)
|
||||||
|
@ -198,6 +298,12 @@ export class ProfilesService {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getRecentProfiles (): PartialProfile<Profile>[] {
|
||||||
|
let recentProfiles: PartialProfile<Profile>[] = JSON.parse(window.localStorage['recentProfiles'] ?? '[]')
|
||||||
|
recentProfiles = recentProfiles.slice(0, this.config.store.terminal.showRecentProfiles)
|
||||||
|
return recentProfiles
|
||||||
|
}
|
||||||
|
|
||||||
async quickConnect (query: string): Promise<PartialProfile<Profile>|null> {
|
async quickConnect (query: string): Promise<PartialProfile<Profile>|null> {
|
||||||
for (const provider of this.getProviders()) {
|
for (const provider of this.getProviders()) {
|
||||||
if (provider.supportsQuickConnect) {
|
if (provider.supportsQuickConnect) {
|
||||||
|
@ -211,25 +317,6 @@ export class ProfilesService {
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
getConfigProxyForProfile <T extends Profile> (profile: PartialProfile<T>, skipUserDefaults = false): T {
|
|
||||||
const defaults = this.getProfileDefaults(profile, skipUserDefaults).reduce(configMerge, {})
|
|
||||||
return new ConfigProxy(profile, defaults) as unknown as T
|
|
||||||
}
|
|
||||||
|
|
||||||
async launchProfile (profile: PartialProfile<Profile>): Promise<void> {
|
|
||||||
await this.openNewTabForProfile(profile)
|
|
||||||
|
|
||||||
let recentProfiles: PartialProfile<Profile>[] = JSON.parse(window.localStorage['recentProfiles'] ?? '[]')
|
|
||||||
if (this.config.store.terminal.showRecentProfiles > 0) {
|
|
||||||
recentProfiles = recentProfiles.filter(x => x.group !== profile.group || x.name !== profile.name)
|
|
||||||
recentProfiles.unshift(profile)
|
|
||||||
recentProfiles = recentProfiles.slice(0, this.config.store.terminal.showRecentProfiles)
|
|
||||||
} else {
|
|
||||||
recentProfiles = []
|
|
||||||
}
|
|
||||||
window.localStorage['recentProfiles'] = JSON.stringify(recentProfiles)
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Methods used to interract with Profile/ProfileGroup/Global defaults
|
* Methods used to interract with Profile/ProfileGroup/Global defaults
|
||||||
*/
|
*/
|
||||||
|
@ -254,6 +341,7 @@ export class ProfilesService {
|
||||||
/**
|
/**
|
||||||
* Return defaults for a given profile
|
* Return defaults for a given profile
|
||||||
* Always return something, empty object if no defaults found
|
* Always return something, empty object if no defaults found
|
||||||
|
* arg: skipUserDefaults -> do not merge global provider defaults in ConfigProxy
|
||||||
*/
|
*/
|
||||||
getProfileDefaults (profile: PartialProfile<Profile>, skipUserDefaults = false): any {
|
getProfileDefaults (profile: PartialProfile<Profile>, skipUserDefaults = false): any {
|
||||||
const provider = this.providerForProfile(profile)
|
const provider = this.providerForProfile(profile)
|
||||||
|
@ -276,7 +364,7 @@ export class ProfilesService {
|
||||||
async getProfileGroups (includeProfiles = false, includeNonUserGroup = false): Promise<PartialProfileGroup<ProfileGroup>[]> {
|
async getProfileGroups (includeProfiles = false, includeNonUserGroup = false): Promise<PartialProfileGroup<ProfileGroup>[]> {
|
||||||
let profiles: PartialProfile<Profile>[] = []
|
let profiles: PartialProfile<Profile>[] = []
|
||||||
if (includeProfiles) {
|
if (includeProfiles) {
|
||||||
profiles = await this.getProfiles()
|
profiles = await this.getProfiles(includeNonUserGroup, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
const profileGroupCollapsed = JSON.parse(window.localStorage.profileGroupCollapsed ?? '{}')
|
const profileGroupCollapsed = JSON.parse(window.localStorage.profileGroupCollapsed ?? '{}')
|
||||||
|
@ -327,7 +415,7 @@ export class ProfilesService {
|
||||||
* arg: saveConfig (default: true) -> invoke after the ProfileGroup was updated
|
* arg: saveConfig (default: true) -> invoke after the ProfileGroup was updated
|
||||||
*/
|
*/
|
||||||
async writeProfileGroup (group: PartialProfileGroup<ProfileGroup>, saveConfig = true): Promise<void> {
|
async writeProfileGroup (group: PartialProfileGroup<ProfileGroup>, saveConfig = true): Promise<void> {
|
||||||
this.deleteProfileGroup(group, false)
|
this.deleteProfileGroup(group, false, false)
|
||||||
|
|
||||||
delete group.profiles
|
delete group.profiles
|
||||||
delete group.editable
|
delete group.editable
|
||||||
|
@ -346,12 +434,13 @@ export class ProfilesService {
|
||||||
async deleteProfileGroup (group: PartialProfileGroup<ProfileGroup>, saveConfig = true, deleteProfiles = true): Promise<void> {
|
async deleteProfileGroup (group: PartialProfileGroup<ProfileGroup>, saveConfig = true, deleteProfiles = true): Promise<void> {
|
||||||
this.config.store.groups = this.config.store.groups.filter(g => g.id !== group.id)
|
this.config.store.groups = this.config.store.groups.filter(g => g.id !== group.id)
|
||||||
if (deleteProfiles) {
|
if (deleteProfiles) {
|
||||||
this.config.store.profiles = this.config.store.profiles.filter(x => x.group !== group.id)
|
await this.deleteBulkProfiles({ group: group.id }, false)
|
||||||
} else {
|
} else {
|
||||||
for (const profile of this.config.store.profiles.filter(x => x.group === group.id)) {
|
for (const profile of this.config.store.profiles.filter(x => x.group === group.id)) {
|
||||||
delete profile.group
|
delete profile.group
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (saveConfig) {
|
if (saveConfig) {
|
||||||
return this.config.save()
|
return this.config.save()
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ import slugify from 'slugify'
|
||||||
import deepClone from 'clone-deep'
|
import deepClone from 'clone-deep'
|
||||||
import { Component, Inject } from '@angular/core'
|
import { Component, Inject } from '@angular/core'
|
||||||
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
|
||||||
import { ConfigService, HostAppService, Profile, SelectorService, ProfilesService, PromptModalComponent, PlatformService, BaseComponent, PartialProfile, ProfileProvider, TranslateService, Platform, AppHotkeyProvider, ProfileGroup, PartialProfileGroup } from 'tabby-core'
|
import { ConfigService, HostAppService, Profile, SelectorService, ProfilesService, PromptModalComponent, PlatformService, BaseComponent, PartialProfile, ProfileProvider, TranslateService, Platform, ProfileGroup, PartialProfileGroup } from 'tabby-core'
|
||||||
import { EditProfileModalComponent } from './editProfileModal.component'
|
import { EditProfileModalComponent } from './editProfileModal.component'
|
||||||
|
|
||||||
_('Filter')
|
_('Filter')
|
||||||
|
@ -94,7 +94,7 @@ export class ProfilesSettingsTabComponent extends BaseComponent {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
Object.assign(profile, result)
|
Object.assign(profile, result)
|
||||||
await this.config.save()
|
await this.profilesService.writeProfile(profile)
|
||||||
}
|
}
|
||||||
|
|
||||||
async showProfileEditModal (profile: PartialProfile<Profile>): Promise<PartialProfile<Profile>|null> {
|
async showProfileEditModal (profile: PartialProfile<Profile>): Promise<PartialProfile<Profile>|null> {
|
||||||
|
@ -137,17 +137,7 @@ export class ProfilesSettingsTabComponent extends BaseComponent {
|
||||||
cancelId: 1,
|
cancelId: 1,
|
||||||
},
|
},
|
||||||
)).response === 0) {
|
)).response === 0) {
|
||||||
this.profilesService.providerForProfile(profile)?.deleteProfile(
|
this.profilesService.deleteProfile(profile)
|
||||||
this.profilesService.getConfigProxyForProfile(profile))
|
|
||||||
this.config.store.profiles = this.config.store.profiles.filter(x => x !== profile)
|
|
||||||
const profileHotkeyName = AppHotkeyProvider.getProfileHotkeyName(profile)
|
|
||||||
if (this.config.store.hotkeys.profile.hasOwnProperty(profileHotkeyName)) {
|
|
||||||
const profileHotkeys = deepClone(this.config.store.hotkeys.profile)
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
|
||||||
delete profileHotkeys[profileHotkeyName]
|
|
||||||
this.config.store.hotkeys.profile = profileHotkeys
|
|
||||||
}
|
|
||||||
await this.config.save()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue