mirror of
https://github.com/alireza0/x-ui.git
synced 2026-03-14 05:23:09 +00:00
789 lines
42 KiB
HTML
789 lines
42 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
{{template "head" .}}
|
|
<style>
|
|
@media (min-width: 769px) {
|
|
.ant-layout-content {
|
|
margin: 24px 16px;
|
|
}
|
|
}
|
|
|
|
@media (max-width: 768px) {
|
|
.ant-tabs-nav .ant-tabs-tab {
|
|
margin: 0;
|
|
padding: 12px .5rem;
|
|
}
|
|
}
|
|
|
|
.ant-tabs-bar {
|
|
margin: 0;
|
|
}
|
|
|
|
.ant-list-item {
|
|
display: block;
|
|
}
|
|
|
|
.collapse-title {
|
|
color: inherit;
|
|
font-weight: bold;
|
|
font-size: 18px;
|
|
padding: 10px 20px;
|
|
border-bottom: 2px solid;
|
|
}
|
|
|
|
.collapse-title > i {
|
|
color: inherit;
|
|
font-size: 24px;
|
|
}
|
|
</style>
|
|
<body>
|
|
<a-layout id="app" v-cloak :class="themeSwitcher.currentTheme">
|
|
{{ template "commonSider" . }}
|
|
<a-layout id="content-layout">
|
|
<a-layout-content>
|
|
<a-spin :spinning="spinning" :delay="500" tip='{{ i18n "loading"}}'>
|
|
<transition name="list" appear>
|
|
<a-alert type="error" v-if="showAlert" style="margin-bottom: 10px"
|
|
message='{{ i18n "secAlertTitle" }}'
|
|
color="red"
|
|
description='{{ i18n "secAlertSsl" }}'
|
|
show-icon closable
|
|
>
|
|
</a-alert>
|
|
</transition>
|
|
<a-space direction="vertical">
|
|
<a-card hoverable style="margin-bottom: .5rem;">
|
|
<a-row>
|
|
<a-col :xs="24" :sm="8" style="padding: 4px;">
|
|
<a-space direction="horizontal">
|
|
<a-button type="primary" :disabled="saveBtnDisable" @click="updateXraySetting">{{ i18n "pages.settings.save" }}</a-button>
|
|
<a-button type="danger" :disabled="!saveBtnDisable" @click="restartPanel">{{ i18n "pages.settings.restartPanel" }}</a-button>
|
|
</a-space>
|
|
</a-col>
|
|
<a-col :xs="24" :sm="16">
|
|
<template>
|
|
<div>
|
|
<a-back-top :target="() => document.getElementById('content-layout')" visibility-height="200">
|
|
</a-back-top>
|
|
<a-alert type="warning" style="float: right; width: fit-content"
|
|
message='{{ i18n "pages.settings.infoDesc" }}'
|
|
show-icon
|
|
>
|
|
</div>
|
|
</template>
|
|
</a-col>
|
|
</a-row>
|
|
</a-card>
|
|
<a-tabs default-active-key="1">
|
|
<a-tab-pane key="tpl-1" tab='{{ i18n "pages.xray.basicTemplate"}}' style="padding-top: 20px;">
|
|
<a-collapse>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.generalConfigs"}}'>
|
|
<a-row :xs="24" :sm="24" :lg="12">
|
|
<a-alert type="warning" style="text-align: center;">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.generalConfigsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<a-list-item>
|
|
<a-row style="padding: 20px">
|
|
<a-col :lg="24" :xl="12">
|
|
<a-list-item-meta
|
|
title='{{ i18n "pages.xray.FreedomStrategy" }}'
|
|
description='{{ i18n "pages.xray.FreedomStrategyDesc" }}'/>
|
|
</a-col>
|
|
<a-col :lg="24" :xl="12">
|
|
<template>
|
|
<a-select
|
|
v-model="freedomStrategy"
|
|
style="width: 100%" :dropdown-class-name="themeSwitcher.currentTheme">
|
|
<a-select-option v-for="s in outboundDomainStrategies" :value="s">[[ s ]]</a-select-option>
|
|
</a-select>
|
|
</template>
|
|
</a-col>
|
|
</a-row>
|
|
</a-list-item>
|
|
<a-row style="padding: 20px">
|
|
<a-col :lg="24" :xl="12">
|
|
<a-list-item-meta
|
|
title='{{ i18n "pages.xray.RoutingStrategy" }}'
|
|
description='{{ i18n "pages.xray.RoutingStrategyDesc" }}'/>
|
|
</a-col>
|
|
<a-col :lg="24" :xl="12">
|
|
<template>
|
|
<a-select
|
|
v-model="routingStrategy"
|
|
style="width: 100%" :dropdown-class-name="themeSwitcher.currentTheme">
|
|
<a-select-option v-for="s in routingDomainStrategies" :value="s">[[ s ]]</a-select-option>
|
|
</a-select>
|
|
</template>
|
|
</a-col>
|
|
</a-row>
|
|
</a-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.blockConfigs"}}'>
|
|
<a-row :xs="24" :sm="24" :lg="12">
|
|
<a-alert type="warning" style="text-align: center;">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.generalConfigsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.Torrent"}}' desc='{{ i18n "pages.xray.TorrentDesc"}}' v-model="torrentSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.PrivateIp"}}' desc='{{ i18n "pages.xray.PrivateIpDesc"}}' v-model="privateIpSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.Ads"}}' desc='{{ i18n "pages.xray.AdsDesc"}}' v-model="AdsSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.Family"}}' desc='{{ i18n "pages.xray.FamilyDesc"}}' v-model="familyProtectSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.blockCountryConfigs"}}'>
|
|
<a-row :xs="24" :sm="24" :lg="12">
|
|
<a-alert type="warning" style="text-align: center;">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.blockCountryConfigsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.IRIp"}}' desc='{{ i18n "pages.xray.IRIpDesc"}}' v-model="IRIpSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.IRDomain"}}' desc='{{ i18n "pages.xray.IRDomainDesc"}}' v-model="IRDomainSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.ChinaIp"}}' desc='{{ i18n "pages.xray.ChinaIpDesc"}}' v-model="ChinaIpSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.ChinaDomain"}}' desc='{{ i18n "pages.xray.ChinaDomainDesc"}}' v-model="ChinaDomainSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.RussiaIp"}}' desc='{{ i18n "pages.xray.RussiaIpDesc"}}' v-model="RussiaIpSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.RussiaDomain"}}' desc='{{ i18n "pages.xray.RussiaDomainDesc"}}' v-model="RussiaDomainSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.directCountryConfigs"}}'>
|
|
<a-row :xs="24" :sm="24" :lg="12">
|
|
<a-alert type="warning" style="text-align: center;">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.directCountryConfigsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectIRIp"}}' desc='{{ i18n "pages.xray.DirectIRIpDesc"}}' v-model="IRIpDirectSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectIRDomain"}}' desc='{{ i18n "pages.xray.DirectIRDomainDesc"}}' v-model="IRDomainDirectSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectChinaIp"}}' desc='{{ i18n "pages.xray.DirectChinaIpDesc"}}' v-model="ChinaIpDirectSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectChinaDomain"}}' desc='{{ i18n "pages.xray.DirectChinaDomainDesc"}}' v-model="ChinaDomainDirectSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectRussiaIp"}}' desc='{{ i18n "pages.xray.DirectRussiaIpDesc"}}' v-model="RussiaIpDirectSettings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.DirectRussiaDomain"}}' desc='{{ i18n "pages.xray.DirectRussiaDomainDesc"}}' v-model="RussiaDomainDirectSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.ipv4Configs"}}'>
|
|
<a-row :xs="24" :sm="24" :lg="12">
|
|
<a-alert type="warning" style="text-align: center;">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.ipv4ConfigsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.GoogleIPv4"}}' desc='{{ i18n "pages.xray.GoogleIPv4Desc"}}' v-model="GoogleIPv4Settings"></setting-list-item>
|
|
<setting-list-item type="switch" title='{{ i18n "pages.xray.NetflixIPv4"}}' desc='{{ i18n "pages.xray.NetflixIPv4Desc"}}' v-model="NetflixIPv4Settings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.settings.resetDefaultConfig"}}'>
|
|
<a-space direction="horizontal" style="padding: 0 20px">
|
|
<a-button type="primary" @click="resetXrayConfigToDefault">{{ i18n "pages.settings.resetDefaultConfig" }}</a-button>
|
|
</a-space>
|
|
</a-collapse-panel>
|
|
</a-collapse>
|
|
</a-tab-pane>
|
|
<a-tab-pane key="tpl-2" tab='{{ i18n "pages.xray.manualLists"}}' style="padding-top: 20px;">
|
|
<a-row :xs="24" :sm="24" :lg="12" style="margin-bottom: 10px;">
|
|
<a-alert type="warning" style="float: left; width: fit-content">
|
|
<template slot="message">
|
|
<a-icon type="exclamation-circle" theme="filled" style="color: #FFA031"></a-icon>
|
|
{{ i18n "pages.xray.manualListsDesc" }}
|
|
</template>
|
|
</a-alert>
|
|
</a-row>
|
|
<a-collapse>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.manualBlockedIPs"}}'>
|
|
<setting-list-item type="textarea" v-model="manualBlockedIPs"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.manualBlockedDomains"}}'>
|
|
<setting-list-item type="textarea" v-model="manualBlockedDomains"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.manualDirectIPs"}}'>
|
|
<setting-list-item type="textarea" v-model="manualDirectIPs"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.manualDirectDomains"}}'>
|
|
<setting-list-item type="textarea" v-model="manualDirectDomains"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.manualIPv4Domains"}}'>
|
|
<setting-list-item type="textarea" v-model="manualIPv4Domains"></setting-list-item>
|
|
</a-collapse-panel>
|
|
</a-collapse>
|
|
</a-tab-pane>
|
|
<a-tab-pane key="tpl-3" tab='{{ i18n "pages.xray.advancedTemplate"}}' style="padding-top: 20px;">
|
|
<a-collapse>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.Inbounds"}}'>
|
|
<setting-list-item type="textarea" title='{{ i18n "pages.xray.Inbounds"}}' desc='{{ i18n "pages.xray.InboundsDesc"}}' v-model="inboundSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.Outbounds"}}'>
|
|
<setting-list-item type="textarea" title='{{ i18n "pages.xray.Outbounds"}}' desc='{{ i18n "pages.xray.OutboundsDesc"}}' v-model="outboundSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
<a-collapse-panel header='{{ i18n "pages.xray.Routings"}}'>
|
|
<setting-list-item type="textarea" title='{{ i18n "pages.xray.Routings"}}' desc='{{ i18n "pages.xray.RoutingsDesc"}}' v-model="routingRuleSettings"></setting-list-item>
|
|
</a-collapse-panel>
|
|
</a-collapse>
|
|
</a-tab-pane>
|
|
<a-tab-pane key="tpl-4" tab='{{ i18n "pages.xray.completeTemplate"}}' style="padding-top: 20px;">
|
|
<setting-list-item type="textarea" title='{{ i18n "pages.xray.Template"}}' desc='{{ i18n "pages.xray.TemplateDesc"}}' v-model="xraySetting"></setting-list-item>
|
|
</a-tab-pane>
|
|
</a-tabs>
|
|
</a-space>
|
|
</a-spin>
|
|
</a-layout-content>
|
|
</a-layout>
|
|
</a-layout>
|
|
{{template "js" .}}
|
|
{{template "component/themeSwitcher" .}}
|
|
{{template "component/setting"}}
|
|
<script>
|
|
|
|
const app = new Vue({
|
|
delimiters: ['[[', ']]'],
|
|
el: '#app',
|
|
data: {
|
|
siderDrawer,
|
|
themeSwitcher,
|
|
spinning: false,
|
|
oldXraySetting: '',
|
|
xraySetting: '',
|
|
saveBtnDisable: true,
|
|
showAlert: false,
|
|
ipv4Settings: {
|
|
tag: "IPv4",
|
|
protocol: "freedom",
|
|
settings: {
|
|
domainStrategy: "UseIPv4"
|
|
}
|
|
},
|
|
directSettings: {
|
|
tag: "direct",
|
|
protocol: "freedom"
|
|
},
|
|
outboundDomainStrategies: ["AsIs", "UseIP", "UseIPv4", "UseIPv6"],
|
|
routingDomainStrategies: ["AsIs", "IPIfNonMatch", "IPOnDemand"],
|
|
settingsData: {
|
|
protocols: {
|
|
bittorrent: ["bittorrent"],
|
|
},
|
|
ips: {
|
|
local: ["geoip:private"],
|
|
cn: ["geoip:cn"],
|
|
ir: ["geoip:ir"],
|
|
ru: ["geoip:ru"],
|
|
},
|
|
domains: {
|
|
ads: [
|
|
"geosite:category-ads-all",
|
|
"ext:iran.dat:ads"
|
|
],
|
|
google: ["geosite:google"],
|
|
netflix: ["geosite:netflix"],
|
|
cn: [
|
|
"geosite:cn",
|
|
"regexp:.*\\.cn$"
|
|
],
|
|
ru: [
|
|
"geosite:category-gov-ru",
|
|
"regexp:.*\\.ru$"
|
|
],
|
|
ir: [
|
|
"regexp:.*\\.ir$",
|
|
"ext:iran.dat:ir",
|
|
"ext:iran.dat:other",
|
|
"geosite:category-ir"
|
|
]
|
|
},
|
|
familyProtectDNS: {
|
|
"servers": [
|
|
"1.1.1.3",
|
|
"1.0.0.3",
|
|
"94.140.14.15",
|
|
"94.140.15.16"
|
|
],
|
|
"queryStrategy": "UseIPv4"
|
|
},
|
|
}
|
|
},
|
|
methods: {
|
|
loading(spinning = true) {
|
|
this.spinning = spinning;
|
|
},
|
|
async getXraySetting() {
|
|
this.loading(true);
|
|
const msg = await HttpUtil.post("/xui/xray/");
|
|
this.loading(false);
|
|
if (msg.success) {
|
|
this.oldXraySetting = msg.obj;
|
|
this.xraySetting = msg.obj;
|
|
this.saveBtnDisable = true;
|
|
}
|
|
},
|
|
async updateXraySetting() {
|
|
this.loading(true);
|
|
const msg = await HttpUtil.post("/xui/xray/update", {xraySetting : this.xraySetting});
|
|
this.loading(false);
|
|
if (msg.success) {
|
|
await this.getXraySetting();
|
|
}
|
|
},
|
|
async restartPanel() {
|
|
await new Promise(resolve => {
|
|
this.$confirm({
|
|
title: '{{ i18n "pages.settings.restartPanel" }}',
|
|
content: '{{ i18n "pages.settings.restartPanelDesc" }}',
|
|
class: themeSwitcher.currentTheme,
|
|
okText: '{{ i18n "sure" }}',
|
|
cancelText: '{{ i18n "cancel" }}',
|
|
onOk: () => resolve(),
|
|
});
|
|
});
|
|
this.loading(true);
|
|
const msg = await HttpUtil.post("/xui/setting/restartPanel");
|
|
this.loading(false);
|
|
},
|
|
async resetXrayConfigToDefault() {
|
|
this.loading(true);
|
|
const msg = await HttpUtil.get("/xui/xray/getDefaultJsonConfig");
|
|
this.loading(false);
|
|
if (msg.success) {
|
|
this.templateSettings = JSON.parse(JSON.stringify(msg.obj, null, 2));
|
|
this.saveBtnDisable = true;
|
|
}
|
|
},
|
|
syncRulesWithOutbound(tag, setting) {
|
|
const newTemplateSettings = this.templateSettings;
|
|
const haveRules = newTemplateSettings.routing.rules.some((r) => r?.outboundTag === tag);
|
|
const outboundIndex = newTemplateSettings.outbounds.findIndex((o) => o.tag === tag);
|
|
if (!haveRules && outboundIndex > 0) {
|
|
newTemplateSettings.outbounds.splice(outboundIndex);
|
|
}
|
|
if (haveRules && outboundIndex < 0) {
|
|
newTemplateSettings.outbounds.push(setting);
|
|
}
|
|
this.templateSettings = newTemplateSettings;
|
|
},
|
|
templateRuleGetter(routeSettings) {
|
|
const { property, outboundTag } = routeSettings;
|
|
let result = [];
|
|
if (this.templateSettings != null) {
|
|
this.templateSettings.routing.rules.forEach(
|
|
(routingRule) => {
|
|
if (
|
|
routingRule.hasOwnProperty(property) &&
|
|
routingRule.hasOwnProperty("outboundTag") &&
|
|
routingRule.outboundTag === outboundTag
|
|
) {
|
|
result.push(...routingRule[property]);
|
|
}
|
|
}
|
|
);
|
|
}
|
|
return result;
|
|
},
|
|
templateRuleSetter(routeSettings) {
|
|
const { data, property, outboundTag } = routeSettings;
|
|
const oldTemplateSettings = this.templateSettings;
|
|
const newTemplateSettings = oldTemplateSettings;
|
|
currentProperty = this.templateRuleGetter({ outboundTag, property })
|
|
if (currentProperty.length == 0) {
|
|
const propertyRule = {
|
|
type: "field",
|
|
outboundTag,
|
|
[property]: data
|
|
};
|
|
newTemplateSettings.routing.rules.push(propertyRule);
|
|
}
|
|
else {
|
|
const newRules = [];
|
|
insertedOnce = false;
|
|
newTemplateSettings.routing.rules.forEach(
|
|
(routingRule) => {
|
|
if (
|
|
routingRule.hasOwnProperty(property) &&
|
|
routingRule.hasOwnProperty("outboundTag") &&
|
|
routingRule.outboundTag === outboundTag
|
|
) {
|
|
if (!insertedOnce && data.length > 0) {
|
|
insertedOnce = true;
|
|
routingRule[property] = data;
|
|
newRules.push(routingRule);
|
|
}
|
|
}
|
|
else {
|
|
newRules.push(routingRule);
|
|
}
|
|
}
|
|
);
|
|
newTemplateSettings.routing.rules = newRules;
|
|
}
|
|
this.templateSettings = newTemplateSettings;
|
|
}
|
|
},
|
|
async mounted() {
|
|
if (window.location.protocol !== "https:") {
|
|
this.showAlert = true;
|
|
}
|
|
await this.getXraySetting();
|
|
while (true) {
|
|
await PromiseUtil.sleep(1000);
|
|
this.saveBtnDisable = this.oldXraySetting === this.xraySetting;
|
|
}
|
|
},
|
|
computed: {
|
|
templateSettings: {
|
|
get: function () { return this.xraySetting ? JSON.parse(this.xraySetting) : null; },
|
|
set: function (newValue) { this.xraySetting = JSON.stringify(newValue, null, 2); },
|
|
},
|
|
inboundSettings: {
|
|
get: function () { return this.templateSettings ? JSON.stringify(this.templateSettings.inbounds, null, 2) : null; },
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
newTemplateSettings.inbounds = JSON.parse(newValue);
|
|
this.templateSettings = newTemplateSettings;
|
|
},
|
|
},
|
|
outboundSettings: {
|
|
get: function () { return this.templateSettings ? JSON.stringify(this.templateSettings.outbounds, null, 2) : null; },
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
newTemplateSettings.outbounds = JSON.parse(newValue);
|
|
this.templateSettings = newTemplateSettings;
|
|
},
|
|
},
|
|
routingRuleSettings: {
|
|
get: function () { return this.templateSettings ? JSON.stringify(this.templateSettings.routing.rules, null, 2) : null; },
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
newTemplateSettings.routing.rules = JSON.parse(newValue);
|
|
this.templateSettings = newTemplateSettings;
|
|
},
|
|
},
|
|
freedomStrategy: {
|
|
get: function () {
|
|
if (!this.templateSettings) return "AsIs";
|
|
freedomOutbound = this.templateSettings.outbounds.find((o) => o.protocol === "freedom" && !o.tag);
|
|
if (!freedomOutbound) return "AsIs";
|
|
if (!freedomOutbound.settings || !freedomOutbound.settings.domainStrategy) return "AsIs";
|
|
return freedomOutbound.settings.domainStrategy;
|
|
},
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
freedomOutboundIndex = newTemplateSettings.outbounds.findIndex((o) => o.protocol === "freedom" && !o.tag);
|
|
if (!newTemplateSettings.outbounds[freedomOutboundIndex].settings) {
|
|
newTemplateSettings.outbounds[freedomOutboundIndex].settings = { "domainStrategy": newValue };
|
|
} else {
|
|
newTemplateSettings.outbounds[freedomOutboundIndex].settings.domainStrategy = newValue;
|
|
}
|
|
this.templateSettings = newTemplateSettings;
|
|
}
|
|
},
|
|
routingStrategy: {
|
|
get: function () {
|
|
if (!this.templateSettings || !this.templateSettings.routing || !this.templateSettings.routing.domainStrategy) return "AsIs";
|
|
return this.templateSettings.routing.domainStrategy;
|
|
},
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
newTemplateSettings.routing.domainStrategy = newValue;
|
|
this.templateSettings = newTemplateSettings;
|
|
}
|
|
},
|
|
blockedIPs: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "blocked", property: "ip" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "blocked", property: "ip", data: newValue });
|
|
}
|
|
},
|
|
blockedDomains: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "blocked", property: "domain" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "blocked", property: "domain", data: newValue });
|
|
}
|
|
},
|
|
blockedProtocols: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "blocked", property: "protocol" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "blocked", property: "protocol", data: newValue });
|
|
}
|
|
},
|
|
directIPs: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "direct", property: "ip" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "direct", property: "ip", data: newValue });
|
|
this.syncRulesWithOutbound("direct", this.directSettings);
|
|
}
|
|
},
|
|
directDomains: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "direct", property: "domain" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "direct", property: "domain", data: newValue });
|
|
this.syncRulesWithOutbound("direct", this.directSettings);
|
|
}
|
|
},
|
|
ipv4Domains: {
|
|
get: function () {
|
|
return this.templateRuleGetter({ outboundTag: "IPv4", property: "domain" });
|
|
},
|
|
set: function (newValue) {
|
|
this.templateRuleSetter({ outboundTag: "IPv4", property: "domain", data: newValue });
|
|
this.syncRulesWithOutbound("IPv4", this.ipv4Settings);
|
|
}
|
|
},
|
|
manualBlockedIPs: {
|
|
get: function () { return JSON.stringify(this.blockedIPs, null, 2); },
|
|
set: debounce(function (value) { this.blockedIPs = JSON.parse(value); }, 1000)
|
|
},
|
|
manualBlockedDomains: {
|
|
get: function () { return JSON.stringify(this.blockedDomains, null, 2); },
|
|
set: debounce(function (value) { this.blockedDomains = JSON.parse(value); }, 1000)
|
|
},
|
|
manualDirectIPs: {
|
|
get: function () { return JSON.stringify(this.directIPs, null, 2); },
|
|
set: debounce(function (value) { this.directIPs = JSON.parse(value); }, 1000)
|
|
},
|
|
manualDirectDomains: {
|
|
get: function () { return JSON.stringify(this.directDomains, null, 2); },
|
|
set: debounce(function (value) { this.directDomains = JSON.parse(value); }, 1000)
|
|
},
|
|
manualIPv4Domains: {
|
|
get: function () { return JSON.stringify(this.ipv4Domains, null, 2); },
|
|
set: debounce(function (value) { this.ipv4Domains = JSON.parse(value); }, 1000)
|
|
},
|
|
torrentSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.protocols.bittorrent, this.blockedProtocols);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedProtocols = [...this.blockedProtocols, ...this.settingsData.protocols.bittorrent];
|
|
} else {
|
|
this.blockedProtocols = this.blockedProtocols.filter(data => !this.settingsData.protocols.bittorrent.includes(data));
|
|
}
|
|
},
|
|
},
|
|
privateIpSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.local, this.blockedIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedIPs = [...this.blockedIPs, ...this.settingsData.ips.local];
|
|
} else {
|
|
this.blockedIPs = this.blockedIPs.filter(data => !this.settingsData.ips.local.includes(data));
|
|
}
|
|
},
|
|
},
|
|
AdsSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.ads, this.blockedDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedDomains = [...this.blockedDomains, ...this.settingsData.domains.ads];
|
|
} else {
|
|
this.blockedDomains = this.blockedDomains.filter(data => !this.settingsData.domains.ads.includes(data));
|
|
}
|
|
},
|
|
},
|
|
familyProtectSettings: {
|
|
get: function () {
|
|
if (!this.templateSettings || !this.templateSettings.dns || !this.templateSettings.dns.servers) return false;
|
|
return doAllItemsExist(this.templateSettings.dns.servers, this.settingsData.familyProtectDNS.servers);
|
|
},
|
|
set: function (newValue) {
|
|
newTemplateSettings = this.templateSettings;
|
|
if (newValue) {
|
|
newTemplateSettings.dns = this.settingsData.familyProtectDNS;
|
|
} else {
|
|
delete newTemplateSettings.dns;
|
|
}
|
|
this.templateSettings = newTemplateSettings;
|
|
},
|
|
},
|
|
GoogleIPv4Settings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.google, this.ipv4Domains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.ipv4Domains = [...this.ipv4Domains, ...this.settingsData.domains.google];
|
|
} else {
|
|
this.ipv4Domains = this.ipv4Domains.filter(data => !this.settingsData.domains.google.includes(data));
|
|
}
|
|
},
|
|
},
|
|
NetflixIPv4Settings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.netflix, this.ipv4Domains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.ipv4Domains = [...this.ipv4Domains, ...this.settingsData.domains.netflix];
|
|
} else {
|
|
this.ipv4Domains = this.ipv4Domains.filter(data => !this.settingsData.domains.netflix.includes(data));
|
|
}
|
|
},
|
|
},
|
|
IRIpSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.ir, this.blockedIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedIPs = [...this.blockedIPs, ...this.settingsData.ips.ir];
|
|
} else {
|
|
this.blockedIPs = this.blockedIPs.filter(data => !this.settingsData.ips.ir.includes(data));
|
|
}
|
|
}
|
|
},
|
|
IRDomainSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.ir, this.blockedDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedDomains = [...this.blockedDomains, ...this.settingsData.domains.ir];
|
|
} else {
|
|
this.blockedDomains = this.blockedDomains.filter(data => !this.settingsData.domains.ir.includes(data));
|
|
}
|
|
}
|
|
},
|
|
ChinaIpSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.cn, this.blockedIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedIPs = [...this.blockedIPs, ...this.settingsData.ips.cn];
|
|
} else {
|
|
this.blockedIPs = this.blockedIPs.filter(data => !this.settingsData.ips.cn.includes(data));
|
|
}
|
|
}
|
|
},
|
|
ChinaDomainSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.cn, this.blockedDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedDomains = [...this.blockedDomains, ...this.settingsData.domains.cn];
|
|
} else {
|
|
this.blockedDomains = this.blockedDomains.filter(data => !this.settingsData.domains.cn.includes(data));
|
|
}
|
|
}
|
|
},
|
|
RussiaIpSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.ru, this.blockedIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedIPs = [...this.blockedIPs, ...this.settingsData.ips.ru];
|
|
} else {
|
|
this.blockedIPs = this.blockedIPs.filter(data => !this.settingsData.ips.ru.includes(data));
|
|
}
|
|
}
|
|
},
|
|
RussiaDomainSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.ru, this.blockedDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.blockedDomains = [...this.blockedDomains, ...this.settingsData.domains.ru];
|
|
} else {
|
|
this.blockedDomains = this.blockedDomains.filter(data => !this.settingsData.domains.ru.includes(data));
|
|
}
|
|
}
|
|
},
|
|
IRIpDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.ir, this.directIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directIPs = [...this.directIPs, ...this.settingsData.ips.ir];
|
|
} else {
|
|
this.directIPs = this.directIPs.filter(data => !this.settingsData.ips.ir.includes(data));
|
|
}
|
|
}
|
|
},
|
|
IRDomainDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.ir, this.directDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directDomains = [...this.directDomains, ...this.settingsData.domains.ir];
|
|
} else {
|
|
this.directDomains = this.directDomains.filter(data => !this.settingsData.domains.ir.includes(data));
|
|
}
|
|
}
|
|
},
|
|
ChinaIpDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.cn, this.directIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directIPs = [...this.directIPs, ...this.settingsData.ips.cn];
|
|
} else {
|
|
this.directIPs = this.directIPs.filter(data => !this.settingsData.ips.cn.includes(data));
|
|
}
|
|
}
|
|
},
|
|
ChinaDomainDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.cn, this.directDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directDomains = [...this.directDomains, ...this.settingsData.domains.cn];
|
|
} else {
|
|
this.directDomains = this.directDomains.filter(data => !this.settingsData.domains.cn.includes(data));
|
|
}
|
|
}
|
|
},
|
|
RussiaIpDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.ips.ru, this.directIPs);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directIPs = [...this.directIPs, ...this.settingsData.ips.ru];
|
|
} else {
|
|
this.directIPs = this.directIPs.filter(data => !this.settingsData.ips.ru.includes(data));
|
|
}
|
|
}
|
|
},
|
|
RussiaDomainDirectSettings: {
|
|
get: function () {
|
|
return doAllItemsExist(this.settingsData.domains.ru, this.directDomains);
|
|
},
|
|
set: function (newValue) {
|
|
if (newValue) {
|
|
this.directDomains = [...this.directDomains, ...this.settingsData.domains.ru];
|
|
} else {
|
|
this.directDomains = this.directDomains.filter(data => !this.settingsData.domains.ru.includes(data));
|
|
}
|
|
}
|
|
},
|
|
},
|
|
});
|
|
</script>
|
|
</body>
|
|
</html> |