Compare commits

..

34 Commits

Author SHA1 Message Date
b788f54c23 Merge branch 'main' of git.asv.ovh:asvstudiosapps/rpigroupplay 2026-02-14 07:23:58 +01:00
e0cbb2a8f1 inserito ADV WRD 2026-02-14 07:23:34 +01:00
110bc1a1c3 rimosso le versioni precedenti in parentesi 2026-02-10 21:45:19 +01:00
3872a61ad6 vers. 2.4.2 2026-02-10 21:37:59 +01:00
821c353638 corretta la visualizzazione della barra di navigazione dei player 2026-02-07 01:17:17 +01:00
573eeee5f5 corretto la visualizzazione del player 2026-02-07 01:14:12 +01:00
2cdd1cb29b vers. 2.4.1 2026-02-07 01:06:06 +01:00
5b3e05726b vers. 2.4.0 2026-02-07 00:31:18 +01:00
71af2b9407 corretta la visualizzazione della pagina about 2026-02-06 23:51:01 +01:00
a96edaa2b9 Inserito pulsante per i feedback 2026-02-06 23:25:50 +01:00
07864334d4 aggiunta la pagina "about" 2026-02-06 23:08:31 +01:00
ee44bfb7a4 rimosso l'alert per dispositivi ios 2026-02-06 22:19:59 +01:00
2b0b457166 vers. 2.3.0 2026-02-06 22:13:10 +01:00
ae5a4a8153 Inserito alert per iOS 26.2.1 2026-02-05 22:40:41 +01:00
02872cf616 ultima modifica al css 2026-02-04 01:12:28 +01:00
f7ddcda130 # 2026-02-04 01:10:20 +01:00
0982a45307 disattivati le animazioni sui pulsanti 2026-02-04 01:08:58 +01:00
b987bf9062 Update data/changelog.xml
update changelog
2026-02-04 01:02:31 +01:00
c3bac71ad2 Update css/style.css
correzione css
2026-02-04 01:02:11 +01:00
0d76cd4794 FIX: la navbar e il footer venivano oscurati dal loader 2026-02-04 00:57:11 +01:00
3619ababf2 FIX: correzione visualizzazione del caricamento 2026-02-04 00:52:23 +01:00
f562f5bfde vers. 2.2.0 2026-02-04 00:50:09 +01:00
91ebc5c883 vers. 2.1.4 2026-01-30 00:08:08 +01:00
ff9c5896f0 Fix player HLS v2 2026-01-28 23:21:18 +01:00
ea36956669 vers. 2.1.3 2026-01-28 23:14:52 +01:00
3c350b17ae aggiunto player hls v.2 2026-01-28 23:13:46 +01:00
008df364fe spero ultimo bugfix hls mobile 2026-01-28 23:08:43 +01:00
6f68bf123d secondo bugfix hls mobile 2026-01-28 23:03:48 +01:00
891feaed5d bugfix hls mobile 2026-01-28 22:59:59 +01:00
bb8d88f60a ver. 2.1.2 2026-01-28 22:45:13 +01:00
d7e098140c Aggiornamento index.php 2026-01-28 20:24:10 +01:00
4d9c7fd328 modifica gitignore 2026-01-28 19:40:11 +01:00
0b6a3ffea4 modifica Readme.md 2026-01-28 19:31:16 +01:00
52e40799d6 vers. 2.1.1 2026-01-28 19:05:44 +01:00
95 changed files with 1575 additions and 1097 deletions

6
.gitignore vendored
View File

