Files
thetool/Layout/default/RaspberryDisplay/Index.php

274 lines
9.9 KiB
PHP

<?php /** @noinspection PhpUndefinedClassInspection
* @var string $mfLayoutPackage
*/
include(realpath(dirname(__FILE__) . "/../../$mfLayoutPackage") . "/header.php"); ?>
<!-- start page title -->
<div class="row">
<div class="col-12">
<div class="page-title-box">
<div class="page-title-right">
<ol class="breadcrumb m-0">
<li class="breadcrumb-item"><a href="<?= self::getUrl("Dashboard") ?>"><?= MFAPPNAME_SLUG ?></a>
</li>
<li class="breadcrumb-item"><a href="<?= self::getUrl("Order") ?>">Raspberry Displays</a></li>
</ol>
</div>
<h4 class="page-title">Raspberry Displays</h4>
</div>
</div>
</div>
<!-- end page title -->
<!-- TODO: export style to css -->
<style>
[v-cloak] {
display: none !important;
}
.display-grid {
display: grid;
grid-template-columns: repeat(8, 11.25vw);
grid-row-gap: 20px;
width: 100vw;
}
.display {
background-color: #f8f9fa;
border: 2px solid #dee2e6;
border-radius: 4px;
overflow: hidden;
font-size: clamp(0.6rem, 0.8rem, 1.1rem);
display: grid;
grid-template-rows: repeat(3, 1fr);
justify-items: center;
}
.display > *:nth-child(1) {
align-self: start;
}
.display > *:nth-child(2) {
align-self: center;
}
.display > *:nth-child(3) {
align-self: end;
}
.small-27-inch {
grid-column: span 1;
margin: 0 0.37vw;
width: calc(10.5vw);
height: calc(10.5vw * 9 / 16)
}
.big-42-inch {
grid-column: span 2;
margin: 0 0.37vw;
width: calc(21vw);
height: calc(21vw * 9 / 16)
}
.overlay {
width: 100%;
height: 100%;
background-color: rgba(10, 10, 10, 0.5); /* Semi-transparent black */
z-index: 9999;
display: flex;
justify-content: center;
align-items: center; /* Center the spinner vertically and horizontally */
position: absolute;
top: 50%;
left: 50%;
opacity: 0.5;
transform: translate(-50%, -50%);
}
label {
margin: 0;
}
/* Define the transition */
.fade-enter-active, .fade-leave-active {
transition: opacity 0.2s ease;
}
.fade-enter, .fade-leave-to /* .fade-leave-active in <2.1.8 */
{
opacity: 0;
}
</style>
<script src="https://cdn.jsdelivr.net/npm/vue@2.7.16/dist/vue.js"></script>
<script src="https://cdn.jsdelivr.net/npm/vue-select@3.20.2/dist/vue-select.min.js"></script>
<link href="https://cdn.jsdelivr.net/npm/vue-select@3.20.2/dist/vue-select.min.css" rel="stylesheet">
<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
<div id="app" class="card">
<transition name="fade">
<div v-if="loading" class="w-100 h-100" style="position: absolute">
<div class="overlay" id="loading">
<div class="d-flex justify-content-center">
<div class="spinner-border" role="status">
<span class="sr-only">Loading...</span>
</div>
</div>
</div>
</div>
</transition>
<div class="p-2">
<h3>8322 Studenzen NOC Displays</h3>
<div class="display-grid">
<div v-for="display in displays2" :key="display.id"
:class="['display', display['display_label'].includes('-B-') ? 'big-42-inch' : 'small-27-inch']"
:style="display['custom_style']" style="">
<div style="display: grid; grid-template-columns: max-content auto max-content; justify-items: center;width:100%; padding: 0 2px">
<div>
<!-- FONT AWESOME ONLINE GREEN CIRCLE -->
<i class="fas fa-circle" data-toggle="tooltip" title="ONLINE" style="color: green"></i>
</div>
<div>
<div @click.prevent="enableDisplayURLEditMode(display.id)" style="cursor: pointer">
<span v-if="displaysURLEditMode !== display.id">{{ display.display_url | cleanupURL }}</span>
<input v-else-if="displaysURLEditMode === display.id"
v-model="display.display_url"
@keyup.enter="disableDisplayURLEditMode(display.id, display.display_url)"
@blur="disableDisplayURLEditMode(display.id, display.display_url)"
ref="displayURLEditInput"
class="form-control"
type="text">
</div>
</div>
<div style="cursor: pointer">
<!-- FONT AWESOME REBOOT ICON -->
<i class="fas fa-red fa-sync-alt" data-toggle="tooltip" title="Reboot this Raspberry"
@click="rebootRaspberry(display.id)"
style="color: green"></i>
</div>
</div>
<div>
<!-- Checkbox for Auto Refresh Enabled -->
<div style="display: inline-block" data-toggle="tooltip"
:title="`Auto refresh is ${display.auto_refresh_enabled ? 'enabled' : 'disabled'}.`">
<input type="checkbox" :id="'auto_refresh_enabled_checkbox_' + display.id"
v-model="display.auto_refresh_enabled"
@change="submitChanges(display.id, 'auto_refresh_enabled', display.auto_refresh_enabled)">
<label :for="'auto_refresh_enabled_checkbox_' + display.id">ARF</label>
</div>
<!-- This will only display if both are true, consider adjusting logic as needed -->
<span style="margin: 0 4px"> | </span>
<!-- Checkbox for Margin Hotfix Enabled -->
<div style="display: inline-block" data-toggle="tooltip"
:title="`Margin Hotfix is ${display.margin_hot_fix_enabled ? 'enabled' : 'disabled'}.`">
<input type="checkbox" :id="'margin_hot_fix_enabled_checkbox_' + display.id"
v-model="display.margin_hot_fix_enabled"
@change="submitChanges(display.id, 'margin_hot_fix_enabled', display.margin_hot_fix_enabled)">
<label :for="'margin_hot_fix_enabled_checkbox_' + display.id">MHF</label>
</div>
</div>
<div>
{{ display.display_label }}
</div>
</div>
</div>
</div>
</div>
<script>
Vue.filter('cleanupURL', function (value) {
value = value.replace(/^(?:https?:\/\/)?(?:www\.)?/i, "").split('/')[0];
return value;
})
new Vue({
el: '#app',
mounted() {
this.fetchDisplays();
},
methods: {
async rebootRaspberry(displayID) {
this.loading = true;
await axios.get('/RaspberryDisplay/api?do=reboot', {
params: {
displayID: displayID
}
});
this.loading = false;
},
async fetchDisplays() {
this.loading = true;
const response = await axios.get('/RaspberryDisplay/api?do=getDisplays');
this.displays2 = response.data.result;
this.loading = false;
Vue.nextTick(() => {
$('[data-toggle="tooltip"]').tooltip('dispose');
$('[data-toggle="tooltip"]').tooltip();
});
},
enableDisplayURLEditMode(displayID) {
this.displaysURLEditMode = displayID;
const _this = this;
// wait for the DOM to update
Vue.nextTick(() => {
_this.$refs['displayURLEditInput'][0].focus();
});
},
disableDisplayURLEditMode(displayID, displayURL) {
this.displaysURLEditMode = null;
this.submitChanges(displayID, 'display_url', displayURL);
},
async submitChanges(displayID, field, value) {
this.loading = true;
await axios.get('/RaspberryDisplay/api?do=change', {
params: {
displayID: displayID,
field: field,
value: value,
}
});
await this.fetchDisplays();
this.loading = false;
}
},
data: {
loading: false,
displaysURLEditMode: null,
displays2: null,
displays: [
{id: 1, size: 'small-27-inch', label: 'NOC-DP-S-1'},
{id: 2, size: 'small-27-inch', label: 'NOC-DP-S-2'},
{id: 3, size: 'big-42-inch', label: 'NOC-DP-B-3'},
{id: 4, size: 'big-42-inch', label: 'NOC-DP-B-4'},
{id: 5, size: 'small-27-inch', label: 'NOC-DP-S-5'},
{id: 6, size: 'small-27-inch', label: 'NOC-DP-S-6'},
{
id: 7,
size: 'small-27-inch',
label: 'NOC-DP-S-7',
customStyle: 'grid-column: 3 / span 2; justify-self: center;'
},
{
id: 8,
size: 'small-27-inch',
label: 'NOC-DP-S-8',
customStyle: 'grid-column: 5 / span 2; justify-self: center;'
}
]
}
});
</script>