Browse Source

chore: add empty screens for empty data (balancers, reverses, dns)

Shishkevich D. 3 weeks ago
parent
commit
d6f9f3f6d3

+ 369 - 329
web/html/xui/xray.html

@@ -338,273 +338,296 @@
                 </a-collapse>
               </a-tab-pane>
               <a-tab-pane key="tpl-routing" tab='{{ i18n "pages.xray.Routings"}}' style="padding-top: 20px;">
-                <a-button type="primary" icon="plus" @click="addRule">{{ i18n "pages.xray.rules.add" }}</a-button>
-                <a-table-sortable :columns="isMobile ? rulesMobileColumns : rulesColumns" bordered
-                    :row-key="r => r.key"
-                    :data-source="routingRuleData"
-                    :scroll="isMobile ? {} : { x: 1000 }"
-                    :pagination="false"
-                    :indent-size="0" 
-                    :style="isMobile ? 'padding: 5px 0' : 'margin-top: 10px;'"
-                    v-on:onSort="replaceRule">
-                  <template slot="action" slot-scope="text, rule, index">
-                    <table-sort-trigger :item-index="index"></table-sort-trigger>
-                    <span class="ant-table-row-index"> [[ index+1 ]] </span>
-                    <a-dropdown :trigger="['click']">
-                      <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
-                      <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                        <a-menu-item v-if="index>0" @click="replaceRule(index,0)">
-                          <a-icon type="vertical-align-top"></a-icon>
-                          {{ i18n "pages.xray.rules.first"}}
-                        </a-menu-item>
-                        <a-menu-item v-if="index>0" @click="replaceRule(index,index-1)">
-                          <a-icon type="arrow-up"></a-icon>
-                          {{ i18n "pages.xray.rules.up"}}
-                        </a-menu-item>
-                        <a-menu-item v-if="index<routingRuleData.length-1" @click="replaceRule(index,index+1)">
-                          <a-icon type="arrow-down"></a-icon>
-                          {{ i18n "pages.xray.rules.down"}}
-                        </a-menu-item>
-                        <a-menu-item v-if="index<routingRuleData.length-1" @click="replaceRule(index,routingRuleData.length-1)">
-                          <a-icon type="vertical-align-bottom"></a-icon>
-                          {{ i18n "pages.xray.rules.last"}}
-                        </a-menu-item>
-                        <a-menu-item @click="editRule(index)">
-                          <a-icon type="edit"></a-icon>
-                          {{ i18n "edit" }}
-                        </a-menu-item>
-                        <a-menu-item @click="deleteRule(index)">
-                          <span style="color: #FF4D4F">
-                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                          </span>
-                        </a-menu-item>
-                      </a-menu>
-                    </a-dropdown>
-                  </template>
-                  <template slot="inbound" slot-scope="text, rule, index">
-                    <a-popover :overlay-class-name="themeSwitcher.currentTheme">
-                      <template slot="content">
-                        <p v-if="rule.inboundTag">Inbound Tag: [[ rule.inboundTag ]]</p>
-                        <p v-if="rule.user">User email: [[ rule.user ]]</p>
-                      </template>
-                      [[ [rule.inboundTag,rule.user].join('\n') ]]
-                    </a-popover>
-                  </template>
-                  <template slot="outbound" slot-scope="text, rule, index">
-                    <a-popover :overlay-class-name="themeSwitcher.currentTheme">
-                      <template slot="content">
-                        <p v-if="rule.outboundTag">Outbound Tag: [[ rule.outboundTag ]]</p>
-                      </template>
-                      [[ rule.outboundTag ]]
-                    </a-popover>
-                  </template>
-                  <template slot="balancer" slot-scope="text, rule, index">
-                    <a-popover :overlay-class-name="themeSwitcher.currentTheme">
-                      <template slot="content">
-                        <p v-if="rule.balancerTag">Balancer Tag: [[ rule.balancerTag ]]</p>
-                      </template>
-                      [[ rule.balancerTag ]]
-                    </a-popover>
-                  </template>
-                  <template slot="info" slot-scope="text, rule, index">
-                    <a-popover placement="bottomRight"
-                        v-if="(rule.source+rule.sourcePort+rule.network+rule.protocol+rule.attrs+rule.ip+rule.domain+rule.port).length>0"
-                        :overlay-class-name="themeSwitcher.currentTheme" trigger="click">
-                      <template slot="content">
-                        <table cellpadding="2" style="max-width: 300px;">
-                          <tr v-if="rule.source">
-                            <td>Source</td>
-                            <td><a-tag color="blue" v-for="r in rule.source.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.sourcePort">
-                            <td>Source Port</td>
-                            <td><a-tag color="green" v-for="r in rule.sourcePort.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.network">
-                            <td>Network</td>
-                            <td><a-tag color="blue" v-for="r in rule.network.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.protocol">
-                            <td>Protocol</td>
-                            <td><a-tag color="green" v-for="r in rule.protocol.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.attrs">
-                            <td>Attrs</td>
-                            <td><a-tag color="blue" v-for="r in rule.attrs.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.ip">
-                            <td>IP</td>
-                            <td><a-tag color="green" v-for="r in rule.ip.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.domain">
-                            <td>Domain</td>
-                            <td><a-tag color="blue" v-for="r in rule.domain.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.port">
-                            <td>Port</td>
-                            <td><a-tag color="green" v-for="r in rule.port.split(',')">[[ r ]]</a-tag></td>
-                          </tr>
-                          <tr v-if="rule.balancerTag">
-                            <td>Balancer Tag</td>
-                            <td><a-tag color="blue">[[ rule.balancerTag ]]</a-tag></td>
-                          </tr>
-                        </table>
-                      </template>
-                      <a-button shape="round" size="small" style="font-size: 14px; padding: 0 10px;">
-                        <a-icon type="info"></a-icon>
-                      </a-button>
-                    </a-popover>
-                  </template>
-                </a-table-sortable>
-              </a-tab-pane>
-              <a-tab-pane key="tpl-outbound" tab='{{ i18n "pages.xray.Outbounds"}}' style="padding-top: 20px;" force-render="true">
-                <a-row>
-                  <a-col :xs="12" :sm="12" :lg="12">
-                    <a-button type="primary" icon="plus" @click="addOutbound()" style="margin-bottom: 10px;">
-                      {{ i18n "pages.xray.outbound.addOutbound" }}
-                    </a-button>
-                    <a-button type="primary" icon="cloud" @click="showWarp()" style="margin-bottom: 10px;">WARP</a-button>
-                  </a-col>
-                  <a-col :xs="12" :sm="12" :lg="12" style="text-align: right;">
-                    <a-icon type="sync" :spin="refreshing" @click="refreshOutboundTraffic()" style="margin: 0 5px;"></a-icon>
-                    <a-popconfirm placement="topRight" @confirm="resetOutboundTraffic(-1)"
-                        title='{{ i18n "pages.inbounds.resetTrafficContent"}}'
-                        :overlay-class-name="themeSwitcher.currentTheme"
-                        ok-text='{{ i18n "reset"}}'
-                        cancel-text='{{ i18n "cancel"}}'>
-                      <a-icon slot="icon" type="question-circle-o" :style="themeSwitcher.isDarkTheme ? 'color: #008771' : 'color: #008771'"></a-icon>
-                      <a-icon type="retweet" style="cursor: pointer;"></a-icon>
-                    </a-popconfirm>
-                  </a-col>
-                </a-row>
-                <a-table :columns="outboundColumns" bordered
-                    :row-key="r => r.key"
-                    :data-source="outboundData"
-                    :scroll="isMobile ? {} : { x: 800 }"
-                    :pagination="false"
-                    :indent-size="0"
-                    :style="isMobile ? 'padding: 5px 5px' : 'margin-right: 1px;'">
-                  <template slot="action" slot-scope="text, outbound, index">
-                    [[ index+1 ]]
-                    <a-dropdown :trigger="['click']">
-                      <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
-                      <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                        <a-menu-item v-if="index>0" @click="setFirstOutbound(index)">
-                          <a-icon type="vertical-align-top"></a-icon>
-                          {{ i18n "pages.xray.rules.first"}}
-                        </a-menu-item>
-                        <a-menu-item @click="editOutbound(index)">
-                          <a-icon type="edit"></a-icon>
-                          {{ i18n "edit" }}
-                        </a-menu-item>
-                        <a-menu-item @click="resetOutboundTraffic(index)">
-                          <span>
-                            <a-icon type="retweet"></a-icon> {{ i18n "pages.inbounds.resetTraffic"}}
-                          </span>
-                        </a-menu-item>
-                        <a-menu-item @click="deleteOutbound(index)">
-                          <span style="color: #FF4D4F">
-                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                          </span>
-                        </a-menu-item>
-                      </a-menu>
-                    </a-dropdown>
-                  </template>
-                  <template slot="address" slot-scope="text, outbound, index">
-                    <p style="margin: 0 5px;" v-for="addr in findOutboundAddress(outbound)">[[ addr ]]</p>
-                  </template>
-                  <template slot="protocol" slot-scope="text, outbound, index">
-                    <a-tag style="margin:0;" color="purple">[[ outbound.protocol ]]</a-tag>
-                    <template v-if="[Protocols.VMess, Protocols.VLESS, Protocols.Trojan, Protocols.Shadowsocks].includes(outbound.protocol)">
-                      <a-tag style="margin:0;" color="blue">[[ outbound.streamSettings.network ]]</a-tag>
-                      <a-tag style="margin:0;" v-if="outbound.streamSettings.security=='tls'" color="green">tls</a-tag>
-                      <a-tag style="margin:0;" v-if="outbound.streamSettings.security=='reality'" color="green">reality</a-tag>
+                <a-space direction="vertical" size="middle">
+                    <a-button type="primary" icon="plus" @click="addRule">{{ i18n "pages.xray.rules.add" }}</a-button>
+                    <a-table-sortable :columns="isMobile ? rulesMobileColumns : rulesColumns" bordered
+                        :row-key="r => r.key"
+                        :data-source="routingRuleData"
+                        :scroll="isMobile ? {} : { x: 1000 }"
+                        :pagination="false"
+                        :indent-size="0" 
+                        v-on:onSort="replaceRule">
+                    <template slot="action" slot-scope="text, rule, index">
+                        <table-sort-trigger :item-index="index"></table-sort-trigger>
+                        <span class="ant-table-row-index"> [[ index+1 ]] </span>
+                        <a-dropdown :trigger="['click']">
+                        <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
+                        <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                            <a-menu-item v-if="index>0" @click="replaceRule(index,0)">
+                            <a-icon type="vertical-align-top"></a-icon>
+                            {{ i18n "pages.xray.rules.first"}}
+                            </a-menu-item>
+                            <a-menu-item v-if="index>0" @click="replaceRule(index,index-1)">
+                            <a-icon type="arrow-up"></a-icon>
+                            {{ i18n "pages.xray.rules.up"}}
+                            </a-menu-item>
+                            <a-menu-item v-if="index<routingRuleData.length-1" @click="replaceRule(index,index+1)">
+                            <a-icon type="arrow-down"></a-icon>
+                            {{ i18n "pages.xray.rules.down"}}
+                            </a-menu-item>
+                            <a-menu-item v-if="index<routingRuleData.length-1" @click="replaceRule(index,routingRuleData.length-1)">
+                            <a-icon type="vertical-align-bottom"></a-icon>
+                            {{ i18n "pages.xray.rules.last"}}
+                            </a-menu-item>
+                            <a-menu-item @click="editRule(index)">
+                            <a-icon type="edit"></a-icon>
+                            {{ i18n "edit" }}
+                            </a-menu-item>
+                            <a-menu-item @click="deleteRule(index)">
+                            <span style="color: #FF4D4F">
+                                <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                            </span>
+                            </a-menu-item>
+                        </a-menu>
+                        </a-dropdown>
                     </template>
