| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 | {{define "modals/qrcodeModal"}}<a-modal id="qrcode-modal" v-model="qrModal.visible" :closable="true" :class="themeSwitcher.currentTheme"  width="fit-content" :dialog-style="isMobile ? { top: '18px' } : {}" :footer="null">  <template #title>    <a-space direction="horizontal">      <span>[[ qrModal.title ]]</span>      <a-popover :overlay-class-name="themeSwitcher.currentTheme" trigger="click" placement="bottom">        <template slot="content">          <a-space direction="vertical">            <template v-for="(row, index) in qrModal.qrcodes">              <b>[[ row.remark ]]</b>              <a-space direction="horizontal">                <a-switch size="small" :checked="row.useIPv4" @click="toggleIPv4(index)"></a-switch>                <span>{{ i18n "useIPv4ForHost" }}</span>              </a-space>            </template>          </a-space>        </template>        <a-icon type="setting"></a-icon>      </a-popover>    </a-space>  </template>  <tr-qr-modal class="qr-modal">    <template v-if="app.subSettings?.enable && qrModal.subId">      <tr-qr-box class="qr-box">        <a-tag color="purple" class="qr-tag"><span>{{ i18n "pages.settings.subSettings"}}</span></a-tag>        <tr-qr-bg class="qr-bg-sub">          <tr-qr-bg-inner class="qr-bg-sub-inner">            <canvas @click="copy(genSubLink(qrModal.client.subId))" id="qrCode-sub" class="qr-cv"></canvas>          </tr-qr-bg-inner>        </tr-qr-bg>      </tr-qr-box>      <tr-qr-box class="qr-box">        <a-tag color="purple" class="qr-tag"><span>{{ i18n "pages.settings.subSettings"}} Json</span></a-tag>        <tr-qr-bg class="qr-bg-sub">          <tr-qr-bg-inner class="qr-bg-sub-inner">            <canvas @click="copy(genSubJsonLink(qrModal.client.subId))" id="qrCode-subJson" class="qr-cv"></canvas>          </tr-qr-bg-inner>        </tr-qr-bg>      </tr-qr-box>    </template>    <template v-for="(row, index) in qrModal.qrcodes">      <tr-qr-box class="qr-box">        <a-tag color="green" class="qr-tag"><span>[[ row.remark ]]</span></a-tag>        <tr-qr-bg class="qr-bg">          <canvas @click="copy(row.link)" :id="'qrCode-'+index" class="qr-cv"></canvas>        </tr-qr-bg>      </tr-qr-box>    </template>  </tr-qr-modal></a-modal><style>  .ant-table:not(.ant-table-expanded-row .ant-table) {    outline: 1px solid #f0f0f0;    outline-offset: -1px;    border-radius: 1rem;    overflow-x: hidden;  }    /* QR code transition effects */  .qr-cv {    transition: all 0.3s ease-in-out;  }    .qr-transition-enter-active, .qr-transition-leave-active {    transition: opacity 0.3s, transform 0.3s;  }    .qr-transition-enter, .qr-transition-leave-to {    opacity: 0;    transform: scale(0.9);  }    .qr-transition-enter-to, .qr-transition-leave {    opacity: 1;    transform: scale(1);  }    .qr-flash {    animation: qr-flash-animation 0.6s;  }    @keyframes qr-flash-animation {    0% {      opacity: 1;      transform: scale(1);    }    50% {      opacity: 0.5;      transform: scale(0.95);    }    100% {      opacity: 1;      transform: scale(1);    }  }</style><script>  const qrModal = {    title: '',    dbInbound: new DBInbound(),    client: null,    qrcodes: [],    visible: false,    subId: '',    show: function (title = '', dbInbound, client) {      this.title = title;      this.dbInbound = dbInbound;      this.inbound = dbInbound.toInbound();      this.client = client;      this.subId = '';      this.qrcodes = [];      // Reset the status fetched flag when showing the modal      if (qrModalApp) qrModalApp.statusFetched = false;      if (this.inbound.protocol == Protocols.WIREGUARD) {        this.inbound.genInboundLinks(dbInbound.remark).split('\r\n').forEach((l, index) => {          this.qrcodes.push({            remark: "Peer " + (index + 1),            link: l,            useIPv4: false,            originalLink: l          });        });      } else {        this.inbound.genAllLinks(this.dbInbound.remark, app.remarkModel, client).forEach(l => {          this.qrcodes.push({            remark: l.remark,            link: l.link,            useIPv4: false,            originalLink: l.link          });        });      }      this.visible = true;    },    close: function () {      this.visible = false;    },  };  const qrModalApp = new Vue({    delimiters: ['[[', ']]'],    el: '#qrcode-modal',    mixins: [MediaQueryMixin],    data: {      qrModal: qrModal,      serverStatus: null,      statusFetched: false,    },    methods: {      async getStatus() {        try {          const msg = await HttpUtil.post('/server/status');          if (msg.success) {            this.serverStatus = msg.obj;          }        } catch (e) {          console.error("Failed to get status:", e);        }      },            toggleIPv4(index) {        const row = qrModal.qrcodes[index];        row.useIPv4 = !row.useIPv4;        this.updateLink(index);      },      updateLink(index) {        const row = qrModal.qrcodes[index];        if (!this.serverStatus || !this.serverStatus.publicIP) {          return;        }                if (row.useIPv4 && this.serverStatus.publicIP.ipv4) {          // Replace the hostname or IP in the link with the IPv4 address          const originalLink = row.originalLink;          const url = new URL(originalLink);          const ipv4 = this.serverStatus.publicIP.ipv4;                    if (qrModal.inbound.protocol == Protocols.WIREGUARD) {            // Special handling for WireGuard config            const endpointRegex = /Endpoint = ([^:]+):(\d+)/;            const match = originalLink.match(endpointRegex);            if (match) {              row.link = originalLink.replace(                `Endpoint = ${match[1]}:${match[2]}`,                `Endpoint = ${ipv4}:${match[2]}`              );            }          } else {            // For other protocols using URL format            url.hostname = ipv4;            row.link = url.toString();          }        } else {          // Restore original link          row.link = row.originalLink;        }                // Update QR code with transition effect        const canvasElement = document.querySelector('#qrCode-' + index);        if (canvasElement) {          // Add flash animation class          canvasElement.classList.add('qr-flash');                    // Remove the class after animation completes          setTimeout(() => {            canvasElement.classList.remove('qr-flash');          }, 600);        }                this.setQrCode("qrCode-" + index, row.link);      },      copy(content) {        ClipboardManager          .copyText(content)          .then(() => {            app.$message.success('{{ i18n "copied" }}')          })      },      setQrCode(elementId, content) {        new QRious({          element: document.querySelector('#' + elementId),          size: 400,          value: content,          background: 'white',          backgroundAlpha: 0,          foreground: 'black',          padding: 2,          level: 'L'        });      },      genSubLink(subID) {        return app.subSettings.subURI + subID;      },      genSubJsonLink(subID) {        return app.subSettings.subJsonURI + subID;      },      revertOverflow() {        const elements = document.querySelectorAll(".qr-tag");        elements.forEach((element) => {          element.classList.remove("tr-marquee");          element.children[0].style.animation = '';          while (element.children.length > 1) {            element.removeChild(element.lastChild);          }        });      }    },    updated() {      if (this.qrModal.visible) {        fixOverflow();        if (!this.statusFetched) {          this.getStatus();          this.statusFetched = true;        }      } else {        this.revertOverflow();        // Reset the flag when modal is closed so it will fetch again next time        this.statusFetched = false;      }      if (qrModal.client && qrModal.client.subId) {        qrModal.subId = qrModal.client.subId;        this.setQrCode("qrCode-sub", this.genSubLink(qrModal.subId));        this.setQrCode("qrCode-subJson", this.genSubJsonLink(qrModal.subId));      }      qrModal.qrcodes.forEach((element, index) => {        this.setQrCode("qrCode-" + index, element.link);        // Update links based on current toggle state        if (element.useIPv4 && this.serverStatus && this.serverStatus.publicIP) {          this.updateLink(index);        }      });    }  });  function fixOverflow() {    const elements = document.querySelectorAll(".qr-tag");    elements.forEach((element) => {      function isElementOverflowing(element) {        const overflowX = element.offsetWidth < element.scrollWidth,          overflowY = element.offsetHeight < element.scrollHeight;        return overflowX || overflowY;      }      function wrapContentsInMarquee(element) {        element.classList.add("tr-marquee");        element.children[0].style.animation = `move-ltr ${(element.children[0].clientWidth / element.clientWidth) * 5          }s ease-in-out infinite`;        const marqueeText = element.children[0];        if (element.children.length < 2) {          for (let i = 0; i < 1; i++) {            const marqueeText = element.children[0].cloneNode(true);            element.children[0].after(marqueeText);          }        }      }      if (isElementOverflowing(element)) {        wrapContentsInMarquee(element);      }    });  }</script>{{end}}
 |