forked from snxraven/peardock
Save connection settings as cookies, add a reset connection button to sidebar
This commit is contained in:
parent
7728e5e5b7
commit
730f356e14
137
app.js
137
app.js
@ -48,34 +48,88 @@ function waitForPeerResponse(expectedMessageFragment, timeout = 900000) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Utility functions for managing cookies
|
||||||
|
function setCookie(name, value, days = 365) {
|
||||||
|
const date = new Date();
|
||||||
|
date.setTime(date.getTime() + days * 24 * 60 * 60 * 1000);
|
||||||
|
const expires = `expires=${date.toUTCString()}`;
|
||||||
|
document.cookie = `${name}=${encodeURIComponent(value)};${expires};path=/`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCookie(name) {
|
||||||
|
const cookies = document.cookie.split('; ');
|
||||||
|
for (let i = 0; i < cookies.length; i++) {
|
||||||
|
const [key, value] = cookies[i].split('=');
|
||||||
|
if (key === name) return decodeURIComponent(value);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function deleteCookie(name) {
|
||||||
|
document.cookie = `${name}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load connections from cookies
|
||||||
|
function loadConnections() {
|
||||||
|
const savedConnections = getCookie('connections');
|
||||||
|
return savedConnections ? JSON.parse(savedConnections) : {};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save connections to cookies
|
||||||
|
function saveConnections() {
|
||||||
|
setCookie('connections', JSON.stringify(connections));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add Reset Connections Button
|
||||||
|
// Toggle Reset Connections Button Visibility
|
||||||
|
function toggleResetButtonVisibility() {
|
||||||
|
const resetConnectionsBtn = document.querySelector('#sidebar .btn-danger');
|
||||||
|
if (!resetConnectionsBtn) return;
|
||||||
|
|
||||||
|
// Show or hide the button based on active connections
|
||||||
|
resetConnectionsBtn.style.display = Object.keys(connections).length > 0 ? 'block' : 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add Reset Connections Button
|
||||||
|
const resetConnectionsBtn = document.createElement('button');
|
||||||
|
resetConnectionsBtn.textContent = 'Reset Connections';
|
||||||
|
resetConnectionsBtn.className = 'btn btn-danger w-100 mt-2';
|
||||||
|
resetConnectionsBtn.addEventListener('click', () => {
|
||||||
|
console.log('[INFO] Resetting connections and clearing cookies.');
|
||||||
|
Object.keys(connections).forEach((topicId) => {
|
||||||
|
disconnectConnection(topicId);
|
||||||
|
});
|
||||||
|
deleteCookie('connections');
|
||||||
|
resetConnectionsView();
|
||||||
|
showWelcomePage();
|
||||||
|
toggleResetButtonVisibility(); // Ensure button visibility is updated
|
||||||
|
});
|
||||||
|
document.getElementById('sidebar').appendChild(resetConnectionsBtn);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Initialize the app
|
// Initialize the app
|
||||||
console.log('[INFO] Client app initialized');
|
console.log('[INFO] Client app initialized');
|
||||||
|
// Load connections from cookies and restore them
|
||||||
document.addEventListener('DOMContentLoaded', () => {
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
console.log('[INFO] Initializing Client App');
|
const savedConnections = loadConnections();
|
||||||
|
console.log('[INFO] Restoring saved connections:', savedConnections);
|
||||||
|
|
||||||
const connectionTitle = document.getElementById('connection-title');
|
// Restore saved connections
|
||||||
if (!connectionTitle) {
|
Object.keys(savedConnections).forEach((topicId) => {
|
||||||
console.error('[ERROR] Connection title element is missing! Creating dynamically...');
|
const topicHex = savedConnections[topicId].topicHex;
|
||||||
const content = document.getElementById('content');
|
addConnection(topicHex); // Add connections using the saved topicHex
|
||||||
const titleElement = document.createElement('h1');
|
});
|
||||||
titleElement.id = 'connection-title';
|
|
||||||
titleElement.textContent = '';
|
|
||||||
titleElement.classList.add('hidden'); // Initially hidden
|
|
||||||
content.insertBefore(titleElement, content.firstChild);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Object.keys(connections).length === 0) {
|
if (Object.keys(connections).length > 0) {
|
||||||
showWelcomePage();
|
|
||||||
} else {
|
|
||||||
hideWelcomePage();
|
hideWelcomePage();
|
||||||
|
} else {
|
||||||
|
showWelcomePage();
|
||||||
}
|
}
|
||||||
|
|
||||||
assertVisibility(); // Ensure visibility is correct after initialization
|
assertVisibility(); // Ensure visibility reflects the restored connections
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
// Show Status Indicator
|
// Show Status Indicator
|
||||||
// Modify showStatusIndicator to recreate it dynamically
|
// Modify showStatusIndicator to recreate it dynamically
|
||||||
function showStatusIndicator(message = 'Processing...') {
|
function showStatusIndicator(message = 'Processing...') {
|
||||||
@ -137,6 +191,10 @@ collapseSidebarBtn.addEventListener('click', () => {
|
|||||||
sidebar.classList.toggle('collapsed');
|
sidebar.classList.toggle('collapsed');
|
||||||
const btn = collapseSidebarBtn;
|
const btn = collapseSidebarBtn;
|
||||||
btn.innerHTML = sidebar.classList.contains('collapsed') ? '>' : '<';
|
btn.innerHTML = sidebar.classList.contains('collapsed') ? '>' : '<';
|
||||||
|
|
||||||
|
// Toggle Reset Connections Button Visibility
|
||||||
|
const resetConnectionsBtn = document.querySelector('#sidebar .btn-danger');
|
||||||
|
resetConnectionsBtn.style.display = sidebar.classList.contains('collapsed') ? 'none' : 'block';
|
||||||
});
|
});
|
||||||
|
|
||||||
function handlePeerData(data, topicId, peer) {
|
function handlePeerData(data, topicId, peer) {
|
||||||
@ -188,20 +246,18 @@ addConnectionForm.addEventListener('submit', (e) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Function to add a new connection
|
|
||||||
function addConnection(topicHex) {
|
function addConnection(topicHex) {
|
||||||
console.log(`[DEBUG] Adding connection with topic: ${topicHex}`);
|
console.log(`[DEBUG] Adding connection with topic: ${topicHex}`);
|
||||||
|
|
||||||
if (Object.keys(connections).length === 0) {
|
if (Object.keys(connections).length === 0) {
|
||||||
console.log('[DEBUG] Hiding welcome page after first connection');
|
hideWelcomePage();
|
||||||
hideWelcomePage(); // Hide the welcome page when the first connection is added
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const topic = b4a.from(topicHex, 'hex');
|
const topic = b4a.from(topicHex, 'hex');
|
||||||
const topicId = topicHex.substring(0, 12);
|
const topicId = topicHex.substring(0, 12);
|
||||||
|
|
||||||
console.log(`[INFO] Adding connection with topic: ${topicHex}`);
|
connections[topicId] = { topic, peer: null, swarm: null, topicHex };
|
||||||
assertVisibility(); // Ensure visibility reflects the added connection
|
saveConnections(); // Save updated connections to cookies
|
||||||
|
|
||||||
const connectionItem = document.createElement('li');
|
const connectionItem = document.createElement('li');
|
||||||
connectionItem.className = 'list-group-item d-flex align-items-center justify-content-between';
|
connectionItem.className = 'list-group-item d-flex align-items-center justify-content-between';
|
||||||
@ -215,23 +271,14 @@ function addConnection(topicHex) {
|
|||||||
</button>
|
</button>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
// Add click event to switch connection
|
|
||||||
connectionItem.querySelector('span').addEventListener('click', () => switchConnection(topicId));
|
connectionItem.querySelector('span').addEventListener('click', () => switchConnection(topicId));
|
||||||
|
connectionItem.querySelector('.disconnect-btn').addEventListener('click', (e) => {
|
||||||
// Add click event to the disconnect button
|
e.stopPropagation();
|
||||||
const disconnectBtn = connectionItem.querySelector('.disconnect-btn');
|
|
||||||
disconnectBtn.addEventListener('click', (e) => {
|
|
||||||
e.stopPropagation(); // Prevent triggering the switch connection event
|
|
||||||
disconnectConnection(topicId, connectionItem);
|
disconnectConnection(topicId, connectionItem);
|
||||||
});
|
});
|
||||||
|
|
||||||
connectionList.appendChild(connectionItem);
|
connectionList.appendChild(connectionItem);
|
||||||
|
|
||||||
connections[topicId] = { topic, peer: null, swarm: null };
|
|
||||||
|
|
||||||
console.log('[DEBUG] Updated connections object:', connections);
|
|
||||||
|
|
||||||
// Create a new swarm for this connection
|
|
||||||
const swarm = new Hyperswarm();
|
const swarm = new Hyperswarm();
|
||||||
connections[topicId].swarm = swarm;
|
connections[topicId].swarm = swarm;
|
||||||
|
|
||||||
@ -239,27 +286,18 @@ function addConnection(topicHex) {
|
|||||||
|
|
||||||
swarm.on('connection', (peer) => {
|
swarm.on('connection', (peer) => {
|
||||||
console.log(`[INFO] Connected to peer for topic: ${topicHex}`);
|
console.log(`[INFO] Connected to peer for topic: ${topicHex}`);
|
||||||
|
|
||||||
if (connections[topicId].peer) {
|
if (connections[topicId].peer) {
|
||||||
console.warn(`[WARN] Duplicate connection detected for topic: ${topicId}. Closing.`);
|
|
||||||
peer.destroy();
|
peer.destroy();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
connections[topicId].peer = peer;
|
connections[topicId].peer = peer;
|
||||||
updateConnectionStatus(topicId, true);
|
updateConnectionStatus(topicId, true);
|
||||||
|
|
||||||
peer.on('data', (data) => {
|
peer.on('data', (data) => handlePeerData(data, topicId, peer));
|
||||||
handlePeerData(data, topicId, peer);
|
|
||||||
});
|
|
||||||
|
|
||||||
peer.on('close', () => {
|
peer.on('close', () => {
|
||||||
console.log(`[INFO] Disconnected from peer for topic: ${topicHex}`);
|
|
||||||
updateConnectionStatus(topicId, false);
|
updateConnectionStatus(topicId, false);
|
||||||
|
|
||||||
if (window.activePeer === peer) {
|
if (window.activePeer === peer) {
|
||||||
window.activePeer = null;
|
window.activePeer = null;
|
||||||
// connectionTitle.textContent = 'Disconnected';
|
|
||||||
dashboard.classList.add('hidden');
|
dashboard.classList.add('hidden');
|
||||||
containerList.innerHTML = '';
|
containerList.innerHTML = '';
|
||||||
}
|
}
|
||||||
@ -270,17 +308,28 @@ function addConnection(topicHex) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Automatically collapse the sidebar when a new connection is added
|
// Collapse the sidebar after adding a connection
|
||||||
const sidebar = document.getElementById('sidebar');
|
const sidebar = document.getElementById('sidebar');
|
||||||
const collapseSidebarBtn = document.getElementById('collapse-sidebar-btn');
|
const collapseSidebarBtn = document.getElementById('collapse-sidebar-btn');
|
||||||
if (!sidebar.classList.contains('collapsed')) {
|
if (!sidebar.classList.contains('collapsed')) {
|
||||||
sidebar.classList.add('collapsed');
|
sidebar.classList.add('collapsed');
|
||||||
collapseSidebarBtn.innerHTML = '>';
|
collapseSidebarBtn.innerHTML = '>';
|
||||||
console.log('[DEBUG] Sidebar auto-collapsed after adding a new connection');
|
console.log('[DEBUG] Sidebar collapsed after adding connection');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Initialize connections from cookies on page load
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const savedConnections = loadConnections();
|
||||||
|
console.log('[INFO] Loading saved connections:', savedConnections);
|
||||||
|
|
||||||
|
Object.keys(savedConnections).forEach((topicId) => {
|
||||||
|
const topicHex = savedConnections[topicId].topic;
|
||||||
|
addConnection(topicHex);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
function disconnectConnection(topicId, connectionItem) {
|
function disconnectConnection(topicId, connectionItem) {
|
||||||
const connection = connections[topicId];
|
const connection = connections[topicId];
|
||||||
|
185
index.html
185
index.html
@ -22,9 +22,10 @@
|
|||||||
color: white;
|
color: white;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.hidden {
|
.hidden {
|
||||||
display: none !important;
|
display: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
#titlebar {
|
#titlebar {
|
||||||
-webkit-app-region: drag;
|
-webkit-app-region: drag;
|
||||||
@ -78,14 +79,18 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#content {
|
#content {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column; /* Keep vertical stacking for child elements */
|
flex-direction: column;
|
||||||
margin-left: 250px; /* Leave space for the sidebar */
|
/* Keep vertical stacking for child elements */
|
||||||
flex: 1; /* Allow the content to grow */
|
margin-left: 250px;
|
||||||
padding: 30px;
|
/* Leave space for the sidebar */
|
||||||
overflow-y: auto; /* Allow scrolling if content overflows */
|
flex: 1;
|
||||||
position: relative;
|
/* Allow the content to grow */
|
||||||
}
|
padding: 30px;
|
||||||
|
overflow-y: auto;
|
||||||
|
/* Allow scrolling if content overflows */
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
#sidebar.collapsed~#content {
|
#sidebar.collapsed~#content {
|
||||||
margin-left: 50px;
|
margin-left: 50px;
|
||||||
@ -171,17 +176,18 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#status-indicator {
|
#status-indicator {
|
||||||
display: none; /* Ensure it's hidden by default */
|
display: none;
|
||||||
position: fixed;
|
/* Ensure it's hidden by default */
|
||||||
top: 0;
|
position: fixed;
|
||||||
left: 0;
|
top: 0;
|
||||||
width: 100%;
|
left: 0;
|
||||||
height: 100%;
|
width: 100%;
|
||||||
align-items: center;
|
height: 100%;
|
||||||
justify-content: center;
|
align-items: center;
|
||||||
background-color: rgba(0, 0, 0, 0.75);
|
justify-content: center;
|
||||||
z-index: 1050;
|
background-color: rgba(0, 0, 0, 0.75);
|
||||||
}
|
z-index: 1050;
|
||||||
|
}
|
||||||
|
|
||||||
#status-indicator .spinner-border {
|
#status-indicator .spinner-border {
|
||||||
width: 3rem;
|
width: 3rem;
|
||||||
@ -195,37 +201,126 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
#welcome-page {
|
#welcome-page {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
/* Stack child elements vertically */
|
||||||
|
justify-content: center;
|
||||||
|
/* Center content vertically */
|
||||||
|
align-items: center;
|
||||||
|
/* Center content horizontally */
|
||||||
|
text-align: center;
|
||||||
|
/* Center-align text */
|
||||||
|
position: absolute;
|
||||||
|
/* Overlay it over the content area */
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
/* Center it perfectly in the content */
|
||||||
|
max-width: 800px;
|
||||||
|
/* Restrict the width of the welcome page */
|
||||||
|
width: 100%;
|
||||||
|
/* Allow it to scale */
|
||||||
|
padding: 20px;
|
||||||
|
background-color: transparent;
|
||||||
|
/* Match the theme */
|
||||||
|
}
|
||||||
|
|
||||||
|
#welcome-page.hidden {
|
||||||
|
display: none !important;
|
||||||
|
/* Completely hide when not needed */
|
||||||
|
}
|
||||||
|
|
||||||
|
#dashboard {
|
||||||
|
display: flex;
|
||||||
|
/* Use flex layout for content within the dashboard */
|
||||||
|
flex-direction: column;
|
||||||
|
/* Stack elements vertically */
|
||||||
|
flex: 1;
|
||||||
|
/* Ensure it uses all available space */
|
||||||
|
width: 100%;
|
||||||
|
/* Take up the full width of the content area */
|
||||||
|
padding: 0;
|
||||||
|
/* Remove extra padding */
|
||||||
|
overflow-y: auto;
|
||||||
|
/* Allow vertical scrolling if needed */
|
||||||
|
position: relative;
|
||||||
|
/* Prevent overlap with other elements */
|
||||||
|
}
|
||||||
|
|
||||||
|
#dashboard.hidden {
|
||||||
|
display: none !important;
|
||||||
|
/* Hide the dashboard completely when not needed */
|
||||||
|
}
|
||||||
|
#alert-container {
|
||||||
|
position: fixed;
|
||||||
|
bottom: 20px;
|
||||||
|
right: 20px;
|
||||||
|
z-index: 1055; /* Ensure it overlays important elements only */
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column; /* Stack child elements vertically */
|
flex-direction: column-reverse; /* Stack alerts upwards */
|
||||||
justify-content: center; /* Center content vertically */
|
gap: 10px; /* Add space between alerts */
|
||||||
align-items: center; /* Center content horizontally */
|
pointer-events: none; /* Prevent container from blocking clicks */
|
||||||
text-align: center; /* Center-align text */
|
|
||||||
position: absolute; /* Overlay it over the content area */
|
|
||||||
top: 50%;
|
|
||||||
left: 50%;
|
|
||||||
transform: translate(-50%, -50%); /* Center it perfectly in the content */
|
|
||||||
max-width: 800px; /* Restrict the width of the welcome page */
|
|
||||||
width: 100%; /* Allow it to scale */
|
|
||||||
padding: 20px;
|
|
||||||
background-color: transparent; /* Match the theme */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#welcome-page.hidden {
|
.alert {
|
||||||
display: none !important; /* Completely hide when not needed */
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
max-width: 80%;
|
||||||
|
padding: 12px 20px;
|
||||||
|
background-color: #2b2b2b;
|
||||||
|
color: #e0e0e0;
|
||||||
|
border-radius: 6px;
|
||||||
|
box-shadow: 0 4px 10px rgba(0, 0, 0, 0.5);
|
||||||
|
font-family: Arial, sans-serif;
|
||||||
|
font-size: 14px;
|
||||||
|
animation: fadeIn 0.3s ease-out, fadeOut 4.5s ease-in forwards;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
pointer-events: auto; /* Allow alerts to be interactive */
|
||||||
}
|
}
|
||||||
|
|
||||||
#dashboard {
|
.alert.success {
|
||||||
display: flex; /* Use flex layout for content within the dashboard */
|
border-left: 6px solid #28a745; /* Green border for success */
|
||||||
flex-direction: column; /* Stack elements vertically */
|
|
||||||
flex: 1; /* Ensure it uses all available space */
|
|
||||||
width: 100%; /* Take up the full width of the content area */
|
|
||||||
padding: 0; /* Remove extra padding */
|
|
||||||
overflow-y: auto; /* Allow vertical scrolling if needed */
|
|
||||||
position: relative; /* Prevent overlap with other elements */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#dashboard.hidden {
|
.alert.danger {
|
||||||
display: none !important; /* Hide the dashboard completely when not needed */
|
border-left: 6px solid #dc3545; /* Red border for danger */
|
||||||
|
}
|
||||||
|
|
||||||
|
.alert .close-btn {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: #e0e0e0;
|
||||||
|
font-size: 16px;
|
||||||
|
cursor: pointer;
|
||||||
|
margin-left: auto; /* Align to the far right */
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
#alert-container {
|
||||||
|
bottom: 10px;
|
||||||
|
right: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.alert {
|
||||||
|
max-width: 90%;
|
||||||
|
font-size: 12px;
|
||||||
|
padding: 10px 15px;
|
||||||
|
white-space: normal; /* Allow wrapping on small screens */
|
||||||
|
text-overflow: clip; /* Disable ellipsis when wrapping */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
.alert {
|
||||||
|
white-space: nowrap; /* Re-enable nowrap for larger screens */
|
||||||
|
text-overflow: ellipsis; /* Add ellipsis for overflowed text */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#sidebar.collapsed .btn-danger {
|
||||||
|
display: none;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user