-                  </template>
-                  <template slot="traffic" slot-scope="text, outbound, index">
-                    <a-tag color="green">[[ findOutboundTraffic(outbound) ]]</a-tag>
-                  </template>
-                </a-table>
+                    <template slot="inbound" slot-scope="text, rule, index">
+                        <a-popover :overlay-class-name="themeSwitcher.currentTheme">
+                        <template slot="content">
+                            <p v-if="rule.inboundTag">Inbound Tag: [[ rule.inboundTag ]]</p>
+                            <p v-if="rule.user">User email: [[ rule.user ]]</p>
+                        </template>
+                        [[ [rule.inboundTag,rule.user].join('\n') ]]
+                        </a-popover>
+                    </template>
+                    <template slot="outbound" slot-scope="text, rule, index">
+                        <a-popover :overlay-class-name="themeSwitcher.currentTheme">
+                        <template slot="content">
+                            <p v-if="rule.outboundTag">Outbound Tag: [[ rule.outboundTag ]]</p>
+                        </template>
+                        [[ rule.outboundTag ]]
+                        </a-popover>
+                    </template>
+                    <template slot="balancer" slot-scope="text, rule, index">
+                        <a-popover :overlay-class-name="themeSwitcher.currentTheme">
+                        <template slot="content">
+                            <p v-if="rule.balancerTag">Balancer Tag: [[ rule.balancerTag ]]</p>
+                        </template>
+                        [[ rule.balancerTag ]]
+                        </a-popover>
+                    </template>
+                    <template slot="info" slot-scope="text, rule, index">
+                        <a-popover placement="bottomRight"
+                            v-if="(rule.source+rule.sourcePort+rule.network+rule.protocol+rule.attrs+rule.ip+rule.domain+rule.port).length>0"
+                            :overlay-class-name="themeSwitcher.currentTheme" trigger="click">
+                        <template slot="content">
+                            <table cellpadding="2" style="max-width: 300px;">
+                            <tr v-if="rule.source">
+                                <td>Source</td>
+                                <td><a-tag color="blue" v-for="r in rule.source.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.sourcePort">
+                                <td>Source Port</td>
+                                <td><a-tag color="green" v-for="r in rule.sourcePort.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.network">
+                                <td>Network</td>
+                                <td><a-tag color="blue" v-for="r in rule.network.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.protocol">
+                                <td>Protocol</td>
+                                <td><a-tag color="green" v-for="r in rule.protocol.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.attrs">
+                                <td>Attrs</td>
+                                <td><a-tag color="blue" v-for="r in rule.attrs.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.ip">
+                                <td>IP</td>
+                                <td><a-tag color="green" v-for="r in rule.ip.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.domain">
+                                <td>Domain</td>
+                                <td><a-tag color="blue" v-for="r in rule.domain.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.port">
+                                <td>Port</td>
+                                <td><a-tag color="green" v-for="r in rule.port.split(',')">[[ r ]]</a-tag></td>
+                            </tr>
+                            <tr v-if="rule.balancerTag">
+                                <td>Balancer Tag</td>
+                                <td><a-tag color="blue">[[ rule.balancerTag ]]</a-tag></td>
+                            </tr>
+                            </table>
+                        </template>
+                        <a-button shape="round" size="small" style="font-size: 14px; padding: 0 10px;">
+                            <a-icon type="info"></a-icon>
+                        </a-button>
+                        </a-popover>
+                    </template>
+                    </a-table-sortable>
+                </a-space>
+              </a-tab-pane>
+              <a-tab-pane key="tpl-outbound" tab='{{ i18n "pages.xray.Outbounds"}}' force-render="true">
+                <a-space direction="vertical" size="middle">
+                    <a-row>
+                        <a-col :xs="12" :sm="12" :lg="12">
+                          <a-space direction="horizontal" size="small">
+                            <a-button type="primary" icon="plus" @click="addOutbound()">
+                                {{ i18n "pages.xray.outbound.addOutbound" }}
+                              </a-button>
+                            <a-button type="primary" icon="cloud" @click="showWarp()">WARP</a-button>
+                          </a-space>
+                        </a-col>
+                        <a-col :xs="12" :sm="12" :lg="12" style="text-align: right;">
+                          <a-icon type="sync" :spin="refreshing" @click="refreshOutboundTraffic()" style="margin: 0 5px;"></a-icon>
+                          <a-popconfirm placement="topRight" @confirm="resetOutboundTraffic(-1)"
+                              title='{{ i18n "pages.inbounds.resetTrafficContent"}}'
+                              :overlay-class-name="themeSwitcher.currentTheme"
+                              ok-text='{{ i18n "reset"}}'
+                              cancel-text='{{ i18n "cancel"}}'>
+                            <a-icon slot="icon" type="question-circle-o" :style="themeSwitcher.isDarkTheme ? 'color: #008771' : 'color: #008771'"></a-icon>
+                            <a-icon type="retweet" style="cursor: pointer;"></a-icon>
+                          </a-popconfirm>
+                        </a-col>
+                      </a-row>
+                      <a-table :columns="outboundColumns" bordered
+                        :row-key="r => r.key"
+                        :data-source="outboundData"
+                        :scroll="isMobile ? {} : { x: 800 }"
+                        :pagination="false"
+                        :indent-size="0">
+                        <template slot="action" slot-scope="text, outbound, index">
+                          [[ index+1 ]]
+                          <a-dropdown :trigger="['click']">
+                            <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
+                            <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                              <a-menu-item v-if="index>0" @click="setFirstOutbound(index)">
+                                <a-icon type="vertical-align-top"></a-icon>
+                                {{ i18n "pages.xray.rules.first"}}
+                              </a-menu-item>
+                              <a-menu-item @click="editOutbound(index)">
+                                <a-icon type="edit"></a-icon>
+                                {{ i18n "edit" }}
+                              </a-menu-item>
+                              <a-menu-item @click="resetOutboundTraffic(index)">
+                                <span>
+                                  <a-icon type="retweet"></a-icon> {{ i18n "pages.inbounds.resetTraffic"}}
+                                </span>
+                              </a-menu-item>
+                              <a-menu-item @click="deleteOutbound(index)">
+                                <span style="color: #FF4D4F">
+                                  <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                                </span>
+                              </a-menu-item>
+                            </a-menu>
+                          </a-dropdown>
+                        </template>
+                        <template slot="address" slot-scope="text, outbound, index">
+                            <p style="margin: 0 5px;" v-for="addr in findOutboundAddress(outbound)">[[ addr ]]</p>
+                        </template>
+                        <template slot="protocol" slot-scope="text, outbound, index">
+                            <a-tag style="margin:0;" color="purple">[[ outbound.protocol ]]</a-tag>
+                            <template v-if="[Protocols.VMess, Protocols.VLESS, Protocols.Trojan, Protocols.Shadowsocks].includes(outbound.protocol)">
+                                <a-tag style="margin:0;" color="blue">[[ outbound.streamSettings.network ]]</a-tag>
+                                <a-tag style="margin:0;" v-if="outbound.streamSettings.security=='tls'" color="green">tls</a-tag>
+                                <a-tag style="margin:0;" v-if="outbound.streamSettings.security=='reality'" color="green">reality</a-tag>
+                            </template>
+                        </template>
+                        <template slot="traffic" slot-scope="text, outbound, index">
+                            <a-tag color="green">[[ findOutboundTraffic(outbound) ]]</a-tag>
+                        </template>
+                    </a-table>
+                </a-space>
               </a-tab-pane>
               <a-tab-pane key="tpl-reverse" tab='{{ i18n "pages.xray.outbound.reverse"}}' style="padding-top: 20px;" force-render="true">
