471 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			471 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| let isSortOptionsOpen = false;
 | |
| 
 | |
| function toggleOpenSubscription(subId) {
 | |
|   const subscriptionElement = document.querySelector('.subscription[data-id="' + subId + '"]');
 | |
|   subscriptionElement.classList.toggle('is-open');
 | |
| }
 | |
| 
 | |
| function toggleSortOptions() {
 | |
|   const sortOptions = document.querySelector("#sort-options");
 | |
|   sortOptions.classList.toggle("is-open");
 | |
|   isSortOptionsOpen = !isSortOptionsOpen;
 | |
| }
 | |
| 
 | |
| function toggleNotificationDays() {
 | |
|   const notifyCheckbox = document.querySelector("#notifications");
 | |
|   const notifyDaysBefore = document.querySelector("#notify_days_before");
 | |
|   notifyDaysBefore.disabled = !notifyCheckbox.checked;
 | |
| }
 | |
| 
 | |
| function resetForm() {
 | |
|     const id = document.querySelector("#id");
 | |
|     id.value = "";
 | |
|     const formTitle = document.querySelector("#form-title");
 | |
|     formTitle.textContent = translate('add_subscription');
 | |
|     const logo = document.querySelector("#form-logo");
 | |
|     logo.src = "";
 | |
|     logo.style = 'display: none';
 | |
|     const logoUrl = document.querySelector("#logo-url");
 | |
|     logoUrl.value = "";
 | |
|     const logoSearchButton = document.querySelector("#logo-search-button");
 | |
|     logoSearchButton.classList.add("disabled");
 | |
|     const submitButton = document.querySelector("#save-button");
 | |
|     submitButton.disabled = false;
 | |
|     const notifyDaysBefore = document.querySelector("#notify_days_before");
 | |
|     notifyDaysBefore.disabled = true;
 | |
|     const form = document.querySelector("#subs-form");
 | |
|     form.reset();
 | |
|     closeLogoSearch();
 | |
|     const deleteButton = document.querySelector("#deletesub");
 | |
|     deleteButton.style = 'display: none';
 | |
|     deleteButton.removeAttribute("onClick");
 | |
| }
 | |
| 
 | |
| function fillEditFormFields(subscription) {
 | |
|   const formTitle = document.querySelector("#form-title");
 | |
|   formTitle.textContent = translate('edit_subscription');
 | |
|   const logo = document.querySelector("#form-logo");
 | |
|   const defaultLogo = window.theme && window.theme == "light" ? "images/siteicons/" + colorTheme + "/wallos.png" : "images/siteicons/" + colorTheme + "/walloswhite.png";
 | |
|   const logoFile = subscription.logo !== null ? "images/uploads/logos/" + subscription.logo : defaultLogo;
 | |
|   logo.src = logoFile;
 | |
|   logo.style = 'display: block';
 | |
|   const logoSearchButton = document.querySelector("#logo-search-button");
 | |
|   logoSearchButton.classList.remove("disabled");
 | |
|   const id = document.querySelector("#id");
 | |
|   id.value = subscription.id;
 | |
|   const name = document.querySelector("#name");
 | |
|   name.value = subscription.name;
 | |
|   const price = document.querySelector("#price");
 | |
|   price.value = subscription.price;
 | |
| 
 | |
|   const currencySelect = document.querySelector("#currency");
 | |
|   currencySelect.value = subscription.currency_id.toString();
 | |
|   const frequencySelect = document.querySelector("#frequency");
 | |
|   frequencySelect.value = subscription.frequency;
 | |
|   const cycleSelect = document.querySelector("#cycle");
 | |
|   cycleSelect.value = subscription.cycle;
 | |
|   const paymentSelect = document.querySelector("#payment_method");
 | |
|   paymentSelect.value = subscription.payment_method_id;
 | |
|   const categorySelect = document.querySelector("#category");
 | |
|   categorySelect.value = subscription.category_id;
 | |
|   const payerSelect = document.querySelector("#payer_user");
 | |
|   payerSelect.value = subscription.payer_user_id;
 | |
| 
 | |
|   const nextPament = document.querySelector("#next_payment");
 | |
|   nextPament.value = subscription.next_payment;
 | |
|   const notes = document.querySelector("#notes");
 | |
|   notes.value = subscription.notes;
 | |
|   const inactive = document.querySelector("#inactive");
 | |
|   inactive.checked = subscription.inactive;
 | |
|   const url = document.querySelector("#url");
 | |
|   url.value = subscription.url;
 | |
| 
 | |
|   const notifications = document.querySelector("#notifications");
 | |
|   if (notifications) {
 | |
|     notifications.checked = subscription.notify;
 | |
|   }
 | |
| 
 | |
|   const notifyDaysBefore = document.querySelector("#notify_days_before");
 | |
|   notifyDaysBefore.value = subscription.notify_days_before;
 | |
|   if (subscription.notify === 1) {
 | |
|     notifyDaysBefore.disabled = false;
 | |
|   }
 | |
| 
 | |
|   const deleteButton = document.querySelector("#deletesub");
 | |
|   deleteButton.style = 'display: block';
 | |
|   deleteButton.setAttribute("onClick", `deleteSubscription(${subscription.id})`);
 | |
| 
 | |
|   const modal = document.getElementById('subscription-form');
 | |
|   modal.classList.add("is-open");
 | |
| }
 | |
