forked from mirrors/amnezia-wg-easy
		
	Add Darkmode
This commit is contained in:
		
							parent
							
								
									e967522f88
								
							
						
					
					
						commit
						042168f0bb
					
				
					 12 changed files with 2566 additions and 206 deletions
				
			
		
							
								
								
									
										1718
									
								
								src/www/css/app.css
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1718
									
								
								src/www/css/app.css
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										1
									
								
								src/www/css/vendor/tailwind.min.css
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								src/www/css/vendor/tailwind.min.css
									
										
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| 
						 | 
				
			
			@ -3,29 +3,57 @@
 | 
			
		|||
 | 
			
		||||
<head>
 | 
			
		||||
  <title>WireGuard</title>
 | 
			
		||||
  <link href="./css/vendor/tailwind.min.css" rel="stylesheet">
 | 
			
		||||
  <script src="https://cdn.tailwindcss.com"></script>
 | 
			
		||||
  <script>
 | 
			
		||||
    tailwind.config = {
 | 
			
		||||
      darkMode: 'class',
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (localStorage.theme === 'dark' || (!('theme' in localStorage) && window.matchMedia('(prefers-color-scheme: dark)').matches)) {
 | 
			
		||||
      document.documentElement.classList.add('dark')
 | 
			
		||||
    } else {
 | 
			
		||||
      document.documentElement.classList.remove('dark')
 | 
			
		||||
    }
 | 
			
		||||
  </script>
 | 
			
		||||
  <!-- <link href="./css/vendor/tailwind.min.css" rel="stylesheet"> -->
 | 
			
		||||
  <link rel="manifest" href="./manifest.json">
 | 
			
		||||
  <link rel="icon" type="image/png" href="./img/favicon.png">
 | 
			
		||||
  <link rel="apple-touch-icon" href="./img/apple-touch-icon.png">
 | 
			
		||||
  <meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover">
 | 
			
		||||
  <meta name="apple-mobile-web-app-capable" content="yes">
 | 
			
		||||
</head>
 | 
			
		||||
 | 
			
		||||
<style>
 | 
			
		||||
  [v-cloak] {
 | 
			
		||||
    display: none;
 | 
			
		||||
  }
 | 
			
		||||
</style>
 | 
			
		||||
 | 
			
		||||
<body class="bg-gray-50">
 | 
			
		||||
<body class="bg-gray-50 dark:bg-neutral-800">
 | 
			
		||||
 | 
			
		||||
  <div id="app">
 | 
			
		||||
 | 
			
		||||
    <div v-cloak class="container mx-auto max-w-3xl">
 | 
			
		||||
    <div class="flex justify-end">
 | 
			
		||||
      <button v-cloak id="theme-toggle" @click="toggleTheme"
 | 
			
		||||
        class="mt-5 mr-5 text-gray-500 dark:text-neutral-200 bg-gray-200 dark:bg-neutral-700 hover:bg-gray-300 dark:hover:bg-neutral-600 focus:outline-none rounded-lg text-sm p-2.5 transition">
 | 
			
		||||
        <svg id="theme-toggle-dark-icon" :class="{ hidden: isDark }" class="w-5 h-5" fill="currentColor"
 | 
			
		||||
          viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
          <path d="M17.293 13.293A8 8 0 016.707 2.707a8.001 8.001 0 1010.586 10.586z"></path>
 | 
			
		||||
        </svg>
 | 
			
		||||
        <svg id="theme-toggle-light-icon" :class="{ hidden: !isDark }" class="w-5 h-5" fill="currentColor"
 | 
			
		||||
          viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
          <path fill-rule="evenodd" clip-rule="evenodd"
 | 
			
		||||
            d="M10 2a1 1 0 011 1v1a1 1 0 11-2 0V3a1 1 0 011-1zm4 8a4 4 0 11-8 0 4 4 0 018 0zm-.464 4.95l.707.707a1 1 0 001.414-1.414l-.707-.707a1 1 0 00-1.414 1.414zm2.12-10.607a1 1 0 010 1.414l-.706.707a1 1 0 11-1.414-1.414l.707-.707a1 1 0 011.414 0zM17 11a1 1 0 100-2h-1a1 1 0 100 2h1zm-7 4a1 1 0 011 1v1a1 1 0 11-2 0v-1a1 1 0 011-1zM5.05 6.464A1 1 0 106.465 5.05l-.708-.707a1 1 0 00-1.414 1.414l.707.707zm1.414 8.486l-.707.707a1 1 0 01-1.414-1.414l.707-.707a1 1 0 011.414 1.414zM4 11a1 1 0 100-2H3a1 1 0 000 2h1z">
 | 
			
		||||
          </path>
 | 
			
		||||
        </svg>
 | 
			
		||||
      </button>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div v-cloak class="container mx-auto max-w-3xl px-5 md:px-0">
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === true">
 | 
			
		||||
        <span v-if="requiresPassword"
 | 
			
		||||
          class="text-sm text-gray-400 mb-10 mr-2 mt-3 cursor-pointer hover:underline float-right" @click="logout">
 | 
			
		||||
          class="text-sm text-gray-400 dark:text-neutral-400 mb-10 mr-2 mt-3 cursor-pointer hover:underline float-right"
 | 
			
		||||
          @click="logout">
 | 
			
		||||
          Logout
 | 
			
		||||
          <svg class="h-3 inline" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
            stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -33,13 +61,14 @@
 | 
			
		|||
              d="M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1" />
 | 
			
		||||
          </svg>
 | 
			
		||||
        </span>
 | 
			
		||||
        <h1 class="text-4xl font-medium mt-10 mb-2">
 | 
			
		||||
          <img src="./img/logo.png" width="32" class="inline align-middle" />
 | 
			
		||||
        <h1 class="text-4xl dark:text-neutral-200 font-medium mt-2 mb-2">
 | 
			
		||||
          <img src="./img/logo.png" width="32" class="inline align-middle dark:bg" />
 | 
			
		||||
          <span class="align-middle">WireGuard</span>
 | 
			
		||||
        </h1>
 | 
			
		||||
        <h2 class="text-sm text-gray-400 mb-10"></h2>
 | 
			
		||||
        <h2 class="text-sm text-gray-400 dark:text-neutral-400 mb-10"></h2>
 | 
			
		||||
 | 
			
		||||
        <div v-if="latestRelease" class="bg-red-800 p-4 text-white text-sm font-small mb-10 rounded-md shadow-lg"
 | 
			
		||||
        <div v-if="latestRelease"
 | 
			
		||||
          class="bg-red-800 dark:bg-red-100 p-4 text-white dark:text-red-600 text-sm font-small mb-10 rounded-md shadow-lg"
 | 
			
		||||
          :title="`v${currentRelease} → v${latestRelease.version}`">
 | 
			
		||||
          <div class="container mx-auto flex flex-row flex-auto items-center">
 | 
			
		||||
            <div class="flex-grow">
 | 
			
		||||
| 
						 | 
				
			
			@ -48,20 +77,20 @@
 | 
			
		|||
            </div>
 | 
			
		||||
 | 
			
		||||
            <a href="https://github.com/WeeJeWel/wg-easy#updating" target="_blank"
 | 
			
		||||
              class="p-3 rounded-md bg-white float-right font-sm font-semibold text-red-800 flex-shrink-0 border-2 border-red-800 hover:border-white hover:text-white hover:bg-red-800 transition-all">
 | 
			
		||||
              class="p-3 rounded-md bg-white dark:bg-red-100 float-right font-sm font-semibold text-red-800 dark:text-red-600 flex-shrink-0 border-2 border-red-800 dark:border-red-600 hover:border-white dark:hover:border-red-600 hover:text-white dark:hover:text-red-100 hover:bg-red-800 dark:hover:bg-red-600 transition-all">
 | 
			
		||||
              Update →
 | 
			
		||||
            </a>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div class="shadow-md rounded-lg bg-white overflow-hidden">
 | 
			
		||||
          <div class="flex flex-row flex-auto items-center p-3 px-5 border border-b-2 border-gray-100">
 | 
			
		||||
        <div class="shadow-md rounded-lg bg-white dark:bg-neutral-700 overflow-hidden">
 | 
			
		||||
          <div class="flex flex-row flex-auto items-center p-3 px-5 border-b-2 border-gray-100 dark:border-neutral-600">
 | 
			
		||||
            <div class="flex-grow">
 | 
			
		||||
              <p class="text-2xl font-medium">Clients</p>
 | 
			
		||||
              <p class="text-2xl font-medium dark:text-neutral-200">Clients</p>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div class="flex-shrink-0">
 | 
			
		||||
              <button @click="clientCreate = true; clientCreateName = '';"
 | 
			
		||||
                class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 border-2 border-gray-100 py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 dark:text-neutral-200 border-2 border-gray-100 dark:border-neutral-600 py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                <svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                  stroke="currentColor">
 | 
			
		||||
                  <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
| 
						 | 
				
			
			@ -75,113 +104,56 @@
 | 
			
		|||
          <div>
 | 
			
		||||
            <!-- Client -->
 | 
			
		||||
            <div v-if="clients && clients.length > 0" v-for="client in clients" :key="client.id"
 | 
			
		||||
              class="relative overflow-hidden border-b border-gray-100 border-solid">
 | 
			
		||||
              class="relative overflow-hidden border-b last:border-b-0 border-gray-100 dark:border-neutral-600 border-solid">
 | 
			
		||||
 | 
			
		||||
              <!-- Chart -->
 | 
			
		||||
              <div class="absolute z-0 bottom-0 left-0 right-0" style="width: 100%; height: 20%;">
 | 
			
		||||
                <!-- Bar -->
 | 
			
		||||
                <div v-for="(_, index) in client.transferTxHistory" :style="{
 | 
			
		||||
                    display: 'inline-flex',
 | 
			
		||||
                    alignItems: 'flex-end',
 | 
			
		||||
                    width: '2%', // 1/100th of client.transferTxHistory.length
 | 
			
		||||
                    height: '100%',
 | 
			
		||||
                    padding: '0 3px',
 | 
			
		||||
                    boxSizing: 'border-box',
 | 
			
		||||
                    fontSize: 0,
 | 
			
		||||
                  }">
 | 
			
		||||
 | 
			
		||||
                  <!-- TX -->
 | 
			
		||||
                  <div :style="{
 | 
			
		||||
                    minHeight: '0px',
 | 
			
		||||
                    minWidth: '2px',
 | 
			
		||||
                    maxWidth: '4px',
 | 
			
		||||
                    width: '50%',
 | 
			
		||||
                    marginRight: '1px',
 | 
			
		||||
                    height: Math.round((client.transferTxHistory[index]/client.transferMax)*100) + '%',
 | 
			
		||||
                    background: client.hoverTx
 | 
			
		||||
                      ? '#992922'
 | 
			
		||||
                      : '#F3F4F6',
 | 
			
		||||
                    transition: 'all 0.2s',
 | 
			
		||||
                    borderRadius: '2px 2px 0 0',
 | 
			
		||||
                  }"></div>
 | 
			
		||||
 | 
			
		||||
                  <!-- RX -->
 | 
			
		||||
                  <div :style="{
 | 
			
		||||
                    minHeight: '0px',
 | 
			
		||||
                    minWidth: '2px',
 | 
			
		||||
                    maxWidth: '4px',
 | 
			
		||||
                    width: '50%',
 | 
			
		||||
                    height: Math.round((client.transferRxHistory[index]/client.transferMax)*100) + '%',
 | 
			
		||||
                    background: client.hoverRx
 | 
			
		||||
                    ? '#992922'
 | 
			
		||||
                    : '#F0F1F3',
 | 
			
		||||
                    transition: 'all 0.2s',
 | 
			
		||||
                    borderRadius: '2px 2px 0 0',
 | 
			
		||||
                  }"></div>
 | 
			
		||||
                </div>
 | 
			
		||||
              <div class="absolute z-0 bottom-0 left-0 right-0" style="top: 60%;">
 | 
			
		||||
                <apexchart width="100%" height="100%" :options="client.chartOptions" :series="client.transferTxSeries">
 | 
			
		||||
                </apexchart>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="absolute z-0 top-0 left-0 right-0" style="bottom: 60%;">
 | 
			
		||||
                <apexchart width="100%" height="100%" :options="client.chartOptions" :series="client.transferRxSeries"
 | 
			
		||||
                  style="transform: scaleY(-1);">
 | 
			
		||||
                </apexchart>
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              <div class="relative p-5 z-10 flex flex-row flex-wrap justify-start">
 | 
			
		||||
                <div class="h-10 w-10 mr-5 rounded-full bg-gray-50 relative">
 | 
			
		||||
                  <svg class="w-6 m-2 text-gray-300" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                    fill="currentColor">
 | 
			
		||||
                    <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z"
 | 
			
		||||
                      clip-rule="evenodd" />
 | 
			
		||||
                  </svg>
 | 
			
		||||
                  <img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" />
 | 
			
		||||
              <div class="relative p-5 z-10 flex flex-col md:flex-row justify-between">
 | 
			
		||||
                <div class="flex items-center pb-2 md:pb-0">
 | 
			
		||||
                  <div class="h-10 w-10 mr-5 rounded-full bg-gray-50 relative">
 | 
			
		||||
                    <svg class="w-6 m-2 text-gray-300" xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                      viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
                      <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z"
 | 
			
		||||
                        clip-rule="evenodd" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                    <img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" />
 | 
			
		||||
 | 
			
		||||
                  <div
 | 
			
		||||
                    v-if="client.latestHandshakeAt && ((new Date() - new Date(client.latestHandshakeAt) < 1000 * 60 * 10))">
 | 
			
		||||
                    <div class="animate-ping w-4 h-4 p-1 bg-red-100 rounded-full absolute -bottom-1 -right-1"></div>
 | 
			
		||||
                    <div class="w-2 h-2 bg-red-800 rounded-full absolute bottom-0 right-0"></div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="sm:flex-grow">
 | 
			
		||||
 | 
			
		||||
                  <!-- Name -->
 | 
			
		||||
                  <div class="text-gray-700 group" :title="'Created on ' + dateTime(new Date(client.createdAt))">
 | 
			
		||||
 | 
			
		||||
                    <!-- Show -->
 | 
			
		||||
                    <input v-show="clientEditNameId === client.id" v-model="clientEditName"
 | 
			
		||||
                      v-on:keyup.enter="updateClientName(client, clientEditName); clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                      v-on:keyup.escape="clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                      :ref="'client-' + client.id + '-name'"
 | 
			
		||||
                      class="rounded px-1 border-2 border-gray-100 focus:border-gray-200 outline-none w-30" />
 | 
			
		||||
                    <span v-show="clientEditNameId !== client.id" style="max-width: 28ch;"
 | 
			
		||||
                      class="inline-block border-t-2 border-b-2 border-transparent align-top sm:w-auto overflow-hidden sm:overflow-visible overflow-ellipsis">{{client.name}}</span>
 | 
			
		||||
 | 
			
		||||
                    <!-- Edit -->
 | 
			
		||||
                    <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                      @click="clientEditName = client.name; clientEditNameId = client.id; setTimeout(() => $refs['client-' + client.id + '-name'][0].select(), 1);"
 | 
			
		||||
                      class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                      <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                        class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
                        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                          d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <div
 | 
			
		||||
                      v-if="client.latestHandshakeAt && ((new Date() - new Date(client.latestHandshakeAt) < 1000 * 60 * 10))">
 | 
			
		||||
                      <div
 | 
			
		||||
                        class="animate-ping w-4 h-4 p-1 bg-red-100 dark:bg-red-100 rounded-full absolute -bottom-1 -right-1">
 | 
			
		||||
                      </div>
 | 
			
		||||
                      <div class="w-2 h-2 bg-red-800 dark:bg-red-600 rounded-full absolute bottom-0 right-0"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                  <!-- Info -->
 | 
			
		||||
                  <div class="text-gray-400 text-xs">
 | 
			
		||||
                  <div class="flex-grow">
 | 
			
		||||
 | 
			
		||||
                    <!-- Address -->
 | 
			
		||||
                    <span class="group">
 | 
			
		||||
                    <!-- Name -->
 | 
			
		||||
                    <div class="text-gray-700 dark:text-neutral-200 group"
 | 
			
		||||
                      :title="'Created on ' + dateTime(new Date(client.createdAt))">
 | 
			
		||||
 | 
			
		||||
                      <!-- Show -->
 | 
			
		||||
                      <input v-show="clientEditAddressId === client.id" v-model="clientEditAddress"
 | 
			
		||||
                        v-on:keyup.enter="updateClientAddress(client, clientEditAddress); clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                        v-on:keyup.escape="clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                        :ref="'client-' + client.id + '-address'"
 | 
			
		||||
                        class="rounded border-2 border-gray-100 focus:border-gray-200 outline-none w-20 text-black" />
 | 
			
		||||
                      <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                        class="inline-block border-t-2 border-b-2 border-transparent">{{client.address}}</span>
 | 
			
		||||
                      <input v-show="clientEditNameId === client.id" v-model="clientEditName"
 | 
			
		||||
                        v-on:keyup.enter="updateClientName(client, clientEditName); clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                        v-on:keyup.escape="clientEditName = null; clientEditNameId = null;"
 | 
			
		||||
                        :ref="'client-' + client.id + '-name'"
 | 
			
		||||
                        class="rounded px-1 border-2 dark:bg-neutral-700 border-gray-100 dark:border-neutral-600 focus:border-gray-200 dark:focus:border-neutral-500 dark:placeholder:text-neutral-500 outline-none w-30" />
 | 
			
		||||
                      <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                        class="inline-block border-t-2 border-b-2 border-transparent">{{client.name}}</span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Edit -->
 | 
			
		||||
                      <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                        @click="clientEditAddress = client.address; clientEditAddressId = client.id; setTimeout(() => $refs['client-' + client.id + '-address'][0].select(), 1);"
 | 
			
		||||
                      <span v-show="clientEditNameId !== client.id"
 | 
			
		||||
                        @click="clientEditName = client.name; clientEditNameId = client.id; setTimeout(() => $refs['client-' + client.id + '-name'][0].select(), 1);"
 | 
			
		||||
                        class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                        <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                          class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none"
 | 
			
		||||