@@ -1 +1,5 @@
*:Zone.Identifier *:Zone.Identifier
*Zone.Identifier
*.old
/api/*
.htaccess

View File

@@ -1,77 +0,0 @@
# Abilita il rewrite engine
RewriteEngine On
# Imposta la directory base (modifica se necessario)
# Se l'app è in una sottocartella, usa: RewriteBase /nome_cartella/
RewriteBase /newapp.rpigroup.it/
# Reindirizza richieste HTTP a HTTPS (opzionale, decommentare se necessario)
# RewriteCond %{HTTPS} off
# RewriteRule ^(.*)$ https://%{HTTP_HOST}%{REQUEST_URI} [L,R=301]
# Non reindirizzare file e cartelle esistenti
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
# Escludi file statici dal rewriting
RewriteCond %{REQUEST_URI} !\.(css|js|jpg|jpeg|png|gif|svg|ico|xml|json|woff|woff2|ttf|eot)$ [NC]
# Reindirizza tutto a index.php mantenendo il path
RewriteRule ^(.*)$ index.php/$1 [L,QSA]
# Impedisci l'accesso diretto a file sensibili
<FilesMatch "^(config|\.htaccess|\.env)">
Order Allow,Deny
Deny from all
</FilesMatch>
# Gestione MIME types
<IfModule mod_mime.c>
AddType application/javascript js
AddType text/css css
AddType image/svg+xml svg
AddType application/vnd.ms-fontobject eot
AddType font/ttf ttf
AddType font/otf otf
AddType font/woff woff
AddType font/woff2 woff2
</IfModule>
# Abilita compressione GZIP (opzionale)
<IfModule mod_deflate.c>
AddOutputFilterByType DEFLATE text/html text/plain text/xml text/css text/javascript application/javascript application/json
</IfModule>
# Cache control (opzionale - commentato perché hai lo script no-cache)
# <IfModule mod_expires.c>
# ExpiresActive On
# ExpiresByType image/jpg "access plus 1 month"
# ExpiresByType image/jpeg "access plus 1 month"
# ExpiresByType image/gif "access plus 1 month"
# ExpiresByType image/png "access plus 1 month"
# ExpiresByType image/svg+xml "access plus 1 month"
# ExpiresByType text/css "access plus 1 week"
# ExpiresByType application/javascript "access plus 1 week"
# </IfModule>
# Header no-cache per sviluppo (rimuovi in produzione se usi la cache)
<IfModule mod_headers.c>
<FilesMatch "\.(html|php)$">
Header set Cache-Control "no-cache, no-store, must-revalidate"
Header set Pragma "no-cache"
Header set Expires "0"
</FilesMatch>
</IfModule>
# Sicurezza aggiuntiva
<IfModule mod_headers.c>
Header set X-Content-Type-Options "nosniff"
Header set X-Frame-Options "SAMEORIGIN"
Header set X-XSS-Protection "1; mode=block"
</IfModule>
# Disabilita directory listing
Options -Indexes
# Abilita follow symlinks (necessario per RewriteRule)
Options +FollowSymLinks

Binary file not shown.

Binary file not shown.

View File

@@ -3,8 +3,76 @@
======================================== */ ======================================== */
/* ========================================
LOADING OVERLAY
======================================== */
/* Overlay di caricamento a schermo intero */
.loading-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(16, 25, 75, 0.95);
backdrop-filter: blur(8px);
display: flex;
justify-content: center;
align-items: center;
z-index: 9999;
opacity: 0;
visibility: hidden;
transition: opacity 0.3s ease, visibility 0.3s ease;
}
/* Quando l'overlay è attivo */
.loading-overlay.active {
opacity: 1;
visibility: visible;
}
/* Contenuto dell'overlay */
.loading-content {
text-align: center;
color: white;
}
/* Spinner grande per il loading overlay */
.spinner-large {
width: 60px;
height: 60px;
border: 5px solid rgba(255, 255, 255, 0.2);
border-top-color: #f7b835;
border-radius: 50%;
animation: spin 0.8s linear infinite;
margin: 0 auto 20px;
}
/* Testo di caricamento */
.loading-text {
font-size: 1.1rem;
font-weight: 500;
color: white;
margin: 0;
animation: pulse 1.5s ease-in-out infinite;
}
/* Animazione pulse per il testo */
@keyframes pulse {
0%,
100% {
opacity: 1;
}
50% {
opacity: 0.5;
}
}
/* Transizione per i link di navigazione */ /* Transizione per i link di navigazione */
.navLink, /* .navLink,
.nav-link, .nav-link,
.station-link, .station-link,
.linkBox { .linkBox {
@@ -13,7 +81,7 @@
} }
/* Effetto hover sui link */ /* Effetto hover sui link */
.navLink:hover, /* .navLink:hover,
.nav-link:hover, .nav-link:hover,
.station-link:hover, .station-link:hover,
.linkBox:hover { .linkBox:hover {
@@ -25,10 +93,13 @@
@keyframes pulseActive { @keyframes pulseActive {
0%, 100% {
0%,
100% {
opacity: 1; opacity: 1;
transform: translateX(-50%) scale(1); transform: translateX(-50%) scale(1);
} }
50% { 50% {
opacity: 0.6; opacity: 0.6;
transform: translateX(-50%) scale(1.2); transform: translateX(-50%) scale(1.2);
@@ -36,7 +107,7 @@
} }
/* Effetto click/tap */ /* Effetto click/tap */
.navLink:active, /* .navLink:active,
.nav-link:active, .nav-link:active,
.station-link:active, .station-link:active,
.linkBox:active { .linkBox:active {
@@ -45,7 +116,7 @@
} }
/* Transizione per le card delle stazioni */ /* Transizione per le card delle stazioni */
.station-card { /*.station-card {
transition: transform 0.3s ease, box-shadow 0.3s ease; transition: transform 0.3s ease, box-shadow 0.3s ease;
} }
@@ -55,7 +126,7 @@
} }
/* Transizione per i clickBox della home */ /* Transizione per i clickBox della home */
.clickBox { /* .clickBox {
transition: all 0.3s ease; transition: all 0.3s ease;
} }
@@ -100,7 +171,7 @@
} }
/* Transizione per elementi che appaiono */ /* Transizione per elementi che appaiono */
.content-page, /*.content-page,
.station-list, .station-list,
.player-container { .player-container {
animation: fadeInContent 0.5s ease-out; animation: fadeInContent 0.5s ease-out;
@@ -116,7 +187,7 @@
} }
/* Transizione per le immagini */ /* Transizione per le immagini */
.station-logo, /* .station-logo,
.station-logo-large { .station-logo-large {
transition: transform 0.3s ease, filter 0.3s ease; transition: transform 0.3s ease, filter 0.3s ease;
} }
@@ -128,7 +199,7 @@
} }
/* Transizione per i pulsanti */ /* Transizione per i pulsanti */
button, /* button,
.submit-btn, .submit-btn,
.play-pause-btn { .play-pause-btn {
transition: all 0.3s ease; transition: all 0.3s ease;
@@ -148,7 +219,7 @@ button:active,
} }
/* Transizione smooth per tutti gli elementi interattivi */ /* Transizione smooth per tutti gli elementi interattivi */
a, button, input, textarea, select { /* a, button, input, textarea, select {
transition: all 0.2s ease; transition: all 0.2s ease;
} }
@@ -156,7 +227,7 @@ a, button, input, textarea, select {
/* Non ci sono più after pseudo-elementi per le underline */ /* Non ci sono più after pseudo-elementi per le underline */
/* Transizione per il back-link */ /* Transizione per il back-link */
.back-link a { /* .back-link a {
transition: all 0.3s ease; transition: all 0.3s ease;
display: inline-block; display: inline-block;
} }
@@ -167,7 +238,7 @@ a, button, input, textarea, select {
} }
/* Animazione per le liste */ /* Animazione per le liste */
.stations-container { /* .stations-container {
display: grid; display: grid;
gap: 1rem; gap: 1rem;
} }
@@ -194,7 +265,7 @@ a, button, input, textarea, select {
} }
/* Transizione per i form */ /* Transizione per i form */
.form-group input, /* .form-group input,
.form-group textarea, .form-group textarea,
.form-group select { .form-group select {
transition: border-color 0.3s ease, box-shadow 0.3s ease; transition: border-color 0.3s ease, box-shadow 0.3s ease;
@@ -208,7 +279,7 @@ a, button, input, textarea, select {
} }
/* Animazione per i messaggi di risposta */ /* Animazione per i messaggi di risposta */
.form-response { /* .form-response {
animation: slideInDown 0.4s ease-out; animation: slideInDown 0.4s ease-out;
} }
@@ -224,7 +295,7 @@ a, button, input, textarea, select {
} }
/* Transizione per video e iframe */ /* Transizione per video e iframe */
video, /* video,
iframe { iframe {
transition: opacity 0.3s ease; transition: opacity 0.3s ease;
} }
@@ -235,17 +306,17 @@ iframe:hover {
} }
/* Performance optimization */ /* Performance optimization */
* { /** {
-webkit-tap-highlight-color: transparent; -webkit-tap-highlight-color: transparent;
} }
/* Smooth scrolling */ /* Smooth scrolling */
html { /* html {
scroll-behavior: smooth; scroll-behavior: smooth;
} }
/* Riduzione movimento per chi ha impostato preferenze di accessibilità */ /* Riduzione movimento per chi ha impostato preferenze di accessibilità */
@media (prefers-reduced-motion: reduce) { /*@media (prefers-reduced-motion: reduce) {
*, *,
*::before, *::before,
*::after { *::after {

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@@ -1,400 +1,421 @@
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Black.ttf') format('truetype'); src: url('fonts/Poppins-Black.ttf') format('truetype');
font-weight: 900; font-weight: 900;
font-style: normal; font-style: normal;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-BlackItalic.ttf') format('truetype'); src: url('fonts/Poppins-BlackItalic.ttf') format('truetype');
font-weight: 900; font-weight: 900;
font-style: italic; font-style: italic;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Bold.ttf') format('truetype'); src: url('fonts/Poppins-Bold.ttf') format('truetype');
font-weight: 700; font-weight: 700;
font-style: normal; font-style: normal;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-BoldItalic.ttf') format('truetype'); src: url('fonts/Poppins-BoldItalic.ttf') format('truetype');
font-weight: 700; font-weight: 700;
font-style: italic; font-style: italic;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Medium.ttf') format('truetype'); src: url('fonts/Poppins-Medium.ttf') format('truetype');
font-weight: 500; font-weight: 500;
font-style: normal; font-style: normal;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-MediumItalic.ttf') format('truetype'); src: url('fonts/Poppins-MediumItalic.ttf') format('truetype');
font-weight: 500; font-weight: 500;
font-style: italic; font-style: italic;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Regular.ttf') format('truetype'); src: url('fonts/Poppins-Regular.ttf') format('truetype');
font-weight: 400; font-weight: 400;
font-style: normal; font-style: normal;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Italic.ttf') format('truetype'); src: url('fonts/Poppins-Italic.ttf') format('truetype');
font-weight: 400; font-weight: 400;
font-style: italic; font-style: italic;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-Light.ttf') format('truetype'); src: url('fonts/Poppins-Light.ttf') format('truetype');
font-weight: 300; font-weight: 300;
font-style: normal; font-style: normal;
} }
@font-face { @font-face {
font-family: 'Poppins'; font-family: 'Poppins';
src: url('fonts/Poppins-LightItalic.ttf') format('truetype'); src: url('fonts/Poppins-LightItalic.ttf') format('truetype');
font-weight: 300; font-weight: 300;
font-style: italic; font-style: italic;
} }
body { body {
font-family: 'Poppins', sans-serif !important; font-family: 'Poppins', sans-serif !important;
} }
* { * {
margin: 0; margin: 0;
padding: 0; padding: 0;
user-select: none; user-select: none;
} }
html, html,
body { body {
overscroll-behavior: none; overscroll-behavior: none;
} }
body { body {
font-family: "Poppins", sans-serif; font-family: "Poppins", sans-serif;
height: 100vh; height: 100vh;
overflow: hidden; overflow: hidden;
} }
/* DESKTOP SECTION */ /* DESKTOP SECTION */
body.desktopBody { body.desktopBody {
background: #2a377e; background: #2a377e;
color: white; color: white;
} }
div.dbox, div.dbox,
div.dbox_mobile { div.dbox_mobile {
position: fixed; position: fixed;
top: 50%; top: 50%;
left: 50%; left: 50%;
transform: translate(-50%, -50%); transform: translate(-50%, -50%);
background: #4358ca8f; background: #4358ca8f;
padding: 25px; padding: 25px;
border-radius: 50px; border-radius: 50px;
text-align: center; text-align: center;
width: 500px; width: 500px;
} }
div.dbox_mobile { div.dbox_mobile {
width: 90%; width: 90%;
max-width: 450px; max-width: 750px;
color: white; color: white;
border-radius: 10px; border-radius: 10px;
max-height: 475px; max-height: 475px;
overflow: auto; overflow: auto;
} }
div.dbox>h1.title, div.dbox>h1.title,
div.dbox_mobile>h1.title { div.dbox_mobile>h1.title {
margin-bottom: 0px; margin-bottom: 0px;
} }
div.dbox>hr, div.dbox>hr,
div.dbox_mobile>hr { div.dbox_mobile>hr {
margin: 2rem 0; margin: 2rem 0;
} }
div.dbox>.dbtn, div.dbox>.dbtn,
div.dbox_mobile>.dbtn { div.dbox_mobile>.dbtn {
border: none; border: none;
border-radius: 50px; border-radius: 50px;
padding: 10px 20px; padding: 10px 20px;
background: #f0f0f0; background: #f0f0f0;
color: black; color: black;
text-decoration: none; text-decoration: none;
font-weight: 600; font-weight: 600;
-webkit-appearance: button; -webkit-appearance: button;
} }
div.dbox>.dbtn:hover { div.dbox>.dbtn:hover {
background: #e0e0e0; background: #e0e0e0;
cursor: pointer; cursor: pointer;
} }
div.dfooter { div.dfooter {
position: fixed; position: fixed;
bottom: 15px; bottom: 15px;
left: 50%; left: 50%;
transform: translateX(-50%); transform: translateX(-50%);
text-align: center; text-align: center;
color: #ffffff90; color: #ffffff90;
font-size: 13px; font-size: 13px;
width: 100%; width: 100%;
} }
/* MOBILE SECTION */ /* MOBILE SECTION */
.appBody { .appBody {
background-color: #10194b; background-color: #10194b;
max-width: 450px; max-width: 750px;
min-width: 330px; min-width: 330px;
margin: auto; margin: auto;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
min-height: 100vh; min-height: 100vh;
} }
.header>.logo-section, .header>.logo-section,
.header>.menu-section { .header>.menu-section {
text-align: center; text-align: center;
} }
.header>.logo-section, .header>.logo-section,
.footer>.copyright-section { .footer>.copyright-section {
background-color: #2a377e; background-color: #2a377e;
padding: 15px 0; padding: 15px 0;
} }
.header>.menu-section { .header>.menu-section {
background-color: #3849a8; background-color: #3849a8;
padding: 13px 0; padding: 13px 0;
} }
.header>.logo-section>img { .header>.logo-section>img {
width: 250px; width: 250px;
} }
.header>.menu-section>.navLink { .header>.menu-section>.navLink {
color: white; color: white;
text-decoration: none; text-decoration: none;
margin: 0 15px; margin: 0 15px;
} }
.header>.menu-section>.navLink.active { .header>.menu-section>.navLink.active {
color: #2a377e; color: #2a377e;
margin: 0; margin: 0;
padding: 3px 15px; padding: 3px 15px;
background: white; background: white;
border-radius: 50px; border-radius: 50px;
font-weight: 500; font-weight: 500;
} }
main { main {
background-color: white; background-color: white;
flex: 1; flex: 1;
overflow: auto; overflow: auto;
} }
.titlePage { .titlePage {
text-align: center; text-align: center;
font-weight: 600; font-weight: 600;
margin: 15px 0 20px; margin: 15px 0 20px;
} }
.subtitlePage { .subtitlePage {
text-align: center; text-align: center;
font-weight: 500; font-weight: 500;
margin: -15px 0 20px; margin: -15px 0 20px;
} }
.linkBox { .linkBox {
text-decoration: none; text-decoration: none;
color: black; color: black;
} }
.clickBox { .clickBox {
width: 100%; width: 100%;
border: 2px solid #2a377e; border: 2px solid #2a377e;
text-align: center; text-align: center;
border-radius: 20px; border-radius: 20px;
font-weight: 500; font-weight: 500;
/* margin-bottom: 10px; */ /* margin-bottom: 10px; */
padding: 4px; padding: 4px;
} }
.clickBox.Squared { .clickBox.Squared {
font-weight: 600; font-weight: 600;
border: solid 4px #2a377e; border: solid 4px #2a377e;
} }
.clickBox>img { .clickBox>img {
width: 80%; width: 80%;
} }
.changelogTitle { .changelogTitle {
font-weight: 600; font-weight: 600;
padding: 0 13px; padding: 0 13px;
margin: 0 0 15px; margin: 0 0 15px;
} }
.changelogList, .changelogList,
.tec, .tec,
.stationList { .stationList {
padding: 0 13px; padding: 0 13px;
margin: 0 0 7px; margin: 0 0 10px;
text-align: justify; text-align: justify;
hyphens: auto; hyphens: auto;
} }
.stationList { .stationList {
text-align: center; text-align: center;
} }
.stationCard>a.stationLink { .stationCard>a.stationLink {
padding: 0; padding: 0;
} }
.stationCard>a.stationLink>img { .stationCard>a.stationLink>img {
width: 100%; width: 100%;
border-radius: 20px; border-radius: 20px;
} }
/* .stationCard>.thematicBadge { /* .stationCard>.thematicBadge {
position: relative; position: relative;
background: #f7b835; background: #f7b835;
padding: 1px 5px; padding: 1px 5px;
font-size: 0.8rem; font-size: 0.8rem;
border-radius: 27px; border-radius: 27px;
font-style: italic; font-style: italic;
font-weight: 600; font-weight: 600;
top: 31px; top: 31px;
left: 35px; left: 35px;
margin: 0; margin: 0;
} */ } */
.stationCard.isthematic{ .stationCard.isthematic {
text-align: right; text-align: right;
margin-top: -24px; margin-top: -24px;
} }
.stationCard.isthematic:before{ .stationCard.isthematic:before {
content: "Tematica"; content: "Tematica";
position: relative; position: relative;
background: #f7b835; background: #f7b835;
padding: 1px 5px; padding: 1px 5px;
font-size: 0.8rem; font-size: 0.8rem;
border-radius: 27px; border-radius: 27px;
font-style: italic; font-style: italic;
font-weight: 600; font-weight: 600;
top: 30px; top: 30px;
right: 7px; right: 7px;
margin: 0; margin: 0;
} }
iframe.contentplayer{ iframe.contentplayer {
position: absolute; height: calc(100vh - 393px);
left: 50%; width: 100%;
height: calc(100vh - 312px); max-width: 750px;
width: 100%; }
max-width: 450px;
transform: translateX(-50%); .footer_player {
} background: #f7b835;
color: #2a377d;
.footer_player{ z-index: 90;
background: #f7b835; width: 100%;
color: #2a377d; max-width: 750px;
position: fixed; height: 100px;
z-index: 90; padding: 20px;
bottom: 70px; border-top: 1px solid #eee;
left: 50%; flex: 1;
transform: translateX(-50%); align-content: end;
width: 100%; }
max-width: 450px;
height: 100px; .footer_player>.row>.col-2>img {
border-top-left-radius: 8px; border-radius: 5px;
border-top-right-radius: 8px; }
padding: 20px;
} button#playPauseBtn,
button#formatToggleBtn {
.footer_player > .row > .col-2 > img{ background: none;
border-radius: 5px; border: none;
} }
button#playPauseBtn{ .footer,
background: none; .header {
border: none; z-index: 10000;
} }
.footer{ .footer>.menu-section {
z-index: 1; background: #3849a8;
} font-size: 0.7rem;
padding: 7px 0;
.footer>.menu-section { text-align: center;
background: #3849a8; }
font-size: 0.7rem;
padding: 7px 0; .footer>.menu-section>a {
text-align: center; color: white;
} text-decoration: none;
margin: 0 5px;
.footer>.menu-section>a { }
color: white;
text-decoration: none; .footer>.copyright-section {
margin: 0 5px; color: white;
} font-size: 0.8rem;
text-align: center;
.footer>.copyright-section { padding: 10px;
color: white; font-weight: 500;
font-size: 0.8rem; }
text-align: center;
padding: 10px; @media (max-width: 450px) {
font-weight: 500; .appBody {
} background-color: #2a377e;
}
@media (max-width: 450px) { }
.appBody {
background-color: #2a377e;
} /* Loading spinner */
} .loading {
display: flex;
justify-content: center;
/* Loading spinner */ align-items: center;
.loading { min-height: 200px;
display: flex; }
justify-content: center;
align-items: center; .spinner {
min-height: 200px; width: 40px;
} height: 40px;
border: 4px solid rgba(0, 0, 0, 0.1);
.spinner { border-radius: 50%;
width: 40px; border-top-color: #3498db;
height: 40px; animation: spin 1s ease-in-out infinite;
border: 4px solid rgba(0, 0, 0, 0.1); }
border-radius: 50%;
border-top-color: #3498db; @keyframes spin {
animation: spin 1s ease-in-out infinite; to {
} transform: rotate(360deg);
}
@keyframes spin { }
to {
transform: rotate(360deg);
} /* Forza orientamento portrait - nasconde contenuto in landscape */
} @media screen and (orientation: landscape) and (max-height: 450px) {
body.appBody::after {
content: "Ruota il dispositivo in verticale";
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: #2a377e;
color: white;
display: flex;
align-items: center;
justify-content: center;
font-size: 1.2rem;
font-weight: 600;
z-index: 9999;
text-align: center;
padding: 20px;
}
body.appBody>* {
display: none !important;
}
}

Binary file not shown.

View File

@@ -1,318 +1,385 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<changelog> <changelog>
<version>
<version> <number>2.4.2</number>
<number>2.1.1</number> <logs>
<logs> <log>Aggiornata la grandezza della finestra dell'applicazione da desktop.</log>
<log>Corretti alcuni bug che impedivano l'accesso al player dal link esterno</log> <log>Inserito il link della repository su ASV Git all'interno della pagina del changelog.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>2.1.0</number> <number>2.4.1</number>
<logs> <logs>
<log>Risoluzione dei problemi minori presenti nel codice, che causava problemi nella navigazione in app</log> <log>Aggiornata la pagiana del player audio, per renderla più coerente con le altre pagine dell'applicazione.</log>
<log>Risoluzione dei problemi minori presenti nel codice, che causava problemi di riproduzione audio al player</log> <log>Aggiunta la visualizzazione dell'artista e del brano in riproduzione.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>2.0.4</number> <number>2.4.0</number>
<logs> <logs>
<log>Blocco rotazione: è stato risolto il problema della rotazione schermo dell'applicazione. Ora rimane fissa in verticale.</log> <log>Implementato il player video dedicato per la riproduzione dei canali visivi.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>2.0.3</number> <number>2.3.0</number>
<logs> <logs>
<log>Corretta la visualizzazione dei contenuti forniti dalle emittenti all'interno del player</log> <log>Ottimizzata la risoluzione dell'applicazione su dispositivi larghi (tablet, iPad e computer).</log>
<log>Corretta la visualizzazione del player audio</log> <log>Implementato lo switch tra il player audio in HLS e il player audio in MP3/AAC.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Rimosso l'avviso per i dispositivi iOS.</log>
</logs> <log>Correzione e bugfix di problematiche varie.</log>
</version> </logs>
</version>
<version>
<number>2.0.2</number> <version>
<logs> <number>2.2.0</number>
<log>Inserita la pagina statica per le emittenti tematiche</log> <logs>
<log>Correzione e bugfix di problematiche varie.</log> <log>E' stato reintrodotto la schermata di caricamento ad ogni selezione di ogni pagina dell'applicazione.</log>
</logs> <log>Correzione e bugfix di problematiche varie.</log>
</version> </logs>
</version>
<version>
<number>2.0.1</number> <version>
<logs> <number>2.1.4</number>
<log>Aggiunta la Visual Radio dell'emittente "Radio Città 105</log> <logs>
<log>Correzione e bugfix di problematiche varie.</log> <log>Corretto la visione verticale sui dispositivi mobili.</log>
</logs> <log>Correzione e bugfix di problematiche varie.</log>
</version> </logs>
</version>
<version>
<number>2.0.0</number> <version>
<logs> <number>2.1.3</number>
<log>Nuova UI/UX: RPIGroup aggiorna la veste grafica della sua applicazione, rendendola più "fumettosa" e "giocattolosa". Un'estetica completamente diversa da tutte le altre varie app radiofoniche</log> <logs>
<log>Nuova Engine: Nuovo motore e struttura dell'applicazione. Lato backend è cambiato completamente rispetto alla versione 1.</log> <log>Implementato il nuovo player audio per la riproduzione dei flussi audio in HLS per il bitrate adattivo.</log>
<log>Nuovo Player Audio/Video: Player più semplice, ma conserva le caratteristiche della precedente versione.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Termini e Condizioni: inserimento per obblighi di legge dei corrispettivi "Termini e Condizioni"</log> </logs>
<log>Policy Privacy: inserimento per obblighi di legge dei corrispettivi "Policy Privacy"</log> </version>
<log>Dalla versione 2 e successive, è stata rimossa la dicitura "stable" in quanto non esistono più varie versioni dell'applicazione.</log>
<log>Correzione e bugfix di problematiche varie.</log> <version>
</logs> <number>2.1.2</number>
</version> <logs>
<log>Implementazione del sistema di qualità adattiva (ABR) per il player audio.</log>
<version> <log>Correzione e bugfix di problematiche varie.</log>
<number>1.2.1 Stable</number> </logs>
<logs> </version>
<log>Aggiunta la pagina sul chiarimento dei "Diritti d'Autore".</log>
<log>Correzione e bugfix di problematiche varie.</log> <version>
</logs> <number>2.1.1</number>
</version> <logs>
<log>Corretti alcuni bug che impedivano l'accesso al player dal link esterno.</log>
<version> <log>Correzione e bugfix di problematiche varie.</log>
<number>1.2.0 Stable</number> </logs>
<logs> </version>
<log>Implementato il "Media Sessions" che permette di visualizzare la radio in riproduzione nel centro notifiche su iOS e Android</log>
<log>Preparazione dell'ottimizzazione del software in occasione della terza versione dell'app.</log> <version>
<log>Correzione e bugfix di problematiche varie.</log> <number>2.1.0</number>
</logs> <logs>
</version> <log>Risoluzione dei problemi minori presenti nel codice, che causava problemi nella navigazione in app.</log>
<log>Risoluzione dei problemi minori presenti nel codice, che causava problemi di riproduzione audio al player.</log>
<version> <log>Correzione e bugfix di problematiche varie.</log>
<number>1.1.1 Stable</number> </logs>
<logs> </version>
<log>Correzione bug che impediva il caricamento del logo nell'icona dell'app sui smartphone.</log>
<log>Risolto il problema dell'overscrolling su smartphone.</log> <version>
<log>Correzione e bugfix di problematiche varie.</log> <number>2.0.4</number>
</logs> <logs>
</version> <log>Blocco rotazione: è stato risolto il problema della rotazione schermo dell'applicazione. Ora rimane fissa in verticale.</log>
<log>Correzione e bugfix di problematiche varie.</log>
<version> </logs>
<number>1.1.0 Stable</number> </version>
<logs>
<log>Aggiunta la nuova stazione radio tematica "RDL Revival 70-80-90"</log> <version>
<log>Migliorata la visualizzazione del selettore radio della pagina home</log> <number>2.0.3</number>
<log>Implementato nel player la visualizzazione della pagina statica per le radio tematiche</log> <logs>
<log>Correzione e bugfix di problematiche varie causate dall'ultima versione "Beta".</log> <log>Corretta la visualizzazione dei contenuti forniti dalle emittenti all'interno del player.</log>
</logs> <log>Corretta la visualizzazione del player audio.</log>
</version> <log>Correzione e bugfix di problematiche varie.</log>
</logs>
<version> </version>
<number>1.0.0 Stable</number>
<logs> <version>
<log>Passaggio alla versione "Stable" dell'applicazione</log> <number>2.0.2</number>
<log>Verifica di ulteriori correzioni dal passaggio della versione stabile</log> <logs>
<log>Leggerimento dell'applicazione a livello backend</log> <log>Inserita la pagina statica per le emittenti tematiche.</log>
<log>Ulteriori analisi di stabilità dal momento del passaggio alla versione stabile</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Correzione e bugfix di problematiche varie causate dall'ultima versione "Beta".</log> </logs>
</logs> </version>
</version>
<version>
<version> <number>2.0.1</number>
<number>0.24.0 Beta</number> <logs>
<logs> <log>Aggiunta la Visual Radio dell'emittente "Radio Città 105".</log>
<log>Aggiormaneto player audio - Riproduzione dei flussi audio in HLS per il bitrate adattivo. Le radio iscritte devono supportare il protocollo HLS.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Supporto all'interscambio della connessione - Al cambiare di tipologia (Wifi o Rete Cellulare), il flusso audio non si interrompe.</log> </logs>
<log>Preparazione del codice alla versione "1.0.0 Stable"</log> </version>
<log>Correzione e bugfix di problematiche varie.</log>
</logs> <version>
</version> <number>2.0.0</number>
<logs>
<version> <log>Nuova UI/UX: RPIGroup aggiorna la veste grafica della sua applicazione, rendendola più "fumettosa" e "giocattolosa". Un'estetica completamente diversa da tutte le altre varie app radiofoniche.</log>
<number>0.23.3 Beta</number> <log>Nuova Engine: Nuovo motore e struttura dell'applicazione. Lato backend è cambiato completamente rispetto alla versione 1.</log>
<logs> <log>Nuovo Player Audio/Video: Player più semplice, ma conserva le caratteristiche della precedente versione.</log>
<log>Inserimento del nuovo flusso video per il canale "RC105 TV".</log> <log>Termini e Condizioni: inserimento per obblighi di legge dei corrispettivi "Termini e Condizioni"</log>
<log>Aggiornamento player video - Inserita l'anteprima del flusso video</log> <log>Policy Privacy: inserimento per obblighi di legge dei corrispettivi "Policy Privacy"</log>
<log>Correzione accesso al player video con l'inserimento automatico dell'anteprima e flusso video del canale selezionato</log> <log>Dalla versione 2 e successive, è stata rimossa la dicitura "stable" in quanto non esistono più varie versioni dell'applicazione.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.23.2 Beta</number> <number>1.2.1 Stable</number>
<logs> <logs>
<log>Inizializzazione cambio interfaccia e modernamento.</log> <log>Aggiunta la pagina sul chiarimento dei "Diritti d'Autore".</log>
<log>Reinserimento della schermata di caricamento nel player.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Correzione e bugfix di problematiche varie.</log> </logs>
</logs> </version>
</version>
<version>
<version> <number>1.2.0 Stable</number>
<number>0.23.1 Beta</number> <logs>
<logs> <log>Implementato il "Media Sessions" che permette di visualizzare la radio in riproduzione nel centro notifiche su iOS e Android.</log>
<log>Rimozione del logo al caricamento di ogni singola pagina (tranne all'avvio dell'app)</log> <log>Preparazione dell'ottimizzazione del software in occasione della terza versione dell'app.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.23.0 Beta</number> <number>1.1.1 Stable</number>
<logs> <logs>
<log>Rilasciato il nuovo player video</log> <log>Correzione bug che impediva il caricamento del logo nell'icona dell'app sui smartphone.</log>
<log>Inserimento dell'emittente RC105TV nella lista delle WebTV</log> <log>Risolto il problema dell'overscrolling su smartphone.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.22.4 Beta</number> <number>1.1.0 Stable</number>
<logs> <logs>
<log>Correzione errori minimi nel sistema</log> <log>Aggiunta la nuova stazione radio tematica "RDL Revival 70-80-90".</log>
<log>Aggiunta indicatore della versione app nella schermata desktop</log> <log>Migliorata la visualizzazione del selettore radio della pagina home.</log>
<log>Preparazione player video - Correzioni minimi player e aggiunta di pagine mancanti</log> <log>Implementato nel player la visualizzazione della pagina statica per le radio tematiche.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie causate dall'ultima versione "Beta".</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.22.3 Beta</number> <number>1.0.0 Stable</number>
<logs> <logs>
<log>Migliorati i tempi di caricamento dei player</log> <log>Passaggio alla versione "Stable" dell'applicazione.</log>
<log>Aggiunta nuovi file di Configurazione</log> <log>Verifica di ulteriori correzioni dal passaggio della versione stabile.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Leggerimento dell'applicazione a livello backend.</log>
</logs> <log>Ulteriori analisi di stabilità dal momento del passaggio alla versione stabile.</log>
</version> <log>Correzione e bugfix di problematiche varie causate dall'ultima versione "Beta".</log>
</logs>
<version> </version>
<number>0.22.2 Beta</number>
<logs> <version>
<log>Corretto la visualizzazione del player da desktop.</log> <number>0.24.0 Beta</number>
<log>Corretto la visualizzazione del font scelto per la webapp "Rubik".</log> <logs>
<log>Varie ottimizzazioni per la valutazione finale di Google Page Speed.</log> <log>Aggiormaneto player audio - Riproduzione dei flussi audio in HLS per il bitrate adattivo. Le radio iscritte devono supportare il protocollo HLS.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Supporto all'interscambio della connessione - Al cambiare di tipologia (Wifi o Rete Cellulare), il flusso audio non si interrompe.</log>
</logs> <log>Preparazione del codice alla versione "1.0.0 Stable"</log>
</version> <log>Correzione e bugfix di problematiche varie.</log>
</logs>
<version> </version>
<number>0.22.1 Beta</number>
<logs> <version>
<log>Corretto il bug del logo all'interno dell'homepage</log> <number>0.23.3 Beta</number>
<log>Corretto la riproduzione audio dell'emittente Radio Città 105</log> <logs>
<log>Correzione e bugfix di problematiche varie.</log> <log>Inserimento del nuovo flusso video per il canale "RC105 TV".</log>
</logs> <log>Aggiornamento player video - Inserita l'anteprima del flusso video</log>
</version> <log>Correzione accesso al player video con l'inserimento automatico dell'anteprima e flusso video del canale selezionato</log>
<log>Correzione e bugfix di problematiche varie.</log>
<version> </logs>
<number>0.22.0 Beta</number> </version>
<logs>
<log>Cambiata la struttura interna dell'app, con unificazione della visualizzazione mobile con quella desktop.</log> <version>
<log>Cambiata la configurazione dell'app. Ottimizzata per ridurre i tempi di attesa e output della pagina.</log> <number>0.23.2 Beta</number>
<log>Le emittenti registrate sono inserite all'interno di un file XML apposito.</log> <logs>
<log>Aggiornato il player interattivo. Ora disponibile di default per tutte le emittenti.</log> <log>Inizializzazione cambio interfaccia e modernamento.</log>
<log>Ottimizzata la homescreen con correzioni di vari bug.</log> <log>Reinserimento della schermata di caricamento nel player.</log>
<log>Correzione e bugfix di problematiche varie.</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.21.0 Beta</number> <number>0.23.1 Beta</number>
<logs> <logs>
<log>Realizzazione del nuovo player interattivo. Disponibile per smartphone e desktop (Al momento, è disponibile solo per l'emittente RDL).</log> <log>Rimozione del logo al caricamento di ogni singola pagina (tranne all'avvio dell'app).</log>
<log>Disattivato il precedente player mobile. Verrà riattivato esclusivamente per "emittenti" sprovviste di pagine dinamiche.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Correzione e bugfix di problematiche varie.</log> </logs>
</logs> </version>
</version>
<version>
<version> <number>0.23.0 Beta</number>
<number>0.20.0 Beta</number> <logs>
<logs> <log>Rilasciato il nuovo player video.</log>
<log>L'app entra ufficialmente nello stato di Beta-testing.</log> <log>Inserimento dell'emittente RC105TV nella lista delle WebTV.</log>
<log>Cambiata la modalità di indicazione della versione, non più basata sulla data di aggiornamento, ma sul numero di update dell'app.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Avviata ufficialmente la "Roadmap" sulla Versione 1.0.0 (indicata a fine pagina CHANGELOG).</log> </logs>
<log>Aggiunto script per iOS e iPadOS sul rilevamento uso browser/app.</log> </version>
<log>Correzione e bugfix script rilevamento app Android.</log>
</logs> <version>
</version> <number>0.22.4 Beta</number>
<logs>
<version> <log>Correzione errori minimi nel sistema.</log>
<number>0.12.6 Alpha (Last Release)</number> <log>Aggiunta indicatore della versione app nella schermata desktop.</log>
<logs> <log>Preparazione player video - Correzioni minimi player e aggiunta di pagine mancanti.</log>
<log>Aggiornata la lista emittenti nel menu in alto sinistra.</log> <log>Correzione e bugfix di problematiche varie.</log>
<log>Aggiornate le pagine "Cosa è RPIGRPUP PLAY", "COME FUNZIONA L'APP" e "HAI BISOGNO DI AIUTO".</log> </logs>
<log>Corretto bug inizializzazione app per iOS e iPadOS.</log> </version>
<log>Aggiunto script di controllo gestione app per iOS e iPadOS.</log>
<log>Correzione e bugfix di problematiche varie.</log> <version>
</logs> <number>0.22.3 Beta</number>
</version> <logs>
<log>Migliorati i tempi di caricamento dei player.</log>
<version> <log>Aggiunta nuovi file di Configurazione.</log>
<number>0.12.5 Aplha</number> <log>Correzione e bugfix di problematiche varie.</log>
<logs> </logs>
<log>Realizzazione file XML per il Changelog.</log> </version>
<log>Aggiunto pulsante per la visualizzazione del Changelog.</log>
<log>Correzione pagina About per la visualizzazione del Changelog.</log> <version>
<log>Correzione visualizzazione del Home Page.</log> <number>0.22.2 Beta</number>
<log>Reso automatico il cambio di versione dell'app in base all'ultima versione disponibile e dichiarata all'intero del Changelog.</log> <logs>
</logs> <log>Corretto la visualizzazione del player da desktop.</log>
</version> <log>Corretto la visualizzazione del font scelto per la webapp "Rubik".</log>
<log>Varie ottimizzazioni per la valutazione finale di Google Page Speed.</log>
<version> <log>Correzione e bugfix di problematiche varie.</log>
<number>0.12.2 Alpha</number> </logs>
<logs> </version>
<log>Corretto il problema del caricamento del "Player Mobile" su iOS e derivati.</log>
<log>Corretto e aggiunto nuovi script del "Player Mobile"</log> <version>
<log>Aggiornato slogan dell'emittente "RDL XMAS"</log> <number>0.22.1 Beta</number>
<log>Aggiunto il Changelog</log> <logs>
</logs> <log>Corretto il bug del logo all'interno dell'homepage.</log>
</version> <log>Corretto la riproduzione audio dell'emittente Radio Città 105.</log>
<log>Correzione e bugfix di problematiche varie.</log>
<version> </logs>
<number>0.11.30 Alpha</number> </version>
<logs>
<log>Corretto la visualizzazione del "Player Mobile"</log> <version>
<log>Aggiornata la lista emittenti nel menu in alto sinistra</log> <number>0.22.0 Beta</number>
<log>Rimosse momentaneamente le WebTV nel menu in alto sinistra</log> <logs>
</logs> <log>Cambiata la struttura interna dell'app, con unificazione della visualizzazione mobile con quella desktop.</log>
</version> <log>Cambiata la configurazione dell'app. Ottimizzata per ridurre i tempi di attesa e output della pagina.</log>
<log>Le emittenti registrate sono inserite all'interno di un file XML apposito.</log>
<version> <log>Aggiornato il player interattivo. Ora disponibile di default per tutte le emittenti.</log>
<number>0.11.29 Alpha</number> <log>Ottimizzata la homescreen con correzioni di vari bug.</log>
<logs> <log>Correzione e bugfix di problematiche varie.</log>
<log>Aggiunta nuova emittente "RDL XMAS"</log> </logs>
<log>Corretta la visualizzazione del "PLAYER DESKTOP"</log> </version>
<log>Corretta la visualizzazione della lista emittenti in "Home Desktop" e "Home Mobile"</log>
<log>Abilitato il "Player Mobile" anche in ambiente Desktop</log> <version>
<log>Aggiornato script che impedisce il refresh della pagina da Mobile</log> <number>0.21.0 Beta</number>
</logs> <logs>
</version> <log>Realizzazione del nuovo player interattivo. Disponibile per smartphone e desktop (Al momento, è disponibile solo per l'emittente RDL).</log>
<log>Disattivato il precedente player mobile. Verrà riattivato esclusivamente per "emittenti" sprovviste di pagine dinamiche.</log>
<version> <log>Correzione e bugfix di problematiche varie.</log>
<number>0.9.26 Alpha</number> </logs>
<logs> </version>
<log>Cambiata l'URL del flusso audio di RDL e RC105</log>
<log>Correzione e bugfix all'interno dello stile dell'app</log> <version>
<log>Correzione e bugfix all'interno dei metatag</log> <number>0.20.0 Beta</number>
<log>Aggiunto script di disabilitazione pulsante F5 da tastiera</log> <logs>
</logs> <log>L'app entra ufficialmente nello stato di Beta-testing.</log>
</version> <log>Cambiata la modalità di indicazione della versione, non più basata sulla data di aggiornamento, ma sul numero di update dell'app.</log>
<log>Avviata ufficialmente la "Roadmap" sulla Versione 1.0.0 (indicata a fine pagina CHANGELOG).</log>
<version> <log>Aggiunto script per iOS e iPadOS sul rilevamento uso browser/app.</log>
<number>0.9.24 Alpha</number> <log>Correzione e bugfix script rilevamento app Android.</log>
<logs> </logs>
<log>Correzione e bugfix player video</log> </version>
</logs>
</version> <version>
<number>0.12.6 Alpha (Last Release)</number>
<version> <logs>
<number>0.9.23 Alpha</number> <log>Aggiornata la lista emittenti nel menu in alto sinistra.</log>
<logs> <log>Aggiornate le pagine "Cosa è RPIGRPUP PLAY", "COME FUNZIONA L'APP" e "HAI BISOGNO DI AIUTO".</log>
<log>Aggiunto player video</log> <log>Corretto bug inizializzazione app per iOS e iPadOS.</log>
<log>Aggiunto collegamenti dei canali video delle emittenti</log> <log>Aggiunto script di controllo gestione app per iOS e iPadOS.</log>
<log>Aggiunto in configurazione gli URL dei canali video delle emittenti</log> <log>Correzione e bugfix di problematiche varie.</log>
</logs> </logs>
</version> </version>
<version> <version>
<number>0.7.31 Alpha</number> <number>0.12.5 Aplha</number>
<logs> <logs>
<log>Correzione e bugfix all'interno della configurazione dell'app</log> <log>Realizzazione file XML per il Changelog.</log>
<log>Rimozione dello script per lo "scroll to update page" da smartphone</log> <log>Aggiunto pulsante per la visualizzazione del Changelog.</log>
<log>Aggiunti i metodi di contatto in About</log> <log>Correzione pagina About per la visualizzazione del Changelog.</log>
</logs> <log>Correzione visualizzazione del Home Page.</log>
</version> <log>Reso automatico il cambio di versione dell'app in base all'ultima versione disponibile e dichiarata all'intero del Changelog.</log>
</logs>
</version>
<version>
<number>0.12.2 Alpha</number>
<logs>
<log>Corretto il problema del caricamento del "Player Mobile" su iOS e derivati.</log>
<log>Corretto e aggiunto nuovi script del "Player Mobile"</log>
<log>Aggiornato slogan dell'emittente "RDL XMAS"</log>
<log>Aggiunto il Changelog</log>
</logs>
</version>
<version>
<number>0.11.30 Alpha</number>
<logs>
<log>Corretto la visualizzazione del "Player Mobile"</log>
<log>Aggiornata la lista emittenti nel menu in alto sinistra</log>
<log>Rimosse momentaneamente le WebTV nel menu in alto sinistra</log>
</logs>
</version>
<version>
<number>0.11.29 Alpha</number>
<logs>
<log>Aggiunta nuova emittente "RDL XMAS"</log>
<log>Corretta la visualizzazione del "PLAYER DESKTOP"</log>
<log>Corretta la visualizzazione della lista emittenti in "Home Desktop" e "Home Mobile"</log>
<log>Abilitato il "Player Mobile" anche in ambiente Desktop</log>
<log>Aggiornato script che impedisce il refresh della pagina da Mobile</log>
</logs>
</version>
<version>
<number>0.9.26 Alpha</number>
<logs>
<log>Cambiata l'URL del flusso audio di RDL e RC105</log>
<log>Correzione e bugfix all'interno dello stile dell'app</log>
<log>Correzione e bugfix all'interno dei metatag</log>
<log>Aggiunto script di disabilitazione pulsante F5 da tastiera</log>
</logs>
</version>
<version>
<number>0.9.24 Alpha</number>
<logs>
<log>Correzione e bugfix player video</log>
</logs>
</version>
<version>
<number>0.9.23 Alpha</number>
<logs>
<log>Aggiunto player video</log>
<log>Aggiunto collegamenti dei canali video delle emittenti</log>
<log>Aggiunto in configurazione gli URL dei canali video delle emittenti</log>
</logs>
</version>
<version>
<number>0.7.31 Alpha</number>
<logs>
<log>Correzione e bugfix all'interno della configurazione dell'app</log>
<log>Rimozione dello script per lo "scroll to update page" da smartphone</log>
<log>Aggiunti i metodi di contatto in About</log>
</logs>
</version>
</changelog> </changelog>

Binary file not shown.

View File

@@ -8,9 +8,10 @@
<slogan>O Sei Fuori, O Sei Dei Nostri</slogan> <slogan>O Sei Fuori, O Sei Dei Nostri</slogan>
<thematic>false</thematic> <thematic>false</thematic>
<logo>https://i0.wp.com/www.radiodiffusionelibera.com/wp-content/uploads/2017/01/RDL-Facebook.png</logo> <logo>https://i0.wp.com/www.radiodiffusionelibera.com/wp-content/uploads/2017/01/RDL-Facebook.png</logo>
<stream>https://asvradiostream.asvstudios.it/radio/8000/radio.mp3</stream> <stream>https://asvradiostream.asvstudios.it/radio/8000/radio.aac</stream>
<streamhls>https://srvone.radio.asvhosting.com/hls/rdlradio/live.m3u8</streamhls> <streamhls>https://srvone.radio.asvhosting.com/hls/rdlradio/live.m3u8</streamhls>
<contentplayer>https://www.radiodiffusionelibera.com/contentrpigplay</contentplayer> <contentplayer>https://www.radiodiffusionelibera.com/contentrpigplay</contentplayer>
<apiradio>https://srvone.radio.asvhosting.com/api/nowplaying/2</apiradio>
</station> </station>
<station> <station>
@@ -22,9 +23,10 @@
<stream>https://asvradiostream.asvstudios.it/radio/8020/auto.aac</stream> <stream>https://asvradiostream.asvstudios.it/radio/8020/auto.aac</stream>
<streamhls>https://srvone.radio.asvhosting.com/hls/radiocitta105/live.m3u8</streamhls> <streamhls>https://srvone.radio.asvhosting.com/hls/radiocitta105/live.m3u8</streamhls>
<contentplayer>https://www.radiocitta105.it/contentrpigplay</contentplayer> <contentplayer>https://www.radiocitta105.it/contentrpigplay</contentplayer>
<apiradio>https://srvone.radio.asvhosting.com/api/nowplaying/1</apiradio>
</station> </station>
<station> <!-- <station>
<id>3</id> <id>3</id>
<name>RadioAI</name> <name>RadioAI</name>
<slogan>Solo musica AI - Powered by RDL </slogan> <slogan>Solo musica AI - Powered by RDL </slogan>
@@ -33,7 +35,7 @@
<stream>https://srvone.radio.asvhosting.com/listen/radioai/radio.mp3</stream> <stream>https://srvone.radio.asvhosting.com/listen/radioai/radio.mp3</stream>
<streamhls>https://srvone.radio.asvhosting.com/hls/radioai/live.m3u8</streamhls> <streamhls>https://srvone.radio.asvhosting.com/hls/radioai/live.m3u8</streamhls>
<contentplayer></contentplayer> <contentplayer></contentplayer>
</station> </station> -->
<!-- <station> <!-- <station>
<id>4</id> <id>4</id>
@@ -46,7 +48,7 @@
<contentplayer></contentplayer> <contentplayer></contentplayer>
</station> --> </station> -->
<station> <!-- <station>
<id>5</id> <id>5</id>
<name>Radio People Italy</name> <name>Radio People Italy</name>
<slogan>La radio della Gente</slogan> <slogan>La radio della Gente</slogan>
@@ -55,6 +57,6 @@
<stream></stream> <stream></stream>
<streamhls></streamhls> <streamhls></streamhls>
<contentplayer>https://www.rpigroup.it/radiopeopleitaly_contentrpigroup/</contentplayer> <contentplayer>https://www.rpigroup.it/radiopeopleitaly_contentrpigroup/</contentplayer>
</station> </station> -->
</radio> </radio>

Binary file not shown.

View File

@@ -14,8 +14,8 @@
<id>1</id> <id>1</id>
<name>Rc105 TV</name> <name>Rc105 TV</name>
<logo>https://www.radiocitta105.it/wp-content/uploads/2020/06/26168468_1590103344416186_7025872599153073152_n-1.png</logo> <logo>https://www.radiocitta105.it/wp-content/uploads/2020/06/26168468_1590103344416186_7025872599153073152_n-1.png</logo>
<stream>https://webtv.rpigroup.it/e1e55a4b-abec-4043-8f08-e2105b48b59b.m3u8</stream> <stream>https://tv.rpigroup.net/memfs/a9699134-efb3-4932-b8db-5a49ae214031.m3u8</stream>
<poster>https://webtv.rpigroup.it/memfs/e1e55a4b-abec-4043-8f08-e2105b48b59b.jpg</poster> <poster>https://tv.rpigroup.net/memfs/a9699134-efb3-4932-b8db-5a49ae214031.jpg</poster>
</station> </station>
</tv> </tv>

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@@ -5,6 +5,9 @@
# --------------------------------------------------------------- # ---------------------------------------------------------------
# Author: A.S.V. Studios APPS # Author: A.S.V. Studios APPS
# Website: https://app.rpigroup.net # Website: https://app.rpigroup.net
# Copyright (c) 2025-202 A.S.V. Studios APPS
# ---------------------------------------------------------------
# Questa app è disponibile all'interno della repository pubblica di RPIGroup Play.
# --------------------------------------------------------------- # ---------------------------------------------------------------
# All Rights is reserved by A.S.V. Studios APPS. # All Rights is reserved by A.S.V. Studios APPS.
# #

Binary file not shown.

546
js/app.js
View File

@@ -6,6 +6,8 @@ document.addEventListener('DOMContentLoaded', function () {
let hlsInstance = null; let hlsInstance = null;
let currentXHR = null; let currentXHR = null;
let isLoading = false; let isLoading = false;
let loadingOverlay = null;
let metadataInterval = null;
// WeakMap per tracciare i listener degli elementi // WeakMap per tracciare i listener degli elementi
const linkListeners = new WeakMap(); const linkListeners = new WeakMap();
@@ -55,6 +57,12 @@ document.addEventListener('DOMContentLoaded', function () {
function cleanupPlayer() { function cleanupPlayer() {
console.log('Pulizia player audio...'); console.log('Pulizia player audio...');
// Stop metadata updates
if (metadataInterval) {
clearInterval(metadataInterval);
metadataInterval = null;
}
// Distruggi istanza HLS // Distruggi istanza HLS
if (hlsInstance) { if (hlsInstance) {
try { try {
@@ -94,6 +102,27 @@ document.addEventListener('DOMContentLoaded', function () {
} }
} }
/**
* Mostra l'overlay di caricamento
*/
function showLoadingOverlay() {
if (!loadingOverlay) {
loadingOverlay = document.getElementById('loadingOverlay');
}
if (loadingOverlay) {
loadingOverlay.classList.add('active');
}
}
/**
* Nasconde l'overlay di caricamento
*/
function hideLoadingOverlay() {
if (loadingOverlay) {
loadingOverlay.classList.remove('active');
}
}
/** /**
* Caricamento pagina con protezione da race condition * Caricamento pagina con protezione da race condition
*/ */
@@ -120,6 +149,9 @@ document.addEventListener('DOMContentLoaded', function () {
return; return;
} }
// Mostra l'overlay di caricamento
showLoadingOverlay();
// Animazione fade out // Animazione fade out
contentContainer.classList.remove('fade-in'); contentContainer.classList.remove('fade-in');
contentContainer.classList.add('fade-out'); contentContainer.classList.add('fade-out');
@@ -138,6 +170,9 @@ document.addEventListener('DOMContentLoaded', function () {
if (this.status === 200) { if (this.status === 200) {
contentContainer.innerHTML = this.responseText; contentContainer.innerHTML = this.responseText;
// Nascondi overlay di caricamento
hideLoadingOverlay();
// Animazione fade in // Animazione fade in
contentContainer.classList.remove('fade-out'); contentContainer.classList.remove('fade-out');
contentContainer.classList.add('fade-in'); contentContainer.classList.add('fade-in');
@@ -175,6 +210,7 @@ document.addEventListener('DOMContentLoaded', function () {
console.log('Pagina caricata:', page); console.log('Pagina caricata:', page);
} else { } else {
hideLoadingOverlay();
contentContainer.innerHTML = '<div class="content-page"><h2>Errore</h2><p>Impossibile caricare la pagina. Codice errore: ' + this.status + '</p></div>'; contentContainer.innerHTML = '<div class="content-page"><h2>Errore</h2><p>Impossibile caricare la pagina. Codice errore: ' + this.status + '</p></div>';
contentContainer.classList.remove('fade-out'); contentContainer.classList.remove('fade-out');
contentContainer.classList.add('fade-in'); contentContainer.classList.add('fade-in');
@@ -185,6 +221,7 @@ document.addEventListener('DOMContentLoaded', function () {
currentXHR.onerror = function () { currentXHR.onerror = function () {
isLoading = false; isLoading = false;
currentXHR = null; currentXHR = null;
hideLoadingOverlay();
contentContainer.innerHTML = '<div class="content-page"><h2>Errore di connessione</h2><p>Controlla la tua connessione internet e riprova.</p></div>'; contentContainer.innerHTML = '<div class="content-page"><h2>Errore di connessione</h2><p>Controlla la tua connessione internet e riprova.</p></div>';
contentContainer.classList.remove('fade-out'); contentContainer.classList.remove('fade-out');
contentContainer.classList.add('fade-in'); contentContainer.classList.add('fade-in');
@@ -194,6 +231,7 @@ document.addEventListener('DOMContentLoaded', function () {
currentXHR.onabort = function () { currentXHR.onabort = function () {
isLoading = false; isLoading = false;
currentXHR = null; currentXHR = null;
hideLoadingOverlay();
console.log('Richiesta XHR annullata'); console.log('Richiesta XHR annullata');
}; };
@@ -309,228 +347,304 @@ document.addEventListener('DOMContentLoaded', function () {
*/ */
function initializePlayer() { function initializePlayer() {
console.log('Inizializzazione player audio...'); console.log('Inizializzazione player audio (v2.1 - Format Toggle)...');
const playPauseBtn = document.getElementById('playPauseBtn'); const playPauseBtn = document.getElementById('playPauseBtn');
const playIcon = document.querySelector('.play-icon'); const playIcon = document.querySelector('.play-icon');
const pauseIcon = document.querySelector('.pause-icon'); const pauseIcon = document.querySelector('.pause-icon');
const playerStatus = document.getElementById('playerStatus'); const playerStatus = document.getElementById('playerStatus');
const artistElement = document.getElementById('artist'); const formatToggleBtn = document.getElementById('formatToggleBtn');
const formatLabel = document.getElementById('formatLabel');
if (!playPauseBtn) { if (!playPauseBtn) return;
console.error('Pulsante play/pause non trovato');
return;
}
// Pulisci il player precedente // 1. Cleanup Preventivo
cleanupPlayer(); cleanupPlayer();
// Ottieni il nuovo audio player dal DOM // 2. Setup Elementi
audioPlayer = document.getElementById('hlsAudioPlayer'); audioPlayer = document.getElementById('hlsAudioPlayer');
if (!audioPlayer) return;
if (!audioPlayer) {
console.error('Elemento audio non trovato');
return;
}
const streamHLS = playPauseBtn.getAttribute('data-stream-hls'); const streamHLS = playPauseBtn.getAttribute('data-stream-hls');
const streamFallback = playPauseBtn.getAttribute('data-stream-fallback'); const streamFallback = playPauseBtn.getAttribute('data-stream-fallback');
const stationName = playPauseBtn.getAttribute('data-station-name'); const stationName = playPauseBtn.getAttribute('data-station-name');
const stationSlogan = playPauseBtn.getAttribute('data-station-slogan');
const stationLogo = playPauseBtn.getAttribute('data-station-logo');
console.log('Stream HLS:', streamHLS);
console.log('Stream Fallback:', streamFallback);
console.log('Stazione:', stationName);
// Stato formato audio (preferenza salvata in localStorage)
const storageKey = 'audioFormat_' + stationName;
let currentFormat = localStorage.getItem(storageKey) || 'hls'; // 'hls' o 'direct'
let isPlaying = false; let isPlaying = false;
let streamType = 'hls'; // 'hls' o 'direct'
// Aggiorna label del formato
function updateFormatLabel() {
if (formatLabel) {
formatLabel.textContent = currentFormat === 'hls' ? 'HLS' : 'MP3';
}
}
updateFormatLabel();
// Configurazione HLS.js Ottimizzata per Qualità
const hlsConfig = {
debug: false,
enableWorker: true,
lowLatencyMode: false, // Disabilitato per stabilità e buffer migliore
backBufferLength: 90,
maxBufferLength: 30, // Aumentato buffer (default 30s)
startLevel: -1, // Auto start
xhrSetup: function (xhr, url) {
xhr.withCredentials = false; // Fix CORS issues sometimes
}
};
/** /**
* Inizializza HLS stream * Logica di Selezione Player
* Priorità: HLS.js (PC/Android) > Nativo (iOS) > Fallback (Direct Stream)
*/ */
function initHLSStream() { function initStream(forceFormat = null) {
if (!window.Hls) { const preferredFormat = forceFormat || currentFormat;
console.error('HLS.js non caricato');
return false; // Se l'utente ha scelto formato diretto, usa subito MP3/AAC
if (preferredFormat === 'direct') {
console.log('User preference: Direct stream (MP3/AAC)');
useDirectStream();
return;
} }
// STRATEGIA 1: HLS.js (Preferita per PC e Android)
if (Hls.isSupported()) { if (Hls.isSupported()) {
console.log('HLS supportato - uso HLS.js'); console.log('Strategy: HLS.js (High Quality Control)');
hlsInstance = new Hls({
debug: false,
enableWorker: true,
lowLatencyMode: true,
backBufferLength: 90
});
hlsInstance = new Hls(hlsConfig);
hlsInstance.loadSource(streamHLS); hlsInstance.loadSource(streamHLS);
hlsInstance.attachMedia(audioPlayer); hlsInstance.attachMedia(audioPlayer);
hlsInstance.on(Hls.Events.MANIFEST_PARSED, function () { hlsInstance.on(Hls.Events.MANIFEST_PARSED, function (event, data) {
console.log('Manifest HLS caricato'); console.log(`HLS Manifest Loaded. Levels: ${data.levels.length}`);
audioPlayer.play().then(() => {
console.log('Riproduzione HLS avviata'); // Log dettagliato di tutti i livelli disponibili per debug
updatePlayState(true); data.levels.forEach((level, index) => {
}).catch(err => { console.log(`Level ${index}: ${level.bitrate} bps (${(level.bitrate / 1000).toFixed(0)} kbps)`);
console.error('Errore avvio riproduzione HLS:', err);
tryFallbackStream();
}); });
// CRITICAL: Forza partenza dal livello massimo per garantire qualità alta
if (data.levels.length > 1) {
const maxLevel = data.levels.length - 1;
hlsInstance.startLevel = maxLevel;
console.log(`✓ Forced START level to MAX: ${maxLevel} (${(data.levels[maxLevel].bitrate / 1000).toFixed(0)} kbps)`);
}
attemptPlay();
});
// Track quality switches per debug
hlsInstance.on(Hls.Events.LEVEL_SWITCHING, function (event, data) {
console.log(`→ Switching to level ${data.level}...`);
});
hlsInstance.on(Hls.Events.LEVEL_SWITCHED, function (event, data) {
const currentLevel = hlsInstance.levels[data.level];
console.log(`✓ Switched to level ${data.level}: ${(currentLevel.bitrate / 1000).toFixed(0)} kbps`);
}); });
hlsInstance.on(Hls.Events.ERROR, function (event, data) { hlsInstance.on(Hls.Events.ERROR, function (event, data) {
console.error('Errore HLS:', data.type, data.details);
if (data.fatal) { if (data.fatal) {
console.warn('HLS Fatal Error:', data.type);
switch (data.type) { switch (data.type) {
case Hls.ErrorTypes.NETWORK_ERROR: case Hls.ErrorTypes.NETWORK_ERROR:
console.log('Errore di rete, tentativo fallback...'); hlsInstance.startLoad();
tryFallbackStream();
break; break;
case Hls.ErrorTypes.MEDIA_ERROR: case Hls.ErrorTypes.MEDIA_ERROR:
console.log('Errore media, tentativo recupero...');
hlsInstance.recoverMediaError(); hlsInstance.recoverMediaError();
break; break;
default: default:
console.log('Errore fatale, tentativo fallback...'); useDirectStream();
tryFallbackStream();
break; break;
} }
} }
}); });
return true; return;
} else if (audioPlayer.canPlayType('application/vnd.apple.mpegurl')) {
console.log('HLS nativo supportato');
audioPlayer.src = streamHLS;
audioPlayer.play().then(() => {
console.log('Riproduzione HLS nativa avviata');
updatePlayState(true);
}).catch(err => {
console.error('Errore HLS nativo:', err);
tryFallbackStream();
});
return true;
} }
return false; // STRATEGIA 2: Nativo (Obbligatorio per iOS)
if (audioPlayer.canPlayType('application/vnd.apple.mpegurl')) {
console.log('Strategy: Native HLS (iOS/Safari)');
audioPlayer.src = streamHLS;
audioPlayer.preload = 'auto'; // Suggerisce al browser di caricare dati
// Hack per iOS: Alcune volte serve un tocco utente, gestito dal click play
attemptPlay();
return;
}
// STRATEGIA 3: Fallback (Stream MP3 diretto)
console.log('Strategy: Direct Fallback');
useDirectStream();
} }
/** function useDirectStream() {
* Usa stream diretto come fallback
*/
function tryFallbackStream() {
console.log('Tentativo stream fallback:', streamFallback);
if (hlsInstance) { if (hlsInstance) {
hlsInstance.destroy(); hlsInstance.destroy();
hlsInstance = null; hlsInstance = null;
} }
console.log('Using direct stream:', streamFallback);
streamType = 'direct';
audioPlayer.src = streamFallback; audioPlayer.src = streamFallback;
audioPlayer.load(); attemptPlay();
audioPlayer.play().then(() => {
console.log('Riproduzione stream diretto avviata');
updatePlayState(true);
}).catch(err => {
console.error('Errore stream diretto:', err);
updatePlayState(false);
if (playerStatus) {
playerStatus.textContent = 'Errore di riproduzione';
}
});
} }
/** // Funzione per cambiare formato
* Aggiorna stato visuale play/pause function toggleFormat() {
*/ const wasPlaying = !audioPlayer.paused;
function updatePlayState(playing) {
isPlaying = playing;
if (playing) { // Cambia formato
playIcon.style.display = 'none'; currentFormat = currentFormat === 'hls' ? 'direct' : 'hls';
pauseIcon.style.display = 'inline-block'; localStorage.setItem(storageKey, currentFormat);
if (playerStatus) { console.log('Switching to format:', currentFormat);
playerStatus.textContent = 'In riproduzione...';
} // Aggiorna label
} else { updateFormatLabel();
playIcon.style.display = 'inline-block';
pauseIcon.style.display = 'none'; // Ferma riproduzione corrente
if (playerStatus) { if (audioPlayer) {
playerStatus.textContent = stationSlogan || 'In pausa'; audioPlayer.pause();
} }
// Pulisci player precedente
if (hlsInstance) {
hlsInstance.destroy();
hlsInstance = null;
}
audioPlayer.src = '';
// Reinizializza con nuovo formato
initStream(currentFormat);
// Riprendi riproduzione se era in corso
if (wasPlaying) {
setTimeout(() => {
togglePlay();
}, 100);
} }
} }
/** function attemptPlay() {
* Gestione click play/pause // Nota: chiamare .play() senza interazione utente fallirà.
*/ // Questa funzione prepara lo stato. Il vero play avviene al click.
playPauseBtn.addEventListener('click', function () { if (playerStatus) playerStatus.textContent = 'Pronto alla riproduzione';
if (!audioPlayer) return; }
if (isPlaying) { function togglePlay() {
// Pausa if (audioPlayer.paused) {
audioPlayer.pause(); const playPromise = audioPlayer.play();
updatePlayState(false); if (playPromise !== undefined) {
console.log('Riproduzione in pausa'); playPromise.then(() => {
} else { updateUI(true);
// Play }).catch(error => {
if (!audioPlayer.src && !hlsInstance) { console.error('Play prevented:', error);
// Prima riproduzione - inizializza lo stream // Se HLS fallisce al play (es. stale manifest), prova fallback
if (streamHLS && !initHLSStream()) { if (!audioPlayer.src.includes(streamFallback) && !hlsInstance) {
// Se HLS fallisce, usa direttamente il fallback fallbackToDirectStream();
tryFallbackStream(); audioPlayer.play().catch(e => console.error('Fallback failed:', e));
} }
} else {
// Riprendi riproduzione
audioPlayer.play().then(() => {
updatePlayState(true);
console.log('Riproduzione ripresa');
}).catch(err => {
console.error('Errore ripresa riproduzione:', err);
updatePlayState(false);
}); });
} }
} else {
audioPlayer.pause();
updateUI(false);
} }
}); }
/** function updateUI(playing) {
* Event listeners audio player isPlaying = playing;
*/ playIcon.style.display = playing ? 'none' : 'block';
audioPlayer.addEventListener('play', function () { pauseIcon.style.display = playing ? 'block' : 'none';
updatePlayState(true);
});
audioPlayer.addEventListener('pause', function () {
updatePlayState(false);
});
audioPlayer.addEventListener('error', function (e) {
console.error('Errore audio player:', e);
updatePlayState(false);
if (playerStatus) { if (playerStatus) {
playerStatus.textContent = 'Errore di riproduzione'; playerStatus.textContent = playing ? 'In riproduzione' : 'In pausa';
} }
}); }
audioPlayer.addEventListener('waiting', function () { // Listeners
if (playerStatus) { playPauseBtn.onclick = (e) => {
playerStatus.textContent = 'Buffering...'; e.preventDefault();
// Se non inizializzato (primo click), avvia stream
if (!audioPlayer.src && !hlsInstance) {
initStream();
// Piccolo timeout per dare tempo al setup prima del play effettivo
setTimeout(togglePlay, 50);
} else {
togglePlay();
} }
}); };
audioPlayer.addEventListener('playing', function () { audioPlayer.onplay = () => updateUI(true);
if (playerStatus) { audioPlayer.onpause = () => updateUI(false);
playerStatus.textContent = 'In riproduzione...';
} audioPlayer.onerror = (e) => {
}); console.error('Audio Error:', audioPlayer.error);
if (playerStatus) playerStatus.textContent = 'Errore riproduzione';
updateUI(false);
};
// Format toggle button listener
if (formatToggleBtn) {
formatToggleBtn.onclick = (e) => {
e.preventDefault();
toggleFormat();
};
}
// Setup Media Session // Setup Media Session
setupMediaSession(stationName, stationSlogan, stationLogo, playIcon, pauseIcon); setupMediaSession(stationName, playPauseBtn.getAttribute('data-station-slogan'), playPauseBtn.getAttribute('data-station-logo'), playIcon, pauseIcon);
console.log('Player audio inizializzato'); // --- GESTIONE METADATI (Now Playing) ---
const apiUrl = playPauseBtn.getAttribute('data-api-url');
if (apiUrl) {
console.log('Starting metadata polling for:', apiUrl);
function updateMetadata() {
fetch(apiUrl)
.then(response => response.json())
.then(data => {
if (data.now_playing && data.now_playing.song) {
const song = data.now_playing.song;
// Aggiorna DOM
const songsEl = document.getElementById('songs');
const artistEl = document.getElementById('artist');
const albumArtEl = document.getElementById('albumsong');
if (songsEl) songsEl.textContent = song.title;
if (artistEl) artistEl.textContent = song.artist;
if (albumArtEl && song.art) {
// Evita refresh se l'immagine è la stessa
if (albumArtEl.src !== song.art) {
albumArtEl.src = song.art;
}
}
// Aggiorna Media Session
if ('mediaSession' in navigator) {
navigator.mediaSession.metadata = new MediaMetadata({
title: song.title,
artist: song.artist,
album: stationName,
artwork: [
{ src: song.art || playPauseBtn.getAttribute('data-station-logo'), sizes: '512x512', type: 'image/png' }
]
});
}
}
})
.catch(err => console.error('Error fetching metadata:', err));
}
// Chiamata immediata
updateMetadata();
// Polling ogni 10 secondi
metadataInterval = setInterval(updateMetadata, 10000);
}
console.log('Player System Ready with format toggle.');
} }
/** /**
@@ -538,7 +652,127 @@ document.addEventListener('DOMContentLoaded', function () {
*/ */
function initializeTVPlayer() { function initializeTVPlayer() {
console.log('TV Player inizializzato'); console.log('TV Player inizializzato');
// Implementare logica specifica per TV se necessario
const video = document.getElementById('tvVideoPlayer');
if (!video) {
console.log('Elemento video TV non trovato');
return;
}
// Cleanup istanza precedente se esiste
if (video.hlsInstance) {
video.hlsInstance.destroy();
delete video.hlsInstance;
}
const videoSrc = video.getAttribute('data-src');
console.log('Video Source:', videoSrc);
if (!videoSrc) {
console.error('Nessuna sorgente video specificata in data-src');
return;
}
// Creazione elemento per errori se non esiste già
let errorDisplay = video.parentNode.querySelector('.tv-error-display');
if (!errorDisplay) {
errorDisplay = document.createElement('div');
errorDisplay.className = 'tv-error-display';
errorDisplay.style.display = 'none';
errorDisplay.style.position = 'absolute';
errorDisplay.style.top = '50%';
errorDisplay.style.left = '50%';
errorDisplay.style.transform = 'translate(-50%, -50%)';
errorDisplay.style.backgroundColor = 'rgba(0, 0, 0, 0.8)';
errorDisplay.style.color = '#fff';
errorDisplay.style.padding = '20px';
errorDisplay.style.borderRadius = '8px';
errorDisplay.style.textAlign = 'center';
errorDisplay.style.zIndex = '2000';
errorDisplay.style.maxWidth = '90%';
if (video.parentNode) {
video.parentNode.appendChild(errorDisplay);
// Assicurati che il parent sia relative per il posizionamento absolute
if (getComputedStyle(video.parentNode).position === 'static') {
video.parentNode.style.position = 'relative';
}
}
}
function showError(msg, details) {
console.error(msg, details);
errorDisplay.innerHTML = '<strong>Errore:</strong><br>' + msg + (details ? '<br><small>' + details + '</small>' : '');
errorDisplay.style.display = 'block';
}
// Controllo libreria HLS
if (typeof Hls === 'undefined') {
// Riprova tra poco se magari sta ancora caricando
setTimeout(() => {
if (typeof Hls === 'undefined') {
showError('Libreria HLS non caricata correttamente.');
} else {
initializeTVPlayer(); // Riprova inizializzazione
}
}, 500);
return;
}
if (Hls.isSupported()) {
console.log('HLS Supported - Loading stream');
var hls = new Hls({
debug: false,
enableWorker: true
});
hls.loadSource(videoSrc);
hls.attachMedia(video);
hls.on(Hls.Events.MANIFEST_PARSED, function () {
console.log('TV Manifest Parsed - Ready to play');
var playPromise = video.play();
if (playPromise !== undefined) {
playPromise.catch(error => {
console.log('Autoplay blocked or failed:', error);
});
}
});
// Gestione errori estesa
hls.on(Hls.Events.ERROR, function (event, data) {
console.warn('HLS Error:', data);
if (data.fatal) {
switch (data.type) {
case Hls.ErrorTypes.NETWORK_ERROR:
console.log('Network error, recovering...');
hls.startLoad();
break;
case Hls.ErrorTypes.MEDIA_ERROR:
console.log('Media error, recovering...');
hls.recoverMediaError();
break;
default:
showError('Errore fatale riproduzione.', data.type);
hls.destroy();
break;
}
}
});
video.hlsInstance = hls;
} else if (video.canPlayType('application/vnd.apple.mpegurl')) {
console.log('Native HLS Supported (Safari/iOS)');
video.src = videoSrc;
video.addEventListener('loadedmetadata', function () {
video.play().catch(e => console.log("Autoplay iOS prevented"));
});
video.addEventListener('error', function (e) {
showError('Errore riproduzione nativa.', e.message);
});
} else {
showError('Il tuo browser non supporta HLS.');
}
} }
/** /**
@@ -676,8 +910,8 @@ document.addEventListener('DOMContentLoaded', function () {
const pathWithoutBase = currentPath.replace(BASE_PATH, '').replace(/^\//, ''); const pathWithoutBase = currentPath.replace(BASE_PATH, '').replace(/^\//, '');
const redirectParam = pathWithoutBase ? '&redirect=' + encodeURIComponent(pathWithoutBase) : ''; const redirectParam = pathWithoutBase ? '&redirect=' + encodeURIComponent(pathWithoutBase) : '';
const width = 375; const width = 400;
const height = 667; const height = 840;
const left = (window.screen.width / 2) - (width / 2); const left = (window.screen.width / 2) - (width / 2);
const top = (window.screen.height / 2) - (height / 2); const top = (window.screen.height / 2) - (height / 2);
@@ -713,7 +947,7 @@ document.addEventListener('DOMContentLoaded', function () {
* Setup modalità standalone (PWA) * Setup modalità standalone (PWA)
*/ */
function setupStandaloneMode() { function setupStandaloneMode() {
const appContainer = document.querySelector('.container'); const appContainer = document.querySelector('.container') || document.querySelector('.container-fluid');
if (!appContainer) return; if (!appContainer) return;
if (window.opener && !window.opener.closed) { if (window.opener && !window.opener.closed) {
@@ -756,7 +990,7 @@ document.addEventListener('DOMContentLoaded', function () {
setupOpenAppButton(); setupOpenAppButton();
if (document.querySelector('.container')) { if (document.querySelector('.container') || document.querySelector('.container-fluid')) {
setupStandaloneMode(); setupStandaloneMode();
attachLinkListeners(); attachLinkListeners();
setupHistoryNavigation(); setupHistoryNavigation();

Binary file not shown.

7
js/bootstrap.js vendored Normal file

File diff suppressed because one or more lines are too long

Binary file not shown.

Binary file not shown.

View File

@@ -15,7 +15,7 @@
</div> </div>
</div> </div>
<main class="container" id="content"> <main class="container-fluid" id="content">
<?php <?php
// Carica il contenuto iniziale in base all'URL // Carica il contenuto iniziale in base all'URL
switch ($page) { switch ($page) {
@@ -93,6 +93,14 @@
?> ?>
</main> </main>
<!-- Loading Overlay -->
<div id="loadingOverlay" class="loading-overlay">
<div class="loading-content">
<div class="spinner-large"></div>
<p class="loading-text">Caricamento...</p>
</div>
</div>
<div class="footer"> <div class="footer">
<div class="menu-section"> <div class="menu-section">
<a href="<?php echo $base_path; ?>/page/termini-condizioni" data-page="page/termini-condizioni" class="navLink">Termini e Condizioni</a> • <a href="<?php echo $base_path; ?>/page/termini-condizioni" data-page="page/termini-condizioni" class="navLink">Termini e Condizioni</a> •

Binary file not shown.

Binary file not shown.

View File

@@ -5,7 +5,29 @@
<hr> <hr>
<div class="tec"> <div class="tec">
<p>La pagina è in fase di realizzazione... Attendi il prossimo aggiornamento per leggere le nostre istruzioni d'uso!</p> <p>RPIGroup Play è l'applicazione che permette di ascoltare le radio e le webtv preferite in un unico luogo.</p>
<p>Per utilizzare l'applicazione, basta semplicemente entrare nell'applicazione dal link app.rpigroup.it, selezionare la sezione Radio o TV e scegliere la radio o la webtv che desideri ascoltare.</p>
<br>
<p>
<b>Come installare l'app?</b></br>
Per installare l'applicazione, basta semplicemente entrare nell'applicazione dal link app.rpigroup.it e cliccare sull'icona "Aggiungi all'Home". Oppure, per gli utenti iOS, entrando da safari dal link app.rpigroup.it, cliccare sull'icona "Convididi", scendere alla voce "Aggiungi all'Home" e cliccare su "Aggiungi all'Home".
</p>
<br>
<br><br>
<h2 class="text-center titlePage">F.A.Q.</h2>
<hr>
<p>
<b>Posso scaricarla da Google Play Store e Apple App Store?</b></br>
Non è possibile al momento scaricare l'applicazione da Google Play Store e Apple App Store.
</p>
<p>
<b>Posso utilizzare l'applicazione su un computer?</b></br>
Sì, dal tuo browser preferito, visita la pagina app.rpigroup.it. L'applicazione è anche installabile, cliccando sulla icona dedicata all'installazione della PWA.
</p>
<p>
<b>Posso aggiungere la mia stazione radio preferita?</b></br>
Sì, cliccando sul pulsante "Aggiungi Radio" nella sezione Radio, potrai aggiungere la tua stazione radio preferita compilando il form dedicato.
</p>
</div> </div>
<a href="<?php echo $base_path; ?>/home" data-page="home" class="linkBox mt-3"> <a href="<?php echo $base_path; ?>/home" data-page="home" class="linkBox mt-3">

Binary file not shown.

View File

@@ -2,6 +2,7 @@
<h1 class="titlePage">Changelog</h1> <h1 class="titlePage">Changelog</h1>
<p class="subtitlePage">Visualizza tutti gli aggiornamenti ricevuti</p> <p class="subtitlePage">Visualizza tutti gli aggiornamenti ricevuti</p>
<p class="text-center mb-4" style="font-size: 0.9rem; font-weight: 300;">Codice dell'app visionabile su <a href="https://git.asv.ovh/asvstudiosapps/rpigroupplay" target="_blank" style="font-weight: 500;">ASV Git</a></p>
<?php <?php
// Verifica che $changelog sia stato caricato correttamente in getStation.inc.php // Verifica che $changelog sia stato caricato correttamente in getStation.inc.php

View File

@@ -22,6 +22,38 @@
</div> </div>
</div> </div>
<style>
.content-adv{
text-align: center;
background: #3849a8;
padding: 20px;
border-radius: 20px;
}
.content-adv:before{
content: "ADV";
position: relative;
background: #f7b835;
padding: 3px 10px;
font-size: 0.8rem;
border-radius: 27px;
font-style: italic;
font-weight: 600;
top: -10px;
margin: 0;
}
.content-adv img{
max-width: 100%;
border-radius: 10px;
margin-top: -20px;
}
</style>
<div class="content-adv mb-4">
<a href="https://www.worldradioday.it" target="_blank" class="">
<img src="https://www.radiodiffusionelibera.com/wp-content/uploads/2026/01/banner300x250.jpg" alt="adv">
</a>
</div>
<div class="row g-2 mb-4"> <div class="row g-2 mb-4">
<div class="col-12"> <div class="col-12">
<a href="https://www.co-municare.it" target="_blank" class=""> <a href="https://www.co-municare.it" target="_blank" class="">
@@ -40,7 +72,7 @@
<div class="col-4"> <div class="col-4">
<a href="<?php echo $base_path; ?>/page/about" data-page="page/about" class="linkBox"> <a href="<?php echo $base_path; ?>/page/about" data-page="page/about" class="linkBox">
<div class="clickBox"> <div class="clickBox">
Come funziona? Come<br>funziona?
</div> </div>
</a> </a>
</div> </div>
@@ -58,4 +90,14 @@
</div> </div>
</a> </a>
</div> </div>
</div>
<div class="row g-2 mb-4">
<div class="col-12">
<a href="https://drive.asv.ovh/apps/forms/s/Zfk8cJk3oCz2Y2dHyWdEcgL7" target="_blank" class="linkBox">
<div class="clickBox p-2">
Lascia un feedback
</div>
</a>
</div>
</div> </div>

Binary file not shown.

View File

@@ -1,62 +1,102 @@
<?php <?php
if($station->contentplayer == ""){
?> $station_id = (string)$station->id;
<style> $station_name = (string)$station->name;
.staticpage{ $station_slogan = (string)$station->slogan;
background: linear-gradient(45deg, #3849a8, #f7b835); $station_logo = (string)$station->logo;
padding: 22px 35px 10px 35px; $station_stream = (string)$station->stream;
display: block; $station_streamhls = (string)$station->streamhls;
z-index: 1; $station_contentplayer = (string)$station->contentplayer;
position: absolute; $station_apiradio = (string)$station->apiradio;
height: calc(100vh - 312px);
color: white;
text-align: center;
width: 100%;
max-width: 450px;
left: 50%;
transform: translateX(-50%);
}
.staticpage > .logo{
width: 200px;
border-radius: 15px;
margin-bottom: 20px;
}
.staticpage > p{
margin: 0;
}
.staticpage > p.titleRadio{
font-weight: 600;
}
</style>
<div class="staticpage">
<img src="<?php echo $station->logo; ?>" class="logo" alt="logo radio">
<p class="titleRadio"><?php echo $station->name; ?></p>
<p class="subtitleRadio"><?php echo $station->slogan; ?></p>
</div>
<?php
}else{
?>
<iframe src="<?php echo $station->contentplayer; ?>" class="contentplayer" frameborder="0"></iframe>
<?php
}
?> ?>
<div class="footer_player" <?php if($is_mobile){ ?> style="bottom: 84px;" <? } ?>> <div class="radio-player-container" style="background-color: #ffffff;height: calc(100% - <?php if($is_mobile){ echo "232px"; }else{ echo "217px"; } ?>);position: absolute;left: calc(50%);display: flex;flex-direction: column;max-width: 750px;transform: translateX(-50%);width: 100%;">
<div class="row align-items-center">
<div class="col-2" style="width: 70px;"> <div class="radio-header" style="padding: 20px; border-bottom: 1px solid #eee; display: flex; align-items: center; justify-content: space-between;">
<img src="<?php echo $station->logo; ?>" alt="<?php echo $station->name; ?>" style="width: 60px;"> <div class="left-controls" style="display: flex; align-items: center;">
<a href="<?php echo $base_path; ?>/radio" data-page="radio" class="linkBox" style="color: #333; display: flex; align-items: center; text-decoration: none;">
<span class="material-icons" style="font-size: 28px;">arrow_back</span>
</a>
<span style="font-size: 18px; font-weight: 600; color: #333; margin-left: 15px;"><?php echo $station_name; ?></span>
</div> </div>
<div class="col">
<div id="artist" style="font-weight: 700; text-overflow: ellipsis; display: -webkit-box; -webkit-line-clamp: 1; -webkit-box-orient: vertical; overflow: hidden;"><?php echo $station->name; ?></div> <?php if(!empty($station_logo)): ?>
<div id="playerStatus" style="text-overflow: ellipsis; display: -webkit-box; -webkit-line-clamp: 1; -webkit-box-orient: vertical ;overflow: hidden;"><?php echo $station->slogan; ?></div> <div class="right-controls">
<img src="<?php echo $station_logo; ?>" alt="Logo" style="height: 30px; width: auto;">
</div> </div>
<div class="col-2" style="text-align: center; margin-top: 1px; margin-right: 5px; width: 67px;"> <?php endif; ?>
<button id="playPauseBtn" data-stream-hls="<?php echo $station->streamhls; ?>" data-stream-fallback="<?php echo $station->stream; ?>" data-station-name="<?php echo htmlspecialchars($station->name); ?>" data-station-logo="<?php echo $station->logo; ?>" data-station-slogan="<?php echo htmlspecialchars($station->slogan); ?>"> </div>
<i class="material-icons play-icon" id="play-pause" style="font-size: 35px; border-radius: 10px; border: 2px solid #ffffff; padding: 4px 5px 4px 4px; margin-top: -2px; background: #f7b835; color: #2a377e;">play_arrow</i>
<i class="material-icons pause-icon" id="play-pause" style="display:none; font-size: 35px; border-radius: 10px; border: 2px solid #ffffff; padding: 4px 5px 4px 4px; margin-top: -2px; background: #f7b835; color: #2a377e;">pause</i> <?php
</button> if($station->contentplayer == ""){
?>
<style>
.staticpage{
background: linear-gradient(45deg, #3849a8, #f7b835);
padding: 22px 35px 10px 35px;
display: block;
z-index: 1;
height: calc(100vh - 393px);
color: white;
text-align: center;
width: 100%;
max-width: 750px;
}
.staticpage > .logo{
width: 200px;
border-radius: 15px;
margin-bottom: 20px;
}
.staticpage > p{
margin: 0;
}
.staticpage > p.titleRadio{
font-weight: 600;
}
</style>
<div class="staticpage">
<img src="<?php echo $station->logo; ?>" class="logo" alt="logo radio">
<p class="titleRadio"><?php echo $station->name; ?></p>
<p class="subtitleRadio"><?php echo $station->slogan; ?></p>
</div>
<?php
}else{
?>
<iframe src="<?=$station_contentplayer?>" class="contentplayer" frameborder="0"></iframe>
<?php
}
?>
<div class="footer_player">
<div class="row align-items-center">
<div class="col-2" style="width: 70px;">
<img id="albumsong" src="<?php echo $station->logo; ?>" alt="<?php echo $station->name; ?>" style="width: 60px;height: 60px;">
</div>
<div class="col">
<div id="songs" style="font-weight: 700; text-overflow: ellipsis; display: -webkit-box; -webkit-line-clamp: 1; -webkit-box-orient: vertical; overflow: hidden;"><?php echo $station->name; ?></div>
<div id="artist" style="text-overflow: ellipsis; display: -webkit-box; -webkit-line-clamp: 1; -webkit-box-orient: vertical ;overflow: hidden;"><?php echo $station->slogan; ?></div>
</div>
<div class="col-auto" style="text-align: center; margin-top: 1px; padding-right: 0;">
<button id="formatToggleBtn" title="Cambia formato audio">
<span id="formatLabel" style="font-size: 12px;font-weight: 600;border-radius: 6px;border: 2px solid #ffffff;padding: 4px 6px;background: rgba(247, 184, 53, 0.3);color: #2a377e;">HLS</span>
</button>
</div>
<div class="col-2" style="text-align: center; margin-top: 1px; margin-right: 5px; width: 67px;">
<button id="playPauseBtn"
data-stream-hls="<?php echo $station->streamhls; ?>"
data-stream-fallback="<?php echo $station->stream; ?>"
data-station-name="<?php echo htmlspecialchars($station->name); ?>"
data-station-logo="<?php echo $station->logo; ?>"
data-station-slogan="<?php echo htmlspecialchars($station->slogan); ?>"
data-api-url="<?php echo $station_apiradio; ?>">
<i class="material-icons play-icon" id="play-pause" style="font-size: 35px; border-radius: 10px; border: 2px solid #ffffff; padding: 4px 5px 4px 4px; margin-top: -2px; background: #f7b835; color: #2a377e;">play_arrow</i>
<i class="material-icons pause-icon" id="play-pause" style="display:none; font-size: 35px; border-radius: 10px; border: 2px solid #ffffff; padding: 4px 5px 4px 4px; margin-top: -2px; background: #f7b835; color: #2a377e;">pause</i>
</button>
</div>
</div> </div>
</div> </div>
</div> </div>
<audio id="hlsAudioPlayer" preload="none"></audio> <audio id="hlsAudioPlayer" preload="none"></audio>

View File

@@ -0,0 +1,50 @@
<?php
// Assicurati che $station sia definito (dovrebbe essere passato da mobile.php)
if (!isset($station) || empty($station)) {
echo '<div class="alert alert-danger">Errore: Stazione TV non trovata.</div>';
return;
}
$station_id = (string)$station->id;
$station_name = (string)$station->name;
$station_logo = (string)$station->logo;
$station_stream = (string)$station->stream;
$station_poster = isset($station->poster) ? (string)$station->poster : '';
?>
<!-- Container player con sfondo bianco -->
<div class="tv-player-container" style="background-color: #ffffff;height: calc(100% - <?php if($is_mobile){ echo "232px"; }else{ echo "217px"; } ?>);position: absolute;left: calc(50%);display: flex;flex-direction: column;max-width: 750px;transform: translateX(-50%);width: 100%;">
<!-- Header semplice con pulsante indietro e nome -->
<div class="tv-header" style="padding: 20px; border-bottom: 1px solid #eee; display: flex; align-items: center; justify-content: space-between;">
<div class="left-controls" style="display: flex; align-items: center;">
<a href="<?php echo $base_path; ?>/tv" data-page="tv" class="linkBox" style="color: #333; display: flex; align-items: center; text-decoration: none;">
<span class="material-icons" style="font-size: 28px;">arrow_back</span>
</a>
<span style="font-size: 18px; font-weight: 600; color: #333; margin-left: 15px;"><?php echo $station_name; ?></span>
</div>
<?php if(!empty($station_logo)): ?>
<div class="right-controls">
<img src="<?php echo $station_logo; ?>" alt="Logo" style="height: 30px; width: auto;">
</div>
<?php endif; ?>
</div>
<!-- Area Video -->
<div class="video-wrapper" style="flex: 1; display: flex; align-items: center; justify-content: center; background-color: #000;">
<video id="tvVideoPlayer"
class="video-js"
controls
playsinline
data-src="<?php echo $station_stream; ?>"
poster="<?php echo $station_poster; ?>"
style="width: 100%; max-width: 100%; max-height: 100vh; aspect-ratio: 16/9;">
<p class="vjs-no-js">
To view this video please enable JavaScript, and consider upgrading to a web browser that
<a href="https://videojs.com/html5-video-support/" target="_blank">supports HTML5 video</a>
</p>
</video>
</div>
</div>

Binary file not shown.

View File

@@ -1,12 +1,27 @@
<?php header('Content-Type: text/html; charset=UTF-8'); ?> <?php header('Content-Type: text/html; charset=UTF-8'); ?>
<?php
// views/home.php - Vista della pagina principale
$stations = loadTvStations();
?>
<h1 class="titlePage">TV</h1> <h1 class="titlePage">TV</h1>
<p class="subtitlePage">Guarda in streaming <b>RC105 TV</b></p> <p class="subtitlePage">Seleziona la webtv che vuoi guardare</p>
<hr> <hr>
<div class="tec"> <div class="stationList">
<iframe src="https://tv.rpigroup.net/a9699134-efb3-4932-b8db-5a49ae214031.html" style="width:100%; height: 225px;" frameborder="no" scrolling="no" allowfullscreen="true"></iframe> <div class="row g-2">
<?php foreach ($stations as $station): ?>
<div class="col-6">
<div class="stationCard <?php if((string)$station->thematic === 'true'){echo "isthematic";} ?>" data-id="<?php echo $station->id; ?>">
<a href="<?php echo $base_path; ?>/playtv/<?php echo $station->id; ?>" data-page="playtv/<?php echo $station->id; ?>" class="nav-link stationLink">
<img src="<?php echo $station->logo; ?>" alt="<?php echo $station->name; ?>" class="stationLogo">
</a>
</div>
</div>
<?php endforeach; ?>
</div>
</div> </div>
<a href="<?php echo $base_path; ?>/home" data-page="home" class="linkBox mt-3"> <a href="<?php echo $base_path; ?>/home" data-page="home" class="linkBox mt-3">

Binary file not shown.

View File

@@ -1,59 +0,0 @@
<?php
// File: process_contact.php
// Imposta header per JSON
header('Content-Type: application/json');
// Verifica se il modulo è stato inviato
if ($_SERVER["REQUEST_METHOD"] === "POST") {
// Raccolta e pulizia dei dati
$name = filter_input(INPUT_POST, 'name', FILTER_SANITIZE_STRING);
$email = filter_input(INPUT_POST, 'email', FILTER_SANITIZE_EMAIL);
$subject = filter_input(INPUT_POST, 'subject', FILTER_SANITIZE_STRING);
$message = filter_input(INPUT_POST, 'message', FILTER_SANITIZE_STRING);
// Validazione dei dati
$errors = [];
if (empty($name)) {
$errors[] = "Il nome è richiesto";
}
if (empty($email) || !filter_var($email, FILTER_VALIDATE_EMAIL)) {
$errors[] = "Email non valida";
}
if (empty($subject)) {
$errors[] = "L'oggetto è richiesto";
}
if (empty($message)) {
$errors[] = "Il messaggio è richiesto";
}
// Se non ci sono errori, procedi con l'invio della mail
if (empty($errors)) {
// Destinatario
$to = "info@tuoaggregatore.it";
// Intestazioni
$headers = "From: $name <$email>" . "\r\n";
$headers .= "Reply-To: $email" . "\r\n";
$headers .= "X-Mailer: PHP/" . phpversion();
// Prova a inviare l'email
$success = mail($to, $subject, $message, $headers);
if ($success) {
echo json_encode(['success' => true, 'message' => 'Messaggio inviato con successo']);
} else {
echo json_encode(['success' => false, 'message' => 'Impossibile inviare il messaggio']);
}
} else {
// Restituisci errori di validazione
echo json_encode(['success' => false, 'message' => implode(', ', $errors)]);
}
} else {
// Se qualcuno prova ad accedere direttamente a questa pagina
echo json_encode(['success' => false, 'message' => 'Metodo non consentito']);
}

View File

@@ -1,2 +1,6 @@
# RPIGroup Play # RPIGroup Play
Il player ufficiale del gruppo RPIGroup Il player ufficiale del gruppo RPIGroup
---
Questa repository è connessa al dominio APP.RPIGROUP.IT, ed è modificabile solo dagli utenti autorizzati

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@@ -32,17 +32,40 @@ header('Referrer-Policy: strict-origin-when-cross-origin');
<meta name="screen-orientation" content="portrait"> <meta name="screen-orientation" content="portrait">
<meta name="x5-orientation" content="portrait"> <meta name="x5-orientation" content="portrait">
<meta name="x5-fullscreen" content="true"> <meta name="x5-fullscreen" content="true">
<meta name="apple-mobile-web-app-orientations" content="portrait">
<meta name="browsermode" content="application"> <meta name="browsermode" content="application">
<link rel="manifest" href="<?=$base_path?>/manifest.json?v=<?=$version_app?>"> <link rel="manifest" href="<?=$base_path?>/manifest.json?v=<?=$version_app?>">
<link rel="stylesheet" href="<?=$base_path?>/css/bootstrap.css"> <link rel="stylesheet" href="<?=$base_path?>/css/bootstrap.css">
<link rel="stylesheet" href="<?=$base_path?>/css/style.css?v=<?=$version_app?>"> <link rel="stylesheet" href="<?=$base_path?>/css/style.css?v=<?=$version_app?>">
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet"> <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
<!-- <link rel="stylesheet" href="<?=$base_path?>/css/animation.css?v=<?=$version_app?>"> <link rel="stylesheet" href="<?=$base_path?>/css/animation.css?v=<?=$version_app?>">
<script src="https://code.jquery.com/jquery-2.2.4.js"></script> <script src="https://code.jquery.com/jquery-2.2.4.js?v=<?=$version_app?>"></script>
<script src="<?=$base_path?>/js/bootstrap.js"></script> --> <script src="https://cdn.jsdelivr.net/npm/hls.js@latest"></script>
<script src="<?=$base_path?>/js/bootstrap.js?v=<?=$version_app?>"></script>
<script> <script>
// Passa il percorso base a JavaScript // Passa il percorso base a JavaScript
var BASE_PATH = "<?=$base_path?>"; var BASE_PATH = "<?=$base_path?>";
// Forza l'orientamento portrait
if (window.screen && window.screen.orientation) {
// Tenta di bloccare l'orientamento in portrait
window.addEventListener('load', function() {
if (screen.orientation && screen.orientation.lock) {
screen.orientation.lock('portrait').catch(function(error) {
console.log('Orientation lock not supported or failed:', error);
});
}
});
}
// Fallback per dispositivi che non supportano l'API
// Mostra un avviso se l'utente ruota il dispositivo
window.addEventListener('orientationchange', function() {
if (window.orientation !== 0 && window.orientation !== 180) {
// Il dispositivo è in landscape, ma non possiamo forzarlo
console.log('Si prega di tenere il dispositivo in posizione verticale');
}
});
</script> </script>
</head> </head>
<body class="<?php echo $show_app ? 'appBody' : 'desktopBody'; ?>"> <body class="<?php echo $show_app ? 'appBody' : 'desktopBody'; ?>">

Binary file not shown.