You've already forked AstralRinth
711 lines
20 KiB
Vue
711 lines
20 KiB
Vue
<template>
|
|
<ModalConfirm
|
|
ref="modal_confirm"
|
|
title="Are you sure you want to delete this instance?"
|
|
description="If you proceed, all data for your instance will be removed. You will not be able to recover it."
|
|
:has-to-type="false"
|
|
proceed-label="Delete"
|
|
:noblur="!themeStore.advancedRendering"
|
|
@proceed="removeProfile"
|
|
/>
|
|
<Modal
|
|
ref="changeVersionsModal"
|
|
header="Change instance versions"
|
|
:noblur="!themeStore.advancedRendering"
|
|
>
|
|
<div class="change-versions-modal universal-body">
|
|
<div class="input-row">
|
|
<p class="input-label">Loader</p>
|
|
<Chips v-model="loader" :items="loaders" :never-empty="false" />
|
|
</div>
|
|
<div class="input-row">
|
|
<p class="input-label">Game Version</p>
|
|
<div class="versions">
|
|
<DropdownSelect v-model="gameVersion" :options="selectableGameVersions" render-up />
|
|
<Checkbox v-model="showSnapshots" class="filter-checkbox" label="Include snapshots" />
|
|
</div>
|
|
</div>
|
|
<div v-if="loader !== 'vanilla'" class="input-row">
|
|
<p class="input-label">Loader Version</p>
|
|
<DropdownSelect
|
|
:model-value="selectableLoaderVersions[loaderVersionIndex]"
|
|
:options="selectableLoaderVersions"
|
|
:display-name="(option) => option?.id"
|
|
render-up
|
|
@change="(value) => (loaderVersionIndex = value.index)"
|
|
/>
|
|
</div>
|
|
<div class="push-right input-group">
|
|
<button class="btn" @click="$refs.changeVersionsModal.hide()">
|
|
<XIcon />
|
|
Cancel
|
|
</button>
|
|
<button
|
|
class="btn btn-primary"
|
|
:disabled="!isValid || editing"
|
|
@click="saveGvLoaderEdits()"
|
|
>
|
|
<SaveIcon />
|
|
{{ editing ? 'Saving...' : 'Save changes' }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</Modal>
|
|
<section class="card">
|
|
<div class="label">
|
|
<h3>
|
|
<span class="label__title size-card-header">Instance</span>
|
|
</h3>
|
|
</div>
|
|
<label for="instance-icon">
|
|
<span class="label__title">Icon</span>
|
|
</label>
|
|
<div class="input-group">
|
|
<Avatar
|
|
:src="!icon || (icon && icon.startsWith('http')) ? icon : convertFileSrc(icon)"
|
|
size="md"
|
|
class="project__icon"
|
|
/>
|
|
<div class="input-stack">
|
|
<button id="instance-icon" class="btn" @click="setIcon">
|
|
<UploadIcon />
|
|
Select icon
|
|
</button>
|
|
<button class="btn" @click="resetIcon">
|
|
<TrashIcon />
|
|
Remove icon
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<label for="project-name">
|
|
<span class="label__title">Name</span>
|
|
</label>
|
|
<input id="profile-name" v-model="title" autocomplete="off" maxlength="80" type="text" />
|
|
|
|
<div class="adjacent-input">
|
|
<label for="edit-versions">
|
|
<span class="label__title">Edit mod loader/game versions</span>
|
|
<span class="label__description">
|
|
Allows you to change the mod loader, loader version, or game version of the instance.
|
|
</span>
|
|
</label>
|
|
<button
|
|
id="edit-versions"
|
|
class="btn"
|
|
@click="$refs.changeVersionsModal.show()"
|
|
:disabled="offline"
|
|
>
|
|
<EditIcon />
|
|
Edit versions
|
|
</button>
|
|
</div>
|
|
|
|
<div class="adjacent-input">
|
|
<label>
|
|
<span class="label__title">Categories</span>
|
|
<span class="label__description">
|
|
Set the categories of this instance, for display in the library page. This is purely
|
|
cosmetic.
|
|
</span>
|
|
</label>
|
|
<multiselect
|
|
v-model="groups"
|
|
:options="availableGroups"
|
|
:multiple="true"
|
|
:searchable="true"
|
|
:show-no-results="false"
|
|
:close-on-select="false"
|
|
:clear-search-on-select="false"
|
|
:show-labels="false"
|
|
:taggable="true"
|
|
tag-placeholder="Add new category"
|
|
placeholder="Select categories..."
|
|
@tag="
|
|
(newTag) => {
|
|
groups.push(newTag.trim().substring(0, 32))
|
|
availableGroups.push(newTag.trim().substring(0, 32))
|
|
}
|
|
"
|
|
/>
|
|
</div>
|
|
</section>
|
|
<Card>
|
|
<div class="label">
|
|
<h3>
|
|
<span class="label__title size-card-header">Java</span>
|
|
</h3>
|
|
</div>
|
|
<div class="settings-group">
|
|
<h3>Installation</h3>
|
|
<Checkbox v-model="overrideJavaInstall" label="Override global java installations" />
|
|
<JavaSelector v-model="javaInstall" :disabled="!overrideJavaInstall" />
|
|
</div>
|
|
<hr class="card-divider" />
|
|
<div class="settings-group">
|
|
<h3>Java arguments</h3>
|
|
<Checkbox v-model="overrideJavaArgs" label="Override global java arguments" />
|
|
<input
|
|
id="java-args"
|
|
v-model="javaArgs"
|
|
autocomplete="off"
|
|
:disabled="!overrideJavaArgs"
|
|
type="text"
|
|
class="installation-input"
|
|
placeholder="Enter java arguments..."
|
|
/>
|
|
</div>
|
|
<div class="settings-group">
|
|
<h3>Environment variables</h3>
|
|
<Checkbox v-model="overrideEnvVars" label="Override global environment variables" />
|
|
<input
|
|
v-model="envVars"
|
|
autocomplete="off"
|
|
:disabled="!overrideEnvVars"
|
|
type="text"
|
|
class="installation-input"
|
|
placeholder="Enter environment variables..."
|
|
/>
|
|
</div>
|
|
<hr class="card-divider" />
|
|
<div class="settings-group">
|
|
<h3>Java memory</h3>
|
|
<Checkbox v-model="overrideMemorySettings" label="Override global memory settings" />
|
|
<Slider
|
|
v-model="memory.maximum"
|
|
:disabled="!overrideMemorySettings"
|
|
:min="256"
|
|
:max="maxMemory"
|
|
:step="1"
|
|
unit="mb"
|
|
/>
|
|
</div>
|
|
</Card>
|
|
<Card>
|
|
<div class="label">
|
|
<h3>
|
|
<span class="label__title size-card-header">Window</span>
|
|
</h3>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<Checkbox v-model="overrideWindowSettings" label="Override global window settings" />
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="fullscreen">
|
|
<span class="label__title">Fullscreen</span>
|
|
<span class="label__description"> Make the game start in full screen when launched. </span>
|
|
</label>
|
|
<Checkbox id="fullscreen" v-model="fullscreenSetting" :disabled="!overrideWindowSettings" />
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="width">
|
|
<span class="label__title">Width</span>
|
|
<span class="label__description"> The width of the game window when launched. </span>
|
|
</label>
|
|
<input
|
|
id="width"
|
|
v-model="resolution[0]"
|
|
autocomplete="off"
|
|
:disabled="!overrideWindowSettings || fullscreenSetting"
|
|
type="number"
|
|
placeholder="Enter width..."
|
|
/>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="height">
|
|
<span class="label__title">Height</span>
|
|
<span class="label__description"> The height of the game window when launched. </span>
|
|
</label>
|
|
<input
|
|
id="height"
|
|
v-model="resolution[1]"
|
|
autocomplete="off"
|
|
:disabled="!overrideWindowSettings || fullscreenSetting"
|
|
type="number"
|
|
class="input"
|
|
placeholder="Enter height..."
|
|
/>
|
|
</div>
|
|
</Card>
|
|
<Card>
|
|
<div class="label">
|
|
<h3>
|
|
<span class="label__title size-card-header">Hooks</span>
|
|
</h3>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<Checkbox v-model="overrideHooks" label="Override global hooks" />
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="pre-launch">
|
|
<span class="label__title">Pre launch</span>
|
|
<span class="label__description"> Ran before the instance is launched. </span>
|
|
</label>
|
|
<input
|
|
id="pre-launch"
|
|
v-model="hooks.pre_launch"
|
|
autocomplete="off"
|
|
:disabled="!overrideHooks"
|
|
type="text"
|
|
placeholder="Enter pre-launch command..."
|
|
/>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="wrapper">
|
|
<span class="label__title">Wrapper</span>
|
|
<span class="label__description"> Wrapper command for launching Minecraft. </span>
|
|
</label>
|
|
<input
|
|
id="wrapper"
|
|
v-model="hooks.wrapper"
|
|
autocomplete="off"
|
|
:disabled="!overrideHooks"
|
|
type="text"
|
|
placeholder="Enter wrapper command..."
|
|
/>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="post-exit">
|
|
<span class="label__title">Post exit</span>
|
|
<span class="label__description"> Ran after the game closes. </span>
|
|
</label>
|
|
<input
|
|
id="post-exit"
|
|
v-model="hooks.post_exit"
|
|
autocomplete="off"
|
|
:disabled="!overrideHooks"
|
|
type="text"
|
|
placeholder="Enter post-exit command..."
|
|
/>
|
|
</div>
|
|
</Card>
|
|
<Card>
|
|
<div class="label">
|
|
<h3>
|
|
<span class="label__title size-card-header">Instance management</span>
|
|
</h3>
|
|
</div>
|
|
<div class="adjacent-input">
|
|
<label for="repair-profile">
|
|
<span class="label__title">Repair instance</span>
|
|
<span class="label__description">
|
|
Reinstalls Minecraft dependencies and checks for corruption. Use this if your game is not
|
|
launching due to launcher-related errors.
|
|
</span>
|
|
</label>
|
|
<button
|
|
id="repair-profile"
|
|
class="btn btn-highlight"
|
|
:disabled="repairing || offline"
|
|
@click="repairProfile"
|
|
>
|
|
<HammerIcon /> Repair
|
|
</button>
|
|
</div>
|
|
<div v-if="props.instance.modrinth_update_version" class="adjacent-input">
|
|
<label for="repair-profile">
|
|
<span class="label__title">Reinstall modpack</span>
|
|
<span class="label__description">
|
|
Reinstalls Modrinth modpack and checks for corruption. Use this if your game is not
|
|
launching due to your instance diverging from the Modrinth modpack.
|
|
</span>
|
|
</label>
|
|
<button
|
|
id="repair-profile"
|
|
class="btn btn-highlight"
|
|
:disabled="repairing || offline"
|
|
@click="repairModpack"
|
|
>
|
|
<DownloadIcon /> Reinstall
|
|
</button>
|
|
</div>
|
|
|
|
<div class="adjacent-input">
|
|
<label for="delete-profile">
|
|
<span class="label__title">Delete instance</span>
|
|
<span class="label__description">
|
|
Fully removes a instance from the disk. Be careful, as once you delete a instance there is
|
|
no way to recover it.
|
|
</span>
|
|
</label>
|
|
<button
|
|
id="delete-profile"
|
|
class="btn btn-danger"
|
|
:disabled="removing"
|
|
@click="$refs.modal_confirm.show()"
|
|
>
|
|
<TrashIcon /> Delete
|
|
</button>
|
|
</div>
|
|
</Card>
|
|
</template>
|
|
|
|
<script setup>
|
|
import {
|
|
Card,
|
|
Slider,
|
|
TrashIcon,
|
|
Checkbox,
|
|
UploadIcon,
|
|
Avatar,
|
|
EditIcon,
|
|
Modal,
|
|
Chips,
|
|
DropdownSelect,
|
|
XIcon,
|
|
SaveIcon,
|
|
HammerIcon,
|
|
DownloadIcon,
|
|
ModalConfirm,
|
|
} from 'omorphia'
|
|
import { Multiselect } from 'vue-multiselect'
|
|
import { useRouter } from 'vue-router'
|
|
import {
|
|
edit,
|
|
edit_icon,
|
|
get_optimal_jre_key,
|
|
install,
|
|
list,
|
|
remove,
|
|
update_repair_modrinth,
|
|
} from '@/helpers/profile.js'
|
|
import { computed, readonly, ref, shallowRef, watch } from 'vue'
|
|
import { get_max_memory } from '@/helpers/jre.js'
|
|
import { get } from '@/helpers/settings.js'
|
|
import JavaSelector from '@/components/ui/JavaSelector.vue'
|
|
import { convertFileSrc } from '@tauri-apps/api/tauri'
|
|
import { open } from '@tauri-apps/api/dialog'
|
|
import { get_fabric_versions, get_forge_versions, get_quilt_versions } from '@/helpers/metadata.js'
|
|
import { get_game_versions, get_loaders } from '@/helpers/tags.js'
|
|
import { handleError } from '@/store/notifications.js'
|
|
import { mixpanel_track } from '@/helpers/mixpanel'
|
|
import { useTheming } from '@/store/theme.js'
|
|
|
|
const router = useRouter()
|
|
|
|
const props = defineProps({
|
|
instance: {
|
|
type: Object,
|
|
required: true,
|
|
},
|
|
offline: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
})
|
|
|
|
const themeStore = useTheming()
|
|
|
|
const title = ref(props.instance.metadata.name)
|
|
const icon = ref(props.instance.metadata.icon)
|
|
const groups = ref(props.instance.metadata.groups)
|
|
|
|
const instancesList = Object.values(await list(true))
|
|
const availableGroups = ref([
|
|
...new Set(
|
|
instancesList.reduce((acc, obj) => {
|
|
return acc.concat(obj.metadata.groups)
|
|
}, [])
|
|
),
|
|
])
|
|
|
|
async function resetIcon() {
|
|
icon.value = null
|
|
await edit_icon(props.instance.path, null).catch(handleError)
|
|
mixpanel_track('InstanceRemoveIcon')
|
|
}
|
|
|
|
async function setIcon() {
|
|
const value = await open({
|
|
multiple: false,
|
|
filters: [
|
|
{
|
|
name: 'Image',
|
|
extensions: ['png', 'jpeg', 'svg', 'webp', 'gif', 'jpg'],
|
|
},
|
|
],
|
|
})
|
|
|
|
if (!value) return
|
|
|
|
icon.value = value
|
|
await edit_icon(props.instance.path, icon.value).catch(handleError)
|
|
|
|
mixpanel_track('InstanceSetIcon')
|
|
}
|
|
|
|
const globalSettings = await get().catch(handleError)
|
|
|
|
const javaSettings = props.instance.java ?? {}
|
|
|
|
const overrideJavaInstall = ref(!!javaSettings.override_version)
|
|
const optimalJava = readonly(await get_optimal_jre_key(props.instance.path).catch(handleError))
|
|
const javaInstall = ref(optimalJava ?? javaSettings.override_version ?? { path: '', version: '' })
|
|
|
|
const overrideJavaArgs = ref(!!javaSettings.extra_arguments)
|
|
const javaArgs = ref((javaSettings.extra_arguments ?? globalSettings.custom_java_args).join(' '))
|
|
|
|
const overrideEnvVars = ref(!!javaSettings.custom_env_args)
|
|
const envVars = ref(
|
|
(javaSettings.custom_env_args ?? globalSettings.custom_env_args).map((x) => x.join('=')).join(' ')
|
|
)
|
|
|
|
const overrideMemorySettings = ref(!!props.instance.memory)
|
|
const memory = ref(props.instance.memory ?? globalSettings.memory)
|
|
const maxMemory = Math.floor((await get_max_memory().catch(handleError)) / 1024)
|
|
|
|
const overrideWindowSettings = ref(!!props.instance.resolution || !!props.instance.fullscreen)
|
|
const resolution = ref(props.instance.resolution ?? globalSettings.game_resolution)
|
|
const overrideHooks = ref(!!props.instance.hooks)
|
|
const hooks = ref(props.instance.hooks ?? globalSettings.hooks)
|
|
|
|
const fullscreenSetting = ref(!!props.instance.fullscreen)
|
|
|
|
watch(
|
|
[
|
|
title,
|
|
groups,
|
|
groups,
|
|
overrideJavaInstall,
|
|
javaInstall,
|
|
overrideJavaArgs,
|
|
javaArgs,
|
|
overrideEnvVars,
|
|
envVars,
|
|
overrideMemorySettings,
|
|
memory,
|
|
overrideWindowSettings,
|
|
resolution,
|
|
fullscreenSetting,
|
|
overrideHooks,
|
|
hooks,
|
|
],
|
|
async () => {
|
|
const editProfile = {
|
|
metadata: {
|
|
name: title.value.trim().substring(0, 16) ?? 'Instance',
|
|
groups: groups.value.map((x) => x.trim().substring(0, 32)).filter((x) => x.length > 0),
|
|
loader_version: props.instance.metadata.loader_version,
|
|
linked_data: props.instance.metadata.linked_data,
|
|
},
|
|
java: {},
|
|
}
|
|
|
|
if (overrideJavaInstall.value) {
|
|
if (javaInstall.value.path !== '') {
|
|
editProfile.java.override_version = javaInstall.value
|
|
editProfile.java.override_version.path = editProfile.java.override_version.path.replace(
|
|
'java.exe',
|
|
'javaw.exe'
|
|
)
|
|
}
|
|
}
|
|
|
|
if (overrideJavaArgs.value) {
|
|
if (javaArgs.value !== '') {
|
|
editProfile.java.extra_arguments = javaArgs.value.trim().split(/\s+/).filter(Boolean)
|
|
}
|
|
}
|
|
|
|
if (overrideEnvVars.value) {
|
|
if (envVars.value !== '') {
|
|
editProfile.java.custom_env_args = envVars.value
|
|
.trim()
|
|
.split(/\s+/)
|
|
.filter(Boolean)
|
|
.map((x) => x.split('=').filter(Boolean))
|
|
}
|
|
}
|
|
|
|
if (overrideMemorySettings.value) {
|
|
editProfile.memory = memory.value
|
|
}
|
|
|
|
if (overrideWindowSettings.value) {
|
|
editProfile.fullscreen = fullscreenSetting.value
|
|
|
|
if (!fullscreenSetting.value) {
|
|
editProfile.resolution = resolution.value
|
|
}
|
|
}
|
|
|
|
if (overrideHooks.value) {
|
|
editProfile.hooks = hooks.value
|
|
}
|
|
|
|
await edit(props.instance.path, editProfile)
|
|
},
|
|
{ deep: true }
|
|
)
|
|
|
|
const repairing = ref(false)
|
|
|
|
async function repairProfile() {
|
|
repairing.value = true
|
|
await install(props.instance.path).catch(handleError)
|
|
repairing.value = false
|
|
|
|
mixpanel_track('InstanceRepair', {
|
|
loader: props.instance.metadata.loader,
|
|
game_version: props.instance.metadata.game_version,
|
|
})
|
|
}
|
|
|
|
async function repairModpack() {
|
|
repairing.value = true
|
|
await update_repair_modrinth(props.instance.path).catch(handleError)
|
|
repairing.value = false
|
|
|
|
mixpanel_track('InstanceRepair', {
|
|
loader: props.instance.metadata.loader,
|
|
game_version: props.instance.metadata.game_version,
|
|
})
|
|
}
|
|
|
|
const removing = ref(false)
|
|
async function removeProfile() {
|
|
removing.value = true
|
|
await remove(props.instance.path).catch(handleError)
|
|
removing.value = false
|
|
|
|
mixpanel_track('InstanceRemove', {
|
|
loader: props.instance.metadata.loader,
|
|
game_version: props.instance.metadata.game_version,
|
|
})
|
|
|
|
await router.push({ path: '/' })
|
|
}
|
|
|
|
const changeVersionsModal = ref(null)
|
|
const showSnapshots = ref(false)
|
|
|
|
const [fabric_versions, forge_versions, quilt_versions, all_game_versions, loaders] =
|
|
await Promise.all([
|
|
get_fabric_versions().then(shallowRef).catch(handleError),
|
|
get_forge_versions().then(shallowRef).catch(handleError),
|
|
get_quilt_versions().then(shallowRef).catch(handleError),
|
|
get_game_versions().then(shallowRef).catch(handleError),
|
|
get_loaders()
|
|
.then((value) =>
|
|
value
|
|
.filter((item) => item.supported_project_types.includes('modpack'))
|
|
.map((item) => item.name.toLowerCase())
|
|
)
|
|
.then(ref)
|
|
.catch(handleError),
|
|
])
|
|
loaders.value.unshift('vanilla')
|
|
|
|
const loader = ref(props.instance.metadata.loader)
|
|
const gameVersion = ref(props.instance.metadata.game_version)
|
|
const selectableGameVersions = computed(() => {
|
|
return all_game_versions.value
|
|
.filter((item) => {
|
|
let defaultVal = item.version_type === 'release' || showSnapshots.value
|
|
if (loader.value === 'fabric') {
|
|
defaultVal &= fabric_versions.value.gameVersions.some((x) => item.version === x.id)
|
|
} else if (loader.value === 'forge') {
|
|
defaultVal &= forge_versions.value.gameVersions.some((x) => item.version === x.id)
|
|
} else if (loader.value === 'quilt') {
|
|
defaultVal &= quilt_versions.value.gameVersions.some((x) => item.version === x.id)
|
|
}
|
|
|
|
return defaultVal
|
|
})
|
|
.map((item) => item.version)
|
|
})
|
|
|
|
const selectableLoaderVersions = computed(() => {
|
|
if (gameVersion.value) {
|
|
if (loader.value === 'fabric') {
|
|
return fabric_versions.value.gameVersions[0].loaders
|
|
} else if (loader.value === 'forge') {
|
|
return forge_versions.value.gameVersions.find((item) => item.id === gameVersion.value).loaders
|
|
} else if (loader.value === 'quilt') {
|
|
return quilt_versions.value.gameVersions[0].loaders
|
|
}
|
|
}
|
|
return []
|
|
})
|
|
const loaderVersionIndex = ref(
|
|
selectableLoaderVersions.value.findIndex(
|
|
(x) => x.id === props.instance.metadata.loader_version?.id
|
|
)
|
|
)
|
|
|
|
const isValid = computed(() => {
|
|
return (
|
|
selectableGameVersions.value.includes(gameVersion.value) &&
|
|
(loaderVersionIndex.value >= 0 || loader.value === 'vanilla')
|
|
)
|
|
})
|
|
|
|
watch(loader, () => (loaderVersionIndex.value = 0))
|
|
|
|
const editing = ref(false)
|
|
async function saveGvLoaderEdits() {
|
|
editing.value = true
|
|
|
|
const editProfile = {
|
|
metadata: {
|
|
game_version: gameVersion.value,
|
|
loader: loader.value,
|
|
},
|
|
}
|
|
|
|
if (loader.value !== 'vanilla') {
|
|
editProfile.metadata.loader_version = selectableLoaderVersions.value[loaderVersionIndex.value]
|
|
}
|
|
await edit(props.instance.path, editProfile).catch(handleError)
|
|
await repairProfile()
|
|
|
|
editing.value = false
|
|
changeVersionsModal.value.hide()
|
|
}
|
|
</script>
|
|
|
|
<style scoped lang="scss">
|
|
.change-versions-modal {
|
|
display: flex;
|
|
flex-direction: column;
|
|
padding: 1rem;
|
|
gap: 1rem;
|
|
|
|
:deep(.animated-dropdown .options) {
|
|
max-height: 13.375rem;
|
|
}
|
|
|
|
.input-label {
|
|
font-size: 1rem;
|
|
font-weight: bolder;
|
|
color: var(--color-contrast);
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
.versions {
|
|
display: flex;
|
|
flex-direction: row;
|
|
gap: 1rem;
|
|
}
|
|
}
|
|
|
|
.settings-group {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 0.5rem;
|
|
margin: 1rem 0;
|
|
|
|
h3 {
|
|
margin: 0;
|
|
}
|
|
}
|
|
|
|
.installation-input {
|
|
width: 100%;
|
|
}
|
|
|
|
:deep(button.checkbox) {
|
|
border: none;
|
|
}
|
|
</style>
|