| 
						 | 
				
			
			@ -190,48 +162,71 @@
 | 
			
		|||
                            d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
 | 
			
		||||
                    <!-- Transfer TX -->
 | 
			
		||||
                    <span v-if="client.transferTx" :title="'Total Download: ' + bytes(client.transferTx)"
 | 
			
		||||
                      @mouseover="client.hoverTx = clientsPersist[client.id].hoverTx = true;"
 | 
			
		||||
                      @mouseleave="client.hoverTx = clientsPersist[client.id].hoverTx = false;"
 | 
			
		||||
                      style="cursor: default;">
 | 
			
		||||
                      ·
 | 
			
		||||
                      <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                        fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
                          d="M16.707 10.293a1 1 0 010 1.414l-6 6a1 1 0 01-1.414 0l-6-6a1 1 0 111.414-1.414L9 14.586V3a1 1 0 012 0v11.586l4.293-4.293a1 1 0 011.414 0z"
 | 
			
		||||
                          clip-rule="evenodd" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                      {{client.transferTxCurrent | bytes}}/s
 | 
			
		||||
                    </span>
 | 
			
		||||
                    <!-- Info -->
 | 
			
		||||
                    <div class="text-gray-400 dark:text-neutral-400 text-xs">
 | 
			
		||||
 | 
			
		||||
                    <!-- Transfer RX -->
 | 
			
		||||
                    <span v-if="client.transferRx" :title="'Total Upload: ' + bytes(client.transferRx)"
 | 
			
		||||
                      @mouseover="client.hoverRx = clientsPersist[client.id].hoverRx = true;"
 | 
			
		||||
                      @mouseleave="client.hoverRx = clientsPersist[client.id].hoverRx = false;"
 | 
			
		||||
                      style="cursor: default;">
 | 
			
		||||
                      ·
 | 
			
		||||
                      <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                        fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
                          d="M3.293 9.707a1 1 0 010-1.414l6-6a1 1 0 011.414 0l6 6a1 1 0 01-1.414 1.414L11 5.414V17a1 1 0 11-2 0V5.414L4.707 9.707a1 1 0 01-1.414 0z"
 | 
			
		||||
                          clip-rule="evenodd" />
 | 
			
		||||
                      </svg>
 | 
			
		||||
                      {{client.transferRxCurrent | bytes}}/s
 | 
			
		||||
                    </span>
 | 
			
		||||
                      <!-- Address -->
 | 
			
		||||
                      <span class="group block md:inline-block pb-1 md:pb-0">
 | 
			
		||||
 | 
			
		||||
                    <!-- Last seen -->
 | 
			
		||||
                    <span v-if="client.latestHandshakeAt"
 | 
			
		||||
                      :title="'Last seen on ' + dateTime(new Date(client.latestHandshakeAt))">
 | 
			
		||||
                      · {{new Date(client.latestHandshakeAt) | timeago}}
 | 
			
		||||
                    </span>
 | 
			
		||||
                        <!-- Show -->
 | 
			
		||||
                        <input v-show="clientEditAddressId === client.id" v-model="clientEditAddress"
 | 
			
		||||
                          v-on:keyup.enter="updateClientAddress(client, clientEditAddress); clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                          v-on:keyup.escape="clientEditAddress = null; clientEditAddressId = null;"
 | 
			
		||||
                          :ref="'client-' + client.id + '-address'"
 | 
			
		||||
                          class="rounded border-2 dark:bg-neutral-700 border-gray-100 dark:border-neutral-600 focus:border-gray-200 dark:focus:border-neutral-500 outline-none w-20 text-black dark:text-neutral-300 dark:placeholder:text-neutral-500" />
 | 
			
		||||
                        <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                          class="inline-block border-t-2 border-b-2 border-transparent">{{client.address}}</span>
 | 
			
		||||
 | 
			
		||||
                        <!-- Edit -->
 | 
			
		||||
                        <span v-show="clientEditAddressId !== client.id"
 | 
			
		||||
                          @click="clientEditAddress = client.address; clientEditAddressId = client.id; setTimeout(() => $refs['client-' + client.id + '-address'][0].select(), 1);"
 | 
			
		||||
                          class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
 | 
			
		||||
                          <svg xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                            class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none"
 | 
			
		||||
                            viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                            <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                              d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
 | 
			
		||||
                          </svg>
 | 
			
		||||
                        </span>
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Transfer TX -->
 | 
			
		||||
                      <span v-if="client.transferTx" :title="'Total Download: ' + bytes(client.transferTx)">
 | 
			
		||||
                        ·
 | 
			
		||||
                        <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                          fill="currentColor">
 | 
			
		||||
                          <path fill-rule="evenodd"
 | 
			
		||||
                            d="M16.707 10.293a1 1 0 010 1.414l-6 6a1 1 0 01-1.414 0l-6-6a1 1 0 111.414-1.414L9 14.586V3a1 1 0 012 0v11.586l4.293-4.293a1 1 0 011.414 0z"
 | 
			
		||||
                            clip-rule="evenodd" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                        {{client.transferTxCurrent | bytes}}/s
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Transfer RX -->
 | 
			
		||||
                      <span v-if="client.transferRx" :title="'Total Upload: ' + bytes(client.transferRx)">
 | 
			
		||||
                        ·
 | 
			
		||||
                        <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
                          fill="currentColor">
 | 
			
		||||
                          <path fill-rule="evenodd"
 | 
			
		||||
                            d="M3.293 9.707a1 1 0 010-1.414l6-6a1 1 0 011.414 0l6 6a1 1 0 01-1.414 1.414L11 5.414V17a1 1 0 11-2 0V5.414L4.707 9.707a1 1 0 01-1.414 0z"
 | 
			
		||||
                            clip-rule="evenodd" />
 | 
			
		||||
                        </svg>
 | 
			
		||||
                        {{client.transferRxCurrent | bytes}}/s
 | 
			
		||||
                      </span>
 | 
			
		||||
 | 
			
		||||
                      <!-- Last seen -->
 | 
			
		||||
                      <span v-if="client.latestHandshakeAt"
 | 
			
		||||
                        :title="'Last seen on ' + dateTime(new Date(client.latestHandshakeAt))">
 | 
			
		||||
                        · {{new Date(client.latestHandshakeAt) | timeago}}
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <div class="text-right w-full sm:w-auto mt-3 sm:mt-0">
 | 
			
		||||
                  <div class="text-gray-400">
 | 
			
		||||
                <div class="flex items-center justify-end">
 | 
			
		||||
                  <div class="text-gray-400 dark:text-neutral-400 flex gap-1 items-center justify-between">
 | 
			
		||||
 | 
			
		||||
                    <!-- Enable/Disable -->
 | 
			
		||||
                    <div @click="disableClient(client)" v-if="client.enabled === true" title="Disable Client"
 | 
			
		||||
