Add translation system (#85)
This commit is contained in:
		
							parent
							
								
									d287f303f0
								
							
						
					
					
						commit
						d7366dcfb0
					
				
							
								
								
									
										10
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								README.md
									
									
									
									
									
								
							| @ -17,6 +17,7 @@ Wallos: Open-Source Personal Subscription Tracker | |||||||
|     - [Docker-Compose](#docker-compose) |     - [Docker-Compose](#docker-compose) | ||||||
| - [Usage](#usage) | - [Usage](#usage) | ||||||
| - [Contributing](#contributing) | - [Contributing](#contributing) | ||||||
|  |   - [Translations](#translations) | ||||||
| - [Screenshots](#screenshots) | - [Screenshots](#screenshots) | ||||||
| - [License](#license) | - [License](#license) | ||||||
| 
 | 
 | ||||||
| @ -139,6 +140,15 @@ Feel free to open Pull requests with bug fixes and features. I'll do my best to | |||||||
| Feel free to open issues with bug reports or feature requests. Bug fixes will take priority.   | Feel free to open issues with bug reports or feature requests. Bug fixes will take priority.   | ||||||
| I welcome contributions from the community and look forward to working with you to improve this project. | I welcome contributions from the community and look forward to working with you to improve this project. | ||||||
| 
 | 
 | ||||||
|  | ### Translations | ||||||
|  | 
 | ||||||
|  | If you want to contribute with a translation of wallos: | ||||||
|  | - Add your language code to `includes/i18n/languages.php` in the format `"en" => "English"`. Please use the original language name and not the english translation. | ||||||
|  | - Create a copy of the file `includes/i18n/en.php` and rename it to the language code you used above. Example: pt.php for "pt" => "Português". | ||||||
|  | - Translate all the values on the language file to the new language. (Incomplete translations will not be accepted). | ||||||
|  | - Create a copy of the file `scripts/i18n/en.php` and rename it to the language code you used above. | ||||||
|  | - Translate all the values on the language file to the new language. (Incomplete translations will not be accepted). | ||||||
|  | 
 | ||||||
| ## License | ## License | ||||||
| 
 | 
 | ||||||
| This project is licensed under the [GNU General Public License, Version 3](LICENSE.md) - see the [LICENSE.md](LICENSE.md) file for details. | This project is licensed under the [GNU General Public License, Version 3](LICENSE.md) - see the [LICENSE.md](LICENSE.md) file for details. | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								about.php
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								about.php
									
									
									
									
									
								
							| @ -6,50 +6,50 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>About and Credits</h2> |             <h2><?= translate('about_and_credits', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="credits-list"> |         <div class="credits-list"> | ||||||
|             <p>Wallos v1.1.0</p> |             <p>Wallos v1.2.0</p> | ||||||
|             <p>License:  |             <p><?= translate('license', $i18n) ?>: 
 | ||||||
|                 <span> |                 <span> | ||||||
|                     GPLv3 |                     GPLv3 | ||||||
|                     <a href="https://www.gnu.org/licenses/gpl-3.0.en.html" target="_blank" title="Visit external url"> |                     <a href="https://www.gnu.org/licenses/gpl-3.0.en.html" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
|             </p> |             </p> | ||||||
|             <p> |             <p> | ||||||
|                 Issues and Request:  |             <?= translate('issues_and_requests', $i18n) ?>: 
 | ||||||
|                 <span> |                 <span> | ||||||
|                     GitHub |                     GitHub | ||||||
|                     <a href="https://github.com/ellite/Wallos/issues" target="_blank" title="Visit external url"> |                     <a href="https://github.com/ellite/Wallos/issues" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
|             </p> |             </p> | ||||||
|             <p> |             <p> | ||||||
|                 The author:  |                 <?= translate('the_author', $i18n) ?>: 
 | ||||||
|                 <span> |                 <span> | ||||||
|                     https://henrique.pt |                     https://henrique.pt | ||||||
|                     <a href="https://henrique.pt/" target="_blank" title="Visit external url"> |                     <a href="https://henrique.pt/" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
|             </p> |             </p> | ||||||
|             <p> |             <p> | ||||||
|                 Icons:  |                 <?= translate('icons', $i18n) ?>: 
 | ||||||
|                 <span> |                 <span> | ||||||
|                     https://www.streamlinehq.com/freebies/plump-flat-free |                     https://www.streamlinehq.com/freebies/plump-flat-free | ||||||
|                     <a href="https://www.streamlinehq.com/freebies/plump-flat-free" target="_blank" title="Visit external url"> |                     <a href="https://www.streamlinehq.com/freebies/plump-flat-free" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
|             </p> |             </p> | ||||||
|             <p> |             <p> | ||||||
|                 Payment Icons:  |                 <?= translate('payment_icons', $i18n) ?>: 
 | ||||||
|                 <span> |                 <span> | ||||||
|                     https://www.figma.com/file/5IMW8JfoXfB5GRlPNdTyeg/Credit-Cards-and-Payment-Methods-Icons-(Community) |                     https://www.figma.com/file/5IMW8JfoXfB5GRlPNdTyeg/Credit-Cards-and-Payment-Methods-Icons-(Community) | ||||||
|                     <a href="https://www.figma.com/file/5IMW8JfoXfB5GRlPNdTyeg/Credit-Cards-and-Payment-Methods-Icons-(Community)" target="_blank" title="Visit external url"> |                     <a href="https://www.figma.com/file/5IMW8JfoXfB5GRlPNdTyeg/Credit-Cards-and-Payment-Methods-Icons-(Community)" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
| @ -58,7 +58,7 @@ | |||||||
|                 Chart.js:  |                 Chart.js:  | ||||||
|                 <span> |                 <span> | ||||||
|                     https://www.chartjs.org/ |                     https://www.chartjs.org/ | ||||||
|                     <a href="https://www.chartjs.org/" target="_blank" title="Visit external url"> |                     <a href="https://www.chartjs.org/" target="_blank" title="<?= translate('external_url', $i18n) ?>"> | ||||||
|                         <i class="fa-solid fa-arrow-up-right-from-square"></i> |                         <i class="fa-solid fa-arrow-up-right-from-square"></i> | ||||||
|                     </a> |                     </a> | ||||||
|                 </span> |                 </span> | ||||||
|  | |||||||
| @ -19,7 +19,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Failed to add category" |                 "errorMessage" => translate('failed_add_category', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -35,20 +35,21 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
| 
 | 
 | ||||||
|             if ($result) { |             if ($result) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => true |                     "success" => true, | ||||||
|  |                     "message" => translate('category_saved', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Failed to edit category" |                     "errorMessage" => translate('failed_edit_category', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Please fill all the fields" |                 "errorMessage" => translate('fill_all_fields', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -65,7 +66,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             if ($count > 0) { |             if ($count > 0) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Category is in use in subscriptions and can't be removed" |                     "errorMessage" => translate('category_in_use', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
| @ -75,13 +76,14 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|                 $result = $stmt->execute(); |                 $result = $stmt->execute(); | ||||||
|                 if ($result) { |                 if ($result) { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => true |                         "success" => true, | ||||||
|  |                         "message" => translate('category_removed', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } else { |                 } else { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => false, |                         "success" => false, | ||||||
|                         "errorMessage" => "Failed to remove category" |                         "errorMessage" => translate('failed_remove_category', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } |                 } | ||||||
| @ -89,15 +91,15 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Failed to remove category" |                 "errorMessage" => translate('failed_remove_category', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         echo "Error"; |         echo translate('error', $i18n); | ||||||
|     } |     } | ||||||
| } else { | } else { | ||||||
|     echo "Error"; |     echo translate('error', $i18n); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ?>
 | ?>
 | ||||||
| @ -19,7 +19,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             $currencyId = $db->lastInsertRowID(); |             $currencyId = $db->lastInsertRowID(); | ||||||
|             echo $currencyId; |             echo $currencyId; | ||||||
|         } else { |         } else { | ||||||
|             echo "Error adding currency entry."; |             echo translate('error_adding_currency', $i18n); | ||||||
|         } |         } | ||||||
|     } else if (isset($_GET['action']) && $_GET['action'] == "edit") { |     } else if (isset($_GET['action']) && $_GET['action'] == "edit") { | ||||||
|         if (isset($_GET['currencyId']) && $_GET['currencyId'] != "" && isset($_GET['name']) && $_GET['name'] != "" && isset($_GET['symbol']) && $_GET['symbol'] != "") { |         if (isset($_GET['currencyId']) && $_GET['currencyId'] != "" && isset($_GET['name']) && $_GET['name'] != "" && isset($_GET['symbol']) && $_GET['symbol'] != "") { | ||||||
| @ -36,18 +36,22 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             $result = $stmt->execute(); |             $result = $stmt->execute(); | ||||||
| 
 | 
 | ||||||
|             if ($result) { |             if ($result) { | ||||||
|                 echo json_encode(["success" => true]); |                 $response = [ | ||||||
|  |                     "success" => true, | ||||||
|  |                     "message" => $name . " " . translate('currency_saved', $i18n) | ||||||
|  |                 ]; | ||||||
|  |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "message" => "Failed to store Currency on the Database" |                     "message" => translate('failed_to_store_currency', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "message" => "Some fields are missing" |                 "message" => translate('fields_missing', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -70,7 +74,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             if ($count > 0) { |             if ($count > 0) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "message" => "Currency is in use in subscriptions and can't be deleted." |                     "message" => translate('currency_in_use', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|                 exit; |                 exit; | ||||||
| @ -78,7 +82,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|                 if ($currencyId == $mainCurrencyId) { |                 if ($currencyId == $mainCurrencyId) { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => false, |                         "success" => false, | ||||||
|                         "message" => "Currency is set as main currency and can't be deleted." |                         "message" => translate('currency_is_main', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                     exit; |                     exit; | ||||||
| @ -88,11 +92,11 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|                     $stmt->bindParam(':currencyId', $currencyId, SQLITE3_INTEGER); |                     $stmt->bindParam(':currencyId', $currencyId, SQLITE3_INTEGER); | ||||||
|                     $result = $stmt->execute(); |                     $result = $stmt->execute(); | ||||||
|                     if ($result) { |                     if ($result) { | ||||||
|                         echo json_encode(["success" => true]); |                         echo json_encode(["success" => true, "message" => translate('currency_removed', $i18n)]); | ||||||
|                     } else { |                     } else { | ||||||
|                         $response = [ |                         $response = [ | ||||||
|                             "success" => false, |                             "success" => false, | ||||||
|                             "message" => "Failed to remove currency from the Database" |                             "message" => translate('failed_to_remove_currency', $i18n) | ||||||
|                         ]; |                         ]; | ||||||
|                         echo json_encode($response); |                         echo json_encode($response); | ||||||
|                     } |                     } | ||||||
| @ -101,7 +105,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "message" => "Some fields are missing." |                 "message" => translate('fields_missing', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -111,7 +115,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
| } else { | } else { | ||||||
|     $response = [ |     $response = [ | ||||||
|         "success" => false, |         "success" => false, | ||||||
|         "message" => "Your session expired. Please login again" |         "message" => translate('session_expired', $i18n) | ||||||
|     ]; |     ]; | ||||||
|     echo json_encode($response); |     echo json_encode($response); | ||||||
| } | } | ||||||
|  | |||||||
| @ -17,21 +17,21 @@ | |||||||
|                     $stmt->bindParam(":api_key", $newApiKey, SQLITE3_TEXT); |                     $stmt->bindParam(":api_key", $newApiKey, SQLITE3_TEXT); | ||||||
|                     $result = $stmt->execute(); |                     $result = $stmt->execute(); | ||||||
|                     if ($result) { |                     if ($result) { | ||||||
|                         echo json_encode(["success" => true]); |                         echo json_encode(["success" => true, "message" => translate('api_key_saved', $i18n)]); | ||||||
|                     } else { |                     } else { | ||||||
|                         $response = [ |                         $response = [ | ||||||
|                             "success" => false, |                             "success" => false, | ||||||
|                             "message" => "Failed to store API Key on the Database" |                             "message" => translate('failed_to_store_api_key', $i18n) | ||||||
|                         ]; |                         ]; | ||||||
|                         echo json_encode($response); |                         echo json_encode($response); | ||||||
|                     } |                     } | ||||||
|                 } else { |                 } else { | ||||||
|                     echo json_encode(["success" => true]); |                     echo json_encode(["success" => true, "message" => translate('apy_key_saved', $i18n)]); | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "message" => "Invalid API Key" |                     "message" => translate('invalid_api_key', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } |             } | ||||||
|  | |||||||
| @ -13,13 +13,13 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             $householdId = $db->lastInsertRowID(); |             $householdId = $db->lastInsertRowID(); | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => true, |                 "success" => true, | ||||||
|                 "householdId" => $householdId |                 "householdId" => $householdId, | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Failed to add household member" |                 "errorMessage" => translate('failed_add_household', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -35,20 +35,21 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
| 
 | 
 | ||||||
|             if ($result) { |             if ($result) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => true |                     "success" => true, | ||||||
|  |                     "message" => translate('member_saved', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Failed to edit household member" |                     "errorMessage" => translate('failed_edit_household', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Please fill all the fields" |                 "errorMessage" => translate('fill_all_fields', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -65,7 +66,7 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|             if ($count > 0) { |             if ($count > 0) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Household member is in use in subscriptions and can't be removed" |                     "errorMessage" => translate('household_in_use', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
| @ -75,13 +76,14 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|                 $result = $stmt->execute(); |                 $result = $stmt->execute(); | ||||||
|                 if ($result) { |                 if ($result) { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => true |                         "success" => true, | ||||||
|  |                         "message" => translate('member_removed', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } else { |                 } else { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => false, |                         "success" => false, | ||||||
|                         "errorMessage" => "Failed to remove household member" |                         "errorMessage" => translate('failed_remove_household', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } |                 } | ||||||
| @ -89,15 +91,15 @@ if (isset($_SESSION['loggedin']) && $_SESSION['loggedin'] === true) { | |||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Failed to remove household member" |                 "errorMessage" => translate('failed_remove_household', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         echo "Error"; |         echo translate('error', $i18n); | ||||||
|     } |     } | ||||||
| } else { | } else { | ||||||
|     echo "Error"; |     echo translate('error', $i18n); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ?>
 | ?>
 | ||||||
| @ -15,7 +15,7 @@ | |||||||
|         ) { |         ) { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Please fill all mandatory fields" |                 "errorMessage" => translate('fill_mandatory_fields', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } else { |         } else { | ||||||
| @ -33,7 +33,7 @@ | |||||||
|             if ($result === false) { |             if ($result === false) { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Error saving notifications data" |                     "errorMessage" => translate('error_saving_notifications', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|             } else { |             } else { | ||||||
| @ -57,13 +57,14 @@ | |||||||
|      |      | ||||||
|                 if ($stmt->execute()) { |                 if ($stmt->execute()) { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => true |                         "success" => true, | ||||||
|  |                         "message" => translate('notifications_settings_saved', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } else { |                 } else { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => false, |                         "success" => false, | ||||||
|                         "errorMessage" => "Error saving notification data" |                         "errorMessage" => translate('error_saving_notifications', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                 } |                 } | ||||||
|  | |||||||
| @ -19,7 +19,7 @@ if ($_SERVER["REQUEST_METHOD"] === "POST") { | |||||||
|     ) { |     ) { | ||||||
|         $response = [ |         $response = [ | ||||||
|             "success" => false, |             "success" => false, | ||||||
|             "errorMessage" => "Please fill all fields" |             "errorMessage" => translate('fill_all_fields', $i18n) | ||||||
|         ]; |         ]; | ||||||
|         echo json_encode($response); |         echo json_encode($response); | ||||||
|     } else { |     } else { | ||||||
| @ -34,6 +34,7 @@ if ($_SERVER["REQUEST_METHOD"] === "POST") { | |||||||
|         $fromEmail = $data["fromemail"] ?? "wallos@wallosapp.com"; |         $fromEmail = $data["fromemail"] ?? "wallos@wallosapp.com"; | ||||||
| 
 | 
 | ||||||
|         $mail = new PHPMailer(true); |         $mail = new PHPMailer(true); | ||||||
|  |         $mail->CharSet="UTF-8"; | ||||||
|         $mail->isSMTP(); |         $mail->isSMTP(); | ||||||
| 
 | 
 | ||||||
|         $mail->Host = $smtpAddress; |         $mail->Host = $smtpAddress; | ||||||
| @ -51,18 +52,19 @@ if ($_SERVER["REQUEST_METHOD"] === "POST") { | |||||||
|         $mail->setFrom($fromEmail, 'Wallos App'); |         $mail->setFrom($fromEmail, 'Wallos App'); | ||||||
|         $mail->addAddress($email, $name); |         $mail->addAddress($email, $name); | ||||||
| 
 | 
 | ||||||
|         $mail->Subject = 'Wallos Notification'; |         $mail->Subject = translate('wallos_notification', $i18n); | ||||||
|         $mail->Body = 'This is a test notification. If you\'re seeing this, the configuration is correct.'; |         $mail->Body = translate('test_notification', $i18n); | ||||||
| 
 | 
 | ||||||
|         if ($mail->send()) { |         if ($mail->send()) { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => true, |                 "success" => true, | ||||||
|  |                 "message" => translate('notification_sent_successfuly', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Error sending email." . $mail->ErrorInfo |                 "errorMessage" => translate('email_error', $i18n) . $mail->ErrorInfo | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -4,14 +4,14 @@ session_start(); | |||||||
| if (!isset($_SESSION['loggedin']) || $_SESSION['loggedin'] !== true) { | if (!isset($_SESSION['loggedin']) || $_SESSION['loggedin'] !== true) { | ||||||
|     die(json_encode([ |     die(json_encode([ | ||||||
|         "success" => false, |         "success" => false, | ||||||
|         "message" => "Your session expired. Please login again" |         "message" => translate('session_expired', $i18n) | ||||||
|     ])); |     ])); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| if (!isset($_GET['paymentId']) || !isset($_GET['enabled'])) { | if (!isset($_GET['paymentId']) || !isset($_GET['enabled'])) { | ||||||
|     die(json_encode([ |     die(json_encode([ | ||||||
|         "success" => false, |         "success" => false, | ||||||
|         "message" => "Some fields are missing." |         "message" => translate('fields_missing', $i18n) | ||||||
|     ])); |     ])); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -21,7 +21,7 @@ $inUse = $db->querySingle('SELECT COUNT(*) as count FROM subscriptions WHERE pay | |||||||
| if ($inUse) { | if ($inUse) { | ||||||
|     die(json_encode([ |     die(json_encode([ | ||||||
|         "success" => false, |         "success" => false, | ||||||
|         "message" => "Can't delete used payment method" |         "message" => translate('payment_in_use', $i18n) | ||||||
|     ])); |     ])); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -33,13 +33,16 @@ $stmtUpdate->bindParam(':enabled', $enabled); | |||||||
| $stmtUpdate->bindParam(':id', $paymentId); | $stmtUpdate->bindParam(':id', $paymentId); | ||||||
| $resultUpdate = $stmtUpdate->execute(); | $resultUpdate = $stmtUpdate->execute(); | ||||||
| 
 | 
 | ||||||
|  | $text = $enabled ? "enabled" : "disabled"; | ||||||
|  | 
 | ||||||
| if ($resultUpdate) { | if ($resultUpdate) { | ||||||
|     die(json_encode([ |     die(json_encode([ | ||||||
|         "success" => true |         "success" => true, | ||||||
|  |         "message" => translate($text, $i18n) | ||||||
|     ])); |     ])); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| die(json_encode([ | die(json_encode([ | ||||||
|     "success" => false, |     "success" => false, | ||||||
|     "message" => "Failed to update payment method in the database" |     "message" => tranlate('failed_update_payment', $i18n) | ||||||
| ])); | ])); | ||||||
|  | |||||||
| @ -25,13 +25,13 @@ | |||||||
|             if (saveLogo($imageData, $uploadFile, $name)) { |             if (saveLogo($imageData, $uploadFile, $name)) { | ||||||
|                 return $fileName; |                 return $fileName; | ||||||
|             } else { |             } else { | ||||||
|                 echo "Error fetching image: " . curl_error($ch); |                 echo translate('error_fetching_image', $i18n) . ": " . curl_error($ch); | ||||||
|                 return ""; |                 return ""; | ||||||
|             } |             } | ||||||
|              |              | ||||||
|             curl_close($ch); |             curl_close($ch); | ||||||
|         } else { |         } else { | ||||||
|             echo "Error fetching image: " . curl_error($ch); |             echo translate('error_fetching_image', $i18n) . ": " . curl_error($ch); | ||||||
|             return ""; |             return ""; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -194,13 +194,13 @@ | |||||||
|             if ($stmt->execute()) { |             if ($stmt->execute()) { | ||||||
|                 $success['status'] = "Success"; |                 $success['status'] = "Success"; | ||||||
|                 $text = $isEdit ? "updated" : "added"; |                 $text = $isEdit ? "updated" : "added"; | ||||||
|                 $success['message'] = "Subscription " . $text . " successfuly"; |                 $success['message'] = translate('subscription_' . $text . '_successfuly', $i18n); | ||||||
|                 $json = json_encode($success); |                 $json = json_encode($success); | ||||||
|                 header('Content-Type: application/json'); |                 header('Content-Type: application/json'); | ||||||
|                 echo $json; |                 echo $json; | ||||||
|                 exit(); |                 exit(); | ||||||
|             } else { |             } else { | ||||||
|                 echo "Error: " . $db->lastErrorMsg(); |                 echo translate('error', $i18n) . ": " . $db->lastErrorMsg(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -12,11 +12,11 @@ | |||||||
|                 http_response_code(204); |                 http_response_code(204); | ||||||
|             } else { |             } else { | ||||||
|                 http_response_code(500); |                 http_response_code(500); | ||||||
|                 echo json_encode(array("message" => "Error deleting the subscription.")); |                 echo json_encode(array("message" => translate('error_deleting_subscription', $i18n))); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             http_response_code(405); |             http_response_code(405); | ||||||
|             echo json_encode(array("message" => "Invalid request method.")); |             echo json_encode(array("message" => translate('invalid_request_method', $i18n))); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     $db->close(); |     $db->close(); | ||||||
|  | |||||||
| @ -31,10 +31,10 @@ | |||||||
|                 header('Content-Type: application/json'); |                 header('Content-Type: application/json'); | ||||||
|                 echo $subscriptionJson; |                 echo $subscriptionJson; | ||||||
|             } else { |             } else { | ||||||
|                 echo "Error"; |                 echo translate('error', $i18n); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             echo "Error"; |             echo translate('error', $i18n); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     $db->close(); |     $db->close(); | ||||||
|  | |||||||
| @ -43,7 +43,7 @@ | |||||||
|           $print[$id]['name']= $subscription['name']; |           $print[$id]['name']= $subscription['name']; | ||||||
|           $cycle = $subscription['cycle']; |           $cycle = $subscription['cycle']; | ||||||
|           $frequency = $subscription['frequency']; |           $frequency = $subscription['frequency']; | ||||||
|           $print[$id]['billing_cycle'] = getBillingCycle($cycle, $frequency); |           $print[$id]['billing_cycle'] = getBillingCycle($cycle, $frequency, $i18n); | ||||||
|           $paymentMethodId = $subscription['payment_method_id']; |           $paymentMethodId = $subscription['payment_method_id']; | ||||||
|           $print[$id]['currency_code'] = $currencies[$subscription['currency_id']]['code']; |           $print[$id]['currency_code'] = $currencies[$subscription['currency_id']]['code']; | ||||||
|           $currencyId = $subscription['currency_id']; |           $currencyId = $subscription['currency_id']; | ||||||
| @ -66,19 +66,19 @@ | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (isset($print)) { |         if (isset($print)) { | ||||||
|           printSubscriptions($print, $sort, $categories, $members); |           printSubscriptions($print, $sort, $categories, $members, $i18n); | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         if (count($subscriptions) == 0) { |         if (count($subscriptions) == 0) { | ||||||
|             ?>
 |             ?>
 | ||||||
|             <div class="empty-page"> |             <div class="empty-page"> | ||||||
|                 <img src="images/siteimages/empty.png" alt="Empty page" /> |                 <img src="images/siteimages/empty.png" alt="<?= translate('empty_page', $i18n) ?>" /> | ||||||
|                 <p> |                 <p> | ||||||
|                   You don't have any subscriptions yet |                   <?= translate('no_subscriptions_yet', $i18n) ?>
 | ||||||
|                 </p> |                 </p> | ||||||
|                 <button class="button" onClick="addSubscription()"> |                 <button class="button" onClick="addSubscription()"> | ||||||
|                   <img class="button-icon" src="images/siteicons/plusicon.png"> |                   <img class="button-icon" src="images/siteicons/plusicon.png"> | ||||||
|                   Add First Subscription |                   <?= translate('add_first_subscription', $i18n) ?>
 | ||||||
|                 </button> |                 </button> | ||||||
|             </div> |             </div> | ||||||
|             <?php |             <?php | ||||||
|  | |||||||
| @ -76,6 +76,7 @@ | |||||||
|         $email = $_POST['email']; |         $email = $_POST['email']; | ||||||
|         $avatar = $_POST['avatar']; |         $avatar = $_POST['avatar']; | ||||||
|         $main_currency = $_POST['main_currency']; |         $main_currency = $_POST['main_currency']; | ||||||
|  |         $language = $_POST['language']; | ||||||
| 
 | 
 | ||||||
|         if (isset($_POST['password']) && $_POST['password'] != "") { |         if (isset($_POST['password']) && $_POST['password'] != "") { | ||||||
|             $password = $_POST['password']; |             $password = $_POST['password']; | ||||||
| @ -84,7 +85,7 @@ | |||||||
|                 if ($password != $confirm) { |                 if ($password != $confirm) { | ||||||
|                     $response = [ |                     $response = [ | ||||||
|                         "success" => false, |                         "success" => false, | ||||||
|                         "errorMessage" => "Passwords do not match" |                         "errorMessage" => translate('passwords_dont_match', $i18n) | ||||||
|                     ]; |                     ]; | ||||||
|                     echo json_encode($response); |                     echo json_encode($response); | ||||||
|                     exit(); |                     exit(); | ||||||
| @ -92,7 +93,7 @@ | |||||||
|             } else { |             } else { | ||||||
|                 $response = [ |                 $response = [ | ||||||
|                     "success" => false, |                     "success" => false, | ||||||
|                     "errorMessage" => "Passwords do not match" |                     "errorMessage" => translate('passwords_dont_match', $i18n) | ||||||
|                 ]; |                 ]; | ||||||
|                 echo json_encode($response); |                 echo json_encode($response); | ||||||
|                 exit(); |                 exit(); | ||||||
| @ -100,9 +101,9 @@ | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (isset($_POST['password']) && $_POST['password'] != "") { |         if (isset($_POST['password']) && $_POST['password'] != "") { | ||||||
|             $sql = "UPDATE user SET avatar = :avatar, username = :username, email = :email, password = :password, main_currency = :main_currency WHERE id = 1"; |             $sql = "UPDATE user SET avatar = :avatar, username = :username, email = :email, password = :password, main_currency = :main_currency, language = :language WHERE id = 1"; | ||||||
|         } else { |         } else { | ||||||
|             $sql = "UPDATE user SET avatar = :avatar, username = :username, email = :email, main_currency = :main_currency WHERE id = 1"; |             $sql = "UPDATE user SET avatar = :avatar, username = :username, email = :email, main_currency = :main_currency, language = :language WHERE id = 1"; | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         $stmt = $db->prepare($sql); |         $stmt = $db->prepare($sql); | ||||||
| @ -110,6 +111,7 @@ | |||||||
|         $stmt->bindParam(':username', $username, SQLITE3_TEXT); |         $stmt->bindParam(':username', $username, SQLITE3_TEXT); | ||||||
|         $stmt->bindParam(':email', $email, SQLITE3_TEXT); |         $stmt->bindParam(':email', $email, SQLITE3_TEXT); | ||||||
|         $stmt->bindParam(':main_currency', $main_currency, SQLITE3_INTEGER); |         $stmt->bindParam(':main_currency', $main_currency, SQLITE3_INTEGER); | ||||||
|  |         $stmt->bindParam(':language', $language, SQLITE3_TEXT); | ||||||
| 
 | 
 | ||||||
|         if (isset($_POST['password']) && $_POST['password'] != "") { |         if (isset($_POST['password']) && $_POST['password'] != "") { | ||||||
|             $hashedPassword = password_hash($password, PASSWORD_DEFAULT); |             $hashedPassword = password_hash($password, PASSWORD_DEFAULT); | ||||||
| @ -119,12 +121,13 @@ | |||||||
|         $result = $stmt->execute(); |         $result = $stmt->execute(); | ||||||
| 
 | 
 | ||||||
|         if ($result) { |         if ($result) { | ||||||
|  |             $cookieExpire = time() + (30 * 24 * 60 * 60); | ||||||
|  |             setcookie('language', $language, $cookieExpire, '/'); | ||||||
|             if ($username != $oldUsername) { |             if ($username != $oldUsername) { | ||||||
|                 $_SESSION['username'] = $username; |                 $_SESSION['username'] = $username; | ||||||
|                 if (isset($_COOKIE['wallos_login'])) { |                 if (isset($_COOKIE['wallos_login'])) { | ||||||
|                     $cookie = explode('|', $_COOKIE['wallos_login'], 2) ; |                     $cookie = explode('|', $_COOKIE['wallos_login'], 2) ; | ||||||
|                     $token = $cookie[1]; |                     $token = $cookie[1]; | ||||||
|                     $cookieExpire = time() + (30 * 24 * 60 * 60); |  | ||||||
|                     $cookieValue = $username . "|" . $token . "|" . $main_currency; |                     $cookieValue = $username . "|" . $token . "|" . $main_currency; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @ -137,12 +140,13 @@ | |||||||
| 
 | 
 | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => true, |                 "success" => true, | ||||||
|  |                 "message" => translate('user_details_saved', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } else { |         } else { | ||||||
|             $response = [ |             $response = [ | ||||||
|                 "success" => false, |                 "success" => false, | ||||||
|                 "errorMessage" => "Error updating user data" |                 "errorMessage" => translate('error_updating_user_data', $i18n) | ||||||
|             ]; |             ]; | ||||||
|             echo json_encode($response); |             echo json_encode($response); | ||||||
|         } |         } | ||||||
| @ -151,7 +155,7 @@ | |||||||
|     } else { |     } else { | ||||||
|         $response = [ |         $response = [ | ||||||
|             "success" => false, |             "success" => false, | ||||||
|             "errorMessage" => "Please fill all fields" |             "errorMessage" => translate('fill_all_fields', $i18n) | ||||||
|         ]; |         ]; | ||||||
|         echo json_encode($response); |         echo json_encode($response); | ||||||
|         exit(); |         exit(); | ||||||
|  | |||||||
| @ -8,4 +8,8 @@ if (!$db) { | |||||||
|     die('Connection to the database failed.'); |     die('Connection to the database failed.'); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | require_once 'i18n/languages.php'; | ||||||
|  | require_once 'i18n/getlang.php'; | ||||||
|  | require_once 'i18n/' . $lang . '.php'; | ||||||
|  | 
 | ||||||
| ?>
 | ?>
 | ||||||
| @ -4,7 +4,7 @@ | |||||||
|     <div class="toast-content"> |     <div class="toast-content"> | ||||||
|       <i class="fas fa-solid fa-x toast-icon error"></i> |       <i class="fas fa-solid fa-x toast-icon error"></i> | ||||||
|       <div class="message"> |       <div class="message"> | ||||||
|         <span class="text text-1">Error</span> |         <span class="text text-1"><?= translate("error", $i18n) ?></span>
 | ||||||
|         <span class="text text-2 errorMessage"></span> |         <span class="text text-2 errorMessage"></span> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
| @ -16,7 +16,7 @@ | |||||||
|     <div class="toast-content"> |     <div class="toast-content"> | ||||||
|       <i class="fas fa-solid fa-check toast-icon success"></i> |       <i class="fas fa-solid fa-check toast-icon success"></i> | ||||||
|       <div class="message"> |       <div class="message"> | ||||||
|         <span class="text text-1">Success</span> |         <span class="text text-1"><?= translate("success", $i18n) ?></span>
 | ||||||
|         <span class="text text-2 successMessage"></span> |         <span class="text text-2 successMessage"></span> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -4,6 +4,10 @@ | |||||||
|   require_once 'checksession.php'; |   require_once 'checksession.php'; | ||||||
|   require_once 'currency_formatter.php'; |   require_once 'currency_formatter.php'; | ||||||
| 
 | 
 | ||||||
|  |   require_once 'i18n/languages.php'; | ||||||
|  |   require_once 'i18n/getlang.php'; | ||||||
|  |   require_once 'i18n/' . $lang . '.php'; | ||||||
|  | 
 | ||||||
|   if ($userCount == 0) { |   if ($userCount == 0) { | ||||||
|     $db->close(); |     $db->close(); | ||||||
|     header("Location: registration.php"); |     header("Location: registration.php"); | ||||||
| @ -32,7 +36,10 @@ | |||||||
|   <script type="text/javascript" src="scripts/common.js"></script> |   <script type="text/javascript" src="scripts/common.js"></script> | ||||||
|   <script type="text/javascript"> |   <script type="text/javascript"> | ||||||
|     window.theme = "<?= $theme ?>"; |     window.theme = "<?= $theme ?>"; | ||||||
|  |     window.lang = "<?=$lang ?>"; | ||||||
|   </script> |   </script> | ||||||
|  |   <script type="text/javascript" src="scripts/i18n/<?= $lang ?>.js"></script> | ||||||
|  |   <script type="text/javascript" src="scripts/i18n/getlang.js"></script> | ||||||
| </head> | </head> | ||||||
| <body> | <body> | ||||||
|   <header> |   <header> | ||||||
| @ -49,10 +56,11 @@ | |||||||
|             <span id="user"><?= $username ?></span>
 |             <span id="user"><?= $username ?></span>
 | ||||||
|           </button> |           </button> | ||||||
|           <div class="dropdown-content"> |           <div class="dropdown-content"> | ||||||
|             <a href="stats.php"><i class="fa-solid fa-chart-simple"></i>Stats</a> |             <a href="/"><i class="fa-solid fa-list"></i><?= translate('subscriptions', $i18n) ?></a>
 | ||||||
|             <a href="settings.php"><i class="fa-solid fa-gear"></i>Settings</a> |             <a href="stats.php"><i class="fa-solid fa-chart-simple"></i><?= translate('stats', $i18n) ?></a>
 | ||||||
|             <a href="about.php"><i class="fa-solid fa-info-circle"></i>About</a> |             <a href="settings.php"><i class="fa-solid fa-gear"></i><?= translate('settings', $i18n) ?></a>
 | ||||||
|             <a href="logout.php"><i class="fa-solid fa-arrow-right-from-bracket"></i>Logout</a> |             <a href="about.php"><i class="fa-solid fa-info-circle"></i><?= translate('about', $i18n) ?></a>
 | ||||||
|  |             <a href="logout.php"><i class="fa-solid fa-arrow-right-from-bracket"></i><?= translate('logout', $i18n) ?></a>
 | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|       </nav> |       </nav> | ||||||
|  | |||||||
							
								
								
									
										195
									
								
								includes/i18n/en.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										195
									
								
								includes/i18n/en.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,195 @@ | |||||||
|  | <?php | ||||||
|  | 
 | ||||||
|  | $i18n = [ | ||||||
|  |     // Registration page
 | ||||||
|  |     "create_account"  => "You need to create an account before you're able to login", | ||||||
|  |     'username'        => "Username", | ||||||
|  |     'password'        => "Password", | ||||||
|  |     "email"           => "Email", | ||||||
|  |     "confirm_password" => "Confirm Password", | ||||||
|  |     "main_currency"   => "Main Currency", | ||||||
|  |     "language"        => "Language", | ||||||
|  |     "passwords_dont_match" => "Passwords do not match", | ||||||
|  |     "registration_failed" => "Registration failed, please try again.", | ||||||
|  |     "register"        => "Register", | ||||||
|  |     // Login Page
 | ||||||
|  |     'please_login'    => "Please login", | ||||||
|  |     'stay_logged_in'  => "Stay logged in (30 days)", | ||||||
|  |     'login'           => "Login", | ||||||
|  |     'login_failed'    => "Login details are incorrect", | ||||||
|  |     // Header
 | ||||||
|  |     'subscriptions'   => "Subscriptions", | ||||||
|  |     'stats'           => "Statistics", | ||||||
|  |     'settings'        => "Settings", | ||||||
|  |     'about'           => "About", | ||||||
|  |     'logout'          => "Logout", | ||||||
|  |     // Subscriptions page
 | ||||||
|  |     "subscription"    => "Subscription", | ||||||
|  |     "no_subscriptions_yet" => "You don't have any subscriptions yet", | ||||||
|  |     "add_first_subscription" => "Add first subscription", | ||||||
|  |     'new_subscription' => "New Subscription", | ||||||
|  |     'sort'            => "Sort", | ||||||
|  |     'name'            => "Nome", | ||||||
|  |     'last_added'      => "Last Added", | ||||||
|  |     'price'           => "Price", | ||||||
|  |     'next_payment'    => "Next Payment", | ||||||
|  |     'member'          => "Member", | ||||||
|  |     'category'        => "Category", | ||||||
|  |     'payment_method'  => "Payment Method", | ||||||
|  |     "Daily"           => "Daily", | ||||||
|  |     "Weekly"          => "Weekly", | ||||||
|  |     "Monthly"         => "Monthly", | ||||||
|  |     "Yearly"          => "Yearly", | ||||||
|  |     "days"            => "days", | ||||||
|  |     "weeks"           => "weeks", | ||||||
|  |     "months"          => "months", | ||||||
|  |     "years"           => "years", | ||||||
|  |     "external_url"    => "Visit Externarl URL", | ||||||
|  |     "empty_page"      => "Empty Page", | ||||||
|  |     // Subscription form
 | ||||||
|  |     "add_subscription" => "Add subscription", | ||||||
|  |     "edit_subscription" => "Edit subscription", | ||||||
|  |     "subscription_name" => "Subscription name", | ||||||
|  |     "logo_preview"    => "Logo Preview", | ||||||
|  |     "search_logo"     => "Search logo on the web", | ||||||
|  |     "web_search"      => "Web search", | ||||||
|  |     "currency"        => "Currency", | ||||||
|  |     "billing_cycle"   => "Billing Cycle", | ||||||
|  |     "frequency"       => "Frequency", | ||||||
|  |     "cycle"           => "Cycle", | ||||||
|  |     "next_payment"    => "Next Payment", | ||||||
|  |     "payment_method"  => "Payment Method", | ||||||
|  |     "no_category"     => "No category", | ||||||
|  |     "paid_by"         => "Paid by", | ||||||
|  |     "url"             => "URL", | ||||||
|  |     "notes"           => "Notes", | ||||||
|  |     "enable_notifications" => "Enable Notifications for this subscription", | ||||||
|  |     "delete"          => "Delete", | ||||||
|  |     "cancel"          => "Cancel", | ||||||
|  |     "upload_logo"     => "Upload Logo", | ||||||
|  |     // Statistics page
 | ||||||
|  |     'general_statistics' => "General Statistics", | ||||||
|  |     'active_subscriptions' => "Active Subscriptions", | ||||||
|  |     'monthly_cost'    => "Monthly Cost", | ||||||
|  |     'yearly_cost'     => "Yearly Cost", | ||||||
|  |     'average_monthly' => "Average Monthly Subscription Cost", | ||||||
|  |     'most_expensive'  => "Most Expensive Subscription Cost", | ||||||
|  |     'amount_due'      => "Amount due this month", | ||||||
|  |     'split_views'     => "Split Views", | ||||||
|  |     'category_split'  => "Category Split", | ||||||
|  |     'household_split' => "Household Split", | ||||||
|  |     // About page
 | ||||||
|  |     'about_and_credits' => "About and Credits", | ||||||
|  |     'license'         => "License", | ||||||
|  |     'issues_and_requests' => "Issues and Requests", | ||||||
|  |     'the_author'      => "The author", | ||||||
|  |     'icons'           => "Icons", | ||||||
|  |     'payment_icons'   => "Payment Icons", | ||||||
|  |     // Settings page
 | ||||||
|  |     'user_details'    => "User Details", | ||||||
|  |     "household"        => "Household", | ||||||
|  |     "save_member"     => "Save Member", | ||||||
|  |     "delete_member"   => "Delete Member", | ||||||
|  |     "cant_delete_member" => "Can't delete main member", | ||||||
|  |     "cant_delete_member_in_use" => "Can't delete member in use in subscription", | ||||||
|  |     "notifications"   => "Notifications", | ||||||
|  |     "enable_email_notifications" => "Enable email notifications", | ||||||
|  |     "notify_me"       => "Notify me", | ||||||
|  |     "day_before"      => "day before", | ||||||
|  |     "days_before"     => "days before", | ||||||
|  |     "smtp_address"    => "SMTP Address", | ||||||
|  |     "port"            => "Port", | ||||||
|  |     "smtp_username"   => "SMTP Username", | ||||||
|  |     "smtp_password"   => "SMTP Password", | ||||||
|  |     "from_email"      => "From email (Optional)", | ||||||
|  |     "smtp_info"       => "SMTP Password is transmitted and stored in plaintext. For security, please create an account just for this.", | ||||||
|  |     "categories"      => "Categories", | ||||||
|  |     "save_category"   => "Save Category", | ||||||
|  |     "delete_category" => "Delete Category", | ||||||
|  |     "cant_delete_category_in_use" => "Can't delete category in use in subscription", | ||||||
|  |     "currencies"      => "Currencies", | ||||||
|  |     "save_currency"   => "Save currency", | ||||||
|  |     "delete_currency" => "Delete currency", | ||||||
|  |     "cant_delete_main_currency" => "Can't delete main currency", | ||||||
|  |     "cant_delete_currency_in_use" => "Can't delete currency in use in subscription", | ||||||
|  |     "exchange_update" => "Exchange rates last updated on", | ||||||
|  |     "currency_info"   => "Find the supported currencies and correct currency codes on", | ||||||
|  |     "currency_performance" => "For improved performance keep only the currencies you use.", | ||||||
|  |     "fixer_api_key"   => "Fixer API Key", | ||||||
|  |     "api_key"         => "API Key", | ||||||
|  |     "fixer_info"      => "If you use multiple currencies, and want accurate statistics and sorting on the subscriptions, a FREE API Key from Fixer is necessary.", | ||||||
|  |     "get_key"         => "Get your key at", | ||||||
|  |     "display_settings" => "Display Settings", | ||||||
|  |     "switch_theme"    => "Switch Light / Dark Theme", | ||||||
|  |     "calculate_monthly_price" => "Calculate and show monthly price for all subscriptions", | ||||||
|  |     "convert_prices"  => "Always convert and show prices on my main currency (slower)", | ||||||
|  |     "experimental_settings" => "Experimental Settings", | ||||||
|  |     "remove_background" => "Attempt to remove background of logos from image search (experimental)", | ||||||
|  |     "experimental_info" => "Experimental settings will probably not work perfectly.", | ||||||
|  |     "payment_methods" => "Payment Methods", | ||||||
|  |     "payment_methods_info" => "Click a payment method to disable / enable it.", | ||||||
|  |     "cant_delete_payment_method_in_use" => "Can't disable used payment method", | ||||||
|  |     "disable"         => "Disable", | ||||||
|  |     "enable"          => "Enable", | ||||||
|  |     "test"            => "Test", | ||||||
|  |     "add"             => "Add", | ||||||
|  |     "save"            => "Save", | ||||||
|  |     // Toast
 | ||||||
|  |     "success"         => "Success", | ||||||
|  |     // Endpoint responses
 | ||||||
|  |     "session_expired" => "Your session expired. Please login again", | ||||||
|  |     "fields_missing"  => "Some fields are missing", | ||||||
|  |     "fill_all_fields" => "Please fill all fields", | ||||||
|  |     "fill_mandatory_fields" => "Please fill all mandatory fields", | ||||||
|  |     "error"           => "Error", | ||||||
|  |     // Category
 | ||||||
|  |     "failed_add_category" => "Failed to add category", | ||||||
|  |     "failed_edit_category" => "Failed to edit category", | ||||||
|  |     "category_in_use" => "Category is in use in subscriptions and can't be removed", | ||||||
|  |     "failed_remove_category" => "Failed to remove category", | ||||||
|  |     "category_saved"  => "Category saved", | ||||||
|  |     "category_removed" => "Category removed", | ||||||
|  |     // Currency
 | ||||||
|  |     "currency_saved"  => "was saved.", | ||||||
|  |     "error_adding_currency" => "Error adding currency entry.", | ||||||
|  |     "failed_to_store_currency" => "Failed to store Currency on the Database.", | ||||||
|  |     "currency_in_use" => "Currency is in use in subscriptions and can't be deleted.", | ||||||
|  |     "currency_is_main" => "Currency is set as main currency and can't be deleted.", | ||||||
|  |     "failed_to_remove_currency" => "Failed to remove currency from the Database.", | ||||||
|  |     "failed_to_store_api_key" => "Failed to store API Key on the Database.", | ||||||
|  |     "invalid_api_key"  => "Invalid API Key.", | ||||||
|  |     "api_key_saved"   => "API key saved successfully", | ||||||
|  |     "currency_removed" => "Currency removed", | ||||||
|  |     // Household
 | ||||||
|  |     "failed_add_household" => "Failed to add household member", | ||||||
|  |     "failed_edit_household" => "Failed to edit household member", | ||||||
|  |     "failed_remove_household" => "Failed to remove household member", | ||||||
|  |     "household_in_use" => "Household member is in use in subscriptions and can't be removed", | ||||||
|  |     "member_saved"     => "Member saved", | ||||||
|  |     "member_removed"   => "Member removed", | ||||||
|  |     // Notifications
 | ||||||
|  |     "error_saving_notifications" => "Error saving notifications data.", | ||||||
|  |     "wallos_notification" => "Wallos Notification", | ||||||
|  |     "test_notification" => "This is a test notification. If you\'re seeing this, the configuration is correct.", | ||||||
|  |     "email_error"      => "Error sending email", | ||||||
|  |     "notification_sent_successfuly" => "Notification sent successfuly", | ||||||
|  |     "notifications_settings_saved" => "Notification settings saved successfully.", | ||||||
|  |     // Payments
 | ||||||
|  |     "payment_in_use"   => "Can't disable used payment method", | ||||||
|  |     "failed_update_payment" => "Failed to update payment method in the database", | ||||||
|  |     "enabled"          => "enabled", | ||||||
|  |     "disabled"         => "disabled", | ||||||
|  |     // Subscription
 | ||||||
|  |     "error_fetching_image" => "Error fetching image", | ||||||
|  |     "subscription_updated_successfuly" => "Subscription updated successfuly", | ||||||
|  |     "subscription_added_successfuly" => "Subscription added successfuly", | ||||||
|  |     "error_deleting_subscription" => "Error deleting subscription.", | ||||||
|  |     "invalid_request_method" => "Invalid request method.", | ||||||
|  |     // User
 | ||||||
|  |     "error_updating_user_data" => "Error updating user data.", | ||||||
|  |     "user_details_saved" => "User details saved", | ||||||
|  | 
 | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ?>
 | ||||||
							
								
								
									
										25
									
								
								includes/i18n/getlang.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								includes/i18n/getlang.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | <?php | ||||||
|  | 
 | ||||||
|  | $lang = "en"; | ||||||
|  | if (isset($_COOKIE['language'])) { | ||||||
|  |     $selectedLanguage = $_COOKIE['language']; | ||||||
|  | 
 | ||||||
|  |     if (array_key_exists($selectedLanguage, $languages)) { | ||||||
|  |         $lang = $selectedLanguage; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function translate($text, $translations) { | ||||||
|  |     if (array_key_exists($text, $translations)) { | ||||||
|  |         return $translations[$text]; | ||||||
|  |     } else { | ||||||
|  |         require_once 'en.php'; | ||||||
|  |         if (array_key_exists($text, $i18n)) { | ||||||
|  |             return $i18n[$text]; | ||||||
|  |         } else { | ||||||
|  |             return "[i18n String Missing]"; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ?>
 | ||||||
							
								
								
									
										9
									
								
								includes/i18n/languages.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								includes/i18n/languages.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | <?php | ||||||
|  |     // File Name => Language Name
 | ||||||
|  | 
 | ||||||
|  |     $languages = [ | ||||||
|  |         "en" => "English", | ||||||
|  |         "pt" => "Português", | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  | ?>
 | ||||||
							
								
								
									
										194
									
								
								includes/i18n/pt.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										194
									
								
								includes/i18n/pt.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,194 @@ | |||||||
|  | <?php | ||||||
|  | 
 | ||||||
|  | $i18n = [ | ||||||
|  |     // Registration page
 | ||||||
|  |     "create_account"  => "Tem que criar uma conta antes de poder iniciar sessão", | ||||||
|  |     'username'        => "Nome de utilizador", | ||||||
|  |     'password'        => "Password", | ||||||
|  |     "email"           => "Email", | ||||||
|  |     "confirm_password" => "Confirmar Password", | ||||||
|  |     "main_currency"   => "Moeda Principal", | ||||||
|  |     "language"        => "Linguagem", | ||||||
|  |     "passwords_dont_match" => "As passwords não coincidem", | ||||||
|  |     "registration_failed" => "O registo falhou. Tente novamente", | ||||||
|  |     "register"        => "Registar", | ||||||
|  |     // Login Page
 | ||||||
|  |     'please_login'    => "Por favor inicie sessão", | ||||||
|  |     'stay_logged_in'  => "Manter sessão (30 dias)", | ||||||
|  |     'login'           => "Iniciar Sessão", | ||||||
|  |     'login_failed'    => "Dados de autenticação incorrectos", | ||||||
|  |     // Header
 | ||||||
|  |     'subscriptions'   => "Subscrições", | ||||||
|  |     'stats'           => "Estatísticas", | ||||||
|  |     'settings'        => "Definições", | ||||||
|  |     'about'           => "Sobre", | ||||||
|  |     'logout'          => "Terminar Sessão", | ||||||
|  |     // Subscriptions page
 | ||||||
|  |     "subscription"    => "Subscrição", | ||||||
|  |     "no_subscriptions_yet" => "Ainda não tem subscrições", | ||||||
|  |     "add_first_subscription" => "Adicionar primeira subscrição", | ||||||
|  |     'new_subscription' => "Nova Subscrição", | ||||||
|  |     'sort'            => "Ordenar", | ||||||
|  |     'name'            => "Nome", | ||||||
|  |     'last_added'      => "Última Adicionada", | ||||||
|  |     'price'           => "Preço", | ||||||
|  |     'next_payment'    => "Próximo Pagamento", | ||||||
|  |     'member'          => "Membro", | ||||||
|  |     'category'        => "Categoria", | ||||||
|  |     'payment_method'  => "Metodo de Pagamento", | ||||||
|  |     "Daily"           => "Diario", | ||||||
|  |     "Weekly"          => "Semanal", | ||||||
|  |     "Monthly"         => "Mensal", | ||||||
|  |     "Yearly"          => "Anual", | ||||||
|  |     "days"            => "dias", | ||||||
|  |     "weeks"           => "semanas", | ||||||
|  |     "months"          => "meses", | ||||||
|  |     "years"           => "anos", | ||||||
|  |     "external_url"    => "Visitar URL Externo", | ||||||
|  |     "empty_page"      => "Página Vazia", | ||||||
|  |     // Subscription form
 | ||||||
|  |     "add_subscription" => "Adicionar subscrição", | ||||||
|  |     "edit_subscription" => "Modificar subscrição", | ||||||
|  |     "subscription_name" => "Nome da subscrição", | ||||||
|  |     "logo_preview"    => "Pre-visualisação do logo", | ||||||
|  |     "search_logo"     => "Pesquisar logo na internet", | ||||||
|  |     "web_search"      => "Pesquisa online", | ||||||
|  |     "currency"        => "Moeda", | ||||||
|  |     "billing_cycle"   => "Ciclo de faturação", | ||||||
|  |     "frequency"       => "Frequencia", | ||||||
|  |     "Cycle"           => "Ciclo", | ||||||
|  |     "next_payment"    => "Próximo Pagamento", | ||||||
|  |     "payment_method"  => "Método de Pagamento", | ||||||
|  |     "no_category"     => "Sem categoria", | ||||||
|  |     "paid_by"         => "Pago por", | ||||||
|  |     "url"             => "URL", | ||||||
|  |     "notes"           => "Notas", | ||||||
|  |     "enable_notifications" => "Activar notificações para esta subscrição", | ||||||
|  |     "delete"          => "Remover", | ||||||
|  |     "cancel"          => "Cancelar", | ||||||
|  |     "upload_logo"     => "Enviar Logo", | ||||||
|  |     // Statistics page
 | ||||||
|  |     'general_statistics' => "Estatísticas Gerais", | ||||||
|  |     'active_subscriptions' => "Subscrições Activas", | ||||||
|  |     'monthly_cost'    => "Custo Mensal", | ||||||
|  |     'yearly_cost'     => "Custo Anual", | ||||||
|  |     'average_monthly' => "Custo Mensal Médio das Subscrições", | ||||||
|  |     'most_expensive'  => "Custo da Subscrição Mais Cara", | ||||||
|  |     'amount_due'      => "Quantia em dívida este mês", | ||||||
|  |     'split_views'     => "Vistas Divididas", | ||||||
|  |     'category_split'  => "Por Categoria", | ||||||
|  |     'household_split' => "Por Membro", | ||||||
|  |     // About page
 | ||||||
|  |     'about_and_credits' => "Sobre e Créditos", | ||||||
|  |     'license'         => "Licença", | ||||||
|  |     'issues_and_requests' => "Problemas e Pedidos", | ||||||
|  |     'the_author'      => "O Autor", | ||||||
|  |     'icons'           => "Ícones", | ||||||
|  |     'payment_icons'   => "Ícones de Pagamentos", | ||||||
|  |     // Settings page
 | ||||||
|  |     'user_details'    => "Detalhes do utilizador", | ||||||
|  |     "household"        => "Agregado", | ||||||
|  |     "save_member"     => "Guardar Membro", | ||||||
|  |     "delete_member"   => "Apagar Membro", | ||||||
|  |     "cant_delete_member" => "Não pode apagar o membro principal", | ||||||
|  |     "cant_delete_member_in_use" => "Não pode apagar membro em uso em subscrição", | ||||||
|  |     "notifications"   => "Notificações", | ||||||
|  |     "enable_email_notifications" => "Activar notificações por email", | ||||||
|  |     "notify_me"       => "Notificar-me", | ||||||
|  |     "day_before"      => "dia antes", | ||||||
|  |     "days_before"     => "dias antes", | ||||||
|  |     "smtp_address"    => "Endereço SMTP", | ||||||
|  |     "port"            => "Porto", | ||||||
|  |     "smtp_username"   => "Utilizador SMTP", | ||||||
|  |     "smtp_password"   => "Password SMTP", | ||||||
|  |     "from_email"      => "Email de envio (Opcional)", | ||||||
|  |     "smtp_info"       => "A Password é armazenada e transmitida em texto. Por segurança, crie uma conta só para esta finalidade.", | ||||||
|  |     "categories"      => "Categorias", | ||||||
|  |     "save_category"   => "Guardar Categoria", | ||||||
|  |     "delete_category" => "Apagar Categoria", | ||||||
|  |     "cant_delete_category_in_use" => "Não pode apagar categoria em uso em subscrição", | ||||||
|  |     "currencies"      => "Moedas", | ||||||
|  |     "save_currency"   => "Guardar moeda", | ||||||
|  |     "delete_currency" => "Apagar moeda", | ||||||
|  |     "cant_delete_main_currency" => "Não pode apagar a moeda principal", | ||||||
|  |     "cant_delete_currency_in_use" => "Não pode apagar moeda em uso em subscrição", | ||||||
|  |     "exchange_update" => "Taxas de conversão actualizadas em", | ||||||
|  |     "currency_info"   => "Encontre a lista de moedas e os respectivos códigos em", | ||||||
|  |     "currency_performance" => "Por motivos de desempenho mantenha apenas as moedas que usa.", | ||||||
|  |     "fixer_api_key"   => "Fixer API Key", | ||||||
|  |     "api_key"         => "API Key", | ||||||
|  |     "fixer_info"      => "Se usa multiplas moedas e deseja estatísticas correctas é necessário uma API Key grátis do Fixer.", | ||||||
|  |     "get_key"         => "Obtenha a sua API Key em", | ||||||
|  |     "display_settings" => "Definições de visualização", | ||||||
|  |     "switch_theme"    => "Trocar Tema Claro / Escuro", | ||||||
|  |     "calculate_monthly_price" => "Calcular e mostrar preço mensal para todas as subscrições", | ||||||
|  |     "convert_prices"  => "Converter e mostrar todas as subscrições na moeda principal (mais lento)", | ||||||
|  |     "experimental_settings" => "Definições Experimentais", | ||||||
|  |     "remove_background" => "Tentar remover o fundo dos logos na pesquisa de imagem (experimental)", | ||||||
|  |     "experimental_info" => "Definições experimentais provavelmente não funcionarão correctamente.", | ||||||
|  |     "payment_methods" => "Métodos de Pagamento", | ||||||
|  |     "payment_methods_info" => "Clique num método de pagamento para o activar / desactivar.", | ||||||
|  |     "cant_delete_payment_method_in_use" => "Não pode desactivar metodo de pagamento em uso", | ||||||
|  |     "disable"         => "Desactivar", | ||||||
|  |     "enable"          => "Activar", | ||||||
|  |     "test"            => "Testar", | ||||||
|  |     "add"             => "Adicionar", | ||||||
|  |     "save"            => "Guardar", | ||||||
|  |     // Toast
 | ||||||
|  |     "success"         => "Sucesso", | ||||||
|  |     // Endpoint responses
 | ||||||
|  |     "session_expired" => "A sessão expirou. Por favor autentique-se.", | ||||||
|  |     "fields_missing"  => "Alguns campos em falta", | ||||||
|  |     "fill_all_fields" => "Por favor preencha todos os campos", | ||||||
|  |     "fill_mandatory_fields" => "Por favor preencha todos os campos obrigatórios", | ||||||
|  |     "error"           => "Erro", | ||||||
|  |     // Category
 | ||||||
|  |     "failed_add_category" => "Erro ao adicionar categoria", | ||||||
|  |     "failed_edit_category" => "Erro ao modificar categoria", | ||||||
|  |     "category_in_use" => "Categoria em uso em subscrição e não pode ser removida", | ||||||
|  |     "failed_remove_category" => "Erro ao remover categoria", | ||||||
|  |     "category_saved"  => "Categoria guardada", | ||||||
|  |     "category_removed" => "Categoria removida", | ||||||
|  |     // Currency
 | ||||||
|  |     "currency_saved"  => "guardada.", | ||||||
|  |     "error_adding_currency" => "Erro ao adicionar moeda.", | ||||||
|  |     "failed_to_store_currency" => "Erro ao guardar a moeda na base de dados.", | ||||||
|  |     "currency_in_use" => "Moeda em uso em subscrição e não pode ser removida.", | ||||||
|  |     "currency_is_main" => "A Moeda principal não pode ser removida.", | ||||||
|  |     "failed_to_remove_currency" => "Erro ao remover a moeda da base de dados.", | ||||||
|  |     "failed_to_store_api_key" => "Erro ao guardar API Key na base de dados.", | ||||||
|  |     "invalid_api_key"  => "API Key inválida.", | ||||||
|  |     "api_key_saved"   => "API key guardada", | ||||||
|  |     "currency_removed" => "Moeda removida", | ||||||
|  |     // Household
 | ||||||
|  |     "failed_add_household" => "Erro ao adicionar membro", | ||||||
|  |     "failed_edit_household" => "Erro ao modificar membro", | ||||||
|  |     "failed_remove_household" => "Erro ao remover membro", | ||||||
|  |     "household_in_use" => "Membro está em uso em subscrição e não pode er removido", | ||||||
|  |     "member_saved" => "Membro guardado", | ||||||
|  |     "member_removed"   => "Membro removido", | ||||||
|  |     // Notifications
 | ||||||
|  |     "error_saving_notifications" => "Erro ao guardar os dados das notificaçoes.", | ||||||
|  |     "wallos_notification" => "Notificação Wallos", | ||||||
|  |     "test_notification" => "Isto é uma notificação de teste. Se está a ver isto a configuração está correcta.", | ||||||
|  |     "email_error"      => "Erro ao enviar email", | ||||||
|  |     "notification_sent_successfuly" => "Notificação enviada com sucesso", | ||||||
|  |     "notifications_settings_saved" => "Configuração de notificações guardada.", | ||||||
|  |     // Payments
 | ||||||
|  |     "payment_in_use"   => "Não pode desactivar método de pagamento em uso", | ||||||
|  |     "failed_update_payment" => "Erro ao actualizar método de pagamento na base de dados", | ||||||
|  |     "enabled"          => "activado", | ||||||
|  |     "disabled"         => "descativado", | ||||||
|  |     // Subscription
 | ||||||
|  |     "error_fetching_image" => "Erro ao obter a imagem", | ||||||
|  |     "subscription_updated_successfuly" => "Subscrição actualizada com sucesso", | ||||||
|  |     "subscription_added_successfuly" => "Subscrição adicionada com sucesso", | ||||||
|  |     "error_deleting_subscription" => "Erro ao remover subscrição.", | ||||||
|  |     "invalid_request_method" => "Método invalido.", | ||||||
|  |     // User
 | ||||||
|  |     "error_updating_user_data" => "Erro ao actualizar dados do utilizador.", | ||||||
|  |     "user_details_saved" => "Dados do utiliador actualizados.", | ||||||
|  | 
 | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | ?>
 | ||||||
| @ -1,18 +1,20 @@ | |||||||
| <?php | <?php | ||||||
| 
 | 
 | ||||||
|     function getBillingCycle($cycle, $frequency) { |     require_once 'i18n/getlang.php'; | ||||||
|  | 
 | ||||||
|  |     function getBillingCycle($cycle, $frequency, $i18n) { | ||||||
|         switch ($cycle) { |         switch ($cycle) { | ||||||
|         case 1: |         case 1: | ||||||
|             return $frequency == 1 ? "Daily" : $frequency . " days"; |             return $frequency == 1 ? translate('Daily', $i18n) : $frequency . " " . translate('days', $i18n); | ||||||
|             break; |             break; | ||||||
|         case 2: |         case 2: | ||||||
|             return $frequency == 1 ? "Weekly" : $frequency . " weeks"; |             return $frequency == 1 ? translate('Weekly', $i18n) : $frequency . " " . translate('weeks', $i18n); | ||||||
|             break; |             break; | ||||||
|         case 3: |         case 3: | ||||||
|             return $frequency == 1 ? "Monthly" : $frequency . " months"; |             return $frequency == 1 ? translate('Monthly', $i18n) : $frequency . " " . translate('months', $i18n); | ||||||
|             break; |             break; | ||||||
|         case 4: |         case 4: | ||||||
|             return $frequency == 1 ? "Yearly" : $frequency . " years"; |             return $frequency == 1 ? translate('YEarly', $i18n) : $frequency . " " . translate('years', $i18n); | ||||||
|             break;   |             break;   | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -54,7 +56,7 @@ | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     function printSubscriptions($subscriptions, $sort, $categories, $members) { |     function printSubscriptions($subscriptions, $sort, $categories, $members, $i18n) { | ||||||
|         if ($sort === "price") { |         if ($sort === "price") { | ||||||
|             usort($subscriptions, function($a, $b) { |             usort($subscriptions, function($a, $b) { | ||||||
|                 return $a['price'] < $b['price'] ? 1 : -1; |                 return $a['price'] < $b['price'] ? 1 : -1; | ||||||
| @ -97,19 +99,19 @@ | |||||||
|                     <span class="cycle"><?= $subscription['billing_cycle'] ?></span>
 |                     <span class="cycle"><?= $subscription['billing_cycle'] ?></span>
 | ||||||
|                     <span class="next"><?= $subscription['next_payment'] ?></span>
 |                     <span class="next"><?= $subscription['next_payment'] ?></span>
 | ||||||
|                     <span class="price"> |                     <span class="price"> | ||||||
|                     <img src="<?= $subscription['payment_method_icon'] ?>" title="Payment Method: <?= $subscription['payment_method_name'] ?>"/> |                     <img src="<?= $subscription['payment_method_icon'] ?>" title="<?= translate('payment_method', $i18n) ?>: <?= $subscription['payment_method_name'] ?>"/> | ||||||
|                     <?= CurrencyFormatter::format($subscription['price'], $subscription['currency_code']) ?>
 |                     <?= CurrencyFormatter::format($subscription['price'], $subscription['currency_code']) ?>
 | ||||||
|                     </span> |                     </span> | ||||||
|                     <span class="actions"> |                     <span class="actions"> | ||||||
|                     <button class="image-button medium" onClick="openEditSubscription(event, <?= $subscription['id'] ?>)" name="edit"> |                     <button class="image-button medium" onClick="openEditSubscription(event, <?= $subscription['id'] ?>)" name="edit"> | ||||||
|                         <img src="images/siteicons/edit.png" title="Edit subscription"> |                         <img src="images/siteicons/edit.png" title="<?= translate('edit_subscription', $i18n) ?>"> | ||||||
|                     </button> |                     </button> | ||||||
|                     </span> |                     </span> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="subscription-secondary"> |                 <div class="subscription-secondary"> | ||||||
|                     <span class="name"><img src="images/siteicons/subscription.png" alt="Subscription" /><?= $subscription['name'] ?></span>
 |                     <span class="name"><img src="images/siteicons/subscription.png" alt="<?= translate('subscription', $i18n) ?>" /><?= $subscription['name'] ?></span>
 | ||||||
|                     <span class="payer_user" title="Paid By"><img src="images/siteicons/payment.png" alt="Paid By" /><?= $members[$subscription['payer_user_id']]['name'] ?></span>
 |                     <span class="payer_user" title="<?= translate('paid_by', $i18n) ?>"><img src="images/siteicons/payment.png" alt="<?= translate('paid_by', $i18n) ?>" /><?= $members[$subscription['payer_user_id']]['name'] ?></span>
 | ||||||
|                     <span class="category" title="Category" ><img src="images/siteicons/category.png" alt="Category" /><?= $categories[$subscription['category_id']]['name'] ?></span>
 |                     <span class="category" title="<?= translate('category', $i18n) ?>" ><img src="images/siteicons/category.png" alt="<?= translate('category', $i18n) ?>" /><?= $categories[$subscription['category_id']]['name'] ?></span>
 | ||||||
|                     <?php |                     <?php | ||||||
|                         if ($subscription['url'] != "") { |                         if ($subscription['url'] != "") { | ||||||
|                             $url = $subscription['url']; |                             $url = $subscription['url']; | ||||||
| @ -117,7 +119,7 @@ | |||||||
|                                 $url = "https://" . $url; |                                 $url = "https://" . $url; | ||||||
|                             } |                             } | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                                 <span class="url" title="External Url"><a href="<?= $url ?>" target="_blank"><img src="images/siteicons/web.png" alt="URL" /></a></span> |                                 <span class="url" title="<?= translate('external_url', $i18n) ?>"><a href="<?= $url ?>" target="_blank"><img src="images/siteicons/web.png" alt="<?= translate('url', $i18n) ?>" /></a></span> | ||||||
|                             <?php |                             <?php | ||||||
|                         } |                         } | ||||||
|                     ?>
 |                     ?>
 | ||||||
|  | |||||||
							
								
								
									
										77
									
								
								index.php
									
									
									
									
									
								
							
							
						
						
									
										77
									
								
								index.php
									
									
									
									
									
								
							| @ -39,25 +39,30 @@ | |||||||
|   $headerClass = count($subscriptions) > 0 ? "main-actions" : "main-actions hidden"; |   $headerClass = count($subscriptions) > 0 ? "main-actions" : "main-actions hidden"; | ||||||
|   $defaultLogo = $theme == "light" ? "images/wallos.png" : "images/walloswhite.png"; |   $defaultLogo = $theme == "light" ? "images/wallos.png" : "images/walloswhite.png"; | ||||||
| ?>
 | ?>
 | ||||||
|  |     <style> | ||||||
|  |       .logo-preview:after { | ||||||
|  |         content: '<?= translate('upload_logo', $i18n) ?>'; | ||||||
|  |       } | ||||||
|  |     </style> | ||||||
|     <section class="contain"> |     <section class="contain"> | ||||||
|       <header class="<?= $headerClass ?>" id="main-actions"> |       <header class="<?= $headerClass ?>" id="main-actions"> | ||||||
|         <button class="button" onClick="addSubscription()"> |         <button class="button" onClick="addSubscription()"> | ||||||
|           <img class="button-icon" src="images/siteicons/plusicon.png"> |           <img class="button-icon" src="images/siteicons/plusicon.png"> | ||||||
|           New Subscription |           <?= translate('new_subscription', $i18n) ?>
 | ||||||
|         </button> |         </button> | ||||||
|         <div class="sort-container"> |         <div class="sort-container"> | ||||||
|           <button class="button" value="Sort" onClick="toggleSortOptions()" id="sort-button"> |           <button class="button" value="Sort" onClick="toggleSortOptions()" id="sort-button"> | ||||||
|             <img src="images/siteicons/sort.png" class="button-icon" /> Sort |             <img src="images/siteicons/sort.png" class="button-icon" /> <?= translate('sort', $i18n) ?>
 | ||||||
|           </button> |           </button> | ||||||
|           <div class="sort-options" id="sort-options"> |           <div class="sort-options" id="sort-options"> | ||||||
|             <ul> |             <ul> | ||||||
|               <li <?= $sort == "name" ? 'class="selected"' : "" ?> onClick="setSortOption('name')" id="sort-name">Name</li>
 |               <li <?= $sort == "name" ? 'class="selected"' : "" ?> onClick="setSortOption('name')" id="sort-name"><?= translate('name', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "id" ? 'class="selected"' : "" ?> onClick="setSortOption('id')" id="sort-id">Last Added</li>
 |               <li <?= $sort == "id" ? 'class="selected"' : "" ?> onClick="setSortOption('id')" id="sort-id"><?= translate('last_added', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "price" ? 'class="selected"' : "" ?> onClick="setSortOption('price')" id="sort-price">Price</li>
 |               <li <?= $sort == "price" ? 'class="selected"' : "" ?> onClick="setSortOption('price')" id="sort-price"><?= translate('price', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "next_payment" ? 'class="selected"' : "" ?> onClick="setSortOption('next_payment')" id="sort-next_payment">Next payment</li>
 |               <li <?= $sort == "next_payment" ? 'class="selected"' : "" ?> onClick="setSortOption('next_payment')" id="sort-next_payment"><?= translate('next_payment', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "payer_user_id" ? 'class="selected"' : "" ?> onClick="setSortOption('payer_user_id')" id="sort-payer_user_id">Member</li>
 |               <li <?= $sort == "payer_user_id" ? 'class="selected"' : "" ?> onClick="setSortOption('payer_user_id')" id="sort-payer_user_id"><?= translate('member', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "category_id" ? 'class="selected"' : "" ?> onClick="setSortOption('category_id')" id="sort-category_id">Category</li>
 |               <li <?= $sort == "category_id" ? 'class="selected"' : "" ?> onClick="setSortOption('category_id')" id="sort-category_id"><?= translate('category', $i18n) ?></li>
 | ||||||
|               <li <?= $sort == "payment_method_id" ? 'class="selected"' : "" ?> onClick="setSortOption('payment_method_id')" id="sort-payment_method_id">Payment Method</li>
 |               <li <?= $sort == "payment_method_id" ? 'class="selected"' : "" ?> onClick="setSortOption('payment_method_id')" id="sort-payment_method_id"><?= translate('payment_method', $i18n) ?></li>
 | ||||||
|             </ul> |             </ul> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
| @ -71,7 +76,7 @@ | |||||||
|             $print[$id]['name']= $subscription['name']; |             $print[$id]['name']= $subscription['name']; | ||||||
|             $cycle = $subscription['cycle']; |             $cycle = $subscription['cycle']; | ||||||
|             $frequency = $subscription['frequency']; |             $frequency = $subscription['frequency']; | ||||||
|             $print[$id]['billing_cycle'] = getBillingCycle($cycle, $frequency); |             $print[$id]['billing_cycle'] = getBillingCycle($cycle, $frequency, $i18n); | ||||||
|             $paymentMethodId = $subscription['payment_method_id']; |             $paymentMethodId = $subscription['payment_method_id']; | ||||||
|             $print[$id]['currency_code'] = $currencies[$subscription['currency_id']]['code']; |             $print[$id]['currency_code'] = $currencies[$subscription['currency_id']]['code']; | ||||||
|             $currencyId = $subscription['currency_id']; |             $currencyId = $subscription['currency_id']; | ||||||
| @ -94,20 +99,20 @@ | |||||||
|           } |           } | ||||||
| 
 | 
 | ||||||
|           if (isset($print)) { |           if (isset($print)) { | ||||||
|             printSubscriptions($print, $sort, $categories, $members); |             printSubscriptions($print, $sort, $categories, $members, $i18n); | ||||||
|           } |           } | ||||||
|           $db->close(); |           $db->close(); | ||||||
| 
 | 
 | ||||||
|           if (count($subscriptions) == 0) { |           if (count($subscriptions) == 0) { | ||||||
|             ?>
 |             ?>
 | ||||||
|               <div class="empty-page"> |               <div class="empty-page"> | ||||||
|                 <img src="images/siteimages/empty.png" alt="Empty page" /> |                 <img src="images/siteimages/empty.png" alt="<?= translate('empty_page', $i18n) ?>" /> | ||||||
|                 <p> |                 <p> | ||||||
|                   You don't have any subscriptions yet |                   <?= translate('no_subscriptions_yet', $i18n) ?>
 | ||||||
|                 </p> |                 </p> | ||||||
|                 <button class="button" onClick="addSubscription()"> |                 <button class="button" onClick="addSubscription()"> | ||||||
|                   <img class="button-icon" src="images/siteicons/plusicon.png"> |                   <img class="button-icon" src="images/siteicons/plusicon.png"> | ||||||
|                   Add First Subscription |                   <?= translate('add_first_subscription', $i18n) ?>
 | ||||||
|                 </button> |                 </button> | ||||||
|               </div> |               </div> | ||||||
|             <?php |             <?php | ||||||
| @ -117,25 +122,25 @@ | |||||||
| 
 | 
 | ||||||
|       <section class="subscription-form" id="subscription-form"> |       <section class="subscription-form" id="subscription-form"> | ||||||
|         <header> |         <header> | ||||||
|           <h3 id="form-title">Add subscription</h3> |           <h3 id="form-title"><?= translate('add_subscription', $i18n) ?></h3>
 | ||||||
|           <span class="fa-solid fa-xmark close-form" onClick="closeAddSubscription()"></span> |           <span class="fa-solid fa-xmark close-form" onClick="closeAddSubscription()"></span> | ||||||
|         </header> |         </header> | ||||||
|         <form action="endpoints/subscription/add.php" method="post" id="subs-form"> |         <form action="endpoints/subscription/add.php" method="post" id="subs-form"> | ||||||
|            |            | ||||||
|           <div class="form-group-inline"> |           <div class="form-group-inline"> | ||||||
|           <input type="text" id="name" name="name" placeholder="Subscription name" onchange="setSearchButtonStatus()" onkeypress="this.onchange();" onpaste="this.onchange();" oninput="this.onchange();" required> |           <input type="text" id="name" name="name" placeholder="<?= translate('subscription_name', $i18n) ?>" onchange="setSearchButtonStatus()" onkeypress="this.onchange();" onpaste="this.onchange();" oninput="this.onchange();" required> | ||||||
|             <label for="logo" class="logo-preview"> |             <label for="logo" class="logo-preview"> | ||||||
|               <img src="" alt="Logo Preview" id="form-logo">  |               <img src="" alt="<?= translate('logo_preview', $i18n) ?>" id="form-logo">  | ||||||
|             </label> |             </label> | ||||||
|             <input type="file" id="logo" name="logo" accept="image/jpeg, image/png" onchange="handleFileSelect(event)" class="hidden-input"> |             <input type="file" id="logo" name="logo" accept="image/jpeg, image/png" onchange="handleFileSelect(event)" class="hidden-input"> | ||||||
|             <input type="hidden" id="logo-url" name="logo-url"> |             <input type="hidden" id="logo-url" name="logo-url"> | ||||||
|             <div id="logo-search-button" class="image-button medium disabled" title="Search logo on the web" onClick="searchLogo()"> |             <div id="logo-search-button" class="image-button medium disabled" title="<?= translate('search_logo', $i18n) ?>" onClick="searchLogo()"> | ||||||
|               <img src="images/siteicons/websearch.png"> |               <img src="images/siteicons/websearch.png"> | ||||||
|             </div> |             </div> | ||||||
|             <input type="hidden" id="id" name="id"> |             <input type="hidden" id="id" name="id"> | ||||||
|             <div id="logo-search-results" class="logo-search"> |             <div id="logo-search-results" class="logo-search"> | ||||||
|               <header> |               <header> | ||||||
|                 Web search |                 <?= translate('web_search', $i18n) ?>
 | ||||||
|                 <span class="fa-solid fa-xmark close-logo-search" onClick="closeLogoSearch()"></span> |                 <span class="fa-solid fa-xmark close-logo-search" onClick="closeLogoSearch()"></span> | ||||||
|               </header> |               </header> | ||||||
|               <div id="logo-search-images"></div> |               <div id="logo-search-images"></div> | ||||||
| @ -143,8 +148,8 @@ | |||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div class="form-group-inline"> |           <div class="form-group-inline"> | ||||||
|             <input type="number" step="0.01" id="price" name="price" placeholder="Price" required> |             <input type="number" step="0.01" id="price" name="price" placeholder="<?= translate('price', $i18n) ?>" required> | ||||||
|             <select id="currency" name="currency_id" placeholder="Currency"> |             <select id="currency" name="currency_id" placeholder="<?= translate('add_subscription', $i18n) ?>"> | ||||||
|               <?php |               <?php | ||||||
|                 foreach ($currencies as $currency) { |                 foreach ($currencies as $currency) { | ||||||
|                   $selected = ($currency['id'] == $main_currency) ? 'selected' : ''; |                   $selected = ($currency['id'] == $main_currency) ? 'selected' : ''; | ||||||
| @ -156,16 +161,12 @@ | |||||||
|             </select> |             </select> | ||||||
|           </div> |           </div> | ||||||
|            |            | ||||||
|           <div class="form-group"> |  | ||||||
|              |  | ||||||
|           </div> |  | ||||||
|            |  | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <div class="inline"> |             <div class="inline"> | ||||||
|               <div class="split66"> |               <div class="split66"> | ||||||
|                 <label for="cycle">Billing Cycle</label> |                 <label for="cycle"><?= translate('billing_cycle', $i18n) ?></label>
 | ||||||
|                 <div class="inline"> |                 <div class="inline"> | ||||||
|                   <select id="frequency" name="frequency" placeholder="Frequency"> |                   <select id="frequency" name="frequency" placeholder="<?= translate('frequency', $i18n) ?>"> | ||||||
|                   <?php |                   <?php | ||||||
|                       foreach ($frequencies as $frequency) { |                       foreach ($frequencies as $frequency) { | ||||||
|                     ?>
 |                     ?>
 | ||||||
| @ -178,7 +179,7 @@ | |||||||
|                   <?php |                   <?php | ||||||
|                       foreach ($cycles as $cycle) { |                       foreach ($cycles as $cycle) { | ||||||
|                     ?>
 |                     ?>
 | ||||||
|                       <option value="<?= $cycle['id'] ?>" <?= $cycle['id'] == 3 ? "selected" : "" ?>><?= $cycle['name'] ?></option>
 |                       <option value="<?= $cycle['id'] ?>" <?= $cycle['id'] == 3 ? "selected" : "" ?>><?= translate($cycle['name'], $i18n) ?></option>
 | ||||||
|                     <?php |                     <?php | ||||||
|                       } |                       } | ||||||
|                     ?>
 |                     ?>
 | ||||||
| @ -186,14 +187,14 @@ | |||||||
|                 </div> |                 </div> | ||||||
|               </div> |               </div> | ||||||
|               <div class="split33"> |               <div class="split33"> | ||||||
|                 <label for="next_payment">Next Payment</label> |                 <label for="next_payment"><?= translate('next_payment', $i18n) ?></label>
 | ||||||
|                 <input type="date" id="next_payment" name="next_payment" required> |                 <input type="date" id="next_payment" name="next_payment" required> | ||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <label for="payment_method">Payment Method</label> |             <label for="payment_method"><?= translate('payment_method', $i18n) ?></label>
 | ||||||
|             <select id="payment_method" name="payment_method_id"> |             <select id="payment_method" name="payment_method_id"> | ||||||
|               <?php |               <?php | ||||||
|                 foreach ($payment_methods as $payment) { |                 foreach ($payment_methods as $payment) { | ||||||
| @ -208,7 +209,7 @@ | |||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <label for="category">Category</label> |             <label for="category"><?= translate('category', $i18n) ?></label>
 | ||||||
|             <select id="category" name="category_id"> |             <select id="category" name="category_id"> | ||||||
|               <?php |               <?php | ||||||
|                 foreach ($categories as $category) { |                 foreach ($categories as $category) { | ||||||
| @ -224,7 +225,7 @@ | |||||||
|            |            | ||||||
| 
 | 
 | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <label for="payer_user">Paid by</label> |             <label for="payer_user"><?= translate('paid_by', $i18n) ?></label>
 | ||||||
|             <select id="payer_user" name="payer_user_id"> |             <select id="payer_user" name="payer_user_id"> | ||||||
|               <?php |               <?php | ||||||
|                 foreach ($members as $member) { |                 foreach ($members as $member) { | ||||||
| @ -237,11 +238,11 @@ | |||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <input type="text" id="url" name="url" placeholder="URL"> |             <input type="text" id="url" name="url" placeholder="<?= translate('url', $i18n) ?>"> | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <div class="form-group"> |           <div class="form-group"> | ||||||
|             <input type="text" id="notes" name="notes" placeholder="Notes"> |             <input type="text" id="notes" name="notes" placeholder="<?= translate('notes', $i18n) ?>"> | ||||||
|           </div> |           </div> | ||||||
| 
 | 
 | ||||||
|           <?php |           <?php | ||||||
| @ -249,16 +250,16 @@ | |||||||
|           ?>
 |           ?>
 | ||||||
|           <div class="form-group-inline"> |           <div class="form-group-inline"> | ||||||
|             <input type="checkbox" id="notifications" name="notifications"> |             <input type="checkbox" id="notifications" name="notifications"> | ||||||
|             <label for="notifications">Enable Notifications for this subscription</label> |             <label for="notifications"><?= translate('enable_notifications', $i18n) ?></label>
 | ||||||
|           </div> |           </div> | ||||||
|           <?php |           <?php | ||||||
|             } |             } | ||||||
|           ?>
 |           ?>
 | ||||||
| 
 | 
 | ||||||
|           <div class="buttons"> |           <div class="buttons"> | ||||||
|                 <input type="button" value="Delete" class="warning-button left" id="deletesub" style="display: none"> |                 <input type="button" value="<?= translate('delete', $i18n) ?>" class="warning-button left" id="deletesub" style="display: none"> | ||||||
|                 <input type="button" value="Cancel" class="secondary-button" onClick="closeAddSubscription()"> |                 <input type="button" value="<?= translate('cancel', $i18n) ?>" class="secondary-button" onClick="closeAddSubscription()"> | ||||||
|                 <input type="submit" value="Save" id="save-button"> |                 <input type="submit" value="<?= translate('save', $i18n) ?>" id="save-button"> | ||||||
|           </div> |           </div> | ||||||
|         </form> |         </form> | ||||||
|       </section> |       </section> | ||||||
|  | |||||||
							
								
								
									
										22
									
								
								login.php
									
									
									
									
									
								
							
							
						
						
									
										22
									
								
								login.php
									
									
									
									
									
								
							| @ -2,6 +2,10 @@ | |||||||
| require_once 'includes/connect.php'; | require_once 'includes/connect.php'; | ||||||
| require_once 'includes/checkuser.php'; | require_once 'includes/checkuser.php'; | ||||||
| 
 | 
 | ||||||
|  | require_once 'includes/i18n/languages.php'; | ||||||
|  | require_once 'includes/i18n/getlang.php'; | ||||||
|  | require_once 'includes/i18n/' . $lang . '.php'; | ||||||
|  | 
 | ||||||
| if ($userCount == 0) { | if ($userCount == 0) { | ||||||
|     header("Location: registration.php"); |     header("Location: registration.php"); | ||||||
|     exit(); |     exit(); | ||||||
| @ -25,7 +29,7 @@ if (isset($_POST['username']) && isset($_POST['password'])) { | |||||||
|     $password = $_POST['password']; |     $password = $_POST['password']; | ||||||
|     $rememberMe = isset($_POST['remember']) ? true : false; |     $rememberMe = isset($_POST['remember']) ? true : false; | ||||||
| 
 | 
 | ||||||
|     $query = "SELECT id, password, main_currency FROM user WHERE username = :username"; |     $query = "SELECT id, password, main_currency, language FROM user WHERE username = :username"; | ||||||
|     $stmt = $db->prepare($query); |     $stmt = $db->prepare($query); | ||||||
|     $stmt->bindValue(':username', $username, SQLITE3_TEXT); |     $stmt->bindValue(':username', $username, SQLITE3_TEXT); | ||||||
|     $result = $stmt->execute(); |     $result = $stmt->execute(); | ||||||
| @ -35,10 +39,13 @@ if (isset($_POST['username']) && isset($_POST['password'])) { | |||||||
|         $hashedPasswordFromDb = $row['password']; |         $hashedPasswordFromDb = $row['password']; | ||||||
|         $userId = $row['id']; |         $userId = $row['id']; | ||||||
|         $main_currency = $row['main_currency']; |         $main_currency = $row['main_currency']; | ||||||
|  |         $language = $row['language']; | ||||||
|         if (password_verify($password, $hashedPasswordFromDb)) { |         if (password_verify($password, $hashedPasswordFromDb)) { | ||||||
|             $_SESSION['username'] = $username; |             $_SESSION['username'] = $username; | ||||||
|             $_SESSION['loggedin'] = true; |             $_SESSION['loggedin'] = true; | ||||||
|             $_SESSION['main_currency'] = $main_currency; |             $_SESSION['main_currency'] = $main_currency; | ||||||
|  |             $cookieExpire = time() + (30 * 24 * 60 * 60); | ||||||
|  |             setcookie('language', $language, $cookieExpire, '/'); | ||||||
|             if ($rememberMe) { |             if ($rememberMe) { | ||||||
|                 $token = bin2hex(random_bytes(32)); |                 $token = bin2hex(random_bytes(32)); | ||||||
|                 $addLoginTokens = "INSERT INTO login_tokens (user_id, token) VALUES (?, ?)"; |                 $addLoginTokens = "INSERT INTO login_tokens (user_id, token) VALUES (?, ?)"; | ||||||
| @ -47,7 +54,6 @@ if (isset($_POST['username']) && isset($_POST['password'])) { | |||||||
|                 $addLoginTokensStmt->bindValue(2, $token, SQLITE3_TEXT); |                 $addLoginTokensStmt->bindValue(2, $token, SQLITE3_TEXT); | ||||||
|                 $addLoginTokensStmt->execute(); |                 $addLoginTokensStmt->execute(); | ||||||
|                 $_SESSION['token'] = $token; |                 $_SESSION['token'] = $token; | ||||||
|                 $cookieExpire = time() + (30 * 24 * 60 * 60); |  | ||||||
|                 $cookieValue = $username . "|" . $token . "|" . $main_currency; |                 $cookieValue = $username . "|" . $token . "|" . $main_currency; | ||||||
|                 setcookie('wallos_login', $cookieValue, $cookieExpire, '/'); |                 setcookie('wallos_login', $cookieValue, $cookieExpire, '/'); | ||||||
|             } |             } | ||||||
| @ -87,33 +93,33 @@ if (isset($_POST['username']) && isset($_POST['password'])) { | |||||||
|                     } |                     } | ||||||
|                 ?>
 |                 ?>
 | ||||||
|                 <p> |                 <p> | ||||||
|                     Please login. |                     <?= translate('please_login', $i18n) ?>
 | ||||||
|                 </p> |                 </p> | ||||||
|             </header> |             </header> | ||||||
|             <form action="login.php" method="post"> |             <form action="login.php" method="post"> | ||||||
|                 <div class="form-group"> |                 <div class="form-group"> | ||||||
|                     <label for="username">Username:</label> |                     <label for="username"><?= translate('username', $i18n) ?>:</label>
 | ||||||
|                     <input type="text" id="username" name="username" required> |                     <input type="text" id="username" name="username" required> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="form-group"> |                 <div class="form-group"> | ||||||
|                     <label for="password">Password:</label> |                     <label for="password"><?= translate('password', $i18n) ?>:</label>
 | ||||||
|                     <input type="password" id="password" name="password" required> |                     <input type="password" id="password" name="password" required> | ||||||
|                 </div> |                 </div> | ||||||
|                 <div class="form-group-inline"> |                 <div class="form-group-inline"> | ||||||
|                     <input type="checkbox" id="remember" name="remember"> |                     <input type="checkbox" id="remember" name="remember"> | ||||||
|                     <label for="remember">Stay logged in (30 days)</label> |                     <label for="remember"><?= translate('stay_logged_in', $i18n) ?></label>
 | ||||||
|                 </div> |                 </div> | ||||||
|                 <?php |                 <?php | ||||||
|                     if ($loginFailed) { |                     if ($loginFailed) { | ||||||
|                         ?>
 |                         ?>
 | ||||||
|                         <sup class="error"> |                         <sup class="error"> | ||||||
|                             Login details are incorrect. |                             <?= translate('login_failed', $i18n) ?>.
 | ||||||
|                         </sup> |                         </sup> | ||||||
|                         <?php |                         <?php | ||||||
|                     } |                     } | ||||||
|                 ?>
 |                 ?>
 | ||||||
|                 <div class="form-group"> |                 <div class="form-group"> | ||||||
|                     <input type="submit" value="Login"> |                     <input type="submit" value="<?= translate('login', $i18n) ?>"> | ||||||
|                 </div> |                 </div> | ||||||
|             </form> |             </form> | ||||||
|         </section> |         </section> | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								migrations/000005.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								migrations/000005.php
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | <?php | ||||||
|  | // This migration adds a "language" column to the user table and sets all values to english.
 | ||||||
|  | 
 | ||||||
|  | /** @noinspection PhpUndefinedVariableInspection */ | ||||||
|  | $columnQuery = $db->query("SELECT * FROM pragma_table_info('user') where name='language'"); | ||||||
|  | $columnRequired = $columnQuery->fetchArray(SQLITE3_ASSOC) === false; | ||||||
|  | 
 | ||||||
|  | if ($columnRequired) { | ||||||
|  |     $db->exec('ALTER TABLE user ADD COLUMN language TEXT DEFAULT "en"'); | ||||||
|  |     $db->exec('UPDATE user SET language = "en"'); | ||||||
|  | } | ||||||
| @ -2,6 +2,10 @@ | |||||||
| require_once 'includes/connect.php'; | require_once 'includes/connect.php'; | ||||||
| require_once 'includes/checkuser.php'; | require_once 'includes/checkuser.php'; | ||||||
| 
 | 
 | ||||||
|  | require_once 'includes/i18n/languages.php'; | ||||||
|  | require_once 'includes/i18n/getlang.php'; | ||||||
|  | require_once 'includes/i18n/' . $lang . '.php'; | ||||||
|  | 
 | ||||||
| if ($userCount > 0) { | if ($userCount > 0) { | ||||||
|     header("Location: login.php"); |     header("Location: login.php"); | ||||||
|     exit(); |     exit(); | ||||||
| @ -28,12 +32,13 @@ if (isset($_POST['username'])) { | |||||||
|     $password = $_POST['password']; |     $password = $_POST['password']; | ||||||
|     $confirm_password = $_POST['confirm_password']; |     $confirm_password = $_POST['confirm_password']; | ||||||
|     $main_currency = $_POST['main_currency']; |     $main_currency = $_POST['main_currency']; | ||||||
|  |     $language = $_POST['language']; | ||||||
|     $avatar = "0"; |     $avatar = "0"; | ||||||
| 
 | 
 | ||||||
|     if ($password != $confirm_password) { |     if ($password != $confirm_password) { | ||||||
|         $passwordMismatch = true; |         $passwordMismatch = true; | ||||||
|     } else { |     } else { | ||||||
|         $query = "INSERT INTO user (username, email, password, main_currency, avatar) VALUES (:username, :email, :password, :main_currency, :avatar)"; |         $query = "INSERT INTO user (username, email, password, main_currency, avatar, language) VALUES (:username, :email, :password, :main_currency, :avatar, :language)"; | ||||||
|         $stmt = $db->prepare($query); |         $stmt = $db->prepare($query); | ||||||
|         $hashedPassword = password_hash($password, PASSWORD_DEFAULT); |         $hashedPassword = password_hash($password, PASSWORD_DEFAULT); | ||||||
|         $stmt->bindValue(':username', $username, SQLITE3_TEXT); |         $stmt->bindValue(':username', $username, SQLITE3_TEXT); | ||||||
| @ -41,6 +46,7 @@ if (isset($_POST['username'])) { | |||||||
|         $stmt->bindValue(':password', $hashedPassword, SQLITE3_TEXT); |         $stmt->bindValue(':password', $hashedPassword, SQLITE3_TEXT); | ||||||
|         $stmt->bindValue(':main_currency', $main_currency, SQLITE3_TEXT); |         $stmt->bindValue(':main_currency', $main_currency, SQLITE3_TEXT); | ||||||
|         $stmt->bindValue(':avatar', $avatar, SQLITE3_TEXT); |         $stmt->bindValue(':avatar', $avatar, SQLITE3_TEXT); | ||||||
|  |         $stmt->bindValue(':language', $language, SQLITE3_TEXT); | ||||||
|         $result = $stmt->execute(); |         $result = $stmt->execute(); | ||||||
| 
 | 
 | ||||||
|         if ($result) { |         if ($result) { | ||||||
| @ -80,6 +86,7 @@ if (isset($_POST['username'])) { | |||||||
|         <link rel="manifest" href="images/icon/site.webmanifest"> |         <link rel="manifest" href="images/icon/site.webmanifest"> | ||||||
|         <link rel="stylesheet" href="styles/login.css"> |         <link rel="stylesheet" href="styles/login.css"> | ||||||
|         <link rel="stylesheet" href="styles/login-dark-theme.css" id="dark-theme" <?= $theme == "light" ? "disabled" : "" ?>>
 |         <link rel="stylesheet" href="styles/login-dark-theme.css" id="dark-theme" <?= $theme == "light" ? "disabled" : "" ?>>
 | ||||||
|  |         <script type="text/javascript" src="scripts/registration.js"></script> | ||||||
|     </head> |     </head> | ||||||
|     <body> |     <body> | ||||||
|         <div class="content"> |         <div class="content"> | ||||||
| @ -93,28 +100,28 @@ if (isset($_POST['username'])) { | |||||||
|                     } |                     } | ||||||
|                 ?>
 |                 ?>
 | ||||||
|                     <p> |                     <p> | ||||||
|                         You need to create an account before you're able to login. |                         <?= translate('create_account', $i18n) ?>
 | ||||||
|                     </p> |                     </p> | ||||||
|                 </header> |                 </header> | ||||||
|                 <form action="registration.php" method="post"> |                 <form action="registration.php" method="post"> | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <label for="username">Username:</label> |                         <label for="username"><?= translate('username', $i18n) ?>:</label>
 | ||||||
|                         <input type="text" id="username" name="username" required> |                         <input type="text" id="username" name="username" required> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <label for="email">Email:</label> |                         <label for="email"><?= translate('email', $i18n) ?>:</label>
 | ||||||
|                         <input type="email" id="email" name="email" required> |                         <input type="email" id="email" name="email" required> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <label for="password">Password:</label> |                         <label for="password"><?= translate('password', $i18n) ?>:</label>
 | ||||||
|                         <input type="password" id="password" name="password" required> |                         <input type="password" id="password" name="password" required> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <label for="confirm_password">Confirm Password:</label> |                         <label for="confirm_password"><?= translate('confirm_password', $i18n) ?>:</label>
 | ||||||
|                         <input type="password" id="confirm_password" name="confirm_password" required> |                         <input type="password" id="confirm_password" name="confirm_password" required> | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <label for="currency">Main Currency:</label> |                         <label for="currency"><?= translate('main_currency', $i18n) ?>:</label>
 | ||||||
|                         <select id="currency" name="main_currency" placeholder="Currency"> |                         <select id="currency" name="main_currency" placeholder="Currency"> | ||||||
|                         <?php |                         <?php | ||||||
|                             foreach ($currencies as $currency) { |                             foreach ($currencies as $currency) { | ||||||
| @ -125,11 +132,24 @@ if (isset($_POST['username'])) { | |||||||
|                         ?>
 |                         ?>
 | ||||||
|                         </select> |                         </select> | ||||||
|                     </div> |                     </div> | ||||||
|  |                     <div class="form-group"> | ||||||
|  |                         <label for="language"><?= translate('language', $i18n) ?>:</label>
 | ||||||
|  |                         <select id="language" name="language" placeholder="Language" onchange="changeLanguage(this.value)"> | ||||||
|  |                         <?php  | ||||||
|  |                             foreach ($languages as $code => $name) { | ||||||
|  |                                 $selected = ($code === $lang) ? 'selected' : ''; | ||||||
|  |                         ?>
 | ||||||
|  |                                 <option value="<?= $code ?>" <?= $selected ?>><?= $name ?></option>
 | ||||||
|  |                         <?php | ||||||
|  |                             } | ||||||
|  |                         ?>
 | ||||||
|  |                         </select> | ||||||
|  |                     </div> | ||||||
|                     <?php |                     <?php | ||||||
|                         if ($passwordMismatch) { |                         if ($passwordMismatch) { | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                             <sup class="error"> |                             <sup class="error"> | ||||||
|                                 Passwords do not match. |                                 <?= translate('passwords_dont_match', $i18n) ?>
 | ||||||
|                             </sup> |                             </sup> | ||||||
|                             <?php |                             <?php | ||||||
|                         } |                         } | ||||||
| @ -138,13 +158,13 @@ if (isset($_POST['username'])) { | |||||||
|                         if ($registrationFailed) { |                         if ($registrationFailed) { | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                             <sup class="error"> |                             <sup class="error"> | ||||||
|                                 Registration failed, please try again. |                                 <?= translate('registration_failed', $i18n) ?>
 | ||||||
|                             </sup> |                             </sup> | ||||||
|                             <?php |                             <?php | ||||||
|                         } |                         } | ||||||
|                     ?>
 |                     ?>
 | ||||||
|                     <div class="form-group"> |                     <div class="form-group"> | ||||||
|                         <input type="submit" value="Register"> |                         <input type="submit" value="<?= translate('register', $i18n) ?>"> | ||||||
|                     </div> |                     </div> | ||||||
|                 </form> |                 </form> | ||||||
|             </section> |             </section> | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ function resetForm() { | |||||||
|     const id = document.querySelector("#id"); |     const id = document.querySelector("#id"); | ||||||
|     id.value = ""; |     id.value = ""; | ||||||
|     const formTitle = document.querySelector("#form-title"); |     const formTitle = document.querySelector("#form-title"); | ||||||
|     formTitle.textContent = "Add subscription"; |     formTitle.textContent = translate('add_subscription'); | ||||||
|     const logo = document.querySelector("#form-logo"); |     const logo = document.querySelector("#form-logo"); | ||||||
|     logo.src = ""; |     logo.src = ""; | ||||||
|     logo.style = 'display: none'; |     logo.style = 'display: none'; | ||||||
| @ -35,7 +35,7 @@ function resetForm() { | |||||||
| 
 | 
 | ||||||
| function fillEditFormFields(subscription) { | function fillEditFormFields(subscription) { | ||||||
|   const formTitle = document.querySelector("#form-title"); |   const formTitle = document.querySelector("#form-title"); | ||||||
|   formTitle.textContent = "Edit subscription"; |   formTitle.textContent = translate('edit_subscription'); | ||||||
|   const logo = document.querySelector("#form-logo"); |   const logo = document.querySelector("#form-logo"); | ||||||
|   const defaultLogo = window.theme && window.theme == "light" ? "images/wallos.png" : "images/walloswhite.png"; |   const defaultLogo = window.theme && window.theme == "light" ? "images/wallos.png" : "images/walloswhite.png"; | ||||||
|   const logoFile = subscription.logo !== null ? "images/uploads/logos/" + subscription.logo : defaultLogo; |   const logoFile = subscription.logo !== null ? "images/uploads/logos/" + subscription.logo : defaultLogo; | ||||||
| @ -91,19 +91,19 @@ function openEditSubscription(event, id) { | |||||||
|       if (response.ok) { |       if (response.ok) { | ||||||
|         return response.json(); |         return response.json(); | ||||||
|       } else { |       } else { | ||||||
|         showErrorMessage("Failed to load subscription"); |         showErrorMessage(translate('failed_to_load_subscription')); | ||||||
|       } |       } | ||||||
|     }) |     }) | ||||||
|     .then((data) => { |     .then((data) => { | ||||||
|       if (data.error || data === "Error") { |       if (data.error || data === "Error") { | ||||||
|         showErrorMessage("Failed to load subscription"); |         showErrorMessage(translate('failed_to_load_subscription')); | ||||||
|       } else { |       } else { | ||||||
|         const subscription = data; |         const subscription = data; | ||||||
|         fillEditFormFields(subscription); |         fillEditFormFields(subscription); | ||||||
|       } |       } | ||||||
|     }) |     }) | ||||||
|     .catch((error) => { |     .catch((error) => { | ||||||
|       showErrorMessage("Failed to load subscription"); |       showErrorMessage(translate('failed_to_load_subscription')); | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -145,11 +145,11 @@ function deleteSubscription(id) { | |||||||
|     }) |     }) | ||||||
|     .then(response => { |     .then(response => { | ||||||
|         if (response.ok) { |         if (response.ok) { | ||||||
|           showSuccessMessage("Subscription deleted"); |           showSuccessMessage(translate('subscription_deleted')); | ||||||
|           fetchSubscriptions(); |           fetchSubscriptions(); | ||||||
|           closeAddSubscription(); |           closeAddSubscription(); | ||||||
|         } else { |         } else { | ||||||
|           alert("Error deleting the subscription"); |           alert(translate('error_deleting_subscription')); | ||||||
|         } |         } | ||||||
|     }) |     }) | ||||||
|     .catch(error => { |     .catch(error => { | ||||||
| @ -188,7 +188,7 @@ function searchLogo() { | |||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|             .catch(error => { |             .catch(error => { | ||||||
|                 console.error("Error fetching image results:", error); |                 console.error(translate('error_fetching_image_results'), error); | ||||||
|             }); |             }); | ||||||
|   } else { |   } else { | ||||||
|     nameInput.focus(); |     nameInput.focus(); | ||||||
| @ -243,7 +243,7 @@ function fetchSubscriptions() { | |||||||
|       } |       } | ||||||
|     }) |     }) | ||||||
|     .catch(error => { |     .catch(error => { | ||||||
|       console.error("Error reloading subscriptions:", error); |       console.error(translate('error_reloading_subscription'), error); | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										34
									
								
								scripts/i18n/en.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								scripts/i18n/en.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | let i18n = { | ||||||
|  |     // Dashboard
 | ||||||
|  |     'error_reloading_subscription': 'Error reloading subscription:', | ||||||
|  |     'error_fetching_image_results': 'Error fetching image results:', | ||||||
|  |     'subscription_deleted': 'Subscription deleted', | ||||||
|  |     'error_deleting_subscription': "Error deleting subscription", | ||||||
|  |     'failed_to_load_subscription': "Failed to load subscription", | ||||||
|  |     'edit_subscription': "Edit subscription", | ||||||
|  |     'add_subscription': "Add subscription", | ||||||
|  |     // Settings
 | ||||||
|  |     'network_response_error': "Network response was not ok", | ||||||
|  |     'failed_add_member': 'Failed to add member', | ||||||
|  |     'member': 'Member', | ||||||
|  |     'save_member': 'Save member', | ||||||
|  |     'delete_member': 'Delete member', | ||||||
|  |     'failed_remove_member': 'Failed to remove member', | ||||||
|  |     'failed_save_member': 'Failed to sabe member', | ||||||
|  |     'failed_add_category': 'Failed to add categpry', | ||||||
|  |     'category': 'Category', | ||||||
|  |     'save_category': 'Save category', | ||||||
|  |     'delete_category': 'Delete category', | ||||||
|  |     'failed_remove_category': 'Failed to remove category', | ||||||
|  |     'currency': 'Currency', | ||||||
|  |     'currency_code': 'Currency code', | ||||||
|  |     'save_currency': 'Save currency', | ||||||
|  |     'delete_currency': 'Delete currency', | ||||||
|  |     'failed_remove_currency': 'Failed to remove currency', | ||||||
|  |     'failed_save_currency': 'Failed to save currency', | ||||||
|  |     'cant_disable_payment_in_use': 'Can\'t disable payment in use', | ||||||
|  |     'failed_save_payment_method': 'Failed to sabe payment method', | ||||||
|  |     'unknown_error': 'Unknown error, please try again.', | ||||||
|  |     'error_saving_notification_data': 'Error saving notification data', | ||||||
|  |     'error_sending_notification': 'Error sending notification', | ||||||
|  | }; | ||||||
							
								
								
									
										7
									
								
								scripts/i18n/getlang.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								scripts/i18n/getlang.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | function translate(key) { | ||||||
|  |     if (i18n[key]) { | ||||||
|  |         return i18n[key]; | ||||||
|  |     } else { | ||||||
|  |         return "[Translation Missing]"; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										34
									
								
								scripts/i18n/pt.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								scripts/i18n/pt.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | let i18n = { | ||||||
|  |     // Dashboard
 | ||||||
|  |     'error_reloading_subscription': 'Erro ao carregar a subscrição:', | ||||||
|  |     'error_fetching_image_results': 'Erro ao obter imagens:', | ||||||
|  |     'subscription_deleted': 'Subscrição eliminada', | ||||||
|  |     'error_deleting_subscription': 'Erro ao eliminar a subscrição', | ||||||
|  |     'failed_to_load_subscription': 'Falha ao carregar a subscrição', | ||||||
|  |     'edit_subscription': 'Editar subscrição', | ||||||
|  |     'add_subscription': 'Adicionar subscrição', | ||||||
|  |     // Settings
 | ||||||
|  |     'network_response_error': 'Erro de resposta de rede', | ||||||
|  |     'failed_add_member': 'Falha ao adicionar membro', | ||||||
|  |     'member': 'Membro', | ||||||
|  |     'save_member': 'Guardar membro', | ||||||
|  |     'delete_member': 'Remover membro', | ||||||
|  |     'failed_remove_member': 'Erro ao remover membro', | ||||||
|  |     'failed_save_member': 'Erro ao guardar membro', | ||||||
|  |     'failed_add_category': 'Erro ao adicionar categoria', | ||||||
|  |     'category': 'Categoria', | ||||||
|  |     'save_category': 'Guardar categoria', | ||||||
|  |     'delete_category': 'Remover categoria', | ||||||
|  |     'failed_remove_category': 'Erro ao remover categoria', | ||||||
|  |     'currency': 'Moeda', | ||||||
|  |     'currency_code': 'Código de moeda', | ||||||
|  |     'save_currency': 'Guardar moeda', | ||||||
|  |     'delete_currency': 'Remover moeda', | ||||||
|  |     'failed_remove_currency': 'Erro ao remover moeda', | ||||||
|  |     'failed_save_currency': 'Erro ao guardar moeda', | ||||||
|  |     'cant_disable_payment_in_use': 'Não é possível desativar pagamento em uso', | ||||||
|  |     'failed_save_payment_method': 'Erro ao guardar método de pagamento', | ||||||
|  |     'unknown_error': 'Erro desconhecido, por favor, tente novamente.', | ||||||
|  |     'error_saving_notification_data': 'Erro ao guardar dados de notificação', | ||||||
|  |     'error_sending_notification': 'Erro ao enviar notificação', | ||||||
|  | }; | ||||||
							
								
								
									
										58
									
								
								scripts/registration.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								scripts/registration.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | function setCookie(name, value, days) { | ||||||
|  |     var expires = ""; | ||||||
|  |     if (days) { | ||||||
|  |         var date = new Date(); | ||||||
|  |         date.setTime(date.getTime() + (days * 24 * 60 * 60 * 1000)); | ||||||
|  |         expires = "; expires=" + date.toUTCString(); | ||||||
|  |     } | ||||||
|  |     document.cookie = name + "=" + value + expires + "; path=/"; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function storeFormFieldValue(fieldId) { | ||||||
|  |     var fieldElement = document.getElementById(fieldId); | ||||||
|  |     if (fieldElement) { | ||||||
|  |         localStorage.setItem(fieldId, fieldElement.value); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function storeFormFields() { | ||||||
|  |     storeFormFieldValue('username'); | ||||||
|  |     storeFormFieldValue('email'); | ||||||
|  |     storeFormFieldValue('password'); | ||||||
|  |     storeFormFieldValue('confirm_password'); | ||||||
|  |     storeFormFieldValue('currency'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function restoreFormFieldValue(fieldId) { | ||||||
|  |     var fieldElement = document.getElementById(fieldId); | ||||||
|  |     if (fieldElement) { | ||||||
|  |         fieldElement.value = localStorage.getItem(fieldId) || ''; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function restoreFormFields() { | ||||||
|  |     restoreFormFieldValue('username'); | ||||||
|  |     restoreFormFieldValue('email'); | ||||||
|  |     restoreFormFieldValue('password'); | ||||||
|  |     restoreFormFieldValue('confirm_password'); | ||||||
|  |     restoreFormFieldValue('currency'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function removeFromStorage() { | ||||||
|  |     localStorage.removeItem('username'); | ||||||
|  |     localStorage.removeItem('email'); | ||||||
|  |     localStorage.removeItem('password'); | ||||||
|  |     localStorage.removeItem('confirm_password'); | ||||||
|  |     localStorage.removeItem('currency'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function changeLanguage(selectedLanguage) { | ||||||
|  |     storeFormFields(); | ||||||
|  |     setCookie("language", selectedLanguage, 365); | ||||||
|  |     location.reload(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | window.onload = function () { | ||||||
|  |     restoreFormFields(); | ||||||
|  |     removeFromStorage(); | ||||||
|  | }; | ||||||
| @ -24,8 +24,8 @@ function addMemberButton(memberId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|   .then(response => { |   .then(response => { | ||||||
|     if (!response.ok) { |     if (!response.ok) { | ||||||
|       throw new Error('Network response was not ok'); |       throw new Error(translate('network_response_error')); | ||||||
|       showErrorMessage("Failed to add member"); |       showErrorMessage(translate('failed_add_member')); | ||||||
|     } |     } | ||||||
|     return response.json(); |     return response.json(); | ||||||
|   }) |   }) | ||||||
| @ -39,9 +39,9 @@ function addMemberButton(memberId) { | |||||||
| 
 | 
 | ||||||
|       let input = document.createElement("input"); |       let input = document.createElement("input"); | ||||||
|       input.type = "text"; |       input.type = "text"; | ||||||
|       input.placeholder = "Member"; |       input.placeholder = translate('member'); | ||||||
|       input.name = "member"; |       input.name = "member"; | ||||||
|       input.value = "Member"; |       input.value = translate('member'); | ||||||
| 
 | 
 | ||||||
|       let editLink = document.createElement("button"); |       let editLink = document.createElement("button"); | ||||||
|       editLink.className = "image-button medium" |       editLink.className = "image-button medium" | ||||||
| @ -52,7 +52,7 @@ function addMemberButton(memberId) { | |||||||
| 
 | 
 | ||||||
|       let editImage = document.createElement("img"); |       let editImage = document.createElement("img"); | ||||||
|       editImage.src = "images/siteicons/save.png"; |       editImage.src = "images/siteicons/save.png"; | ||||||
|       editImage.title = "Save Member"; |       editImage.title = translate('save_member'); | ||||||
| 
 | 
 | ||||||
|       editLink.appendChild(editImage); |       editLink.appendChild(editImage); | ||||||
| 
 | 
 | ||||||
| @ -65,7 +65,7 @@ function addMemberButton(memberId) { | |||||||
| 
 | 
 | ||||||
|       let deleteImage = document.createElement("img"); |       let deleteImage = document.createElement("img"); | ||||||
|       deleteImage.src = "images/siteicons/delete.png"; |       deleteImage.src = "images/siteicons/delete.png"; | ||||||
|       deleteImage.title = "Delete Member"; |       deleteImage.title = translate('delete_member'); | ||||||
| 
 | 
 | ||||||
|       deleteLink.appendChild(deleteImage); |       deleteLink.appendChild(deleteImage); | ||||||
| 
 | 
 | ||||||
| @ -80,7 +80,7 @@ function addMemberButton(memberId) { | |||||||
|     document.getElementById("addMember").disabled = false; |     document.getElementById("addMember").disabled = false; | ||||||
|   }) |   }) | ||||||
|   .catch(error => { |   .catch(error => { | ||||||
|     showErrorMessage("Failed to add member"); |     showErrorMessage(translate('failed_add_member')); | ||||||
|     document.getElementById("addMember").disabled = false; |     document.getElementById("addMember").disabled = false; | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
| @ -91,7 +91,7 @@ function removeMember(memberId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|     .then(response => { |     .then(response => { | ||||||
|       if (!response.ok) { |       if (!response.ok) { | ||||||
|         throw new Error('Network response was not ok'); |         throw new Error(translate('network_response_error')); | ||||||
|       } |       } | ||||||
|       return response.json(); |       return response.json(); | ||||||
|     }) |     }) | ||||||
| @ -101,13 +101,13 @@ function removeMember(memberId) { | |||||||
|       if (divToRemove) { |       if (divToRemove) { | ||||||
|         divToRemove.parentNode.removeChild(divToRemove); |         divToRemove.parentNode.removeChild(divToRemove); | ||||||
|       } |       } | ||||||
|       showSuccessMessage("Member removed"); |       showSuccessMessage(responseData.message); | ||||||
|      } else { |      } else { | ||||||
|       showErrorMessage(responseData.errorMessage || "Failed to remove member"); |       showErrorMessage(responseData.errorMessage || translate('failed_remove_member')); | ||||||
|      } |      } | ||||||
|     }) |     }) | ||||||
|     .catch(error => { |     .catch(error => { | ||||||
|       showErrorMessage("Failed to remove member"); |       showErrorMessage(translate('failed_remove_member')); | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -124,19 +124,19 @@ function editMember(memberId) { | |||||||
|       .then(response => { |       .then(response => { | ||||||
|         saveButton.classList.remove("disabled"); |         saveButton.classList.remove("disabled"); | ||||||
|         if (!response.ok) { |         if (!response.ok) { | ||||||
|           showErrorMessage("Failed to save member"); |           showErrorMessage(translate('failed_save_member')); | ||||||
|         } |         } | ||||||
|         return response.json(); |         return response.json(); | ||||||
|       }) |       }) | ||||||
|       .then(responseData => { |       .then(responseData => { | ||||||
|         if (responseData.success) { |         if (responseData.success) { | ||||||
|           showSuccessMessage("Member saved"); |           showSuccessMessage(responseData.message); | ||||||
|         } else { |         } else { | ||||||
|           showErrorMessage(responseData.errorMessage || "Failed to save member"); |           showErrorMessage(responseData.errorMessage || translate('failed_save_member')); | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|       .catch(error => { |       .catch(error => { | ||||||
|         showErrorMessage("Failed to save member"); |         showErrorMessage(translate('failed_save_member')); | ||||||
|       }); |       }); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -147,8 +147,8 @@ function addCategoryButton(categoryId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|   .then(response => { |   .then(response => { | ||||||
|     if (!response.ok) { |     if (!response.ok) { | ||||||
|       throw new Error('Network response was not ok'); |       throw new Error(translate('network_response_error')); | ||||||
|       showErrorMessage("Failed to add category"); |       showErrorMessage(translate('failed_add_category')); | ||||||
|     } |     } | ||||||
|     return response.json(); |     return response.json(); | ||||||
|   }) |   }) | ||||||
| @ -162,9 +162,9 @@ function addCategoryButton(categoryId) { | |||||||
| 
 | 
 | ||||||
|       let input = document.createElement("input"); |       let input = document.createElement("input"); | ||||||
|       input.type = "text"; |       input.type = "text"; | ||||||
|       input.placeholder = "Category"; |       input.placeholder = translate('category'); | ||||||
|       input.name = "category"; |       input.name = "category"; | ||||||
|       input.value = "Category"; |       input.value = translate('category'); | ||||||
| 
 | 
 | ||||||
|       let editLink = document.createElement("button"); |       let editLink = document.createElement("button"); | ||||||
|       editLink.className = "image-button medium" |       editLink.className = "image-button medium" | ||||||
| @ -175,7 +175,7 @@ function addCategoryButton(categoryId) { | |||||||
| 
 | 
 | ||||||
|       let editImage = document.createElement("img"); |       let editImage = document.createElement("img"); | ||||||
|       editImage.src = "images/siteicons/save.png"; |       editImage.src = "images/siteicons/save.png"; | ||||||
|       editImage.title = "Save Category"; |       editImage.title = translate('save_category'); | ||||||
| 
 | 
 | ||||||
|       editLink.appendChild(editImage); |       editLink.appendChild(editImage); | ||||||
| 
 | 
 | ||||||
| @ -188,7 +188,7 @@ function addCategoryButton(categoryId) { | |||||||
| 
 | 
 | ||||||
|       let deleteImage = document.createElement("img"); |       let deleteImage = document.createElement("img"); | ||||||
|       deleteImage.src = "images/siteicons/delete.png"; |       deleteImage.src = "images/siteicons/delete.png"; | ||||||
|       deleteImage.title = "Delete Category"; |       deleteImage.title = translate('delete_category'); | ||||||
| 
 | 
 | ||||||
|       deleteLink.appendChild(deleteImage); |       deleteLink.appendChild(deleteImage); | ||||||
| 
 | 
 | ||||||
| @ -203,7 +203,7 @@ function addCategoryButton(categoryId) { | |||||||
|     document.getElementById("addCategory").disabled = false; |     document.getElementById("addCategory").disabled = false; | ||||||
|   }) |   }) | ||||||
|   .catch(error => { |   .catch(error => { | ||||||
|     showErrorMessage("Failed to add category"); |     showErrorMessage(translate('failed_add_category')); | ||||||
|     document.getElementById("addCategory").disabled = false; |     document.getElementById("addCategory").disabled = false; | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
| @ -214,7 +214,7 @@ function removeCategory(categoryId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|     .then(response => { |     .then(response => { | ||||||
|       if (!response.ok) { |       if (!response.ok) { | ||||||
|         throw new Error('Network response was not ok'); |         throw new Error(translate('network_response_error')); | ||||||
|       } |       } | ||||||
|       return response.json(); |       return response.json(); | ||||||
|     }) |     }) | ||||||
| @ -224,13 +224,13 @@ function removeCategory(categoryId) { | |||||||
|       if (divToRemove) { |       if (divToRemove) { | ||||||
|         divToRemove.parentNode.removeChild(divToRemove); |         divToRemove.parentNode.removeChild(divToRemove); | ||||||
|       } |       } | ||||||
|       showSuccessMessage("Category removed"); |       showSuccessMessage(responseData.message); | ||||||
|      } else { |      } else { | ||||||
|       showErrorMessage(responseData.errorMessage || "Failed to remove category"); |       showErrorMessage(responseData.errorMessage || translate('failed_remove_category')); | ||||||
|      } |      } | ||||||
|     }) |     }) | ||||||
|     .catch(error => { |     .catch(error => { | ||||||
|       showErrorMessage("Failed to remove category"); |       showErrorMessage(translate('failed_remove_category')); | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -247,19 +247,19 @@ function editCategory(categoryId) { | |||||||
|       .then(response => { |       .then(response => { | ||||||
|         saveButton.classList.remove("disabled"); |         saveButton.classList.remove("disabled"); | ||||||
|         if (!response.ok) { |         if (!response.ok) { | ||||||
|           showErrorMessage("Failed to save category"); |           showErrorMessage(translate('failed_save_category')); | ||||||
|         } |         } | ||||||
|         return response.json(); |         return response.json(); | ||||||
|       }) |       }) | ||||||
|       .then(responseData => { |       .then(responseData => { | ||||||
|         if (responseData.success) { |         if (responseData.success) { | ||||||
|           showSuccessMessage("Category saved"); |           showSuccessMessage(responseData.message); | ||||||
|         } else { |         } else { | ||||||
|           showErrorMessage(responseData.errorMessage || "Failed to save category"); |           showErrorMessage(responseData.errorMessage || translate('failed_save_category')); | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|       .catch(error => { |       .catch(error => { | ||||||
|         showErrorMessage("Failed to save category"); |         showErrorMessage(translate('failed_save_category')); | ||||||
|       }); |       }); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -270,7 +270,7 @@ function addCurrencyButton(currencyId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|   .then(response => { |   .then(response => { | ||||||
|     if (!response.ok) { |     if (!response.ok) { | ||||||
|       throw new Error('Network response was not ok'); |       throw new Error(translate('network_response_error')); | ||||||
|       showErrorMessage(response.text()); |       showErrorMessage(response.text()); | ||||||
|     } |     } | ||||||
|     return response.text(); |     return response.text(); | ||||||
| @ -292,13 +292,13 @@ function addCurrencyButton(currencyId) { | |||||||
| 
 | 
 | ||||||
|       let inputName = document.createElement("input"); |       let inputName = document.createElement("input"); | ||||||
|       inputName.type = "text"; |       inputName.type = "text"; | ||||||
|       inputName.placeholder = "Currency"; |       inputName.placeholder = translate('currency'); | ||||||
|       inputName.name = "currency"; |       inputName.name = "currency"; | ||||||
|       inputName.value = "Currency"; |       inputName.value = translate('currency'); | ||||||
| 
 | 
 | ||||||
|       let inputCode = document.createElement("input"); |       let inputCode = document.createElement("input"); | ||||||
|       inputCode.type = "text"; |       inputCode.type = "text"; | ||||||
|       inputCode.placeholder = "Currency Code"; |       inputCode.placeholder = translate('currency_code'); | ||||||
|       inputCode.name = "code"; |       inputCode.name = "code"; | ||||||
|       inputCode.value = "CODE"; |       inputCode.value = "CODE"; | ||||||
| 
 | 
 | ||||||
| @ -311,7 +311,7 @@ function addCurrencyButton(currencyId) { | |||||||
| 
 | 
 | ||||||
|       let editImage = document.createElement("img"); |       let editImage = document.createElement("img"); | ||||||
|       editImage.src = "images/siteicons/save.png"; |       editImage.src = "images/siteicons/save.png"; | ||||||
|       editImage.title = "Save Currency"; |       editImage.title = translate('save_currency'); | ||||||
| 
 | 
 | ||||||
|       editLink.appendChild(editImage); |       editLink.appendChild(editImage); | ||||||
| 
 | 
 | ||||||
| @ -324,7 +324,7 @@ function addCurrencyButton(currencyId) { | |||||||
| 
 | 
 | ||||||
|       let deleteImage = document.createElement("img"); |       let deleteImage = document.createElement("img"); | ||||||
|       deleteImage.src = "images/siteicons/delete.png"; |       deleteImage.src = "images/siteicons/delete.png"; | ||||||
|       deleteImage.title = "Delete Currency"; |       deleteImage.title = translate('delete_currency'); | ||||||
| 
 | 
 | ||||||
|       deleteLink.appendChild(deleteImage); |       deleteLink.appendChild(deleteImage); | ||||||
| 
 | 
 | ||||||
| @ -352,23 +352,23 @@ function removeCurrency(currencyId) { | |||||||
|   fetch(url) |   fetch(url) | ||||||
|   .then(response => { |   .then(response => { | ||||||
|     if (!response.ok) { |     if (!response.ok) { | ||||||
|       throw new Error("There was an error removing the currency"); |       throw new Error(translate('network_response_error')); | ||||||
|     } |     } | ||||||
|     return response.json(); |     return response.json(); | ||||||
|   }) |   }) | ||||||
|   .then(data => { |   .then(data => { | ||||||
|     if (data.success) { |     if (data.success) { | ||||||
|       showSuccessMessage("Currency removed"); |       showSuccessMessage(data.message); | ||||||
|       let divToRemove = document.querySelector(`[data-currencyid="${currencyId}"]`); |       let divToRemove = document.querySelector(`[data-currencyid="${currencyId}"]`); | ||||||
|       if (divToRemove) { |       if (divToRemove) { | ||||||
|         divToRemove.parentNode.removeChild(divToRemove); |         divToRemove.parentNode.removeChild(divToRemove); | ||||||
|       } |       } | ||||||
|     } else { |     } else { | ||||||
|       showErrorMessage(data.message || "Failed to remove currency"); |       showErrorMessage(data.message || translate('failed_remove_currency')); | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
|   .catch(error => { |   .catch(error => { | ||||||
|     showErrorMessage(error.message || "There was an error removing the currency"); |     showErrorMessage(error.message || translate('failed_remove_currency')); | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -388,7 +388,7 @@ function editCurrency(currencyId) { | |||||||
|     fetch(url) |     fetch(url) | ||||||
|       .then(response => { |       .then(response => { | ||||||
|         if (!response.ok) { |         if (!response.ok) { | ||||||
|           throw new Error("There was an error saving the currency"); |           throw new Error(translate('network_response_error')); | ||||||
|         } |         } | ||||||
|         return response.json(); |         return response.json(); | ||||||
|       }) |       }) | ||||||
| @ -396,17 +396,17 @@ function editCurrency(currencyId) { | |||||||
|         if (data.success) { |         if (data.success) { | ||||||
|           saveButton.classList.remove("disabled"); |           saveButton.classList.remove("disabled"); | ||||||
|           saveButton.disabled = false; |           saveButton.disabled = false; | ||||||
|           showSuccessMessage(currencyName + " was saved"); |           showSuccessMessage(decodeURI(data.message)); | ||||||
|         } else { |         } else { | ||||||
|           saveButton.classList.remove("disabled"); |           saveButton.classList.remove("disabled"); | ||||||
|           saveButton.disabled = false; |           saveButton.disabled = false; | ||||||
|           showErrorMessage(data.message || "Failed to save currency"); |           showErrorMessage(data.message || translate('failed_save_currency')); | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|       .catch(error => { |       .catch(error => { | ||||||
|         saveButton.classList.remove("disabled"); |         saveButton.classList.remove("disabled"); | ||||||
|         saveButton.disabled = false; |         saveButton.disabled = false; | ||||||
|         showErrorMessage(error.message || "There was an error saving the currency"); |         showErrorMessage(error.message || translate('failed_save_currency')); | ||||||
|       }); |       }); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -415,7 +415,7 @@ function togglePayment(paymentId) { | |||||||
|     const element = document.querySelector(`div[data-paymentid="${paymentId}"]`); |     const element = document.querySelector(`div[data-paymentid="${paymentId}"]`); | ||||||
| 
 | 
 | ||||||
|     if (element.dataset.inUse === 'yes') { |     if (element.dataset.inUse === 'yes') { | ||||||
|       return showErrorMessage('Can\'t delete used payment method'); |       return showErrorMessage(translate(cant_disable_payment_in_use)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const newEnabledState = element.dataset.enabled === '1' ? '0' : '1'; |     const newEnabledState = element.dataset.enabled === '1' ? '0' : '1'; | ||||||
| @ -425,18 +425,18 @@ function togglePayment(paymentId) { | |||||||
| 
 | 
 | ||||||
|     fetch(url).then(response => { |     fetch(url).then(response => { | ||||||
|         if (!response.ok) { |         if (!response.ok) { | ||||||
|           throw new Error("There was an error saving the payments method"); |           throw new Error(translate('network_response_error')); | ||||||
|         } |         } | ||||||
|         return response.json(); |         return response.json(); | ||||||
|     }).then(data => { |     }).then(data => { | ||||||
|         if (data.success) { |         if (data.success) { | ||||||
|             element.dataset.enabled = newEnabledState; |             element.dataset.enabled = newEnabledState; | ||||||
|             showSuccessMessage(`${paymentMethodName} was saved`); |             showSuccessMessage(`${paymentMethodName} ${data.message}`); | ||||||
|         } else { |         } else { | ||||||
|             showErrorMessage(data.message || "Failed to save payments method"); |             showErrorMessage(data.message || translate('failed_save_payment_method')); | ||||||
|         } |         } | ||||||
|     }).catch(error => { |     }).catch(error => { | ||||||
|         showErrorMessage(error.message || "There was an error saving the payments method"); |         showErrorMessage(error.message || translate('failed_save_payment_method')); | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -457,14 +457,14 @@ document.addEventListener('DOMContentLoaded', function() { | |||||||
|             document.getElementById("avatar").src = "images/avatars/" + newAvatar + ".svg"; |             document.getElementById("avatar").src = "images/avatars/" + newAvatar + ".svg"; | ||||||
|             var newUsername = document.getElementById("username").value; |             var newUsername = document.getElementById("username").value; | ||||||
|             document.getElementById("user").textContent = newUsername; |             document.getElementById("user").textContent = newUsername; | ||||||
|             showSuccessMessage("User details saved"); |             showSuccessMessage(data.message); | ||||||
|           } else { |           } else { | ||||||
|             showErrorMessage(data.errorMessage); |             showErrorMessage(data.errorMessage); | ||||||
|           } |           } | ||||||
|           document.getElementById("userSubmit").disabled = false; |           document.getElementById("userSubmit").disabled = false; | ||||||
|         }) |         }) | ||||||
|         .catch(error => { |         .catch(error => { | ||||||
|           showErrorMessage("Unknown error, please try again"); |           showErrorMessage(translate('unknown_error')); | ||||||
|         }); |         }); | ||||||
|       });         |       });         | ||||||
| 
 | 
 | ||||||
| @ -484,7 +484,7 @@ function addFixerKeyButton() { | |||||||
|   .then(response => response.json()) |   .then(response => response.json()) | ||||||
|   .then(data => { |   .then(data => { | ||||||
|       if (data.success) { |       if (data.success) { | ||||||
|           showSuccessMessage("API key saved successfully"); |           showSuccessMessage(data.message); | ||||||
|           document.getElementById("addFixerKey").disabled = false; |           document.getElementById("addFixerKey").disabled = false; | ||||||
|       } else { |       } else { | ||||||
|           showErrorMessage(data.message); |           showErrorMessage(data.message); | ||||||
| @ -529,14 +529,14 @@ function saveNotificationsButton() { | |||||||
|   .then(response => response.json()) |   .then(response => response.json()) | ||||||
|   .then(data => { |   .then(data => { | ||||||
|       if (data.success) { |       if (data.success) { | ||||||
|           showSuccessMessage("Notification settings saved successfully."); |           showSuccessMessage(data.message); | ||||||
|       } else { |       } else { | ||||||
|           showErrorMessage(data.errorMessage); |           showErrorMessage(data.errorMessage); | ||||||
|       } |       } | ||||||
|       button.disabled = false; |       button.disabled = false; | ||||||
|   }) |   }) | ||||||
|   .catch(error => { |   .catch(error => { | ||||||
|       showErrorMessage("Error saving notification data"); |       showErrorMessage(translate('error_saving_notification_data')); | ||||||
|       button.disabled = false; |       button.disabled = false; | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
| @ -569,14 +569,14 @@ function testNotificationButton()  { | |||||||
|   .then(response => response.json()) |   .then(response => response.json()) | ||||||
|   .then(data => { |   .then(data => { | ||||||
|       if (data.success) { |       if (data.success) { | ||||||
|           showSuccessMessage("Notification sent successfully."); |           showSuccessMessage(data.message); | ||||||
|       } else { |       } else { | ||||||
|           showErrorMessage(data.errorMessage); |           showErrorMessage(data.errorMessage); | ||||||
|       } |       } | ||||||
|       button.disabled = false; |       button.disabled = false; | ||||||
|   }) |   }) | ||||||
|   .catch(error => { |   .catch(error => { | ||||||
|       showErrorMessage("Error sending notification"); |       showErrorMessage(translate('error_sending_notification')); | ||||||
|       button.disabled = false; |       button.disabled = false; | ||||||
|   }); |   }); | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										119
									
								
								settings.php
									
									
									
									
									
								
							
							
						
						
									
										119
									
								
								settings.php
									
									
									
									
									
								
							| @ -4,7 +4,7 @@ | |||||||
| <section class="contain settings"> | <section class="contain settings"> | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>User details</h2> |             <h2><?= translate('user_details', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|             <form action="endpoints/user/saveuser.php" method="post" id="userForm"> |             <form action="endpoints/user/saveuser.php" method="post" id="userForm"> | ||||||
|                 <div class="user-form"> |                 <div class="user-form"> | ||||||
| @ -35,19 +35,19 @@ | |||||||
|                         </div> |                         </div> | ||||||
|                         <div class="grow"> |                         <div class="grow"> | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                             <label for="username">Username:</label> |                                 <label for="username"><?= translate('username', $i18n) ?>:</label>
 | ||||||
|                                 <input type="text" id="username" name="username" value="<?= $userData['username'] ?>" required> |                                 <input type="text" id="username" name="username" value="<?= $userData['username'] ?>" required> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                             <label for="email">Email:</label> |                                 <label for="email"><?= translate('email', $i18n) ?>:</label>
 | ||||||
|                                 <input type="email" id="email" name="email" value="<?= $userData['email'] ?>" required> |                                 <input type="email" id="email" name="email" value="<?= $userData['email'] ?>" required> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                             <label for="password">Password:</label> |                                 <label for="password"><?= translate('password', $i18n) ?>:</label>
 | ||||||
|                                 <input type="password" id="password" name="password"> |                                 <input type="password" id="password" name="password"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                             <label for="confirm_password">Confirm Password:</label> |                                 <label for="confirm_password"><?= translate('confirm_password', $i18n) ?>:</label>
 | ||||||
|                                 <input type="password" id="confirm_password" name="confirm_password"> |                                 <input type="password" id="confirm_password" name="confirm_password"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <?php |                             <?php | ||||||
| @ -60,7 +60,7 @@ | |||||||
|                                 } |                                 } | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                             <label for="currency">Main Currency:</label> |                                 <label for="currency"><?= translate('main_currency', $i18n) ?>:</label>
 | ||||||
|                                 <select id="currency" name="main_currency" placeholder="Currency"> |                                 <select id="currency" name="main_currency" placeholder="Currency"> | ||||||
|                                 <?php |                                 <?php | ||||||
|                                     foreach ($currencies as $currency) { |                                     foreach ($currencies as $currency) { | ||||||
| @ -72,10 +72,23 @@ | |||||||
|                                 ?>
 |                                 ?>
 | ||||||
|                                 </select> |                                 </select> | ||||||
|                             </div> |                             </div> | ||||||
|  |                             <div class="form-group"> | ||||||
|  |                                 <label for="language"><?= translate('language', $i18n) ?>:</label>
 | ||||||
|  |                                 <select id="language" name="language" placeholder="Language"> | ||||||
|  |                                 <?php  | ||||||
|  |                                     foreach ($languages as $code => $name) { | ||||||
|  |                                         $selected = ($code === $lang) ? 'selected' : ''; | ||||||
|  |                                 ?>
 | ||||||
|  |                                         <option value="<?= $code ?>" <?= $selected ?>><?= $name ?></option>
 | ||||||
|  |                                 <?php | ||||||
|  |                                     } | ||||||
|  |                                 ?>
 | ||||||
|  |                                 </select> | ||||||
|  |                             </div> | ||||||
|                         </div>   |                         </div>   | ||||||
|                     </div> |                     </div> | ||||||
|                     <div class="buttons"> |                     <div class="buttons"> | ||||||
|                         <input type="submit" value="Save" id="userSubmit"/> |                         <input type="submit" value="<?= translate('save', $i18n) ?>" id="userSubmit"/> | ||||||
|                     </div> |                     </div> | ||||||
|                 </div> |                 </div> | ||||||
|             </form> |             </form> | ||||||
| @ -96,7 +109,7 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Household</h2> |             <h2><?= translate('household', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-members"> |         <div class="account-members"> | ||||||
|             <div  id="householdMembers"> |             <div  id="householdMembers"> | ||||||
| @ -106,19 +119,19 @@ | |||||||
|                     <div class="form-group-inline" data-memberid="<?= $member['id'] ?>"> |                     <div class="form-group-inline" data-memberid="<?= $member['id'] ?>"> | ||||||
|                         <input type="text" name="member" value="<?= $member['name'] ?>" placeholder="Member"> |                         <input type="text" name="member" value="<?= $member['name'] ?>" placeholder="Member"> | ||||||
|                         <button class="image-button medium"  onClick="editMember(<?= $member['id'] ?>)" name="save"> |                         <button class="image-button medium"  onClick="editMember(<?= $member['id'] ?>)" name="save"> | ||||||
|                             <img src="images/siteicons/save.png" title="Save Member"> |                             <img src="images/siteicons/save.png" title="<?= translate('save_member', $i18n) ?>"> | ||||||
|                         </button> |                         </button> | ||||||
|                         <?php |                         <?php | ||||||
|                             if ($member['id'] != 1) { |                             if ($member['id'] != 1) { | ||||||
|                                 ?>
 |                                 ?>
 | ||||||
|                                     <button class="image-button medium" onClick="removeMember(<?= $member['id'] ?>)"> |                                     <button class="image-button medium" onClick="removeMember(<?= $member['id'] ?>)"> | ||||||
|                                         <img src="images/siteicons/delete.png" title="Delete Member"> |                                         <img src="images/siteicons/delete.png" title="<?= translate('delete_member', $i18n) ?>"> | ||||||
|                                     </button> |                                     </button> | ||||||
|                                 <?php |                                 <?php | ||||||
|                             } else { |                             } else { | ||||||
|                                 ?>
 |                                 ?>
 | ||||||
|                                     <button class="image-button medium disabled"> |                                     <button class="image-button medium disabled"> | ||||||
|                                         <img src="images/siteicons/delete.png" title="Can't delete main member"> |                                         <img src="images/siteicons/delete.png" title="<?= translate('cant_delete_member', $i18n) ?>"> | ||||||
|                                     </button> |                                     </button> | ||||||
|                                 <?php |                                 <?php | ||||||
|                             } |                             } | ||||||
| @ -129,7 +142,7 @@ | |||||||
|             ?>
 |             ?>
 | ||||||
|             </div> |             </div> | ||||||
|             <div class="buttons"> |             <div class="buttons"> | ||||||
|                 <input type="submit" value="Add" id="addMember" onClick="addMemberButton()"/> |                 <input type="submit" value="<?= translate('add', $i18n) ?>" id="addMember" onClick="addMemberButton()"/> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </section> |     </section> | ||||||
| @ -158,23 +171,23 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Notifications</h2> |             <h2><?= translate('notifications', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-notifications"> |         <div class="account-notifications"> | ||||||
|             <div class="form-group-inline"> |             <div class="form-group-inline"> | ||||||
|                 <input type="checkbox" id="notifications" name="notifications" <?= $notifications['enabled'] ? "checked" : "" ?>>
 |                 <input type="checkbox" id="notifications" name="notifications" <?= $notifications['enabled'] ? "checked" : "" ?>>
 | ||||||
|                 <label for="notifications">Enable email notifications</label> |                 <label for="notifications"><?= translate('enable_email_notifications', $i18n) ?></label>
 | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-group"> |             <div class="form-group"> | ||||||
|                 <label for="days">Notify me: </label> |                 <label for="days"><?= translate('notify_me', $i18n) ?>:</label>
 | ||||||
|                 <select name="days" id="days"> |                 <select name="days" id="days"> | ||||||
|                 <?php |                 <?php | ||||||
|                     for ($i = 1; $i <= 7; $i++) { |                     for ($i = 1; $i <= 7; $i++) { | ||||||
|                         $dayText = $i > 1 ? "days" : "day"; |                         $dayText = $i > 1 ? translate('days_before', $i18n) : translate('day_before', $i18n); | ||||||
|                         $selected = $i == $notifications['days'] ? "selected" : ""; |                         $selected = $i == $notifications['days'] ? "selected" : ""; | ||||||
|                         ?>
 |                         ?>
 | ||||||
|                             <option value="<?= $i ?>" <?= $selected ?>>
 |                             <option value="<?= $i ?>" <?= $selected ?>>
 | ||||||
|                                 <?= $i ?> <?= $dayText ?> before
 |                                 <?= $i ?> <?= $dayText ?>
 | ||||||
|                             </option> |                             </option> | ||||||
|                         <?php |                         <?php | ||||||
|                     } |                     } | ||||||
| @ -182,27 +195,26 @@ | |||||||
|                 </select> |                 </select> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-group-inline"> |             <div class="form-group-inline"> | ||||||
|                 <input type="text" name="smtpaddress" id="smtpaddress" placeholder="SMTP Address" value="<?= $notifications['smtp_address'] ?>" /> |                 <input type="text" name="smtpaddress" id="smtpaddress" placeholder="<?= translate('smtp_address', $i18n) ?>" value="<?= $notifications['smtp_address'] ?>" /> | ||||||
|                 <input type="text" name="smtpport" id="smtpport" placeholder="Port" class="one-third"  value="<?= $notifications['smtp_port'] ?>" /> |                 <input type="text" name="smtpport" id="smtpport" placeholder="<?= translate('port', $i18n) ?>" class="one-third"  value="<?= $notifications['smtp_port'] ?>" /> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-group-inline"> |             <div class="form-group-inline"> | ||||||
|                 <input type="text" name="smtpusername" id="smtpusername" placeholder="SMTP Username"  value="<?= $notifications['smtp_username'] ?>" /> |                 <input type="text" name="smtpusername" id="smtpusername" placeholder="<?= translate('smtp_username', $i18n) ?>"  value="<?= $notifications['smtp_username'] ?>" /> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-group-inline"> |             <div class="form-group-inline"> | ||||||
|                 <input type="password" name="smtppassword" id="smtppassword" placeholder="SMTP Password"  value="<?= $notifications['smtp_password'] ?>" /> |                 <input type="password" name="smtppassword" id="smtppassword" placeholder="<?= translate('smtp_password', $i18n) ?>"  value="<?= $notifications['smtp_password'] ?>" /> | ||||||
|             </div> |             </div> | ||||||
|             <div class="form-group-inline"> |             <div class="form-group-inline"> | ||||||
|                 <input type="text" name="fromemail" id="fromemail" placeholder="From email (Optional)"  value="<?= $notifications['from_email'] ?>" /> |                 <input type="text" name="fromemail" id="fromemail" placeholder="From email (Optional)"  value="<?= $notifications['from_email'] ?>" /> | ||||||
|             </div> |             </div> | ||||||
|             <div class="settings-notes"> |             <div class="settings-notes"> | ||||||
|                 <p> |                 <p> | ||||||
|                     <i class="fa-solid fa-circle-info"></i> SMTP Password is transmitted and stored in plaintext.  |                     <i class="fa-solid fa-circle-info"></i> <?= translate('smtp_info', $i18n) ?></p>
 | ||||||
|                     For security, please create an account just for this.</p> |  | ||||||
|                 <p> |                 <p> | ||||||
|             </div> |             </div> | ||||||
|             <div class="buttons"> |             <div class="buttons"> | ||||||
|                 <input type="button" class="secondary-button" value="Test" id="testNotifications" onClick="testNotificationButton()"/> |                 <input type="button" class="secondary-button" value="<?= translate('test', $i18n) ?>" id="testNotifications" onClick="testNotificationButton()"/> | ||||||
|                 <input type="submit" value="Save" id="saveNotifications" onClick="saveNotificationsButton()"/> |                 <input type="submit" value="<?= translate('save', $i18n) ?>" id="saveNotifications" onClick="saveNotificationsButton()"/> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </section> |     </section> | ||||||
| @ -221,7 +233,7 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Categories</h2> |             <h2><?= translate('categories', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-categories"> |         <div class="account-categories"> | ||||||
|             <div  id="categories"> |             <div  id="categories"> | ||||||
| @ -244,19 +256,19 @@ | |||||||
|                     <div class="form-group-inline" data-categoryid="<?= $category['id'] ?>"> |                     <div class="form-group-inline" data-categoryid="<?= $category['id'] ?>"> | ||||||
|                         <input type="text" name="category" value="<?= $category['name'] ?>" placeholder="Category"> |                         <input type="text" name="category" value="<?= $category['name'] ?>" placeholder="Category"> | ||||||
|                         <button class="image-button medium"  onClick="editCategory(<?= $category['id'] ?>)" name="save"> |                         <button class="image-button medium"  onClick="editCategory(<?= $category['id'] ?>)" name="save"> | ||||||
|                             <img src="images/siteicons/save.png" title="Save Category"> |                             <img src="images/siteicons/save.png" title="<?= translate('save_category', $i18n) ?>"> | ||||||
|                         </button> |                         </button> | ||||||
|                         <?php |                         <?php | ||||||
|                             if ($canDelete) { |                             if ($canDelete) { | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                                 <button class="image-button medium" onClick="removeCategory(<?= $category['id'] ?>)"> |                                 <button class="image-button medium" onClick="removeCategory(<?= $category['id'] ?>)"> | ||||||
|                                     <img src="images/siteicons/delete.png" title="Delete Category"> |                                     <img src="images/siteicons/delete.png" title="<?= translate('delete_category', $i18n) ?>"> | ||||||
|                                 </button> |                                 </button> | ||||||
|                             <?php |                             <?php | ||||||
|                             } else { |                             } else { | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                                 <button class="image-button medium disabled"> |                                 <button class="image-button medium disabled"> | ||||||
|                                     <img src="images/siteicons/delete.png" title="Can't delete category in use in subscription"> |                                     <img src="images/siteicons/delete.png" title="<?= translate('cant_delete_category_in_use', $i18n) ?>"> | ||||||
|                                 </button> |                                 </button> | ||||||
|                             <?php |                             <?php | ||||||
|                             } |                             } | ||||||
| @ -268,7 +280,7 @@ | |||||||
|             ?>
 |             ?>
 | ||||||
|             </div> |             </div> | ||||||
|             <div class="buttons"> |             <div class="buttons"> | ||||||
|                 <input type="submit" value="Add" id="addCategory" onClick="addCategoryButton()"/> |                 <input type="submit" value="<?= translate('add', $i18n) ?>" id="addCategory" onClick="addCategoryButton()"/> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </section> |     </section> | ||||||
| @ -309,7 +321,7 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Currencies</h2> |             <h2><?= translate('currencies', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-currencies"> |         <div class="account-currencies"> | ||||||
|             <div id="currencies"> |             <div id="currencies"> | ||||||
| @ -339,20 +351,20 @@ | |||||||
|                         <input type="text" name="currency" value="<?= $currency['name'] ?>" placeholder="Currency Name"> |                         <input type="text" name="currency" value="<?= $currency['name'] ?>" placeholder="Currency Name"> | ||||||
|                         <input type="text" name="code" value="<?= $currency['code'] ?>" placeholder="Currency Code"> |                         <input type="text" name="code" value="<?= $currency['code'] ?>" placeholder="Currency Code"> | ||||||
|                         <button class="image-button medium"  onClick="editCurrency(<?= $currency['id'] ?>)" name="save"> |                         <button class="image-button medium"  onClick="editCurrency(<?= $currency['id'] ?>)" name="save"> | ||||||
|                             <img src="images/siteicons/save.png" title="Save Currency"> |                             <img src="images/siteicons/save.png" title="<?= translate('save_currency', $i18n) ?>"> | ||||||
|                         </button> |                         </button> | ||||||
|                         <?php |                         <?php | ||||||
|                             if ($canDelete) { |                             if ($canDelete) { | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                                 <button class="image-button medium" onClick="removeCurrency(<?= $currency['id'] ?>)"> |                                 <button class="image-button medium" onClick="removeCurrency(<?= $currency['id'] ?>)"> | ||||||
|                                     <img src="images/siteicons/delete.png" title="Delete Currency"> |                                     <img src="images/siteicons/delete.png" title="<?= translate('delete_currency', $i18n) ?>"> | ||||||
|                                 </button> |                                 </button> | ||||||
|                             <?php |                             <?php | ||||||
|                             } else { |                             } else { | ||||||
|                                 $cantDeleteMessage = $isMainCurrency ? "main currency" : "used currency"; |                                 $cantDeleteMessage = $isMainCurrency ? translate('cant_delete_main_currency', $i18n) : translate('cant_delete_currency_in_use', $i18n); | ||||||
|                             ?>
 |                             ?>
 | ||||||
|                                 <button class="image-button medium disabled"> |                                 <button class="image-button medium disabled"> | ||||||
|                                     <img src="images/siteicons/delete.png" title="Can't delete <?= $cantDeleteMessage ?>"> |                                     <img src="images/siteicons/delete.png" title="<?= $cantDeleteMessage ?>"> | ||||||
|                                 </button> |                                 </button> | ||||||
|                             <?php |                             <?php | ||||||
|                             } |                             } | ||||||
| @ -364,19 +376,19 @@ | |||||||
|             ?>
 |             ?>
 | ||||||
|             </div> |             </div> | ||||||
|             <div class="buttons"> |             <div class="buttons"> | ||||||
|                 <input type="submit" value="Add" id="addCurrency" onClick="addCurrencyButton()"/> |                 <input type="submit" value="<?= translate('add', $i18n) ?>" id="addCurrency" onClick="addCurrencyButton()"/> | ||||||
|             </div> |             </div> | ||||||
|             <div class="settings-notes"> |             <div class="settings-notes"> | ||||||
|                 <p> |                 <p> | ||||||
|                     <i class="fa-solid fa-circle-info"></i> |                     <i class="fa-solid fa-circle-info"></i> | ||||||
|                     Exchange rates last updated on  |                     <?= translate('exchange_update', $i18n) ?>
 | ||||||
|                     <span> |                     <span> | ||||||
|                         <?= $exchange_rates_last_updated ?>
 |                         <?= $exchange_rates_last_updated ?>
 | ||||||
|                     </span> |                     </span> | ||||||
|                 </p> |                 </p> | ||||||
|                 <p> |                 <p> | ||||||
|                     <i class="fa-solid fa-circle-info"></i> |                     <i class="fa-solid fa-circle-info"></i> | ||||||
|                     Find the supported currencies and correct currency codes on  |                     <?= translate('currency_info', $i18n) ?>
 | ||||||
|                     <span> |                     <span> | ||||||
|                         fixer.io  |                         fixer.io  | ||||||
|                         <a href="https://fixer.io/symbols" target="_blank" title="Currency codes"> |                         <a href="https://fixer.io/symbols" target="_blank" title="Currency codes"> | ||||||
| @ -385,7 +397,7 @@ | |||||||
|                     </span> |                     </span> | ||||||
|                 </p> |                 </p> | ||||||
|                 <p> |                 <p> | ||||||
|                     For improved performance keep only the currencies you use. |                     <?= translate('currency_performance', $i18n) ?>
 | ||||||
|                 </p> |                 </p> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @ -409,12 +421,11 @@ | |||||||
|         </header> |         </header> | ||||||
|         <div class="account-fixer"> |         <div class="account-fixer"> | ||||||
|             <div class="form-group"> |             <div class="form-group"> | ||||||
|                 <input type="text" name="fixer-key" id="fixerKey" value="<?= $apiKey ?>" placeholder="ApiKey"> |                 <input type="text" name="fixer-key" id="fixerKey" value="<?= $apiKey ?>" placeholder="<?= translate('api_key', $i18n) ?>"> | ||||||
|             </div> |             </div> | ||||||
|             <div class="settings-notes"> |             <div class="settings-notes"> | ||||||
|                 <p><i class="fa-solid fa-circle-info"></i> If you use multiple currencies, and want accurate statistics and sorting on the subscriptions,  |                 <p><i class="fa-solid fa-circle-info"></i><?= translate('fixer_info', $i18n) ?></p>
 | ||||||
|                     a FREE API Key from Fixer is necessary.</p> |                 <p><?= translate('get_key', $i18n) ?>: 
 | ||||||
|                 <p>Get your key at:  |  | ||||||
|                     <span> |                     <span> | ||||||
|                         https://fixer.io/  |                         https://fixer.io/  | ||||||
|                         <a href="https://fixer.io/#pricing_plan" title="Get free fixer api key" target="_blank"> |                         <a href="https://fixer.io/#pricing_plan" title="Get free fixer api key" target="_blank"> | ||||||
| @ -424,18 +435,18 @@ | |||||||
|                 </p>     |                 </p>     | ||||||
|             </div> |             </div> | ||||||
|             <div class="buttons"> |             <div class="buttons"> | ||||||
|                 <input type="submit" value="Save" id="addFixerKey" onClick="addFixerKeyButton()"/> |                 <input type="submit" value="<?= translate('save', $i18n) ?>" id="addFixerKey" onClick="addFixerKeyButton()"/> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|     </section>     |     </section>     | ||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Display settings</h2> |             <h2><?= translate('display_settings', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-settings-list"> |         <div class="account-settings-list"> | ||||||
|             <div> |             <div> | ||||||
|                 <input type="button" value="Switch Light / Dark Theme" onClick="switchTheme()"> |                 <input type="button" value="<?= translate('switch_theme', $i18n) ?>" onClick="switchTheme()"> | ||||||
|             </div> |             </div> | ||||||
|             <?php |             <?php | ||||||
|                 $monthlyprice = isset($_COOKIE['showMonthlyPrice']) && $_COOKIE['showMonthlyPrice'] === 'true'; |                 $monthlyprice = isset($_COOKIE['showMonthlyPrice']) && $_COOKIE['showMonthlyPrice'] === 'true'; | ||||||
| @ -445,13 +456,13 @@ | |||||||
|             <div> |             <div> | ||||||
|                 <div class="form-group-inline"> |                 <div class="form-group-inline"> | ||||||
|                     <input type="checkbox" id="monthlyprice" name="monthlyprice" onChange="setShowMonthlyPriceCookie()" <?php if ($monthlyprice) echo 'checked'; ?>>
 |                     <input type="checkbox" id="monthlyprice" name="monthlyprice" onChange="setShowMonthlyPriceCookie()" <?php if ($monthlyprice) echo 'checked'; ?>>
 | ||||||
|                     <label for="monthlyprice">Calculate and show monthly price for all subscriptions</label> |                     <label for="monthlyprice"><?= translate('calculate_monthly_price', $i18n) ?></label>
 | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|             <div> |             <div> | ||||||
|                 <div class="form-group-inline"> |                 <div class="form-group-inline"> | ||||||
|                     <input type="checkbox" id="convertcurrency" name="convertcurrency" onChange="setConvertCurrencyCookie()" <?php if ($convertcurrency) echo 'checked'; ?>>
 |                     <input type="checkbox" id="convertcurrency" name="convertcurrency" onChange="setConvertCurrencyCookie()" <?php if ($convertcurrency) echo 'checked'; ?>>
 | ||||||
|                     <label for="convertcurrency">Always convert and show prices on my main currency (slower).</label> |                     <label for="convertcurrency"><?= translate('convert_prices', $i18n) ?></label>
 | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| @ -459,27 +470,27 @@ | |||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Experimental settings</h2> |             <h2><?= translate('experimental_settings', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="account-settings-list"> |         <div class="account-settings-list"> | ||||||
|             <div> |             <div> | ||||||
|                 <div class="form-group-inline"> |                 <div class="form-group-inline"> | ||||||
|                     <input type="checkbox" id="removebackground" name="removebackground" onChange="setRemoveBackgroundCookie()" <?php if ($removebackground) echo 'checked'; ?>>
 |                     <input type="checkbox" id="removebackground" name="removebackground" onChange="setRemoveBackgroundCookie()" <?php if ($removebackground) echo 'checked'; ?>>
 | ||||||
|                     <label for="removebackground">Attempt to remove background of logos from image search (experimental).</label> |                     <label for="removebackground"><?= translate('remove_background', $i18n) ?></label>
 | ||||||
|                 </div> |                 </div> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="settings-notes"> |         <div class="settings-notes"> | ||||||
|             <p> |             <p> | ||||||
|                 <i class="fa-solid fa-circle-info"></i> |                 <i class="fa-solid fa-circle-info"></i> | ||||||
|                 Experimental settings will probably not work perfectly. |                 <?= translate('experimental_info', $i18n) ?>
 | ||||||
|             </p> |             </p> | ||||||
|         </div> |         </div> | ||||||
|     </section>     |     </section>     | ||||||
| 
 | 
 | ||||||
|     <section class="account-section"> |     <section class="account-section"> | ||||||
|         <header> |         <header> | ||||||
|             <h2>Payment Methods</h2> |             <h2><?= translate('payment_methods', $i18n) ?></h2>
 | ||||||
|         </header> |         </header> | ||||||
|         <div class="payments-list"> |         <div class="payments-list"> | ||||||
|             <?php |             <?php | ||||||
| @ -496,7 +507,7 @@ | |||||||
|                              data-enabled="<?= $payment['enabled']; ?>" |                              data-enabled="<?= $payment['enabled']; ?>" | ||||||
|                              data-in-use="<?= $inUse ? 'yes' : 'no' ?>" |                              data-in-use="<?= $inUse ? 'yes' : 'no' ?>" | ||||||
|                              data-paymentid="<?= $payment['id'] ?>" |                              data-paymentid="<?= $payment['id'] ?>" | ||||||
|                              title="<?= $inUse ? 'Can\'t delete used payment method' : '' ?>" |                              title="<?= $inUse ? translate('cant_delete_payment_method_in_use', $i18n) : ($payment['enabled'] ? translate('disable', $i18n) : translate('enable', $i18n)) ?>" | ||||||
|                              onClick="togglePayment(<?= $payment['id'] ?>)"> |                              onClick="togglePayment(<?= $payment['id'] ?>)"> | ||||||
|                             <img src="images/uploads/icons/<?= $payment['icon'] ?>"  alt="Logo" /> |                             <img src="images/uploads/icons/<?= $payment['icon'] ?>"  alt="Logo" /> | ||||||
|                             <span class="payment-name"> |                             <span class="payment-name"> | ||||||
| @ -510,7 +521,7 @@ | |||||||
|         <div class="settings-notes"> |         <div class="settings-notes"> | ||||||
|             <p> |             <p> | ||||||
|                 <i class="fa-solid fa-circle-info"></i> |                 <i class="fa-solid fa-circle-info"></i> | ||||||
|                 Click a payment method to disable / enable it. |                 <?= translate('payment_methods_info', $i18n) ?>
 | ||||||
|             </p> |             </p> | ||||||
|         </div> |         </div> | ||||||
|     </section> |     </section> | ||||||
|  | |||||||
							
								
								
									
										24
									
								
								stats.php
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								stats.php
									
									
									
									
									
								
							| @ -144,31 +144,31 @@ if ($result) { | |||||||
|   |   | ||||||
| ?>
 | ?>
 | ||||||
| <section class="contain"> | <section class="contain"> | ||||||
|   <h2>General Statistics</h2> |   <h2><?= translate('general_statistics', $i18n) ?></h2>
 | ||||||
|   <div class="statistics"> |   <div class="statistics"> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= $activeSubscriptions ?></span>
 |       <span><?= $activeSubscriptions ?></span>
 | ||||||
|       <div class="title">Active Subscriptions</div> |       <div class="title"><?= translate('active_subscriptions', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= CurrencyFormatter::format($totalCostPerMonth, $code) ?></span>
 |       <span><?= CurrencyFormatter::format($totalCostPerMonth, $code) ?></span>
 | ||||||
|       <div class="title">Monthly Cost</div> |       <div class="title"><?= translate('monthly_cost', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= CurrencyFormatter::format($totalCostPerYear, $code) ?></span>
 |       <span><?= CurrencyFormatter::format($totalCostPerYear, $code) ?></span>
 | ||||||
|       <div class="title">Yearly Cost</div> |       <div class="title"><?= translate('yearly_cost', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= CurrencyFormatter::format($averageSubscriptionCost, $code) ?></span>
 |       <span><?= CurrencyFormatter::format($averageSubscriptionCost, $code) ?></span>
 | ||||||
|       <div class="title">Average Monthly Subscription Cost</div> |       <div class="title"><?= translate('average_monthly', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= CurrencyFormatter::format($mostExpensiveSubscription, $code) ?></span>
 |       <span><?= CurrencyFormatter::format($mostExpensiveSubscription, $code) ?></span>
 | ||||||
|       <div class="title">Most Expensive Subscription Cost</div> |       <div class="title"><?= translate('most_expensive', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <div class="statistic"> |     <div class="statistic"> | ||||||
|       <span><?= CurrencyFormatter::format($amountDueThisMonth, $code) ?></span>
 |       <span><?= CurrencyFormatter::format($amountDueThisMonth, $code) ?></span>
 | ||||||
|       <div class="title">Amount due this month</div> |       <div class="title"><?= translate('amount_due', $i18n) ?></div>
 | ||||||
|     </div> |     </div> | ||||||
|     <?php |     <?php | ||||||
|       $numberOfElements = 6; |       $numberOfElements = 6; | ||||||
| @ -179,7 +179,7 @@ if ($result) { | |||||||
|       } |       } | ||||||
|     ?>  
 |     ?>  
 | ||||||
|   </div> |   </div> | ||||||
|   <h2>Split Views</h2> |   <h2><?= translate('split_views', $i18n) ?></h2>
 | ||||||
|   <div class="graphs"> |   <div class="graphs"> | ||||||
|       <?php |       <?php | ||||||
|         $categoryDataPoints = []; |         $categoryDataPoints = []; | ||||||
| @ -211,8 +211,8 @@ if ($result) { | |||||||
|           ?>
 |           ?>
 | ||||||
|           <section class="graph"> |           <section class="graph"> | ||||||
|             <header> |             <header> | ||||||
|               Household Split |               <?= translate('household_split', $i18n) ?>
 | ||||||
|               <div class="sub-header">(Monthly cost)</div> |               <div class="sub-header">(<?= translate('monthly_cost', $i18n) ?>)</div>
 | ||||||
|             </header> |             </header> | ||||||
|             <canvas id="memberSplitChart"></canvas> |             <canvas id="memberSplitChart"></canvas> | ||||||
|         </section> |         </section> | ||||||
| @ -223,8 +223,8 @@ if ($result) { | |||||||
|           ?>
 |           ?>
 | ||||||
|           <section class="graph"> |           <section class="graph"> | ||||||
|             <header> |             <header> | ||||||
|               Category Split |               <?= translate('category_split', $i18n) ?>
 | ||||||
|               <div class="sub-header">(Monthly cost)</div> |               <div class="sub-header">(<?= translate('monthly_cost', $i18n) ?>)</div>
 | ||||||
|             </header> |             </header> | ||||||
|             <canvas id="categorySplitChart" style="height: 370px; width: 100%;"></canvas> |             <canvas id="categorySplitChart" style="height: 370px; width: 100%;"></canvas> | ||||||
|           </section> |           </section> | ||||||
|  | |||||||
| @ -87,6 +87,7 @@ header .logo .logo-image { | |||||||
|   min-width: 130px; |   min-width: 130px; | ||||||
|   box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); |   box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); | ||||||
|   z-index: 5; |   z-index: 5; | ||||||
|  |   width: max-content; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .dropdown-content a { | .dropdown-content a { | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user