-                <a-button type="primary" icon="plus" @click="addReverse()" style="margin-bottom: 10px;">
-                  {{ i18n "pages.xray.outbound.addReverse" }}
-                </a-button>
-                <a-table :columns="reverseColumns" bordered v-if="reverseData.length>0"
-                    :row-key="r => r.key"
-                    :data-source="reverseData"
-                    :scroll="isMobile ? {} : { x: 200 }"
-                    :pagination="false"
-                    :indent-size="0"
-                    :style="isMobile ? 'padding: 5px 0' : 'margin-left: 1px;'">
-                  <template slot="action" slot-scope="text, reverse, index">
-                    [[ index+1 ]]
-                    <a-dropdown :trigger="['click']">
-                      <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
-                      <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                        <a-menu-item @click="editReverse(index)">
-                          <a-icon type="edit"></a-icon>
-                          {{ i18n "edit" }}
-                        </a-menu-item>
-                        <a-menu-item @click="deleteReverse(index)">
-                          <span style="color: #FF4D4F">
-                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                          </span>
-                        </a-menu-item>
-                      </a-menu>
-                    </a-dropdown>
-                  </template>
-                </a-table>
+                <template v-if="reverseData.length > 0">
+                    <a-space direction="vertical" size="middle">
+                        <a-button type="primary" icon="plus" @click="addReverse()">
+                            {{ i18n "pages.xray.outbound.addReverse" }}
+                        </a-button>
+                        <a-table :columns="reverseColumns" bordered
+                            :row-key="r => r.key"
+                            :data-source="reverseData"
+                            :scroll="isMobile ? {} : { x: 200 }"
+                            :pagination="false"
+                            :indent-size="0">
+                            <template slot="action" slot-scope="text, reverse, index">
+                                [[ index+1 ]]
+                                <a-dropdown :trigger="['click']">
+                                <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
+                                <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                                    <a-menu-item @click="editReverse(index)">
+                                    <a-icon type="edit"></a-icon>
+                                    {{ i18n "edit" }}
+                                    </a-menu-item>
+                                    <a-menu-item @click="deleteReverse(index)">
+                                    <span style="color: #FF4D4F">
+                                        <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                                    </span>
+                                    </a-menu-item>
+                                </a-menu>
+                                </a-dropdown>
+                            </template>
+                        </a-table>
+                    </a-space>
+                </template>
+                <template v-else>
+                    <a-empty description='{{ i18n "emptyReverseDesc" }}' style="margin: 10px;">
+                        <a-button type="primary" icon="plus" @click="addReverse()" style="margin-top: 10px;">
+                            {{ i18n "pages.xray.outbound.addReverse" }}
+                        </a-button>
+                    </a-empty>
+                </template>
               </a-tab-pane>
               <a-tab-pane key="tpl-balancer" tab='{{ i18n "pages.xray.Balancers"}}' style="padding-top: 20px;" force-render="true">