| 
						 | 
				
			
			@ -239,12 +234,13 @@
 | 
			
		|||
                      <div class="rounded-full w-4 h-4 m-1 ml-5 bg-white"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <div @click="enableClient(client)" v-if="client.enabled === false" title="Enable Client"
 | 
			
		||||
                      class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-gray-200 cursor-pointer hover:bg-gray-300 transition-all">
 | 
			
		||||
                      class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-gray-200 dark:bg-neutral-400 cursor-pointer hover:bg-gray-300 dark:hover:bg-neutral-500 transition-all">
 | 
			
		||||
                      <div class="rounded-full w-4 h-4 m-1 bg-white"></div>
 | 
			
		||||
                    </div>
 | 
			
		||||
 | 
			
		||||
                    <!-- Show QR-->
 | 
			
		||||
                    <button class="align-middle bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                    <button
 | 
			
		||||
                      class="align-middle bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Show QR Code" @click="qrcode = `./api/wireguard/client/${client.id}/qrcode.svg`">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -255,7 +251,7 @@
 | 
			
		|||
 | 
			
		||||
                    <!-- Download Config -->
 | 
			
		||||
                    <a :href="'./api/wireguard/client/' + client.id + '/configuration'" download
 | 
			
		||||
                      class="align-middle inline-block bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                      class="align-middle inline-block bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Download Configuration">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                        stroke="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -265,7 +261,8 @@
 | 
			
		|||
                    </a>
 | 
			
		||||
 | 
			
		||||
                    <!-- Delete -->
 | 
			
		||||
                    <button class="align-middle bg-gray-100 hover:bg-red-800 hover:text-white p-2 rounded transition"
 | 
			
		||||
                    <button
 | 
			
		||||
                      class="align-middle bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white p-2 rounded transition"
 | 
			
		||||
                      title="Delete Client" @click="clientDelete = client">
 | 
			
		||||
                      <svg class="w-5" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
                        <path fill-rule="evenodd"
 | 
			
		||||