| 
 | |
| function openEditSubscription(event, id) {
 | |
|     event.stopPropagation();
 | |
|     const body = document.querySelector('body');
 | |
|     body.classList.add('no-scroll');
 | |
|     const url = `endpoints/subscription/get.php?id=${id}`;
 | |
|     fetch(url)
 | |
|     .then((response) => {
 | |
|       if (response.ok) {
 | |
|         return response.json();
 | |
|       } else {
 | |
|         showErrorMessage(translate('failed_to_load_subscription'));
 | |
|       }
 | |
|     })
 | |
|     .then((data) => {
 | |
|       if (data.error || data === "Error") {
 | |
|         showErrorMessage(translate('failed_to_load_subscription'));
 | |
|       } else {
 | |
|         const subscription = data;
 | |
|         fillEditFormFields(subscription);
 | |
|       }
 | |
|     })
 | |
|     .catch((error) => {
 | |
|       showErrorMessage(translate('failed_to_load_subscription'));
 | |
|     });
 | |
| }
 | |
| 
 | |
| function addSubscription() {
 | |
|     resetForm();
 | |
|     const modal = document.getElementById('subscription-form');
 | |
|     modal.classList.add("is-open"); 
 | |
|     const body = document.querySelector('body');
 | |
|     body.classList.add('no-scroll');
 | |
| }
 | |
| 
 | |
| function closeAddSubscription() {
 | |
|     const modal = document.getElementById('subscription-form');
 | |
|     modal.classList.remove("is-open"); 
 | |
|     const body = document.querySelector('body');
 | |
|     body.classList.remove('no-scroll');
 | |
|     resetForm();
 | |
| }
 | |
| 
 | |
| function handleFileSelect(event) {
 | |
|     const fileInput = event.target;
 | |
|     const logoPreview = document.querySelector('.logo-preview');
 | |
|     const logoImg = logoPreview.querySelector('img');
 | |
|     const logoUrl = document.querySelector("#logo-url");
 | |
|     logoUrl.value = "";
 | |
| 
 | |
|     if (fileInput.files && fileInput.files[0]) {
 | |
|         const reader = new FileReader();
 | |
| 
 | |
|         reader.onload = function (e) {
 | |
|             logoImg.src = e.target.result;
 | |
|             logoImg.style.display = 'block';
 | |
|         };
 | |
| 
 | |
|         reader.readAsDataURL(fileInput.files[0]);
 | |
|     }
 | |
| }
 | |
| 
 | |
| function deleteSubscription(id) {
 | |
|   if (confirm(translate('confirm_delete_subscription'))) {
 | |
|     fetch(`endpoints/subscription/delete.php?id=${id}`, {
 | |
|       method: 'DELETE',
 | |
|     })
 | |
|     .then(response => {
 | |
|         if (response.ok) {
 | |
|           showSuccessMessage(translate('subscription_deleted'));
 | |
|           fetchSubscriptions();
 | |
|           closeAddSubscription();
 | |
|         } else {
 | |
|           alert(translate('error_deleting_subscription'));
 | |
|         }
 | |
|     })
 | |
|     .catch(error => {
 | |
|         console.error('Error:', error);
 | |
|     });
 | |
|   }
 | |
| }
 | |
| 
 | |