-                <a-button type="primary" icon="plus" @click="addBalancer()" style="margin-bottom: 10px;">
-                  {{ i18n "pages.xray.balancer.addBalancer"}}
-                </a-button>
-                <a-table :columns="balancerColumns" bordered v-if="balancersData.length>0"
-                    :row-key="r => r.key"
-                    :data-source="balancersData"
-                    :scroll="isMobile ? {} : { x: 200 }"
-                    :pagination="false"
-                    :indent-size="0"
-                    :style="isMobile ? 'padding: 5px 0' : 'margin-left: 1px;'">
-                  <template slot="action" slot-scope="text, balancer, index">
-                    [[ index+1 ]]
-                    <a-dropdown :trigger="['click']">
-                      <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
-                      <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                        <a-menu-item @click="editBalancer(index)">
-                          <a-icon type="edit"></a-icon>
-                          {{ i18n "edit" }}
-                        </a-menu-item>
-                        <a-menu-item @click="deleteBalancer(index)">
-                          <span style="color: #FF4D4F">
-                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                          </span>
-                        </a-menu-item>
-                      </a-menu>
-                    </a-dropdown>
-                  </template>
-                  <template slot="strategy" slot-scope="text, balancer, index">
-                    <a-tag style="margin:0;" v-if="balancer.strategy=='random'" color="purple">Random</a-tag>
-                    <a-tag style="margin:0;" v-if="balancer.strategy=='roundRobin'" color="green">Round Robin</a-tag>
-                    <a-tag style="margin:0;" v-if="balancer.strategy=='leastLoad'" color="green">Least Load</a-tag>
-                    <a-tag style="margin:0;" v-if="balancer.strategy=='leastPing'" color="green">Least Ping</a-tag>
-                  </template>
-                  <template slot="selector" slot-scope="text, balancer, index">
-                    <a-tag class="info-large-tag" style="margin:1;" v-for="sel in balancer.selector">[[ sel ]]</a-tag>
-                  </template>
-                </a-table>
-                <a-radio-group
-                    v-if="observatoryEnable || burstObservatoryEnable"
-                    v-model="obsSettings"
-                    @change="changeObsCode"
-                    button-style="solid"
-                    style="margin: 10px 0;"
-                    :size="isMobile ? 'small' : ''">
-                  <a-radio-button value="observatory" v-if="observatoryEnable">Observatory</a-radio-button>
-                  <a-radio-button value="burstObservatory" v-if="burstObservatoryEnable">Burst Observatory</a-radio-button>
-                </a-radio-group>
-                <textarea style="position:absolute; left: -800px;" id="obsSetting"></textarea>
+                <template v-if="balancersData.length > 0">
+                    <a-space direction="vertical" size="middle">
+                        <a-button type="primary" icon="plus" @click="addBalancer()">
+                            {{ i18n "pages.xray.balancer.addBalancer"}}
+                        </a-button>
+                        <a-table :columns="balancerColumns" bordered 
+                            :row-key="r => r.key"
+                            :data-source="balancersData"
+                            :scroll="isMobile ? {} : { x: 200 }"
+                            :pagination="false"
+                            :indent-size="0">
+                            <template slot="action" slot-scope="text, balancer, index">
+                                [[ index+1 ]]
+                                <a-dropdown :trigger="['click']">
+                                <a-icon @click="e => e.preventDefault()" type="more" style="font-size: 16px; text-decoration: bold;"></a-icon>
+                                <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                                    <a-menu-item @click="editBalancer(index)">
+                                    <a-icon type="edit"></a-icon>
+                                    {{ i18n "edit" }}
+                                    </a-menu-item>
+                                    <a-menu-item @click="deleteBalancer(index)">
+                                    <span style="color: #FF4D4F">
+                                        <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                                    </span>
+                                    </a-menu-item>
+                                </a-menu>
+                                </a-dropdown>
+                            </template>
+                            <template slot="strategy" slot-scope="text, balancer, index">
+                                <a-tag style="margin:0;" v-if="balancer.strategy=='random'" color="purple">Random</a-tag>
+                                <a-tag style="margin:0;" v-if="balancer.strategy=='roundRobin'" color="green">Round Robin</a-tag>
+                                <a-tag style="margin:0;" v-if="balancer.strategy=='leastLoad'" color="green">Least Load</a-tag>
+                                <a-tag style="margin:0;" v-if="balancer.strategy=='leastPing'" color="green">Least Ping</a-tag>
+                            </template>
+                            <template slot="selector" slot-scope="text, balancer, index">
+                                <a-tag class="info-large-tag" style="margin:1;" v-for="sel in balancer.selector">[[ sel ]]</a-tag>
+                            </template>
+                        </a-table>
+                        <a-radio-group
+                            v-if="observatoryEnable || burstObservatoryEnable"
+                            v-model="obsSettings"
+                            @change="changeObsCode"
+                            button-style="solid"
+                            :size="isMobile ? 'small' : ''">
+                            <a-radio-button value="observatory" v-if="observatoryEnable">Observatory</a-radio-button>
+                            <a-radio-button value="burstObservatory" v-if="burstObservatoryEnable">Burst Observatory</a-radio-button>
+                        </a-radio-group>
+                        <textarea style="position:absolute; left: -800px;" id="obsSetting"></textarea>
+                    </a-space>
+                </template>
+                <template v-else>
+                    <a-empty description='{{ i18n "emptyBalancersDesc" }}' style="margin: 10px;">
+                        <a-button type="primary" icon="plus" @click="addBalancer()" style="margin-top: 10px;">
+                            {{ i18n "pages.xray.balancer.addBalancer"}}
+                        </a-button>
+                    </a-empty>
+                </template>
               </a-tab-pane>
             <a-tab-pane key="tpl-dns" tab='DNS' style="padding-top: 20px;" force-render="true">
                 <a-collapse>