| 
						 | 
				
			
			@ -280,9 +277,10 @@
 | 
			
		|||
 | 
			
		||||
            </div>
 | 
			
		||||
            <div v-if="clients && clients.length === 0">
 | 
			
		||||
              <p class="text-center m-10 text-gray-400 text-sm">There are no clients yet.<br /><br />
 | 
			
		||||
              <p class="text-center m-10 text-gray-400 dark:text-neutral-400 text-sm">
 | 
			
		||||
                There are no clients yet.<br /><br />
 | 
			
		||||
                <button @click="clientCreate = true; clientCreateName = '';"
 | 
			
		||||
                  class="bg-red-800 text-white hover:bg-red-700 border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                  class="bg-red-800 hover:bg-red-700 text-white border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
 | 
			
		||||
                  <svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                    stroke="currentColor">
 | 
			
		||||
                    <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
| 
						 | 
				
			
			@ -292,7 +290,7 @@
 | 
			
		|||
                </button>
 | 
			
		||||
              </p>
 | 
			
		||||
            </div>
 | 
			
		||||
            <div v-if="clients === null" class="text-gray-200 p-5">
 | 
			
		||||
            <div v-if="clients === null" class="text-gray-200 dark:text-red-300 p-5">
 | 
			
		||||
              <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
                fill="currentColor">
 | 
			
		||||
                <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
 | 
			
		||||