| function setSearchButtonStatus() {
 | |
| 
 | |
|   const nameInput = document.querySelector("#name");
 | |
|   const hasSearchTerm = nameInput.value.trim().length > 0;
 | |
|   const logoSearchButton = document.querySelector("#logo-search-button");
 | |
|   if (hasSearchTerm) {
 | |
|     logoSearchButton.classList.remove("disabled");
 | |
|   } else {
 | |
|     logoSearchButton.classList.add("disabled");
 | |
|   }
 | |
| 
 | |
| }
 | |
| 
 | |
| function searchLogo() {
 | |
|   const nameInput = document.querySelector("#name");
 | |
|   const searchTerm = nameInput.value.trim();
 | |
|   if (searchTerm !== "") {
 | |
|     const logoSearchPopup = document.querySelector("#logo-search-results");
 | |
|     logoSearchPopup.classList.add("is-open");
 | |
|     const imageSearchUrl = `endpoints/logos/search.php?search=${searchTerm}`;
 | |
|     fetch(imageSearchUrl)
 | |
|             .then(response => response.json())
 | |
|             .then(data => {
 | |
|                 if (data.imageUrls) {
 | |
|                     displayImageResults(data.imageUrls);
 | |
|                 } else if (data.error) {
 | |
|                     console.error(data.error);
 | |
|                 }
 | |
|             })
 | |
|             .catch(error => {
 | |
|                 console.error(translate('error_fetching_image_results'), error);
 | |
|             });
 | |
|   } else {
 | |
|     nameInput.focus();
 | |
|   }
 | |
| }
 | |
| 
 | |
| function displayImageResults(imageSources) {
 | |
|   const logoResults = document.querySelector("#logo-search-images");
 | |
|   logoResults.innerHTML = "";
 | |
| 
 | |
|   imageSources.forEach(src => {
 | |
|       const img = document.createElement("img");
 | |
|       img.src = src;
 | |
|       img.onclick = function() {
 | |
|         selectWebLogo(src);
 | |
|       };
 | |
|       img.onerror = function() {
 | |
|         this.parentNode.removeChild(this);
 | |
|       };
 | |
|       logoResults.appendChild(img);
 | |
|   });
 | |
| }
 | |
| 
 | |
| function selectWebLogo(url) {
 | |
|   closeLogoSearch();
 | |
|   const logoPreview = document.querySelector("#form-logo");
 | |
|   const logoUrl = document.querySelector("#logo-url");
 | |
|   logoPreview.src = url;
 | |
|   logoPreview.style.display = 'block';
 | |
|   logoUrl.value = url;
 | |
| }
 | |
| 
 | |
| function closeLogoSearch() {
 | |
|   const logoSearchPopup = document.querySelector("#logo-search-results");
 | |
|   logoSearchPopup.classList.remove("is-open");
 | |
|   const logoResults = document.querySelector("#logo-search-images");
 | |
|   logoResults.innerHTML = "";
 | |
| }
 | |
| 
 | |
| function fetchSubscriptions() {
 | |
|   const subscriptionsContainer = document.querySelector("#subscriptions");
 | |
|   let getSubscriptions = "endpoints/subscriptions/get.php";
 | |
| 
 | |
|   if (activeFilters['category'] !== "") {
 | |
|     getSubscriptions += `?category=${activeFilters['category']}`;
 | |
|   }
 | |
|   if (activeFilters['member'] !== "") {
 | |
|     getSubscriptions += getSubscriptions.includes("?") ? `&member=${activeFilters['member']}` : `?member=${activeFilters['member']}`;
 | |
|   }
 | |
|   if (activeFilters['payment'] !== "") {
 | |
|     getSubscriptions += getSubscriptions.includes("?") ? `&payment=${activeFilters['payment']}` : `?payment=${activeFilters['payment']}`;
 | |
|   }
 | |
| 
 | |
|   fetch(getSubscriptions)
 | |
|     .then(response => response.text())
 | |
|     .then(data => {
 | |
|       if (data) {
 | |
|         subscriptionsContainer.innerHTML = data;
 | |
|         const mainActions = document.querySelector("#main-actions");
 | |
|         if (data.includes("no-matching-subscriptions")) {
 | |
|           // mainActions.classList.add("hidden");
 | |
|         } else {
 | |
|           mainActions.classList.remove("hidden");
 | |
|         }
 | |
|       }
 | |
|     })
 | |
|     .catch(error => {
 | |
|       console.error(translate('error_reloading_subscription'), error);
 | |
|     });
 | |
| }
 | |