@@ -667,79 +690,96 @@
                     </a-collapse-panel>
                     <template v-if="enableDNS">
                         <a-collapse-panel header='DNS'>
-                            <a-button type="primary" icon="plus" @click="addDNSServer()" style="margin: 10px;">{{ i18n
-                                "pages.xray.dns.add" }}</a-button>
-                            <a-table :columns="dnsColumns" bordered v-if="dnsServers.length>0" :row-key="r => r.key"
-                                :data-source="dnsServers" :scroll="isMobile ? {} : { x: 200 }" :pagination="false" :indent-size="0"
-                                style="margin: 10px; margin-top: 10px;">
-                                <template slot="action" slot-scope="text,dns,index">
-                                    [[ index+1 ]]
-                                    <a-dropdown :trigger="['click']">
-                                        <a-icon @click="e => e.preventDefault()" type="more"
-                                            style="font-size: 16px; text-decoration: bold;"></a-icon>
-                                        <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                                            <a-menu-item @click="editDNSServer(index)">
-                                                <a-icon type="edit"></a-icon>
-                                                {{ i18n "edit" }}
-                                            </a-menu-item>
-                                            <a-menu-item @click="deleteDNSServer(index)">
-                                                <span style="color: #FF4D4F">
-                                                    <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                                                </span>
-                                            </a-menu-item>
-                                        </a-menu>
-                                    </a-dropdown>
-                                </template>
-                                <template slot="address" slot-scope="dns,index">
-                                    <span v-if="typeof dns == 'object'">[[ dns.address ]]</span>
-                                    <span v-else>[[ dns ]]</span>
-                                </template>
-                                <template slot="domain" slot-scope="dns,index">
-                                    <span v-if="typeof dns == 'object'">[[ dns.domains.join(",") ]]</span>
-                                </template>
-                                <template slot="expectIPs" slot-scope="dns,index">
-                                    <span v-if="typeof dns == 'object'">[[ dns.expectIPs.join(",") ]]</span>
-                                </template>
-                            </a-table>
+                            <template v-if="dnsServers.length > 0">
+                                <a-space direction="vertical" size="middle">
+                                    <a-button type="primary" icon="plus" @click="addDNSServer()">{{ i18n
+                                        "pages.xray.dns.add" }}</a-button>
+                                    <a-table :columns="dnsColumns" bordered :row-key="r => r.key"
+                                        :data-source="dnsServers" :scroll="isMobile ? {} : { x: 200 }" :pagination="false" :indent-size="0">
+                                        <template slot="action" slot-scope="text,dns,index">
+                                            [[ index+1 ]]
+                                            <a-dropdown :trigger="['click']">
+                                                <a-icon @click="e => e.preventDefault()" type="more"
+                                                    style="font-size: 16px; text-decoration: bold;"></a-icon>
+                                                <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                                                    <a-menu-item @click="editDNSServer(index)">
+                                                        <a-icon type="edit"></a-icon>
+                                                        {{ i18n "edit" }}
+                                                    </a-menu-item>
+                                                    <a-menu-item @click="deleteDNSServer(index)">
+                                                        <span style="color: #FF4D4F">
+                                                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                                                        </span>
+                                                    </a-menu-item>
+                                                </a-menu>
+                                            </a-dropdown>
+                                        </template>
+                                        <template slot="address" slot-scope="dns,index">
+                                            <span v-if="typeof dns == 'object'">[[ dns.address ]]</span>
+                                            <span v-else>[[ dns ]]</span>
+                                        </template>
+                                        <template slot="domain" slot-scope="dns,index">
+                                            <span v-if="typeof dns == 'object'">[[ dns.domains.join(",") ]]</span>
+                                        </template>
+                                        <template slot="expectIPs" slot-scope="dns,index">
+                                            <span v-if="typeof dns == 'object'">[[ dns.expectIPs.join(",") ]]</span>
+                                        </template>
+                                    </a-table>
+                                </a-space>
+                            </template>
+                            <template v-else>
+                                <a-empty description='{{ i18n "emptyDnsDesc" }}' style="margin: 10px;">
+                                    <a-button type="primary" icon="plus" @click="addDNSServer()" style="margin-top: 10px;">{{ i18n "pages.xray.dns.add" }}</a-button>
+                                </a-empty>
+                            </template>
                         </a-collapse-panel>