| 
						 | 
				
			
			@ -308,7 +306,8 @@
 | 
			
		|||
        <div v-if="qrcode">
 | 
			
		||||
          <div class="bg-black bg-opacity-50 fixed top-0 right-0 left-0 bottom-0 flex items-center justify-center z-20">
 | 
			
		||||
            <div class="bg-white rounded-md shadow-lg relative p-8">
 | 
			
		||||
              <button @click="qrcode = null" class="absolute right-4 top-4 text-gray-600 hover:text-gray-800">
 | 
			
		||||
              <button @click="qrcode = null"
 | 
			
		||||
                class="absolute right-4 top-4 text-gray-600 dark:text-neutral-500 hover:text-gray-800 dark:hover:text-neutral-700">
 | 
			
		||||
                <svg class="w-8" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
 | 
			
		||||
                  stroke="currentColor">
 | 
			
		||||
                  <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
 | 
			
		||||
| 
						 | 
				
			
			@ -321,7 +320,7 @@
 | 
			
		|||
 | 
			
		||||
        <!-- Create Dialog -->
 | 
			
		||||
        <div v-if="clientCreate" class="fixed z-10 inset-0 overflow-y-auto">
 | 
			
		||||
          <div class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
          <div class="flex items-center justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
            <!--
 | 
			
		||||
        Background overlay, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -333,7 +332,7 @@
 | 
			
		|||
          To: "opacity-0"
 | 
			
		||||
      -->
 | 
			
		||||
            <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 opacity-75"></div>
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 dark:bg-black opacity-75 dark:opacity-50"></div>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <!-- This element is to trick the browser into centering the modal contents. -->
 | 
			
		||||
| 
						 | 
				
			
			@ -342,49 +341,50 @@
 | 
			
		|||
        Modal panel, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
        Entering: "ease-out duration-300"
 | 
			
		||||
          From: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          From: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
        Leaving: "ease-in duration-200"
 | 
			
		||||
          From: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          From: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
      -->
 | 
			
		||||
            <div
 | 
			
		||||
              class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full"
 | 
			
		||||
              class="inline-block align-bottom bg-white dark:bg-neutral-700 rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg w-full"
 | 
			
		||||
              role="dialog" aria-modal="true" aria-labelledby="modal-headline">
 | 
			
		||||
              <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
              <div class="bg-white dark:bg-neutral-700 px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
                <div class="sm:flex sm:items-start">
 | 
			
		||||
                  <div
 | 
			
		||||
                    class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-red-800 sm:mx-0 sm:h-10 sm:w-10">
 | 
			
		||||
                    <svg class="h-6 w-6 text-white" inline xmlns="http://www.w3.org/2000/svg" fill="none"
 | 
			
		||||
                      viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                    <svg class="h-6 w-6 text-white" inline xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
                      fill="none" viewBox="0 0 24 24" stroke="currentColor">
 | 
			
		||||
                      <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
 | 
			
		||||
                        d="M12 6v6m0 0v6m0-6h6m-6 0H6" />
 | 
			
		||||
                    </svg>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="flex-grow mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900" id="modal-headline">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900 dark:text-neutral-200" id="modal-headline">
 | 
			
		||||
                      New Client
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    <div class="mt-2">
 | 
			
		||||
                      <p class="text-sm text-gray-500">
 | 
			
		||||
                        <input class="rounded p-2 border-2 border-gray-100 focus:border-gray-200 outline-none w-full"
 | 
			
		||||
                        <input
 | 
			
		||||
                          class="rounded p-2 border-2 dark:bg-neutral-700 dark:text-neutral-200 border-gray-100 dark:border-neutral-600 focus:border-gray-200 focus:dark:border-neutral-500 dark:placeholder:text-neutral-400 outline-none w-full"
 | 
			
		||||
                          type="text" v-model.trim="clientCreateName" placeholder="Name" />
 | 
			
		||||
                      </p>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
              <div class="bg-gray-50 dark:bg-neutral-700 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
                <button v-if="clientCreateName.length" type="button" @click="createClient(); clientCreate = null"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-800 text-base font-medium text-white hover:bg-red-700 focus:outline-none sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Create
 | 
			
		||||
                </button>
 | 
			
		||||
                <button v-else type="button"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-200 text-base font-medium text-white sm:ml-3 sm:w-auto sm:text-sm cursor-not-allowed">
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-gray-200 dark:bg-neutral-400 text-base font-medium text-white dark:text-neutral-300 sm:ml-3 sm:w-auto sm:text-sm cursor-not-allowed">
 | 
			
		||||
                  Create
 | 
			
		||||
                </button>
 | 
			
		||||
                <button type="button" @click="clientCreate = null"
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 dark:border-neutral-500 shadow-sm px-4 py-2 bg-white dark:bg-neutral-500 text-base font-medium text-gray-700 dark:text-neutral-50 hover:bg-gray-50 dark:hover:bg-neutral-600 dark:hover:border-neutral-600 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Cancel
 | 
			
		||||
                </button>
 | 
			
		||||
              </div>
 | 
			
		||||
