woah
This commit is contained in:
@@ -24,34 +24,48 @@
|
||||
d="M9 3.5a5.5 5.5 0 100 11 5.5 5.5 0 000-11zM2 9a7 7 0 1112.452 4.391l3.328 3.329a.75.75 0 11-1.06 1.06l-3.329-3.328A7 7 0 012 9z"
|
||||
clip-rule="evenodd" />
|
||||
</svg>
|
||||
<input type="text" v-model="search" ref="searchInput"
|
||||
<input type="text" v-model="search" ref="searchInput" @keydown.down="focusNext(magic.length)"
|
||||
@keydown.up="focusPrev(magic.length)" @keyup.enter="callAction"
|
||||
class="w-full h-10 pr-4 text-white rounded outline-none bg-coolgray-400 pl-11 placeholder:text-neutral-700 sm:text-sm focus:outline-none"
|
||||
placeholder="Search, jump or create... magically... 🪄" role="combobox"
|
||||
aria-expanded="false" aria-controls="options">
|
||||
</div>
|
||||
|
||||
<ul class="px-4 pb-2 overflow-y-auto max-h-80 scroll-py-10 scroll-pb-2 scrollbar" id="options"
|
||||
role="listbox">
|
||||
<li v-if="state.showNew">
|
||||
<ul class="px-4 pb-2 overflow-y-auto lg:max-h-screen max-h-80 scroll-py-10 scroll-pb-2 scrollbar"
|
||||
id="options" role="listbox">
|
||||
<li v-if="sequenceState.sequence.length !== 0">
|
||||
<h2 v-if="sequenceState.sequence[sequenceState.currentActionIndex] && possibleSequences[sequenceState.sequence[sequenceState.currentActionIndex]]"
|
||||
class="mt-4 mb-2 text-xs font-semibold text-neutral-500">{{
|
||||
possibleSequences[sequenceState.sequence[sequenceState.currentActionIndex]].newTitle }}
|
||||
</h2>
|
||||
<ul class="mt-2 -mx-4 text-sm text-white ">
|
||||
<li class="flex items-center px-4 py-2 cursor-pointer select-none group hover:bg-coolgray-400"
|
||||
id="option-1" role="option" tabindex="-1" @click="next('redirect', -1, state.icon)">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="w-6 h-6" viewBox="0 0 24 24"
|
||||
id="option-1" role="option" tabindex="-1"
|
||||
@click="addNew(sequenceState.sequence[sequenceState.currentActionIndex])">
|
||||
<svg xmlns=" http://www.w3.org/2000/svg" class="w-6 h-6" viewBox="0 0 24 24"
|
||||
stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
|
||||
stroke-linejoin="round">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M12 5l0 14" />
|
||||
<path d="M5 12l14 0" />
|
||||
</svg>
|
||||
<span class="flex-auto ml-3 truncate">Add new {{ state.icon }}: <span
|
||||
class="text-xs text-warning" v-if="search">{{ search }}</span>
|
||||
<span v-else class="text-xs text-warning">with random name (or type
|
||||
one)</span></span>
|
||||
<span class="flex-auto ml-3 truncate">
|
||||
<span v-if="search"><span class="capitalize ">{{
|
||||
sequenceState.sequence[sequenceState.currentActionIndex] }}</span> name
|
||||
will be:
|
||||
<span class="text-warning">{{ search
|
||||
}}</span></span>
|
||||
<span v-else><span class="capitalize ">{{
|
||||
sequenceState.sequence[sequenceState.currentActionIndex] }}</span> name
|
||||
will be:
|
||||
<span class="text-warning">randomly generated (type to change)</span>
|
||||
</span>
|
||||
</span>
|
||||
</li>
|
||||
</ul>
|
||||
</li>
|
||||
<li>
|
||||
<ul v-if="data.length == 0" class="mt-2 -mx-4 text-sm text-white">
|
||||
<ul v-if="magic.length == 0" class="mt-2 -mx-4 text-sm text-white">
|
||||
<li class="flex items-center px-4 py-2 select-none group">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="w-6 h-6 icon" viewBox="0 0 24 24"
|
||||
stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
|
||||
@@ -65,67 +79,94 @@
|
||||
<span class="flex-auto ml-3 truncate">Nothing found. Ooops.</span>
|
||||
</li>
|
||||
</ul>
|
||||
<h2 v-if="data.length != 0 && state.title"
|
||||
<h2 v-if="magic.length !== 0 && sequenceState.sequence[sequenceState.currentActionIndex] && possibleSequences[sequenceState.sequence[sequenceState.currentActionIndex]]"
|
||||
class="mt-4 mb-2 text-xs font-semibold text-neutral-500">{{
|
||||
state.title }}
|
||||
possibleSequences[sequenceState.sequence[sequenceState.currentActionIndex]].title }}
|
||||
</h2>
|
||||
<ul v-if="data.length != 0" class="mt-2 -mx-4 text-sm text-white">
|
||||
<li class="flex items-center px-4 py-2 cursor-pointer select-none group hover:bg-coolgray-400"
|
||||
id="option-1" role="option" tabindex="-1" v-for="action, index in data"
|
||||
@click="next(state.next ?? action.next, index, action.newAction)">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="w-6 h-6 icon" viewBox="0 0 24 24"
|
||||
stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
|
||||
stroke-linejoin="round">
|
||||
<template v-if="action.icon === 'git' || state.icon === 'git'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M16 12m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 8m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 16m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 15v-6" />
|
||||
<path d="M15 11l-2 -2" />
|
||||
<path d="M11 7l-1.9 -1.9" />
|
||||
<path
|
||||
d="M13.446 2.6l7.955 7.954a2.045 2.045 0 0 1 0 2.892l-7.955 7.955a2.045 2.045 0 0 1 -2.892 0l-7.955 -7.955a2.045 2.045 0 0 1 0 -2.892l7.955 -7.955a2.045 2.045 0 0 1 2.892 0z" />
|
||||
</template>
|
||||
<template v-if="action.icon === 'server' || state.icon === 'server'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M3 4m0 3a3 3 0 0 1 3 -3h12a3 3 0 0 1 3 3v2a3 3 0 0 1 -3 3h-12a3 3 0 0 1 -3 -3z" />
|
||||
<path d="M15 20h-9a3 3 0 0 1 -3 -3v-2a3 3 0 0 1 3 -3h12" />
|
||||
<path d="M7 8v.01" />
|
||||
<path d="M7 16v.01" />
|
||||
<path d="M20 15l-2 3h3l-2 3" />
|
||||
</template>
|
||||
<template v-if="action.icon === 'destination' || state.icon === 'destination'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M22 12.54c-1.804 -.345 -2.701 -1.08 -3.523 -2.94c-.487 .696 -1.102 1.568 -.92 2.4c.028 .238 -.32 1 -.557 1h-14c0 5.208 3.164 7 6.196 7c4.124 .022 7.828 -1.376 9.854 -5c1.146 -.101 2.296 -1.505 2.95 -2.46z" />
|
||||
<path d="M5 10h3v3h-3z" />
|
||||
<path d="M8 10h3v3h-3z" />
|
||||
<path d="M11 10h3v3h-3z" />
|
||||
<path d="M8 7h3v3h-3z" />
|
||||
<path d="M11 7h3v3h-3z" />
|
||||
<path d="M11 4h3v3h-3z" />
|
||||
<path d="M4.571 18c1.5 0 2.047 -.074 2.958 -.78" />
|
||||
<path d="M10 16l0 .01" />
|
||||
</template>
|
||||
<template v-if="action.icon === 'project' || state.icon === 'project'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M9 4h3l2 2h5a2 2 0 0 1 2 2v7a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-9a2 2 0 0 1 2 -2" />
|
||||
<path d="M17 17v2a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-9a2 2 0 0 1 2 -2h2" />
|
||||
</template>
|
||||
<template v-if="action.icon === 'environment' || state.icon === 'environment'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M16 5l3 3l-2 1l4 4l-3 1l4 4h-9" />
|
||||
<path d="M15 21l0 -3" />
|
||||
<path d="M8 13l-2 -2" />
|
||||
<path d="M8 12l2 -2" />
|
||||
<path d="M8 21v-13" />
|
||||
<path
|
||||
d="M5.824 16a3 3 0 0 1 -2.743 -3.69a3 3 0 0 1 .304 -4.833a3 3 0 0 1 4.615 -3.707a3 3 0 0 1 4.614 3.707a3 3 0 0 1 .305 4.833a3 3 0 0 1 -2.919 3.695h-4z" />
|
||||
</template>
|
||||
</svg>
|
||||
<ul v-if="magic.length != 0" class="mt-2 -mx-4 text-sm text-white">
|
||||
<li class="flex items-center px-4 py-2 transition-all cursor-pointer select-none group hover:bg-coolgray-400"
|
||||
:class="{ 'bg-coolgray-400': currentFocus === index }" id="option-1" role="option"
|
||||
tabindex="-1" v-for="action, index in magic" @click="goThroughSequence(action.id)">
|
||||
<div class="relative">
|
||||
<svg xmlns="http://www.w3.org/2000/svg" class="w-6 h-6 icon" viewBox="0 0 24 24"
|
||||
stroke-width="1.5" stroke="currentColor" fill="none" stroke-linecap="round"
|
||||
stroke-linejoin="round">
|
||||
<template
|
||||
v-if="action.icon === 'git' || sequenceState.sequence[sequenceState.currentActionIndex] === 'git'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M16 12m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 8m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 16m-1 0a1 1 0 1 0 2 0a1 1 0 1 0 -2 0" />
|
||||
<path d="M12 15v-6" />
|
||||
<path d="M15 11l-2 -2" />
|
||||
<path d="M11 7l-1.9 -1.9" />
|
||||
<path
|
||||
d="M13.446 2.6l7.955 7.954a2.045 2.045 0 0 1 0 2.892l-7.955 7.955a2.045 2.045 0 0 1 -2.892 0l-7.955 -7.955a2.045 2.045 0 0 1 0 -2.892l7.955 -7.955a2.045 2.045 0 0 1 2.892 0z" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'server' || sequenceState.sequence[sequenceState.currentActionIndex] === 'server'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M3 4m0 3a3 3 0 0 1 3 -3h12a3 3 0 0 1 3 3v2a3 3 0 0 1 -3 3h-12a3 3 0 0 1 -3 -3z" />
|
||||
<path d="M15 20h-9a3 3 0 0 1 -3 -3v-2a3 3 0 0 1 3 -3h12" />
|
||||
<path d="M7 8v.01" />
|
||||
<path d="M7 16v.01" />
|
||||
<path d="M20 15l-2 3h3l-2 3" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'destination' || sequenceState.sequence[sequenceState.currentActionIndex] === 'destination'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M22 12.54c-1.804 -.345 -2.701 -1.08 -3.523 -2.94c-.487 .696 -1.102 1.568 -.92 2.4c.028 .238 -.32 1 -.557 1h-14c0 5.208 3.164 7 6.196 7c4.124 .022 7.828 -1.376 9.854 -5c1.146 -.101 2.296 -1.505 2.95 -2.46z" />
|
||||
<path d="M5 10h3v3h-3z" />
|
||||
<path d="M8 10h3v3h-3z" />
|
||||
<path d="M11 10h3v3h-3z" />
|
||||
<path d="M8 7h3v3h-3z" />
|
||||
<path d="M11 7h3v3h-3z" />
|
||||
<path d="M11 4h3v3h-3z" />
|
||||
<path d="M4.571 18c1.5 0 2.047 -.074 2.958 -.78" />
|
||||
<path d="M10 16l0 .01" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'project' || sequenceState.sequence[sequenceState.currentActionIndex] === 'project'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path
|
||||
d="M9 4h3l2 2h5a2 2 0 0 1 2 2v7a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-9a2 2 0 0 1 2 -2" />
|
||||
<path
|
||||
d="M17 17v2a2 2 0 0 1 -2 2h-10a2 2 0 0 1 -2 -2v-9a2 2 0 0 1 2 -2h2" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'environment' || sequenceState.sequence[sequenceState.currentActionIndex] === 'environment'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M16 5l3 3l-2 1l4 4l-3 1l4 4h-9" />
|
||||
<path d="M15 21l0 -3" />
|
||||
<path d="M8 13l-2 -2" />
|
||||
<path d="M8 12l2 -2" />
|
||||
<path d="M8 21v-13" />
|
||||
<path
|
||||
d="M5.824 16a3 3 0 0 1 -2.743 -3.69a3 3 0 0 1 .304 -4.833a3 3 0 0 1 4.615 -3.707a3 3 0 0 1 4.614 3.707a3 3 0 0 1 .305 4.833a3 3 0 0 1 -2.919 3.695h-4z" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'key' || sequenceState.sequence[sequenceState.currentActionIndex] === 'key'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M14 10m-2 0a2 2 0 1 0 4 0a2 2 0 1 0 -4 0" />
|
||||
<path d="M21 12a9 9 0 1 1 -18 0a9 9 0 0 1 18 0z" />
|
||||
<path d="M12.5 11.5l-4 4l1.5 1.5" />
|
||||
<path d="M12 15l-1.5 -1.5" />
|
||||
</template>
|
||||
<template
|
||||
v-if="action.icon === 'goto' || sequenceState.sequence[sequenceState.currentActionIndex] === 'goto'">
|
||||
<path stroke="none" d="M0 0h24v24H0z" fill="none" />
|
||||
<path d="M10 18h4" />
|
||||
<path
|
||||
d="M3 8a9 9 0 0 1 9 9v1l1.428 -4.285a12 12 0 0 1 6.018 -6.938l.554 -.277" />
|
||||
<path d="M15 6h5v5" />
|
||||
</template>
|
||||
</svg>
|
||||
<div v-if="action.new"
|
||||
class="absolute top-0 right-0 -mt-2 -mr-2 font-bold text-warning">+
|
||||
</div>
|
||||
</div>
|
||||
<span class="flex-auto ml-3 truncate">{{ action.name }}</span>
|
||||
</li>
|
||||
</ul>
|
||||
@@ -140,51 +181,140 @@
|
||||
<script setup>
|
||||
import { ref, computed, onMounted, onUnmounted, watch, nextTick } from 'vue'
|
||||
import axios from "axios";
|
||||
|
||||
const currentFocus = ref()
|
||||
function focusNext(length) {
|
||||
if (currentFocus.value === undefined) {
|
||||
currentFocus.value = 0
|
||||
} else {
|
||||
if (length > currentFocus.value + 1) {
|
||||
currentFocus.value = currentFocus.value + 1
|
||||
}
|
||||
}
|
||||
}
|
||||
function focusPrev(length) {
|
||||
if (currentFocus.value === undefined) {
|
||||
currentFocus.value = length - 1
|
||||
} else {
|
||||
if (currentFocus.value > 0) {
|
||||
currentFocus.value = currentFocus.value - 1
|
||||
}
|
||||
}
|
||||
}
|
||||
async function callAction() {
|
||||
await goThroughSequence(currentFocus.value)
|
||||
}
|
||||
const showCommandPalette = ref(false)
|
||||
const search = ref()
|
||||
const searchInput = ref()
|
||||
const baseUrl = '/magic'
|
||||
let selected = {};
|
||||
|
||||
const appActions = [{
|
||||
const baseUrl = '/magic'
|
||||
|
||||
const uuidSelector = ['project', 'destination']
|
||||
const nameSelector = ['environment']
|
||||
const possibleSequences = {
|
||||
server: {
|
||||
newTitle: 'Add a new server',
|
||||
title: 'Select a server'
|
||||
},
|
||||
destination: {
|
||||
newTitle: 'Add a new destination',
|
||||
title: 'Select a destination'
|
||||
},
|
||||
project: {
|
||||
newTitle: 'Add a new project',
|
||||
title: 'Select a project'
|
||||
},
|
||||
environment: {
|
||||
newTitle: 'Add a new environment',
|
||||
title: 'Select an environment'
|
||||
},
|
||||
}
|
||||
const magicActions = [{
|
||||
id: 0,
|
||||
name: 'Public Repository',
|
||||
tags: 'application,public,repository,github,gitlab,bitbucket,git',
|
||||
name: 'Deploy a Public Repository',
|
||||
tags: 'git,github,public',
|
||||
icon: 'git',
|
||||
next: 'server'
|
||||
new: true,
|
||||
sequence: ['main', 'server', 'destination', 'project', 'environment', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 1,
|
||||
name: 'Private Repository (with GitHub Apps)',
|
||||
tags: 'application,private,repository,github,gitlab,bitbucket,git',
|
||||
name: 'Deploy a Private Repository (with GitHub Apps)',
|
||||
tags: 'git,github,private',
|
||||
icon: 'git',
|
||||
next: 'server'
|
||||
new: true,
|
||||
sequence: ['main', 'server', 'destination', 'project', 'environment', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Private Repository (with Deploy Key)',
|
||||
tags: 'application,private,repository,github,gitlab,bitbucket,git',
|
||||
name: 'Deploy a Private Repository (with Deploy Key)',
|
||||
tags: 'git,github,private,deploy,key',
|
||||
icon: 'git',
|
||||
next: 'server'
|
||||
new: true,
|
||||
sequence: ['main', 'server', 'destination', 'project', 'environment', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'Add a Private Key',
|
||||
tags: 'key,private,ssh',
|
||||
icon: 'key',
|
||||
new: true,
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Add a Server',
|
||||
tags: 'server,ssh,new,create',
|
||||
icon: 'server',
|
||||
new: true,
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
name: 'Add a Destination',
|
||||
tags: 'destination,docker,network,new,create',
|
||||
icon: 'destination',
|
||||
new: true,
|
||||
sequence: ['main', 'server', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 6,
|
||||
name: 'Goto Dashboard',
|
||||
icon: 'goto',
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 7,
|
||||
name: 'Goto Servers',
|
||||
icon: 'goto',
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 8,
|
||||
name: 'Goto Projects',
|
||||
icon: 'goto',
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 9,
|
||||
name: 'Goto Settings',
|
||||
icon: 'goto',
|
||||
sequence: ['main', 'redirect']
|
||||
},
|
||||
{
|
||||
id: 10,
|
||||
name: 'Goto Command Center',
|
||||
icon: 'goto',
|
||||
sequence: ['main', 'redirect']
|
||||
}
|
||||
]
|
||||
const initialState = {
|
||||
title: null,
|
||||
icon: null,
|
||||
next: null,
|
||||
current: null,
|
||||
showNew: false,
|
||||
data: appActions
|
||||
sequence: [],
|
||||
currentActionIndex: 0,
|
||||
magicActions,
|
||||
selected: {}
|
||||
}
|
||||
const state = ref({ ...initialState })
|
||||
|
||||
const data = computed(() => {
|
||||
if (search?.value) {
|
||||
return state.value.data.filter(item => item.name.toLowerCase().includes(search.value?.toLowerCase() ?? ''))
|
||||
}
|
||||
return state.value.data
|
||||
})
|
||||
const sequenceState = ref({ ...initialState })
|
||||
|
||||
function focusSearch(event) {
|
||||
if (event.target.nodeName === 'BODY') {
|
||||
@@ -208,135 +338,166 @@ watch(showCommandPalette, async (value) => {
|
||||
searchInput.value.focus();
|
||||
}
|
||||
})
|
||||
|
||||
const magic = computed(() => {
|
||||
if (search.value) {
|
||||
return sequenceState.value.magicActions.filter(action => {
|
||||
return action.name.toLowerCase().includes(search.value.toLowerCase()) || action.tags?.toLowerCase().includes(search.value.toLowerCase())
|
||||
})
|
||||
}
|
||||
return sequenceState.value.magicActions
|
||||
})
|
||||
async function addNew(name) {
|
||||
let targetUrl = new URL(window.location.origin)
|
||||
let newUrl = new URL(`${window.location.origin}${baseUrl}/${name}/new`);
|
||||
if (search.value) {
|
||||
targetUrl.searchParams.append('name', search.value)
|
||||
newUrl.searchParams.append('name', search.value)
|
||||
}
|
||||
switch (name) {
|
||||
case 'server':
|
||||
targetUrl.pathname = '/server/new'
|
||||
window.location.href = targetUrl.href
|
||||
break;
|
||||
case 'destination':
|
||||
targetUrl.pathname = '/destination/new'
|
||||
window.location.href = targetUrl.href
|
||||
break;
|
||||
case 'project':
|
||||
const { data: { project_uuid } } = await axios(newUrl.href)
|
||||
search.value = ''
|
||||
sequenceState.value.selected['project'] = project_uuid
|
||||
sequenceState.value.magicActions = await getEnvironments(project_uuid)
|
||||
sequenceState.value.currentActionIndex += 1
|
||||
break;
|
||||
case 'environment':
|
||||
newUrl.searchParams.append('project_uuid', sequenceState.value.selected.project)
|
||||
const { data: { environment_name } } = await axios(newUrl.href)
|
||||
search.value = ''
|
||||
sequenceState.value.selected['environment'] = environment_name
|
||||
redirect()
|
||||
break;
|
||||
}
|
||||
}
|
||||
function resetState() {
|
||||
showCommandPalette.value = false
|
||||
state.value = { ...initialState }
|
||||
selected = {}
|
||||
sequenceState.value = { ...initialState }
|
||||
search.value = ''
|
||||
}
|
||||
async function next(nextAction, index, newAction = null) {
|
||||
console.log({ nextAction, index, newAction })
|
||||
if (newAction) {
|
||||
let targetUrl = new URL(window.location.origin)
|
||||
let newUrl = new URL(`${window.location.origin}${baseUrl}/${newAction}/new`);
|
||||
if (search.value) newUrl.searchParams.append('name', search.value)
|
||||
switch (newAction) {
|
||||
case 'server':
|
||||
targetUrl.pathname = '/server/new'
|
||||
window.location.href = targetUrl.href
|
||||
break;
|
||||
case 'destination':
|
||||
targetUrl.pathname = '/destination/new'
|
||||
window.location.href = targetUrl.href
|
||||
break;
|
||||
case 'project':
|
||||
const { data: { new_project_uuid, new_project_id } } = await axios(newUrl.href)
|
||||
selected.project = new_project_uuid
|
||||
await getEnvironments(new_project_id)
|
||||
state.value.title = 'Select an Environment'
|
||||
state.value.icon = 'environment'
|
||||
break;
|
||||
case 'environment':
|
||||
if (selected.project) newUrl.searchParams.append('project_uuid', selected.project)
|
||||
const { data: { new_environment_name } } = await axios(newUrl.href)
|
||||
selected.environment = new_environment_name
|
||||
await redirect();
|
||||
break;
|
||||
async function goThroughSequence(actionId) {
|
||||
let currentSequence = null;
|
||||
let nextSequence = null;
|
||||
if (sequenceState.value.selected.main === undefined) {
|
||||
const { sequence, id } = magic.value[actionId];
|
||||
currentSequence = sequence[sequenceState.value.currentActionIndex]
|
||||
nextSequence = sequence[sequenceState.value.currentActionIndex + 1]
|
||||
sequenceState.value.sequence = sequence
|
||||
sequenceState.value.selected = {
|
||||
main: id
|
||||
}
|
||||
|
||||
} else {
|
||||
if (state.value.current) {
|
||||
if (state.value.current === 'environment') {
|
||||
selected[state.value.current] = state.value.data[index].name
|
||||
} else {
|
||||
selected[state.value.current] = state.value.data[index].uuid
|
||||
}
|
||||
currentSequence = sequenceState.value.sequence[sequenceState.value.currentActionIndex]
|
||||
nextSequence = sequenceState.value.sequence[sequenceState.value.currentActionIndex + 1]
|
||||
let selectedId = sequenceState.value.magicActions[actionId].id
|
||||
if (uuidSelector.includes(currentSequence)) {
|
||||
selectedId = sequenceState.value.magicActions[actionId].uuid
|
||||
}
|
||||
else selected['action'] = appActions[index].id
|
||||
console.log({ selected })
|
||||
switch (nextAction) {
|
||||
case 'server':
|
||||
await getServers()
|
||||
state.value.title = 'Select a server'
|
||||
state.value.icon = 'server'
|
||||
state.value.showNew = true
|
||||
break;
|
||||
case 'destination':
|
||||
await getDestinations(state.value.data[index].id)
|
||||
state.value.title = 'Select a destination'
|
||||
state.value.icon = 'destination'
|
||||
state.value.showNew = true
|
||||
break;
|
||||
case 'project':
|
||||
await getProjects()
|
||||
state.value.title = 'Select a project'
|
||||
state.value.icon = 'project'
|
||||
state.value.showNew = true
|
||||
break;
|
||||
case 'environment':
|
||||
await getEnvironments(state.value.data[index].id)
|
||||
state.value.title = 'Select an environment'
|
||||
state.value.icon = 'environment'
|
||||
state.value.showNew = true
|
||||
break;
|
||||
case 'redirect':
|
||||
await redirect();
|
||||
state.value.showNew = false
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
if (nameSelector.includes(currentSequence)) {
|
||||
selectedId = sequenceState.value.magicActions[actionId].name
|
||||
}
|
||||
sequenceState.value.selected = {
|
||||
...sequenceState.value.selected,
|
||||
[currentSequence]: selectedId
|
||||
}
|
||||
}
|
||||
|
||||
switch (nextSequence) {
|
||||
case 'server':
|
||||
sequenceState.value.magicActions = await getServers();
|
||||
break;
|
||||
case 'destination':
|
||||
sequenceState.value.magicActions = await getDestinations(sequenceState.value.selected[currentSequence]);
|
||||
break;
|
||||
case 'project':
|
||||
sequenceState.value.magicActions = await getProjects()
|
||||
break;
|
||||
case 'environment':
|
||||
sequenceState.value.magicActions = await getEnvironments(sequenceState.value.selected[currentSequence])
|
||||
break;
|
||||
case 'redirect':
|
||||
redirect()
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
sequenceState.value.currentActionIndex += 1
|
||||
search.value = ''
|
||||
searchInput.value.focus()
|
||||
}
|
||||
async function getServers() {
|
||||
const { data: { servers } } = await axios.get(`${baseUrl}/servers`);
|
||||
return servers;
|
||||
}
|
||||
async function getDestinations(serverId) {
|
||||
const { data: { destinations } } = await axios.get(`${baseUrl}/destinations?server_id=${serverId}`);
|
||||
return destinations;
|
||||
}
|
||||
async function getProjects() {
|
||||
const { data: { projects } } = await axios.get(`${baseUrl}/projects`);
|
||||
return projects;
|
||||
}
|
||||
async function getEnvironments(project_uuid) {
|
||||
const { data: { environments } } = await axios.get(`${baseUrl}/environments?project_uuid=${project_uuid}`);
|
||||
return environments;
|
||||
}
|
||||
|
||||
async function redirect() {
|
||||
let targetUrl = new URL(window.location.origin)
|
||||
switch (selected.action) {
|
||||
const selected = sequenceState.value.selected
|
||||
const { main, destination = null, project = null, environment = null, server = null } = selected
|
||||
switch (main) {
|
||||
case 0:
|
||||
targetUrl.pathname = `/project/${selected.project}/${selected.environment}/new`
|
||||
targetUrl.pathname = `/project/${project}/${environment}/new`
|
||||
targetUrl.searchParams.append('type', 'public')
|
||||
targetUrl.searchParams.append('destination', selected.destination)
|
||||
targetUrl.searchParams.append('destination', destination)
|
||||
break;
|
||||
case 1:
|
||||
targetUrl.pathname = `/project/${selected.project}/${selected.environment}/new`
|
||||
targetUrl.pathname = `/project/${project}/${environment}/new`
|
||||
targetUrl.searchParams.append('type', 'private-gh-app')
|
||||
targetUrl.searchParams.append('destination', selected.destination)
|
||||
targetUrl.searchParams.append('destination', destination)
|
||||
break;
|
||||
case 2:
|
||||
targetUrl.pathname = `/project/${selected.project}/${selected.environment}/new`
|
||||
targetUrl.pathname = `/project/${project}/${environment}/new`
|
||||
targetUrl.searchParams.append('type', 'private-deploy-key')
|
||||
targetUrl.searchParams.append('destination', selected.destination)
|
||||
targetUrl.searchParams.append('destination', destination)
|
||||
break;
|
||||
case 3:
|
||||
targetUrl.pathname = `/server/${selected.server}/`
|
||||
targetUrl.pathname = `/private-key/new/`
|
||||
break;
|
||||
case 4:
|
||||
targetUrl.pathname = `/server/new/`
|
||||
break;
|
||||
case 5:
|
||||
targetUrl.pathname = `/destination/new/`
|
||||
targetUrl.searchParams.append('server', server)
|
||||
break;
|
||||
case 6:
|
||||
targetUrl.pathname = `/`
|
||||
break;
|
||||
case 7:
|
||||
targetUrl.pathname = `/servers`
|
||||
break;
|
||||
case 8:
|
||||
targetUrl.pathname = `/projects`
|
||||
break;
|
||||
case 9:
|
||||
targetUrl.pathname = `/settings`
|
||||
break;
|
||||
case 10:
|
||||
targetUrl.pathname = `/command-center`
|
||||
break;
|
||||
|
||||
}
|
||||
window.location.href = targetUrl;
|
||||
}
|
||||
async function getServers() {
|
||||
const { data } = await axios.get(`${baseUrl}/servers`);
|
||||
state.value.data = data.servers
|
||||
state.value.current = 'server'
|
||||
state.value.next = 'destination'
|
||||
}
|
||||
async function getDestinations(serverId) {
|
||||
const { data } = await axios.get(`${baseUrl}/destinations?server_id=${serverId}`);
|
||||
state.value.data = data.destinations
|
||||
state.value.current = 'destination'
|
||||
state.value.next = 'project'
|
||||
}
|
||||
async function getProjects() {
|
||||
const { data } = await axios.get(`${baseUrl}/projects`);
|
||||
state.value.data = data.projects
|
||||
state.value.current = 'project'
|
||||
state.value.next = 'environment'
|
||||
}
|
||||
async function getEnvironments(projectId) {
|
||||
const { data } = await axios.get(`${baseUrl}/environments?project_id=${projectId}`);
|
||||
state.value.data = data.environments
|
||||
state.value.current = 'environment'
|
||||
state.value.next = 'redirect'
|
||||
}
|
||||
</script>
|
||||
|
||||
@@ -1,8 +1,17 @@
|
||||
<x-layout>
|
||||
<h1 class="pb-2">Command Center</h1>
|
||||
<h1>Command Center</h1>
|
||||
<div class="pb-10 text-sm breadcrumbs">
|
||||
<ul>
|
||||
<li>
|
||||
Execute commands on your servers without leaving the browser.
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
@if ($servers->count() > 0)
|
||||
<livewire:run-command :servers="$servers" />
|
||||
@else
|
||||
<div>No validated servers found.</div>
|
||||
<div>No validated servers found.
|
||||
<x-use-magic-bar />
|
||||
</div>
|
||||
@endif
|
||||
</x-layout>
|
||||
|
||||
@@ -41,7 +41,7 @@
|
||||
</li>
|
||||
|
||||
|
||||
@if (auth()->user()->isPartOfRootTeam())
|
||||
@if (auth()->user()->isAdmin())
|
||||
<li title="Command Center">
|
||||
<a class="hover:bg-transparent" @if (!request()->is('command-center')) href="/command-center" @endif>
|
||||
<svg xmlns="http://www.w3.org/2000/svg"
|
||||
|
||||
2
resources/views/components/use-magic-bar.blade.php
Normal file
2
resources/views/components/use-magic-bar.blade.php
Normal file
@@ -0,0 +1,2 @@
|
||||
Use the magic
|
||||
bar (press <span class="kbd-custom">/</span>) to create a new one.
|
||||
@@ -1,7 +1,11 @@
|
||||
<x-layout>
|
||||
<h1 class="pb-2">Dashboard</h1>
|
||||
<div class="pb-10 text-sm">
|
||||
Something (more) useful will be here.
|
||||
<h1>Dashboard</h1>
|
||||
<div class="pb-10 text-sm breadcrumbs">
|
||||
<ul>
|
||||
<li>
|
||||
Something (more) useful will be here.
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="w-full rounded shadow stats stats-vertical lg:stats-horizontal">
|
||||
<div class="stat">
|
||||
|
||||
@@ -1,5 +1,16 @@
|
||||
<x-layout>
|
||||
<div class="flex items-center justify-center">
|
||||
<div><a class="underline" href="{{ route('dashboard') }}">Go home</a></div>
|
||||
</div>
|
||||
<main class="grid min-h-full px-6 place-items-center lg:px-8">
|
||||
<div class="text-center">
|
||||
<p class="text-6xl font-semibold text-warning">404</p>
|
||||
<h1 class="mt-4 text-3xl font-bold tracking-tight text-white sm:text-5xl">Page not found</h1>
|
||||
<p class="mt-6 text-base leading-7 text-neutral-300">Sorry, we couldn’t find the page you’re looking for.</p>
|
||||
<div class="flex items-center justify-center mt-10 gap-x-6">
|
||||
<a href="/"
|
||||
class="rounded-md bg-coollabs px-3.5 py-2.5 text-sm font-semibold text-white shadow-sm hover:bg-coollabs-100 focus-visible:outline focus-visible:outline-2 focus-visible:outline-offset-2 focus-visible:outline-indigo-600 hover:no-underline">Go
|
||||
back home</a>
|
||||
<a href="https://docs.coollabs.io/contact.html" class="text-sm font-semibold text-white">Contact support
|
||||
<span aria-hidden="true">→</span></a>
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
</x-layout>
|
||||
|
||||
@@ -2,11 +2,11 @@
|
||||
<x-naked-modal show="deleteEnvironment" message='Are you sure you would like to delete this environment?' />
|
||||
@if ($resource_count > 0)
|
||||
<x-forms.button tooltip="First delete all resources." disabled>
|
||||
Delete
|
||||
Delete Environment
|
||||
</x-forms.button>
|
||||
@else
|
||||
<x-forms.button x-on:click.prevent="deleteEnvironment = true">
|
||||
Delete
|
||||
Delete Environment
|
||||
</x-forms.button>
|
||||
@endif
|
||||
</div>
|
||||
|
||||
@@ -2,11 +2,11 @@
|
||||
<x-naked-modal show="deleteProject" message='Are you sure you would like to delete this project?' />
|
||||
@if ($resource_count > 0)
|
||||
<x-forms.button disabled="First delete all resources.">
|
||||
Delete
|
||||
Delete Project
|
||||
</x-forms.button>
|
||||
@else
|
||||
<x-forms.button x-on:click.prevent="deleteProject = true">
|
||||
Delete
|
||||
Delete Project
|
||||
</x-forms.button>
|
||||
@endif
|
||||
</div>
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
<div>
|
||||
<div class="pb-4 text-sm">Outputs are not saved at the moment, only available until you refresh or navigate.</div>
|
||||
<form class="flex items-end justify-center gap-2" wire:submit.prevent='runCommand'>
|
||||
<x-forms.input placeholder="ls -l" autofocus noDirty id="command" label="Command" required />
|
||||
<x-forms.select label="Server" id="server" required>
|
||||
|
||||
@@ -9,9 +9,16 @@
|
||||
Delete
|
||||
</x-forms.button>
|
||||
@endif
|
||||
@if (!$server->settings->is_validated)
|
||||
<div class="w-full">
|
||||
<x-forms.button isHighlighted wire:click.prevent='validateServer'>
|
||||
Validate Server
|
||||
</x-forms.button>
|
||||
</div>
|
||||
@endif
|
||||
</div>
|
||||
<div class="flex flex-col gap-2 xl:flex-row">
|
||||
<div class="flex flex-col w-96">
|
||||
<div class="flex flex-col w-full">
|
||||
@if ($server->id === 0)
|
||||
<x-forms.input id="server.name" label="Name" readonly required />
|
||||
<x-forms.input id="server.description" label="Description" readonly />
|
||||
@@ -23,7 +30,7 @@
|
||||
{{-- <x-forms.checkbox disabled type="checkbox" id="server.settings.is_part_of_swarm"
|
||||
label="Is it part of a Swarm cluster?" /> --}}
|
||||
</div>
|
||||
<div class="flex flex-col">
|
||||
<div class="flex flex-col w-full">
|
||||
@if ($server->id === 0)
|
||||
<x-forms.input id="server.ip" label="IP Address" readonly required />
|
||||
<x-forms.input id="server.user" label="User" readonly required />
|
||||
@@ -37,13 +44,7 @@
|
||||
@endif
|
||||
</div>
|
||||
</div>
|
||||
@if (!$server->settings->is_validated)
|
||||
<div class="w-full pt-4">
|
||||
<x-forms.button isHighlighted wire:click.prevent='validateServer'>
|
||||
Validate Server
|
||||
</x-forms.button>
|
||||
</div>
|
||||
@endif
|
||||
|
||||
@if ($server->settings->is_validated)
|
||||
<h3 class="pt-8 pb-4">Quick Actions</h3>
|
||||
<div class="flex items-center gap-2">
|
||||
@@ -53,17 +54,18 @@
|
||||
{{-- <x-forms.button wire:click.prevent='installDocker'>Install Docker</x-forms.button> --}}
|
||||
</div>
|
||||
@endif
|
||||
<div class="pt-3 text-sm">
|
||||
@isset($uptime)
|
||||
@isset($uptime)
|
||||
<h3>Server Info</h3>
|
||||
<div class="pt-3 text-sm">
|
||||
<p>Uptime: {{ $uptime }}</p>
|
||||
@endisset
|
||||
@isset($dockerVersion)
|
||||
<p>Docker Engine {{ $dockerVersion }}</p>
|
||||
@endisset
|
||||
@isset($dockerComposeVersion)
|
||||
<p>{{ $dockerComposeVersion }}</p>
|
||||
@endisset
|
||||
</div>
|
||||
@isset($dockerVersion)
|
||||
<p>Docker Engine {{ $dockerVersion }}</p>
|
||||
@endisset
|
||||
@isset($dockerComposeVersion)
|
||||
<p>{{ $dockerComposeVersion }}</p>
|
||||
@endisset
|
||||
</div>
|
||||
@endisset
|
||||
</form>
|
||||
<div class="flex items-center gap-2 py-4">
|
||||
<h3>Private Key</h3>
|
||||
|
||||
@@ -1,11 +1,6 @@
|
||||
<div>
|
||||
<form class="flex flex-col gap-1" wire:submit.prevent='submit'>
|
||||
<div class="flex items-center gap-2">
|
||||
<h1>New Server</h1>
|
||||
<x-forms.button type="submit">
|
||||
Save
|
||||
</x-forms.button>
|
||||
</div>
|
||||
<h1>New Server</h1>
|
||||
<x-forms.input id="name" label="Name" required />
|
||||
<x-forms.input id="description" label="Description" />
|
||||
<x-forms.input id="ip" label="IP Address" required
|
||||
@@ -24,6 +19,8 @@
|
||||
@endforeach
|
||||
</x-forms.select>
|
||||
<x-forms.checkbox instantSave noDirty id="is_part_of_swarm" label="Is it part of a Swarm cluster?" />
|
||||
|
||||
<x-forms.button type="submit">
|
||||
Save
|
||||
</x-forms.button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
@@ -1,10 +1,10 @@
|
||||
<div class="flex flex-wrap gap-2">
|
||||
<div class="grid grid-cols-2">
|
||||
@forelse ($private_keys as $private_key)
|
||||
<div class="w-64 box">
|
||||
<button wire:click='setPrivateKey({{ $private_key->id }})'>{{ $private_key->name }}
|
||||
</button>
|
||||
</div>
|
||||
<x-forms.button wire:click='setPrivateKey({{ $private_key->id }})'>{{ $private_key->name }}
|
||||
</x-forms.button>
|
||||
@empty
|
||||
<p>No private keys found</p>
|
||||
<div>No private keys found.
|
||||
<x-use-magic-bar />
|
||||
</div>
|
||||
@endforelse
|
||||
</div>
|
||||
|
||||
@@ -1,7 +1,13 @@
|
||||
<div>
|
||||
<form wire:submit.prevent='submit' class="flex flex-col">
|
||||
<h1>Settings</h1>
|
||||
<div class="pt-2 pb-4 text-sm">Instance wide settings for Coolify. </div>
|
||||
<div class="pb-10 text-sm breadcrumbs">
|
||||
<ul>
|
||||
<li>
|
||||
Instance wide settings for Coolify.
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="flex items-center gap-2">
|
||||
<h3>General</h3>
|
||||
<x-forms.button type="submit">
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
<x-layout>
|
||||
<h1>New Private Key</h1>
|
||||
<h1>Add Private Key</h1>
|
||||
<livewire:private-key.create />
|
||||
</x-layout>
|
||||
|
||||
@@ -20,7 +20,7 @@
|
||||
@if ($environment->applications->count() === 0)
|
||||
<p>No resources found.</p>
|
||||
@endif
|
||||
<div class="flex flex-col gap-2">
|
||||
<div class="grid grid-cols-2 gap-2">
|
||||
@foreach ($environment->applications->sortBy('name') as $application)
|
||||
<a class="box"
|
||||
href="{{ route('project.application.configuration', [$project->uuid, $environment->name, $application->uuid]) }}">
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
<li>{{ $project->name }} </li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="flex flex-col gap-2">
|
||||
<div class="grid grid-cols-2 gap-2">
|
||||
@forelse ($project->environments as $environment)
|
||||
<a class="box" href="{{ route('project.resources', [$project->uuid, $environment->name]) }}">
|
||||
{{ $environment->name }}
|
||||
|
||||
@@ -7,15 +7,14 @@
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="flex flex-col gap-2">
|
||||
<div class="grid grid-cols-2 gap-2">
|
||||
@forelse ($projects as $project)
|
||||
<a href="{{ route('project.show', ['project_uuid' => data_get($project, 'uuid')]) }}"
|
||||
class="box">{{ $project->name }}</a>
|
||||
@empty
|
||||
<div>
|
||||
No project found. Use the magic
|
||||
bar (press <span class="kbd-custom">/</span>) to create a new
|
||||
project.
|
||||
No project found.
|
||||
<x-use-magic-bar />
|
||||
</div>
|
||||
<div>
|
||||
If you do not have a project yet, just create a resource (application, database, etc.) first, it will
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
<x-layout>
|
||||
<h1 class="py-0">Servers</h1>
|
||||
<h1>Servers</h1>
|
||||
<div class="pb-10 text-sm breadcrumbs">
|
||||
<ul>
|
||||
<li>
|
||||
@@ -7,7 +7,7 @@
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="grid grid-cols-2">
|
||||
<div class="grid grid-cols-2 gap-2">
|
||||
@forelse ($servers as $server)
|
||||
<a class="text-center hover:no-underline box group"
|
||||
href="{{ route('server.show', ['server_uuid' => data_get($server, 'uuid')]) }}">
|
||||
@@ -21,7 +21,7 @@
|
||||
@empty
|
||||
<div class="flex flex-col">
|
||||
<div>Without a server, you won't be able to do much.</div>
|
||||
<div>Let's <a class="text-lg underline text-warning" href="{{ route('server.new') }}">create</a> your
|
||||
<div>Let's <a class="text-lg underline text-warning" href="{{ route('server.create') }}">create</a> your
|
||||
first one.</div>
|
||||
</div>
|
||||
@endforelse
|
||||
@@ -1,4 +1,4 @@
|
||||
<x-layout>
|
||||
<h1>Select a private Key</h1>
|
||||
<h1 class="pb-2">Select a private Key</h1>
|
||||
<livewire:server.private-key />
|
||||
</x-layout>
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
<livewire:settings.email :settings="$settings" />
|
||||
|
||||
<h3 class='pb-4'>Actions</h3>
|
||||
@if (auth()->user()->isPartOfRootTeam())
|
||||
@if (auth()->user()->isAdmin())
|
||||
<livewire:force-upgrade />
|
||||
@endif
|
||||
</x-layout>
|
||||
|
||||
Reference in New Issue
Block a user