-                        <a-collapse-panel header='FakeDNS'>
-                            <a-button type="primary" icon="plus" @click="addFakedns()" style="margin: 10px;">{{ i18n
-                                "pages.xray.fakedns.add" }}</a-button>
-                            <a-table :columns="fakednsColumns" bordered v-if="fakeDns && fakeDns.length>0" :row-key="r => r.key"
-                                :data-source="fakeDns" :scroll="isMobile ? {} : { x: 200 }" :pagination="false" :indent-size="0"
-                                style="margin: 10px; margin-top: 10px;">
-                                <template slot="action" slot-scope="text,fakedns,index">
-                                    [[ index+1 ]]
-                                    <a-dropdown :trigger="['click']">
-                                        <a-icon @click="e => e.preventDefault()" type="more"
-                                            style="font-size: 16px; text-decoration: bold;"></a-icon>
-                                        <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
-                                            <a-menu-item @click="editFakedns(index)">
-                                                <a-icon type="edit"></a-icon>
-                                                {{ i18n "edit" }}
-                                            </a-menu-item>
-                                            <a-menu-item @click="deleteFakedns(index)">
-                                                <span style="color: #FF4D4F">
-                                                    <a-icon type="delete"></a-icon> {{ i18n "delete"}}
-                                                </span>
-                                            </a-menu-item>
-                                        </a-menu>
-                                    </a-dropdown>
-                                </template>
-                            </a-table>
+                        <a-collapse-panel header='Fake DNS'>
+                            <template v-if="fakeDns && fakeDns.length > 0">
+                                <a-space direction="vertical" size="middle">
+                                    <a-button type="primary" icon="plus" @click="addFakedns()">{{ i18n "pages.xray.fakedns.add" }}</a-button>
+                                    <a-table :columns="fakednsColumns" bordered :row-key="r => r.key"
+                                        :data-source="fakeDns" :scroll="isMobile ? {} : { x: 200 }" :pagination="false" :indent-size="0">
+                                        <template slot="action" slot-scope="text,fakedns,index">
+                                            [[ index+1 ]]
+                                            <a-dropdown :trigger="['click']">
+                                                <a-icon @click="e => e.preventDefault()" type="more"
+                                                    style="font-size: 16px; text-decoration: bold;"></a-icon>
+                                                <a-menu slot="overlay" :theme="themeSwitcher.currentTheme">
+                                                    <a-menu-item @click="editFakedns(index)">
+                                                        <a-icon type="edit"></a-icon>
+                                                        {{ i18n "edit" }}
+                                                    </a-menu-item>
+                                                    <a-menu-item @click="deleteFakedns(index)">
+                                                        <span style="color: #FF4D4F">
+                                                            <a-icon type="delete"></a-icon> {{ i18n "delete"}}
+                                                        </span>
+                                                    </a-menu-item>
+                                                </a-menu>
+                                            </a-dropdown>
+                                        </template>
+                                    </a-table>
+                                </a-space>
+                            </template>
+                            <template v-else>
+                                <a-empty description='{{ i18n "emptyFakeDnsDesc" }}' style="margin: 20px;">
+                                    <a-button type="primary" icon="plus" @click="addFakedns()" style="margin-top: 10px;">{{ i18n "pages.xray.fakedns.add" }}</a-button>
+                                </a-empty>
+                            </template>
                         </a-collapse-panel>
                     </template>
                 </a-collapse>
             </a-tab-pane>
               <a-tab-pane key="tpl-advanced" tab='{{ i18n "pages.xray.advancedTemplate"}}' style="padding-top: 20px;" force-render="true">