| 
 | |
| function setSortOption(sortOption) {
 | |
|   const sortOptionsContainer = document.querySelector("#sort-options");
 | |
|   const sortOptionsList = sortOptionsContainer.querySelectorAll("li");
 | |
|   sortOptionsList.forEach((option) => { 
 | |
|     if (option.getAttribute("id") === "sort-" + sortOption) {
 | |
|       option.classList.add("selected");
 | |
|     } else {
 | |
|       option.classList.remove("selected");
 | |
|     }
 | |
|   });
 | |
|   const daysToExpire = 30;
 | |
|   const expirationDate = new Date();
 | |
|   expirationDate.setDate(expirationDate.getDate() + daysToExpire);
 | |
|   const cookieValue = encodeURIComponent(sortOption) + '; expires=' + expirationDate.toUTCString();
 | |
|   document.cookie = 'sortOrder=' + cookieValue;
 | |
|   fetchSubscriptions();
 | |
|   toggleSortOptions();
 | |
| }
 | |
| 
 | |
| document.addEventListener('DOMContentLoaded', function() {
 | |
|     const subscriptionForm = document.querySelector("#subs-form");
 | |
|     const submitButton = document.querySelector("#save-button");
 | |
|     const endpoint = "endpoints/subscription/add.php";
 | |
| 
 | |
|     subscriptionForm.addEventListener("submit", function (e) {
 | |
|     e.preventDefault();
 | |
|     
 | |
|     submitButton.disabled = true;
 | |
|     const formData = new FormData(subscriptionForm);
 | |
| 
 | |
|     fetch(endpoint, {
 | |
|         method: "POST",
 | |
|         body: formData,
 | |
|     })
 | |
|     .then((response) => response.json())
 | |
|     .then((data) => {
 | |
|       if (data.status === "Success") {
 | |
|         showSuccessMessage(data.message);
 | |
|         fetchSubscriptions();
 | |
|         closeAddSubscription();
 | |
|       }
 | |
|     })
 | |
|     .catch((error) => {
 | |
|         showErrorMessage(error);
 | |
|         submitButton.disabled = false;
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     document.addEventListener('mousedown', function(event) {
 | |
|       const sortOptions = document.querySelector('#sort-options');
 | |
|       const sortButton = document.querySelector("#sort-button");
 | |
| 
 | |
|       if (!sortOptions.contains(event.target) && !sortButton.contains(event.target) && isSortOptionsOpen) {
 | |
|         sortOptions.classList.remove('is-open');
 | |
|         isSortOptionsOpen = false;
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     document.querySelector('#sort-options').addEventListener('focus', function() {
 | |
|         isSortOptionsOpen = true;
 | |
|     });
 | |
| });
 | |
| 
 | |
| function searchSubscriptions() {
 | |
|     const searchInput = document.querySelector("#search");
 | |
|     const searchTerm = searchInput.value.trim().toLowerCase();
 | |
| 
 | |
|     const subscriptions = document.querySelectorAll(".subscription");
 | |
|     subscriptions.forEach(subscription => {
 | |
|         const name = subscription.getAttribute('data-name').toLowerCase();
 | |
|         if (!name.includes(searchTerm)) {
 | |
|             subscription.classList.add("hide");
 | |
|         } else {
 | |
|             subscription.classList.remove("hide");
 | |
|         }
 | |
|     });
 | |
| }
 | |
| 
 | |
| function closeSubMenus() {
 | |
|   var subMenus = document.querySelectorAll('.filtermenu-submenu-content');
 | |
|   subMenus.forEach(subMenu => {
 | |
|       subMenu.classList.remove('is-open');
 | |
|   });
 | |
| 
 | |
| }
 | |
| 
 | |
| const activeFilters = [];
 | |
| activeFilters['category'] = "";
 | |
| activeFilters['member'] = "";
 | |
| activeFilters['payment'] = "";
 | |
| 
 | |
| document.addEventListener("DOMContentLoaded", function() {
 | |
|   var filtermenu = document.querySelector('#filtermenu-button');
 | |
|   filtermenu.addEventListener('click', function() {
 | |
|       this.parentElement.querySelector('.filtermenu-content').classList.toggle('is-open');
 | |
|       closeSubMenus();
 | |
|   });
 | |
| 
 | |
|   document.addEventListener('click', function(e) {
 | |
|       var filtermenuContent = document.querySelector('.filtermenu-content');
 | |
|       if (filtermenuContent.classList.contains('is-open')) {
 | |
|           var subMenus = document.querySelectorAll('.filtermenu-submenu');
 | |
|           var clickedInsideSubmenu = Array.from(subMenus).some(subMenu => subMenu.contains(e.target) || subMenu === e.target);
 | |
| 
 | |
|           if (!filtermenu.contains(e.target) && !clickedInsideSubmenu) {
 | |
|               closeSubMenus();
 | |
|               filtermenuContent.classList.remove('is-open');
 | |
|           }
 | |
|       }
 | |
|   });
 | |
| });
 | |
| 
 | |
| function toggleSubMenu(subMenu) {
 | |
|   var subMenu = document.getElementById("filter-" + subMenu);
 | |
|   if (subMenu.classList.contains("is-open")) {
 | |
|       closeSubMenus();
 | |
|   } else {
 | |
|       closeSubMenus();
 | |
|       subMenu.classList.add("is-open");
 | |
|   }
 | |
| }
 | |
| 
 | |
| document.querySelectorAll('.filter-item').forEach(function(item) {
 | |
|   item.addEventListener('click', function(e) {
 | |
|     const searchInput = document.querySelector("#search");
 | |
|     searchInput.value = "";
 | |
| 
 | |
|     if (this.hasAttribute('data-categoryid')) {
 | |
|         const categoryId = this.getAttribute('data-categoryid');
 | |
|         if (activeFilters['category'] === categoryId) {
 | |
|             activeFilters['category'] = "";
 | |
|             this.classList.remove('selected');
 | |
|         } else {
 | |
|             activeFilters['category'] = categoryId;
 | |
|             Array.from(this.parentNode.children).forEach(sibling => {
 | |
|               sibling.classList.remove('selected');
 | |
|             });
 | |
|             this.classList.add('selected');
 | |
|         }
 | |
|     } else if (this.hasAttribute('data-memberid')) {
 | |
|         const memberId = this.getAttribute('data-memberid');
 | |
|         if (activeFilters['member'] === memberId) {
 | |
|             activeFilters['member'] = "";
 | |
|             this.classList.remove('selected');
 | |
|         } else {
 | |
|             activeFilters['member'] = memberId;
 | |
|             Array.from(this.parentNode.children).forEach(sibling => {
 | |
|               sibling.classList.remove('selected');
 | |
|             });
 | |
|             this.classList.add('selected');
 | |
|         }
 | |
|     } else if (this.hasAttribute('data-paymentid')) {
 | |
|         const paymentId = this.getAttribute('data-paymentid');
 | |
|         if (activeFilters['payment'] === paymentId) {
 | |
|             activeFilters['payment'] = "";
 | |
|             this.classList.remove('selected');
 | |
|         } else {
 | |
|             activeFilters['payment'] = paymentId;
 | |
|             Array.from(this.parentNode.children).forEach(sibling => {
 | |
|               sibling.classList.remove('selected');
 | |
|             }); 
 | |
|             this.classList.add('selected');
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (activeFilters['category'] !== "" || activeFilters['member'] !== "" || activeFilters['payment'] !== "") {
 | |
|         document.querySelector('#clear-filters').classList.remove('hide');
 | |
|     } else {
 | |
|         document.querySelector('#clear-filters').classList.add('hide');
 | |
|     }
 | |
| 
 | |
|     fetchSubscriptions();
 | |
|   });
 | |
| });
 | |
| 
 | |
| function clearFilters() {
 | |
|   const searchInput = document.querySelector("#search");
 | |
|   searchInput.value = "";
 | |
|   activeFilters['category'] = "";
 | |
|   activeFilters['member'] = "";
 | |
|   activeFilters['payment'] = "";
 | |
|   document.querySelectorAll('.filter-item').forEach(function(item) {
 | |
|     item.classList.remove('selected');
 | |
|   });
 | |
|   document.querySelector('#clear-filters').classList.add('hide');
 | |
|   fetchSubscriptions();
 | |
| } |