| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 |
- <script setup>
- import { computed } from 'vue';
- import { useI18n } from 'vue-i18n';
- import { ExclamationCircleFilled, CloudOutlined, ApiOutlined } from '@ant-design/icons-vue';
- import { Modal } from 'ant-design-vue';
- import { OutboundDomainStrategies } from '@/models/outbound.js';
- import SettingListItem from '@/components/SettingListItem.vue';
- const { t } = useI18n();
- // Phase 6-ii: structured editor for the most-touched fields of the
- // xray template — outbound strategy, routing strategy, log levels,
- // stat counters, and the "basic routing" lists (block IPs/domains/
- // torrent + direct IPs/domains + IPv4 forced + warp/nord domains).
- //
- // Mutates the parent's templateSettings reactive directly. The
- // useXraySetting composable's deep watch on templateSettings re-
- // stringifies into xraySetting so the Advanced JSON tab and the
- // dirty-poll see every edit.
- const props = defineProps({
- templateSettings: { type: Object, default: null },
- outboundTestUrl: { type: String, default: '' },
- warpExist: { type: Boolean, default: false },
- nordExist: { type: Boolean, default: false },
- });
- const emit = defineEmits(['update:outbound-test-url', 'show-warp', 'show-nord', 'reset-default']);
- function confirmResetDefault() {
- Modal.confirm({
- title: t('pages.settings.resetDefaultConfig'),
- okText: t('reset'),
- okType: 'danger',
- cancelText: t('cancel'),
- onOk: () => { emit('reset-default'); },
- });
- }
- // === Static option lists (mirror legacy) =============================
- const ROUTING_DOMAIN_STRATEGIES = ['AsIs', 'IPIfNonMatch', 'IPOnDemand'];
- const LOG_LEVELS = ['none', 'debug', 'info', 'warning', 'error'];
- const ACCESS_LOG = ['none', './access.log'];
- const ERROR_LOG = ['none', './error.log'];
- const MASK_ADDRESS = ['quarter', 'half', 'full'];
- const BITTORRENT_PROTOCOLS = ['bittorrent'];
- // Country / service lists mirror the legacy panel's settingsData
- // (web/html/xray.html on main). Keep additions in sync with that file
- // so Vue 3 + legacy stay swappable while the migration finishes.
- const IPS_OPTIONS = [
- { label: 'Private IPs', value: 'geoip:private' },
- { label: '🇮🇷 Iran', value: 'ext:geoip_IR.dat:ir' },
- { label: '🇨🇳 China', value: 'geoip:cn' },
- { label: '🇷🇺 Russia', value: 'ext:geoip_RU.dat:ru' },
- { label: '🇻🇳 Vietnam', value: 'geoip:vn' },
- { label: '🇪🇸 Spain', value: 'geoip:es' },
- { label: '🇮🇩 Indonesia', value: 'geoip:id' },
- { label: '🇺🇦 Ukraine', value: 'geoip:ua' },
- { label: '🇹🇷 Türkiye', value: 'geoip:tr' },
- { label: '🇧🇷 Brazil', value: 'geoip:br' },
- ];
- const DOMAINS_OPTIONS = [
- { label: '🇮🇷 Iran', value: 'ext:geosite_IR.dat:ir' },
- { label: '🇮🇷 .ir', value: 'regexp:.*\\.ir$' },
- { label: '🇮🇷 .ایران', value: 'regexp:.*\\.xn--mgba3a4f16a$' },
- { label: '🇨🇳 China', value: 'geosite:cn' },
- { label: '🇨🇳 .cn', value: 'regexp:.*\\.cn$' },
- { label: '🇷🇺 Russia', value: 'ext:geosite_RU.dat:ru-available-only-inside' },
- { label: '🇷🇺 .ru', value: 'regexp:.*\\.ru$' },
- { label: '🇷🇺 .su', value: 'regexp:.*\\.su$' },
- { label: '🇷🇺 .рф', value: 'regexp:.*\\.xn--p1ai$' },
- { label: '🇻🇳 .vn', value: 'regexp:.*\\.vn$' },
- ];
- const BLOCK_DOMAINS_OPTIONS = [
- { label: 'Ads All', value: 'geosite:category-ads-all' },
- { label: 'Ads IR 🇮🇷', value: 'ext:geosite_IR.dat:category-ads-all' },
- { label: 'Ads RU 🇷🇺', value: 'ext:geosite_RU.dat:category-ads-all' },
- { label: 'Malware 🇮🇷', value: 'ext:geosite_IR.dat:malware' },
- { label: 'Phishing 🇮🇷', value: 'ext:geosite_IR.dat:phishing' },
- { label: 'Cryptominers 🇮🇷', value: 'ext:geosite_IR.dat:cryptominers' },
- { label: 'Adult +18', value: 'geosite:category-porn' },
- { label: '🇮🇷 Iran', value: 'ext:geosite_IR.dat:ir' },
- { label: '🇮🇷 .ir', value: 'regexp:.*\\.ir$' },
- { label: '🇮🇷 .ایران', value: 'regexp:.*\\.xn--mgba3a4f16a$' },
- { label: '🇨🇳 China', value: 'geosite:cn' },
- { label: '🇨🇳 .cn', value: 'regexp:.*\\.cn$' },
- { label: '🇷🇺 Russia', value: 'ext:geosite_RU.dat:ru-available-only-inside' },
- { label: '🇷🇺 .ru', value: 'regexp:.*\\.ru$' },
- { label: '🇷🇺 .su', value: 'regexp:.*\\.su$' },
- { label: '🇷🇺 .рф', value: 'regexp:.*\\.xn--p1ai$' },
- { label: '🇻🇳 .vn', value: 'regexp:.*\\.vn$' },
- ];
- const SERVICES_OPTIONS = [
- { label: 'Apple', value: 'geosite:apple' },
- { label: 'Meta', value: 'geosite:meta' },
- { label: 'Google', value: 'geosite:google' },
- { label: 'OpenAI', value: 'geosite:openai' },
- { label: 'Spotify', value: 'geosite:spotify' },
- { label: 'Netflix', value: 'geosite:netflix' },
- { label: 'Reddit', value: 'geosite:reddit' },
- { label: 'Speedtest', value: 'geosite:speedtest' },
- ];
- // === Routing-rule helpers (matches legacy templateRule{Getter,Setter}) ==
- function ruleGetter(outboundTag, property) {
- if (!props.templateSettings?.routing?.rules) return [];
- const out = [];
- for (const rule of props.templateSettings.routing.rules) {
- if (
- rule
- && Object.prototype.hasOwnProperty.call(rule, property)
- && Object.prototype.hasOwnProperty.call(rule, 'outboundTag')
- && rule.outboundTag === outboundTag
- ) {
- out.push(...rule[property]);
- }
- }
- return out;
- }
- function ruleSetter(outboundTag, property, data) {
- if (!props.templateSettings?.routing) return;
- const current = ruleGetter(outboundTag, property);
- if (current.length === 0) {
- props.templateSettings.routing.rules.push({
- type: 'field',
- outboundTag,
- [property]: data,
- });
- return;
- }
- // Replace the property on the FIRST matching rule and drop any later
- // duplicates with the same (outboundTag, property) pair (matches the
- // legacy single-write-then-filter behavior).
- const next = [];
- let inserted = false;
- for (const rule of props.templateSettings.routing.rules) {
- const matches =
- rule
- && Object.prototype.hasOwnProperty.call(rule, property)
- && Object.prototype.hasOwnProperty.call(rule, 'outboundTag')
- && rule.outboundTag === outboundTag;
- if (matches) {
- if (!inserted && data.length > 0) {
- rule[property] = data;
- next.push(rule);
- inserted = true;
- }
- } else {
- next.push(rule);
- }
- }
- props.templateSettings.routing.rules = next;
- }
- function syncOutbound(tag, settings) {
- // After editing direct/IPv4/warp/nord rules, ensure the matching
- // outbound exists when the rule list has any entries, and is
- // pruned when none remain (legacy syncRulesWithOutbound).
- const t = props.templateSettings;
- if (!t) return;
- const haveRules = t.routing.rules.some((r) => r?.outboundTag === tag);
- const idx = t.outbounds.findIndex((o) => o.tag === tag);
- if (!haveRules && idx > 0) t.outbounds.splice(idx, 1);
- if (haveRules && idx < 0) t.outbounds.push(settings);
- }
- // === Computed v-models for every Basics field ========================
- function rule(tag, property, syncFn) {
- return computed({
- get: () => ruleGetter(tag, property),
- set: (next) => { ruleSetter(tag, property, next); if (syncFn) syncFn(); },
- });
- }
- const directSettings = { tag: 'direct', protocol: 'freedom' };
- const ipv4Settings = { tag: 'IPv4', protocol: 'freedom', settings: { domainStrategy: 'UseIPv4' } };
- const freedomStrategy = computed({
- get: () => {
- const ob = props.templateSettings?.outbounds?.find(
- (o) => o.protocol === 'freedom' && o.tag === 'direct',
- );
- return ob?.settings?.domainStrategy ?? 'AsIs';
- },
- set: (next) => {
- const t = props.templateSettings;
- if (!t) return;
- const idx = t.outbounds.findIndex((o) => o.protocol === 'freedom' && o.tag === 'direct');
- if (idx < 0) {
- t.outbounds.push({ protocol: 'freedom', tag: 'direct', settings: { domainStrategy: next } });
- } else {
- t.outbounds[idx].settings = t.outbounds[idx].settings || {};
- t.outbounds[idx].settings.domainStrategy = next;
- }
- },
- });
- const routingStrategy = computed({
- get: () => props.templateSettings?.routing?.domainStrategy ?? 'AsIs',
- set: (next) => { if (props.templateSettings?.routing) props.templateSettings.routing.domainStrategy = next; },
- });
- function logField(field, fallback) {
- return computed({
- get: () => props.templateSettings?.log?.[field] ?? fallback,
- set: (next) => { if (props.templateSettings?.log) props.templateSettings.log[field] = next; },
- });
- }
- const logLevel = logField('loglevel', 'warning');
- const accessLog = logField('access', '');
- const errorLog = logField('error', '');
- const maskAddressLog = logField('maskAddress', '');
- const dnslog = logField('dnsLog', false);
- function policyField(field) {
- return computed({
- get: () => !!props.templateSettings?.policy?.system?.[field],
- set: (next) => {
- if (!props.templateSettings?.policy?.system) return;
- props.templateSettings.policy.system[field] = next;
- },
- });
- }
- const statsInboundUplink = policyField('statsInboundUplink');
- const statsInboundDownlink = policyField('statsInboundDownlink');
- const statsOutboundUplink = policyField('statsOutboundUplink');
- const statsOutboundDownlink = policyField('statsOutboundDownlink');
- const blockedIPs = rule('blocked', 'ip');
- const blockedDomains = rule('blocked', 'domain');
- const blockedProtocols = rule('blocked', 'protocol');
- const directIPs = rule('direct', 'ip', () => syncOutbound('direct', directSettings));
- const directDomains = rule('direct', 'domain', () => syncOutbound('direct', directSettings));
- const ipv4Domains = rule('IPv4', 'domain', () => syncOutbound('IPv4', ipv4Settings));
- const warpDomains = rule('warp', 'domain');
- const nordTag = computed(() => {
- const ob = props.templateSettings?.outbounds?.find((o) => o.tag?.startsWith?.('nord-'));
- return ob?.tag || 'nord';
- });
- const nordDomains = computed({
- get: () => ruleGetter(nordTag.value, 'domain'),
- set: (next) => ruleSetter(nordTag.value, 'domain', next),
- });
- const torrentSettings = computed({
- get: () => BITTORRENT_PROTOCOLS.every((p) => blockedProtocols.value.includes(p)),
- set: (next) => {
- if (next) {
- blockedProtocols.value = [...blockedProtocols.value, ...BITTORRENT_PROTOCOLS];
- } else {
- blockedProtocols.value = blockedProtocols.value.filter((d) => !BITTORRENT_PROTOCOLS.includes(d));
- }
- },
- });
- const localOutboundTestUrl = computed({
- get: () => props.outboundTestUrl,
- set: (next) => emit('update:outbound-test-url', next),
- });
- </script>
- <template>
- <a-collapse default-active-key="1">
- <a-collapse-panel key="1" :header="t('pages.xray.generalConfigs')">
- <a-alert type="warning" class="mb-12 hint-alert" :message="t('pages.xray.generalConfigsDesc')">
- <template #icon>
- <ExclamationCircleFilled style="color: #FFA031;" />
- </template>
- </a-alert>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.FreedomStrategy') }}</template>
- <template #description>{{ t('pages.xray.FreedomStrategyDesc') }}</template>
- <template #control>
- <a-select v-model:value="freedomStrategy" :style="{ width: '100%' }">
- <a-select-option v-for="s in OutboundDomainStrategies" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.RoutingStrategy') }}</template>
- <template #description>{{ t('pages.xray.RoutingStrategyDesc') }}</template>
- <template #control>
- <a-select v-model:value="routingStrategy" :style="{ width: '100%' }">
- <a-select-option v-for="s in ROUTING_DOMAIN_STRATEGIES" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.outboundTestUrl') }}</template>
- <template #description>{{ t('pages.xray.outboundTestUrlDesc') }}</template>
- <template #control>
- <a-input v-model:value="localOutboundTestUrl" placeholder="https://www.google.com/generate_204" />
- </template>
- </SettingListItem>
- </a-collapse-panel>
- <a-collapse-panel key="2" :header="t('pages.xray.statistics')">
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.statsInboundUplink') }}</template>
- <template #control><a-switch v-model:checked="statsInboundUplink" /></template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.statsInboundDownlink') }}</template>
- <template #control><a-switch v-model:checked="statsInboundDownlink" /></template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.statsOutboundUplink') }}</template>
- <template #control><a-switch v-model:checked="statsOutboundUplink" /></template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>Outbound downlink stats</template>
- <template #control><a-switch v-model:checked="statsOutboundDownlink" /></template>
- </SettingListItem>
- </a-collapse-panel>
- <a-collapse-panel key="3" :header="t('pages.xray.logConfigs')">
- <a-alert type="warning" class="mb-12 hint-alert" :message="t('pages.xray.logConfigsDesc')">
- <template #icon>
- <ExclamationCircleFilled style="color: #FFA031;" />
- </template>
- </a-alert>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.logLevel') }}</template>
- <template #description>{{ t('pages.xray.logLevelDesc') }}</template>
- <template #control>
- <a-select v-model:value="logLevel" :style="{ width: '100%' }">
- <a-select-option v-for="s in LOG_LEVELS" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.accessLog') }}</template>
- <template #description>{{ t('pages.xray.accessLogDesc') }}</template>
- <template #control>
- <a-select v-model:value="accessLog" :style="{ width: '100%' }">
- <a-select-option value="">{{ t('none') }}</a-select-option>
- <a-select-option v-for="s in ACCESS_LOG" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.errorLog') }}</template>
- <template #description>{{ t('pages.xray.errorLogDesc') }}</template>
- <template #control>
- <a-select v-model:value="errorLog" :style="{ width: '100%' }">
- <a-select-option value="">{{ t('none') }}</a-select-option>
- <a-select-option v-for="s in ERROR_LOG" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.maskAddress') }}</template>
- <template #description>{{ t('pages.xray.maskAddressDesc') }}</template>
- <template #control>
- <a-select v-model:value="maskAddressLog" :style="{ width: '100%' }">
- <a-select-option value="">{{ t('none') }}</a-select-option>
- <a-select-option v-for="s in MASK_ADDRESS" :key="s" :value="s">{{ s }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.dnsLog') }}</template>
- <template #description>{{ t('pages.xray.dnsLogDesc') }}</template>
- <template #control><a-switch v-model:checked="dnslog" /></template>
- </SettingListItem>
- </a-collapse-panel>
- <a-collapse-panel key="4" :header="t('pages.xray.basicRouting')">
- <a-alert type="warning" class="mb-12 hint-alert" :message="t('pages.xray.blockConnectionsConfigsDesc')">
- <template #icon>
- <ExclamationCircleFilled style="color: #FFA031;" />
- </template>
- </a-alert>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.Torrent') }}</template>
- <template #control><a-switch v-model:checked="torrentSettings" /></template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.blockips') }}</template>
- <template #control>
- <a-select v-model:value="blockedIPs" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in IPS_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.blockdomains') }}</template>
- <template #control>
- <a-select v-model:value="blockedDomains" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in BLOCK_DOMAINS_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{
- p.label }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <a-alert type="warning" class="mb-12 hint-alert" :message="t('pages.xray.directConnectionsConfigsDesc')">
- <template #icon>
- <ExclamationCircleFilled style="color: #FFA031;" />
- </template>
- </a-alert>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.directips') }}</template>
- <template #control>
- <a-select v-model:value="directIPs" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in IPS_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.directdomains') }}</template>
- <template #control>
- <a-select v-model:value="directDomains" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in DOMAINS_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.ipv4Routing') }}</template>
- <template #description>{{ t('pages.xray.ipv4RoutingDesc') }}</template>
- <template #control>
- <a-select v-model:value="ipv4Domains" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in SERVICES_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.warpRouting') }}</template>
- <template #description>{{ t('pages.xray.warpRoutingDesc') }}</template>
- <template #control>
- <a-select v-if="warpExist" v-model:value="warpDomains" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in SERVICES_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- <a-button v-else type="primary" @click="emit('show-warp')">
- <template #icon>
- <CloudOutlined />
- </template>
- WARP
- </a-button>
- </template>
- </SettingListItem>
- <SettingListItem paddings="small">
- <template #title>{{ t('pages.xray.nordRouting') }}</template>
- <template #description>{{ t('pages.xray.nordRoutingDesc') }}</template>
- <template #control>
- <a-select v-if="nordExist" v-model:value="nordDomains" mode="tags" :style="{ width: '100%' }">
- <a-select-option v-for="p in SERVICES_OPTIONS" :key="p.value" :value="p.value" :label="p.label">{{ p.label
- }}</a-select-option>
- </a-select>
- <a-button v-else type="primary" @click="emit('show-nord')">
- <template #icon>
- <ApiOutlined />
- </template>
- NordVPN
- </a-button>
- </template>
- </SettingListItem>
- </a-collapse-panel>
- <a-collapse-panel key="reset" :header="t('pages.settings.resetDefaultConfig')">
- <a-space direction="horizontal" :style="{ padding: '0 20px' }">
- <a-button danger @click="confirmResetDefault">
- {{ t('pages.settings.resetDefaultConfig') }}
- </a-button>
- </a-space>
- </a-collapse-panel>
- </a-collapse>
- </template>
- <style scoped>
- .mb-12 {
- margin-bottom: 12px;
- }
- .hint-alert {
- text-align: center;
- }
- </style>
|