-                <a-list-item-meta title='{{ i18n "pages.xray.Template"}}' description='{{ i18n "pages.xray.TemplateDesc"}}'></a-list-item-meta>
-                <a-radio-group v-model="advSettings" @change="changeCode" button-style="solid" style="margin: 10px 0;" :size="isMobile ? 'small' : ''">
-                  <a-radio-button value="xraySetting">{{ i18n "pages.xray.completeTemplate"}}</a-radio-button>
-                  <a-radio-button value="inboundSettings">{{ i18n "pages.xray.Inbounds" }}</a-radio-button>
-                  <a-radio-button value="outboundSettings">{{ i18n "pages.xray.Outbounds" }}</a-radio-button>
-                  <a-radio-button value="routingRuleSettings">{{ i18n "pages.xray.Routings" }}</a-radio-button>
-                </a-radio-group>
-                <textarea style="position:absolute; left: -800px;" id="xraySetting"></textarea>
+                <a-space direction="vertical" size="small">
+                    <a-list-item-meta title='{{ i18n "pages.xray.Template"}}' description='{{ i18n "pages.xray.TemplateDesc"}}'></a-list-item-meta>
+                    <a-radio-group v-model="advSettings" @change="changeCode" button-style="solid" style="margin: 10px 0;" :size="isMobile ? 'small' : ''">
+                        <a-radio-button value="xraySetting">{{ i18n "pages.xray.completeTemplate"}}</a-radio-button>
+                        <a-radio-button value="inboundSettings">{{ i18n "pages.xray.Inbounds" }}</a-radio-button>
+                        <a-radio-button value="outboundSettings">{{ i18n "pages.xray.Outbounds" }}</a-radio-button>
+                        <a-radio-button value="routingRuleSettings">{{ i18n "pages.xray.Routings" }}</a-radio-button>
+                    </a-radio-group>
+                    <textarea style="position:absolute; left: -800px;" id="xraySetting"></textarea>
+                </a-space>
               </a-tab-pane>
             </a-tabs>
           </a-space>

+ 4 - 0
web/translation/translate.en_US.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "Panel default URI path is insecure. Please configure a complex URI path."
 "secAlertSubURI" = "Subscription default URI path is insecure. Please configure a complex URI path."
 "secAlertSubJsonURI" = "Subscription JSON default URI path is insecure. Please configure a complex URI path."
+"emptyDnsDesc" = "No added DNS servers."
+"emptyFakeDnsDesc" = "No added Fake DNS servers."
+"emptyBalancersDesc" = "No added balancers."
+"emptyReverseDesc" = "No added reverse proxies."
 
 [menu]
 "dashboard" = "Overview"

+ 4 - 0
web/translation/translate.es_ES.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "La ruta URI predeterminada del panel no es segura. Por favor, configure una ruta URI compleja."
 "secAlertSubURI" = "La ruta URI predeterminada de la suscripción no es segura. Por favor, configure una ruta URI compleja."
 "secAlertSubJsonURI" = "La ruta URI JSON predeterminada de la suscripción no es segura. Por favor, configure una ruta URI compleja."
+"emptyDnsDesc" = "No hay servidores DNS añadidos."
+"emptyFakeDnsDesc" = "No hay servidores Fake DNS añadidos."
+"emptyBalancersDesc" = "No hay balanceadores añadidos."
+"emptyReverseDesc" = "No hay proxies inversos añadidos."
 
 [menu]
 "dashboard" = "Estado del Sistema"

+ 4 - 0
web/translation/translate.fa_IR.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "مسیر پیش‌فرض لینک پنل ناامن است. لطفاً یک مسیر پیچیده تنظیم کنید"
 "secAlertSubURI" = "مسیر پیش‌فرض لینک سابسکریپشن ناامن است. لطفاً یک مسیر پیچیده تنظیم کنید"
 "secAlertSubJsonURI" = "مسیر پیش‌فرض لینک سابسکریپشن جیسون ناامن است. لطفاً یک مسیر پیچیده تنظیم کنید"
+"emptyDnsDesc" = "هیچ سرور DNS اضافه نشده است."
+"emptyFakeDnsDesc" = "هیچ سرور Fake DNS اضافه نشده است."
+"emptyBalancersDesc" = "هیچ بالانسر اضافه نشده است."
+"emptyReverseDesc" = "هیچ پروکسی معکوس اضافه نشده است."
 
 [menu]
 "dashboard" = "نمای کلی"

+ 4 - 0
web/translation/translate.id_ID.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "Jalur URI default panel tidak aman. Harap konfigurasi jalur URI kompleks."
 "secAlertSubURI" = "Jalur URI default langganan tidak aman. Harap konfigurasi jalur URI kompleks."
 "secAlertSubJsonURI" = "Jalur URI default JSON langganan tidak aman. Harap konfigurasikan jalur URI kompleks."
+"emptyDnsDesc" = "Tidak ada server DNS yang ditambahkan."
+"emptyFakeDnsDesc" = "Tidak ada server Fake DNS yang ditambahkan."
+"emptyBalancersDesc" = "Tidak ada penyeimbang yang ditambahkan."
+"emptyReverseDesc" = "Tidak ada proxy terbalik yang ditambahkan."
 
 [menu]
 "dashboard" = "Ikhtisar"

+ 4 - 0
web/translation/translate.ja_JP.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "デフォルトのURIパスは安全ではありません。複雑なURIパスを設定してください。"
 "secAlertSubURI" = "サブスクリプションのデフォルトURIパスは安全ではありません。複雑なURIパスを設定してください。"
 "secAlertSubJsonURI" = "JSONサブスクリプションのデフォルトURIパスは安全ではありません。複雑なURIパスを設定してください。"