| 
						 | 
				
			
			@ -394,7 +394,7 @@
 | 
			
		|||
 | 
			
		||||
        <!-- Delete Dialog -->
 | 
			
		||||
        <div v-if="clientDelete" class="fixed z-10 inset-0 overflow-y-auto">
 | 
			
		||||
          <div class="flex items-end justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
          <div class="flex items-center justify-center min-h-screen pt-4 px-4 pb-20 text-center sm:block sm:p-0">
 | 
			
		||||
            <!--
 | 
			
		||||
        Background overlay, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -406,7 +406,7 @@
 | 
			
		|||
          To: "opacity-0"
 | 
			
		||||
      -->
 | 
			
		||||
            <div class="fixed inset-0 transition-opacity" aria-hidden="true">
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 opacity-75"></div>
 | 
			
		||||
              <div class="absolute inset-0 bg-gray-500 dark:bg-black opacity-75 dark:opacity-50"></div>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <!-- This element is to trick the browser into centering the modal contents. -->
 | 
			
		||||
| 
						 | 
				
			
			@ -415,16 +415,16 @@
 | 
			
		|||
        Modal panel, show/hide based on modal state.
 | 
			
		||||
 | 
			
		||||
        Entering: "ease-out duration-300"
 | 
			
		||||
          From: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          From: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
          To: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
        Leaving: "ease-in duration-200"
 | 
			
		||||
          From: "opacity-100 translate-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
 | 
			
		||||
          From: "opacity-100 tranneutral-y-0 sm:scale-100"
 | 
			
		||||
          To: "opacity-0 tranneutral-y-4 sm:tranneutral-y-0 sm:scale-95"
 | 
			
		||||
      -->
 | 
			
		||||
            <div
 | 
			
		||||
              class="inline-block align-bottom bg-white rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg sm:w-full"
 | 
			
		||||
              class="inline-block align-bottom bg-white dark:bg-neutral-700 rounded-lg text-left overflow-hidden shadow-xl transform transition-all sm:my-8 sm:align-middle sm:max-w-lg w-full"
 | 
			
		||||
              role="dialog" aria-modal="true" aria-labelledby="modal-headline">
 | 
			
		||||
              <div class="bg-white px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
              <div class="bg-white dark:bg-neutral-700 px-4 pt-5 pb-4 sm:p-6 sm:pb-4">
 | 
			
		||||
                <div class="sm:flex sm:items-start">
 | 
			
		||||
                  <div
 | 
			
		||||
                    class="mx-auto flex-shrink-0 flex items-center justify-center h-12 w-12 rounded-full bg-red-100 sm:mx-0 sm:h-10 sm:w-10">
 | 
			
		||||
| 
						 | 
				
			
			@ -436,11 +436,11 @@
 | 
			
		|||
                    </svg>
 | 
			
		||||
                  </div>
 | 
			
		||||
                  <div class="mt-3 text-center sm:mt-0 sm:ml-4 sm:text-left">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900" id="modal-headline">
 | 
			
		||||
                    <h3 class="text-lg leading-6 font-medium text-gray-900 dark:text-neutral-200" id="modal-headline">
 | 
			
		||||
                      Delete Client
 | 
			
		||||
                    </h3>
 | 
			
		||||
                    <div class="mt-2">
 | 
			
		||||
                      <p class="text-sm text-gray-500">
 | 
			
		||||
                      <p class="text-sm text-gray-500 dark:text-neutral-300">
 | 
			
		||||
                        Are you sure you want to delete <strong>{{clientDelete.name}}</strong>?
 | 
			
		||||
                        This action cannot be undone.
 | 
			
		||||
                      </p>
 | 
			
		||||
| 
						 | 
				
			
			@ -448,13 +448,13 @@
 | 
			
		|||
                  </div>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
              <div class="bg-gray-50 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
              <div class="bg-gray-50 dark:bg-neutral-600 px-4 py-3 sm:px-6 sm:flex sm:flex-row-reverse">
 | 
			
		||||
                <button type="button" @click="deleteClient(clientDelete); clientDelete = null"
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 text-base font-medium text-white hover:bg-red-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-red-500 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 bg-red-600 dark:bg-red-600 text-base font-medium text-white dark:text-white hover:bg-red-700 dark:hover:bg-red-700 focus:outline-none sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Delete
 | 
			
		||||
                </button>
 | 
			
		||||
                <button type="button" @click="clientDelete = null"
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 shadow-sm px-4 py-2 bg-white text-base font-medium text-gray-700 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  class="mt-3 w-full inline-flex justify-center rounded-md border border-gray-300 dark:border-neutral-500 shadow-sm px-4 py-2 bg-white dark:bg-neutral-500 text-base font-medium text-gray-700 dark:text-neutral-50 hover:bg-gray-50 dark:hover:bg-neutral-600 dark:hover:border-neutral-600 focus:outline-none sm:mt-0 sm:ml-3 sm:w-auto sm:text-sm">
 | 
			
		||||
                  Cancel
 | 
			
		||||
                </button>
 | 
			
		||||
              </div>
 | 
			
		||||
| 
						 | 
				
			
			@ -464,22 +464,23 @@
 | 
			
		|||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === false">
 | 
			
		||||
        <h1 class="text-4xl font-medium my-16 text-gray-700 text-center">WireGuard</h1>
 | 
			
		||||
        <h1 class="text-4xl font-medium my-16 text-gray-700 dark:text-neutral-200 text-center">WireGuard</h1>
 | 
			
		||||
 | 
			
		||||
        <form @submit="login" class="shadow rounded-md bg-white mx-auto w-64 p-5 overflow-hidden mt-10">
 | 
			
		||||
        <form @submit="login"
 | 
			
		||||
          class="shadow rounded-md bg-white dark:bg-neutral-700 mx-auto w-64 p-5 overflow-hidden mt-10">
 | 
			
		||||
          <!-- Avatar -->
 | 
			
		||||
          <div class="h-20 w-20 mb-10 mt-5 mx-auto rounded-full bg-red-800 relative overflow-hidden">
 | 
			
		||||
            <svg class="w-10 h-10 m-5 text-white" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
 | 
			
		||||
              fill="currentColor">
 | 
			
		||||
          <div class="h-20 w-20 mb-10 mt-5 mx-auto rounded-full bg-red-800 dark:bg-red-800 relative overflow-hidden">
 | 
			
		||||
            <svg class="w-10 h-10 m-5 text-white dark:text-white" xmlns="http://www.w3.org/2000/svg"
 | 
			
		||||
              viewBox="0 0 20 20" fill="currentColor">
 | 
			
		||||
              <path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z" clip-rule="evenodd" />
 | 
			
		||||
            </svg>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <input type="password" name="password" placeholder="Password" v-model="password"
 | 
			
		||||
            class="px-3 py-2 text-sm text-gray-500 mb-5 border-2 border-gray-100 rounded-lg w-full focus:border-red-800 outline-none" />
 | 
			
		||||
            class="px-3 py-2 text-sm dark:bg-neutral-700 text-gray-500 dark:text-gray-500 mb-5 border-2 border-gray-100 dark:border-neutral-800 rounded-lg w-full focus:border-red-800 dark:focus:border-red-800 dark:placeholder:text-neutral-400 outline-none" />
 | 
			
		||||
 | 
			
		||||
          <button v-if="authenticating"
 | 
			
		||||
            class="bg-red-800 w-full rounded shadow py-2 text-sm text-white cursor-not-allowed">
 | 
			
		||||
            class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed">
 | 
			
		||||
            <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
              fill="currentColor">
 | 
			
		||||
              <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
 | 
			
		||||
| 
						 | 
				
			
			@ -489,14 +490,15 @@
 | 
			
		|||
            </svg>
 | 
			
		||||
          </button>
 | 
			
		||||
          <input v-if="!authenticating && password" type="submit"
 | 
			
		||||
            class="bg-red-800 w-full rounded shadow py-2 text-sm text-white hover:bg-red-700 transition cursor-pointer"
 | 
			
		||||
            class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white hover:bg-red-700 dark:hover:bg-red-700 transition cursor-pointer"
 | 
			
		||||
            value="Sign In">
 | 
			
		||||
          <input v-if="!authenticating && !password" type="submit"
 | 
			
		||||
            class="bg-gray-200 w-full rounded shadow py-2 text-sm text-white cursor-not-allowed" value="Sign In">
 | 
			
		||||
            class="bg-gray-200 dark:bg-neutral-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed"
 | 
			
		||||
            value="Sign In">
 | 
			
		||||
        </form>
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      <div v-if="authenticated === null" class="text-gray-300 pt-24 pb-12">
 | 
			
		||||
      <div v-if="authenticated === null" class="text-gray-300 dark:text-red-300 pt-24 pb-12">
 | 
			
		||||
 | 
			
		||||
        <svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
 | 
			
		||||
          fill="currentColor">
 | 
			
		||||