+"emptyDnsDesc" = "追加されたDNSサーバーはありません。"
+"emptyFakeDnsDesc" = "追加されたFake DNSサーバーはありません。"
+"emptyBalancersDesc" = "追加されたバランサーはありません。"
+"emptyReverseDesc" = "追加されたリバースプロキシはありません。"
 
 [menu]
 "dashboard" = "ダッシュボード"

+ 4 - 0
web/translation/translate.pt_BR.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "O caminho URI padrão do painel não é seguro. Configure um caminho URI complexo."
 "secAlertSubURI" = "O caminho URI padrão de inscrição não é seguro. Configure um caminho URI complexo."
 "secAlertSubJsonURI" = "O caminho URI JSON de inscrição padrão não é seguro. Configure um caminho URI complexo."
+"emptyDnsDesc" = "Nenhum servidor DNS adicionado."
+"emptyFakeDnsDesc" = "Nenhum servidor Fake DNS adicionado."
+"emptyBalancersDesc" = "Nenhum balanceador adicionado."
+"emptyReverseDesc" = "Nenhum proxy reverso adicionado."
 
 [menu]
 "dashboard" = "Visão Geral"

+ 4 - 0
web/translation/translate.ru_RU.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "URI-путь по умолчанию панели небезопасен. Пожалуйста, настройте сложный URI-путь."
 "secAlertSubURI" = "URI-путь по умолчанию подписки небезопасен. Пожалуйста, настройте сложный URI-путь."
 "secAlertSubJsonURI" = "URI-путь по умолчанию для JSON подписки небезопасен. Пожалуйста, настройте сложный URI-путь."
+"emptyDnsDesc" = "Нет добавленных DNS-серверов."
+"emptyFakeDnsDesc" = "Нет добавленных Fake DNS-серверов."
+"emptyBalancersDesc" = "Нет добавленных балансировщиков."
+"emptyReverseDesc" = "Нет добавленных обратных прокси."
 
 [menu]
 "dashboard" = "Статус системы"

+ 4 - 0
web/translation/translate.tr_TR.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "Panel varsayılan URI yolu güvensiz. Karmaşık bir URI yolu yapılandırın."
 "secAlertSubURI" = "Abonelik varsayılan URI yolu güvensiz. Karmaşık bir URI yolu yapılandırın."
 "secAlertSubJsonURI" = "Abonelik JSON varsayılan URI yolu güvensiz. Karmaşık bir URI yolu yapılandırın."
+"emptyDnsDesc" = "Eklenmiş DNS sunucusu yok."
+"emptyFakeDnsDesc" = "Eklenmiş Fake DNS sunucusu yok."
+"emptyBalancersDesc" = "Eklenmiş dengeleyici yok."
+"emptyReverseDesc" = "Eklenmiş ters proxy yok."
 
 [menu]
 "dashboard" = "Genel Bakış"

+ 4 - 0
web/translation/translate.uk_UA.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "Стандартний URI-шлях панелі небезпечний. Будь ласка, сконфігуруйте складний URI-шлях."
 "secAlertSubURI" = "Стандартний URI-шлях підписки небезпечний. Будь ласка, сконфігуруйте складний URI-шлях."
 "secAlertSubJsonURI" = "Стандартний URI-шлях JSON підписки небезпечний. Будь ласка, сконфігуруйте складний URI-шлях."
+"emptyDnsDesc" = "Немає доданих DNS-серверів."
+"emptyFakeDnsDesc" = "Немає доданих Fake DNS-серверів."
+"emptyBalancersDesc" = "Немає доданих балансувальників."
+"emptyReverseDesc" = "Немає доданих зворотних проксі."
 
 [menu]
 "dashboard" = "Огляд"

+ 4 - 0
web/translation/translate.vi_VN.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "Đường dẫn URI mặc định của bảng điều khiển không an toàn. Vui lòng cấu hình một đường dẫn URI phức tạp."
 "secAlertSubURI" = "Đường dẫn URI mặc định của đăng ký không an toàn. Vui lòng cấu hình một đường dẫn URI phức tạp."
 "secAlertSubJsonURI" = "Đường dẫn URI JSON mặc định của đăng ký không an toàn. Vui lòng cấu hình một đường dẫn URI phức tạp."
+"emptyDnsDesc" = "Không có máy chủ DNS nào được thêm."
+"emptyFakeDnsDesc" = "Không có máy chủ Fake DNS nào được thêm."
+"emptyBalancersDesc" = "Không có bộ cân bằng tải nào được thêm."
+"emptyReverseDesc" = "Không có proxy ngược nào được thêm."
 
 [menu]
 "dashboard" = "Trạng thái hệ thống"

+ 4 - 0
web/translation/translate.zh_CN.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "面板默认 URI 路径不安全。请配置复杂的 URI 路径。"
 "secAlertSubURI" = "订阅默认 URI 路径不安全。请配置复杂的 URI 路径。"
 "secAlertSubJsonURI" = "订阅 JSON 默认 URI 路径不安全。请配置复杂的 URI 路径。"
+"emptyDnsDesc" = "未添加DNS服务器。"
+"emptyFakeDnsDesc" = "未添加Fake DNS服务器。"
+"emptyBalancersDesc" = "未添加负载均衡器。"
+"emptyReverseDesc" = "未添加反向代理。"
 
 [menu]
 "dashboard" = "系统状态"

+ 4 - 0
web/translation/translate.zh_TW.toml

@@ -61,6 +61,10 @@
 "secAlertPanelURI" = "面板預設 URI 路徑不安全。請配置複雜的 URI 路徑。"
 "secAlertSubURI" = "訂閱預設 URI 路徑不安全。請配置複雜的 URI 路徑。"
 "secAlertSubJsonURI" = "訂閱 JSON 預設 URI 路徑不安全。請配置複雜的 URI 路徑。"
+"emptyDnsDesc" = "未添加DNS伺服器。"
+"emptyFakeDnsDesc" = "未添加Fake DNS伺服器。"
+"emptyBalancersDesc" = "未添加負載平衡器。"
+"emptyReverseDesc" = "未添加反向代理。"
 
 [menu]
 "dashboard" = "系統狀態"