| 
						 | 
				
			
			@ -510,16 +512,18 @@
 | 
			
		|||
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <p v-cloak class="text-center m-10 text-gray-300 text-xs">Made by <a target="_blank" class="hover:underline"
 | 
			
		||||
        href="https://emilenijssen.nl/?ref=wg-easy">Emile Nijssen</a> · Fork by <a target="_blank" class="hover:underline"
 | 
			
		||||
        href="https://hub.docker.com/r/pheiduck/wg-easy">Philip Heiduck</a> · <a class="hover:underline"
 | 
			
		||||
        href="https://github.com/sponsors/WeeJeWel" target="_blank">Donate to Emile</a> · <a class="hover:underline"
 | 
			
		||||
        href="https://github.com/pheiduck/wg-easy" target="_blank">GitHub</a></p>
 | 
			
		||||
    <p v-cloak class="text-center m-10 text-gray-300 dark:text-neutral-600 text-xs">Made by <a target="_blank"
 | 
			
		||||
        class="hover:underline" href="https://emilenijssen.nl/?ref=wg-easy">Emile Nijssen</a> · Fork by <a target="_blank"
 | 
			
		||||
        class="hover:underline" href="https://hub.docker.com/r/pheiduck/wg-easy">Philip Heiduck</a> · <a
 | 
			
		||||
        class="hover:underline" href="https://github.com/sponsors/WeeJeWel" target="_blank">Donate to Emile</a> · <a
 | 
			
		||||
        class="hover:underline" href="https://github.com/pheiduck/wg-easy" target="_blank">GitHub</a></p>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
  <script src="./js/vendor/vue.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/apexcharts.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/vue-apexcharts.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/md5.min.js"></script>
 | 
			
		||||
  <script src="./js/vendor/timeago.min.js"></script>
 | 
			
		||||
  <script src="./js/api.js"></script>
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -45,6 +45,8 @@ new Vue({
 | 
			
		|||
    currentRelease: null,
 | 
			
		||||
    latestRelease: null,
 | 
			
		||||
 | 
			
		||||
    isDark: null,
 | 
			
		||||
 | 
			
		||||
    chartOptions: {
 | 
			
		||||
      chart: {
 | 
			
		||||
        background: 'transparent',
 | 
			
		||||
| 
						 | 
				
			
			@ -239,6 +241,16 @@ new Vue({
 | 
			
		|||
        .catch((err) => alert(err.message || err.toString()))
 | 
			
		||||
        .finally(() => this.refresh().catch(console.error));
 | 
			
		||||
    },
 | 
			
		||||
    toggleTheme() {
 | 
			
		||||
      if (this.isDark) {
 | 
			
		||||
        localStorage.theme = 'light';
 | 
			
		||||
        document.documentElement.classList.remove('dark');
 | 
			
		||||
      } else {
 | 
			
		||||
        localStorage.theme = 'dark';
 | 
			
		||||
        document.documentElement.classList.add('dark');
 | 
			
		||||
      }
 | 
			
		||||
      this.isDark = !this.isDark;
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  filters: {
 | 
			
		||||
    bytes,
 | 
			
		||||
| 
						 | 
				
			
			@ -247,6 +259,11 @@ new Vue({
 | 
			
		|||
    },
 | 
			
		||||
  },
 | 
			
		||||
  mounted() {
 | 
			
		||||
    this.isDark = false;
 | 
			
		||||
    if (localStorage.theme === 'dark') {
 | 
			
		||||
      this.isDark = true;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.api = new API();
 | 
			
		||||
    this.api.getSession()
 | 
			
		||||
      .then((session) => {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										14
									
								
								src/www/js/vendor/apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/www/js/vendor/apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										7
									
								
								src/www/js/vendor/vue-apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/www/js/vendor/vue-apexcharts.min.js
									
										
									
									
										vendored
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,7 @@
 | 
			
		|||
/**
 | 
			
		||||
 * Minified by jsDelivr using Terser v5.7.1.
 | 
			
		||||
 * Original file: /npm/vue-apexcharts@1.6.2/dist/vue-apexcharts.js
 | 
			
		||||
 *
 | 
			
		||||
 * Do NOT use SRI with dynamically generated files! More information: https://www.jsdelivr.com/using-sri-with-dynamic-files
 | 
			
		||||
 */
 | 
			
		||||
!function (t, e) { "object" == typeof exports && "undefined" != typeof module ? module.exports = e(require("apexcharts/dist/apexcharts.min")) : "function" == typeof define && define.amd ? define(["apexcharts/dist/apexcharts.min"], e) : t.VueApexCharts = e(t.ApexCharts) }(this, (function (t) { "use strict"; function e(t) { return (e = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (t) { return typeof t } : function (t) { return t && "function" == typeof Symbol && t.constructor === Symbol && t !== Symbol.prototype ? "symbol" : typeof t })(t) } function n(t, e, n) { return e in t ? Object.defineProperty(t, e, { value: n, enumerable: !0, configurable: !0, writable: !0 }) : t[e] = n, t } t = t && t.hasOwnProperty("default") ? t.default : t; var i = { props: { options: { type: Object }, type: { type: String }, series: { type: Array, required: !0, default: function () { return [] } }, width: { default: "100%" }, height: { default: "auto" } }, data: function () { return { chart: null } }, beforeMount: function () { window.ApexCharts = t }, mounted: function () { this.init() }, created: function () { var t = this; this.$watch("options", (function (e) { !t.chart && e ? t.init() : t.chart.updateOptions(t.options) })), this.$watch("series", (function (e) { !t.chart && e ? t.init() : t.chart.updateSeries(t.series) }));["type", "width", "height"].forEach((function (e) { t.$watch(e, (function () { t.refresh() })) })) }, beforeDestroy: function () { this.chart && this.destroy() }, render: function (t) { return t("div") }, methods: { init: function () { var e = this, n = { chart: { type: this.type || this.options.chart.type || "line", height: this.height, width: this.width, events: {} }, series: this.series }; Object.keys(this.$listeners).forEach((function (t) { n.chart.events[t] = e.$listeners[t] })); var i = this.extend(this.options, n); return this.chart = new t(this.$el, i), this.chart.render() }, isObject: function (t) { return t && "object" === e(t) && !Array.isArray(t) && null != t }, extend: function (t, e) { var i = this; "function" != typeof Object.assign && (Object.assign = function (t) { if (null == t) throw new TypeError("Cannot convert undefined or null to object"); for (var e = Object(t), n = 1; n < arguments.length; n++) { var i = arguments[n]; if (null != i) for (var r in i) i.hasOwnProperty(r) && (e[r] = i[r]) } return e }); var r = Object.assign({}, t); return this.isObject(t) && this.isObject(e) && Object.keys(e).forEach((function (o) { i.isObject(e[o]) && o in t ? r[o] = i.extend(t[o], e[o]) : Object.assign(r, n({}, o, e[o])) })), r }, refresh: function () { return this.destroy(), this.init() }, destroy: function () { this.chart.destroy() }, updateSeries: function (t, e) { return this.chart.updateSeries(t, e) }, updateOptions: function (t, e, n, i) { return this.chart.updateOptions(t, e, n, i) }, toggleSeries: function (t) { return this.chart.toggleSeries(t) }, showSeries: function (t) { this.chart.showSeries(t) }, hideSeries: function (t) { this.chart.hideSeries(t) }, appendSeries: function (t, e) { return this.chart.appendSeries(t, e) }, resetSeries: function () { this.chart.resetSeries() }, zoomX: function (t, e) { this.chart.zoomX(t, e) }, toggleDataPointSelection: function (t, e) { this.chart.toggleDataPointSelection(t, e) }, appendData: function (t) { return this.chart.appendData(t) }, addText: function (t) { this.chart.addText(t) }, addImage: function (t) { this.chart.addImage(t) }, addShape: function (t) { this.chart.addShape(t) }, dataURI: function () { return this.chart.dataURI() }, setLocale: function (t) { return this.chart.setLocale(t) }, addXaxisAnnotation: function (t, e) { this.chart.addXaxisAnnotation(t, e) }, addYaxisAnnotation: function (t, e) { this.chart.addYaxisAnnotation(t, e) }, addPointAnnotation: function (t, e) { this.chart.addPointAnnotation(t, e) }, removeAnnotation: function (t, e) { this.chart.removeAnnotation(t, e) }, clearAnnotations: function () { this.chart.clearAnnotations() } } }; return window.ApexCharts = t, i.install = function (e) { e.ApexCharts = t, window.ApexCharts = t, Object.defineProperty(e.prototype, "$apexcharts", { get: function () { return t } }) }, i }));
 | 
			
		||||
							
								
								
									
										3
									
								
								src/www/src/css/app.css
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/www/src/css/app.css
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,3 @@
 | 
			
		|||
@tailwind base;
 | 
			
		||||
@tailwind components;
 | 
			
		||||
@tailwind utilities;
 | 
			
		||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue