| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843 |
- @extends('admin.layouts')
- @section('css')
- <link href="/assets/global/vendor/bootstrap-select/bootstrap-select.min.css" rel="stylesheet">
- <link href="/assets/global/vendor/bootstrap-datepicker/bootstrap-datepicker.min.css" rel="stylesheet">
- <link href="/assets/global/vendor/switchery/switchery.min.css" rel="stylesheet">
- <style>
- .hidden {
- display: none
- }
- .bootstrap-select .dropdown-menu {
- max-height: 50vh !important;
- }
- </style>
- @endsection
- @section('content')
- <div class="page-content container-fluid">
- <div class="panel">
- <div class="panel-heading">
- <h2 class="panel-title">
- {{ isset($node) ? trans('admin.action.edit_item', ['attribute' => trans('model.node.attribute')]) : trans('admin.action.add_item', ['attribute' => trans('model.node.attribute')]) }}
- </h2>
- </div>
- <div class="alert alert-info" role="alert">
- <button class="close" data-dismiss="alert" aria-label="{{ trans('common.close') }}">
- <span aria-hidden="true">×</span>
- <span class="sr-only">{{ trans('common.close') }}</span>
- </button>
- {!! trans('admin.node.info.hint') !!}
- </div>
- <div class="panel-body">
- <form class="form-horizontal" id="nodeForm">
- @csrf
- <div class="row">
- <div class="col-lg-6">
- <div class="example-wrap">
- <h4 class="example-title">{{ trans('admin.node.info.basic') }}</h4>
- <div class="example">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="is_ddns">{{ trans('model.node.ddns') }}</label>
- <div class="col-md-9">
- <input id="is_ddns" name="is_ddns" data-plugin="switchery" type="checkbox" onchange="switchSetting('is_ddns')">
- </div>
- <div class="text-help offset-md-3">
- {!! trans('admin.node.info.ddns_hint') !!}
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="name"> {{ trans('model.node.name') }} </label>
- <input class="form-control col-md-4" id="name" name="name" type="text" required>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="server"> {{ trans('model.node.domain') }} </label>
- <input class="form-control col-md-4" id="server" name="server" type="text"
- placeholder="{{ trans('admin.node.info.domain_placeholder') }}">
- <span class="text-help offset-md-3">{{ trans('admin.node.info.domain_hint') }}</span>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="ip"> {{ trans('model.node.ipv4') }} </label>
- <input class="form-control col-md-4" id="ip" name="ip" type="text"
- placeholder="{{ trans('admin.node.info.ipv4_placeholder') }}" required>
- <span class="text-help offset-md-3">{{ trans('admin.node.info.ipv4_hint') }}</span>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="ipv6"> {{ trans('model.node.ipv6') }} </label>
- <input class="form-control col-md-4" id="ipv6" name="ipv6" type="text"
- placeholder="{{ trans('admin.node.info.ipv6_placeholder') }}">
- <span class="text-help offset-md-3">{{ trans('admin.node.info.ipv6_hint') }}</span>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="push_port"> {{ trans('model.node.push_port') }} </label>
- <input class="form-control col-md-4" id="push_port" name="push_port" type="number" value="1080">
- <span class="text-help offset-md-3">{{ trans('admin.node.info.push_port_hint') }}</span>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="traffic_rate"> {{ trans('model.node.data_rate') }} </label>
- <input class="form-control col-md-4" id="traffic_rate" name="traffic_rate" type="number" value="1.0" step="0.01"
- required>
- <div class="text-help offset-md-3">{{ trans('admin.node.info.data_rate_hint') }}</div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="level">{{ trans('model.common.level') }}</label>
- <select class="col-md-5 form-control show-tick" id="level" name="level" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- @foreach ($levels as $level)
- <option value="{{ $level->level }}">{{ $level->name }}</option>
- @endforeach
- </select>
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.level_hint') }}</div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="rule_group_id">{{ trans('model.rule_group.attribute') }}</label>
- <select class="col-md-5 form-control show-tick" id="rule_group_id" name="rule_group_id" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="">{{ trans('common.none') }}</option>
- @foreach ($ruleGroups as $ruleGroup)
- <option value="{{ $ruleGroup->id }}">{{ $ruleGroup->name }}</option>
- @endforeach
- </select>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="speed_limit">{{ trans('model.node.traffic_limit') }}</label>
- <div class="col-md-4 input-group p-0">
- <input class="form-control" id="speed_limit" name="speed_limit" type="number" value="1000" required>
- <span class="input-group-text">Mbps</span>
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="client_limit">{{ trans('model.node.client_limit') }}</label>
- <input class="form-control col-md-4" id="client_limit" name="client_limit" type="number" value="1000" required>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="sort">{{ trans('model.common.sort') }}</label>
- <input class="form-control col-md-4" id="sort" name="sort" type="text" value="1" required />
- <span class="col-md-5"></span>
- <div class="text-help offset-md-3"> {{ trans('admin.sort_asc') }}</div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="labels">{{ trans('model.node.label') }}</label>
- <select class="col-md-5 form-control show-tick" id="labels" name="labels" data-plugin="selectpicker"
- data-style="btn-outline btn-primary" multiple>
- @foreach ($labels as $label)
- <option value="{{ $label->id }}">{{ $label->name }}</option>
- @endforeach
- </select>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="country_code"> {{ trans('model.node.country') }} </label>
- <select class="col-md-5 form-control" id="country_code" name="country_code" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- @foreach ($countries as $country)
- <option value="{{ $country->code }}">{{ $country->code . ' - ' . $country->name }}</option>
- @endforeach
- </select>
- </div>
- <!-- 节点 细则部分 -->
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="next_renewal_date">{{ trans('model.node.next_renewal_date') }}</label>
- <div class="col-md-6 input-group p-0">
- <input class="form-control" id="next_renewal_date" name="next_renewal_date" data-plugin="datepicker" type="text"
- autocomplete="off" />
- <input class="form-control" id="next_next_renewal_date" type="text" readonly />
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="subscription_term_value">
- {{ trans('model.node.subscription_term') }}
- </label>
- <div class="col-md-4 input-group p-0">
- <input class="form-control" id="subscription_term_value" type="number" min="1" />
- <select class="form-control" id="subscription_term_unit" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="days" selected>{{ ucfirst(trans('validation.attributes.day')) }}</option>
- <option value="months">{{ ucfirst(trans('validation.attributes.month')) }}</option>
- <option value="years">{{ ucfirst(trans('validation.attributes.year')) }}</option>
- </select>
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="renewal_cost">{{ trans('model.node.renewal_cost') }}</label>
- <div class="col-md-4 input-group p-0">
- <div class="input-group-prepend">
- <span class="input-group-text">
- {{ array_column(config('common.currency'), 'symbol', 'code')[sysConfig('standard_currency')] }}
- </span>
- </div>
- <input class="form-control" id="renewal_cost" name="renewal_cost" type="number" step="0.01" />
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="description"> {{ trans('model.common.description') }} </label>
- <input class="form-control col-md-6" id="description" name="description" type="text">
- </div>
- </div>
- </div>
- </div>
- <div class="col-lg-6">
- <div class="example-wrap">
- <h4 class="example-title">{{ trans('admin.node.info.extend') }}</h4>
- <div class="example">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="is_display">{{ trans('model.node.display') }}</label>
- <ul class="col-md-9 list-unstyled list-inline">
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="invisible" name="is_display" type="radio" value="0" />
- <label for="invisible">{{ trans('admin.node.info.display.invisible') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="page_only" name="is_display" type="radio" value="1" />
- <label for="page_only">{{ trans('admin.node.info.display.node') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="sub_only" name="is_display" type="radio" value="2" />
- <label for="sub_only">{{ trans('admin.node.info.display.sub') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="visible" name="is_display" type="radio" value="3" checked />
- <label for="visible">{{ trans('admin.node.info.display.all') }}</label>
- </div>
- </li>
- </ul>
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.display.hint') }}</div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="detection_type">{{ trans('model.node.detection') }}</label>
- <ul class="col-md-9 list-unstyled list-inline">
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="detect_disable" name="detection_type" type="radio" value="0" checked />
- <label for="detect_disable">{{ trans('common.close') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="detect_tcp" name="detection_type" type="radio" value="1" />
- <label for="detect_tcp">{{ trans('admin.node.info.detection.tcp') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="detect_icmp" name="detection_type" type="radio" value="2" />
- <label for="detect_icmp">{{ trans('admin.node.info.detection.icmp') }}</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="detect_all" name="detection_type" type="radio" value="3" />
- <label for="detect_all">{{ trans('admin.node.info.detection.all') }}</label>
- </div>
- </li>
- </ul>
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.detection.hint') }}</div>
- </div>
- <!-- 中转 设置部分 -->
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="relay_node_id">{{ trans('model.node.transfer') }}</label>
- <select class="col-md-5 form-control show-tick" id="relay_node_id" name="relay_node_id" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="">{{ trans('common.none') }}</option>
- @foreach ($nodes as $name => $id)
- <option value="{{ $id }}">{{ $id }} - {{ $name }}</option>
- @endforeach
- </select>
- </div>
- <hr />
- <!-- 代理 设置部分 -->
- <div class="proxy-config">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="type">{{ trans('model.common.type') }}</label>
- <ul class="col-md-9 list-unstyled list-inline">
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="shadowsocks" name="type" type="radio" value="0">
- <label for="shadowsocks">Shadowsocks</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="shadowsocksR" name="type" type="radio" value="1">
- <label for="shadowsocksR">ShadowsocksR</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="v2ray" name="type" type="radio" value="2">
- <label for="v2ray">V2Ray</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="trojan" name="type" type="radio" value="3">
- <label for="trojan">Trojan</label>
- </div>
- </li>
- <li class="list-inline-item">
- <div class="radio-custom radio-primary">
- <input id="vnet" name="type" type="radio" value="4">
- <label for="vnet">VNET</label>
- </div>
- </li>
- </ul>
- </div>
- <hr />
- <!-- SS/SSR 设置部分 -->
- <div class="ss-setting">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="method">{{ trans('model.node.method') }}</label>
- <select class="col-md-5 form-control" id="method" name="method" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- @foreach (Helpers::methodList() as $method)
- <option value="{{ $method->name }}" @if (!isset($node) && $method->is_default) selected @endif>{{ $method->name }}
- </option>
- @endforeach
- </select>
- </div>
- <div class="ssr-setting">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="protocol">{{ trans('model.node.protocol') }}</label>
- <select class="col-md-5 form-control" id="protocol" name="protocol" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- @foreach (Helpers::protocolList() as $protocol)
- <option value="{{ $protocol->name }}" @if (!isset($node) && $protocol->is_default) selected @endif>
- {{ $protocol->name }}</option>
- @endforeach
- </select>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="protocol_param"> {{ trans('model.node.protocol_param') }}
- </label>
- <input class="form-control col-md-4" id="protocol_param" name="protocol_param" type="text">
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="obfs">{{ trans('model.node.obfs') }}</label>
- <select class="col-md-5 form-control" id="obfs" name="obfs" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- @foreach (Helpers::obfsList() as $obfs)
- <option value="{{ $obfs->name }}" @if (!isset($node) && $obfs->is_default) selected @endif>
- {{ $obfs->name }}</option>
- @endforeach
- </select>
- </div>
- <div class="form-group row obfs_param">
- <label class="col-md-3 col-form-label" for="obfs_param"> {{ trans('model.node.obfs_param') }} </label>
- <textarea class="form-control col-md-8" id="obfs_param" name="obfs_param" rows="5" placeholder="{!! trans('admin.node.info.obfs_param_hint') !!}"></textarea>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label">{{ trans('admin.node.proxy_info') }}</label>
- <div class="text-help col-md-9">
- {!! trans('admin.node.proxy_info_hint') !!}
- </div>
- </div>
- </div>
- <hr />
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="single">{{ trans('model.node.single') }}</label>
- <div class="col-md-9">
- <input id="single" name="single" data-plugin="switchery" type="checkbox"
- onchange="switchSetting('single')">
- </div>
- <div class="text-help offset-md-3">
- {!! trans('admin.node.info.additional_ports_hint') !!}
- </div>
- </div>
- <div class="single-setting">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="single_port">{{ trans('model.node.service_port') }}</label>
- <input class="form-control col-md-4" id="single_port" name="port" type="number" value="443"
- hidden />
- <span class="text-help offset-md-3"> {!! trans('admin.node.info.single_hint') !!}</span>
- </div>
- <div class="form-group row ssr-setting">
- <label class="col-md-3 col-form-label" for="passwd">{{ trans('model.node.single_passwd') }}</label>
- <input class="form-control col-md-4" id="passwd" name="passwd" type="text"
- placeholder="{{ ucfirst(trans('validation.attributes.password')) }}">
- </div>
- </div>
- </div>
- <!-- V2ray 设置部分 -->
- <div class="v2ray-setting">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_alter_id">{{ trans('model.node.v2_alter_id') }}</label>
- <input class="form-control col-md-4" id="v2_alter_id" name="v2_alter_id" type="text" value="16" />
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_port">{{ trans('model.node.service_port') }}</label>
- <input class="form-control col-md-4" id="v2_port" name="port" type="number" value="10053" hidden />
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_method">{{ trans('model.node.method') }}</label>
- <select class="col-md-5 form-control" id="v2_method" name="v2_method" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="none">none</option>
- <option value="auto">auto</option>
- <option value="aes-128-cfb">aes-128-cfb</option>
- <option value="aes-128-gcm">aes-128-gcm</option>
- <option value="chacha20-poly1305">chacha20-poly1305</option>
- </select>
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.v2_method_hint') }}</div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_net">{{ trans('model.node.v2_net') }}</label>
- <select class="col-md-5 form-control" id="v2_net" name="v2_net" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="tcp">TCP</option>
- <option value="http">HTTP/2</option>
- <option value="ws">WebSocket</option>
- <option value="kcp">mKCP</option>
- <option value="domainsocket">DomainSocket</option>
- <option value="quic">QUIC</option>
- </select>
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.v2_net_hint') }}</div>
- </div>
- <div class="form-group row v2_type">
- <label class="col-md-3 col-form-label" for="v2_type">{{ trans('model.node.v2_cover') }}</label>
- <select class="col-md-5 form-control" id="v2_type" name="v2_type" data-plugin="selectpicker"
- data-style="btn-outline btn-primary">
- <option value="none">{{ trans('admin.node.info.v2_cover.none') }}</option>
- <option value="http">{{ trans('admin.node.info.v2_cover.http') }}</option>
- <optgroup id="type_option" label="">
- <option value="srtp">{{ trans('admin.node.info.v2_cover.srtp') }}</option>
- <option value="utp">{{ trans('admin.node.info.v2_cover.utp') }}</option>
- <option value="wechat-video">{{ trans('admin.node.info.v2_cover.wechat') }}</option>
- <option value="dtls">{{ trans('admin.node.info.v2_cover.dtls') }}</option>
- <option value="wireguard">{{ trans('admin.node.info.v2_cover.wireguard') }}</option>
- </optgroup>
- </select>
- </div>
- <div class="form-group row v2_host">
- <label class="col-md-3 col-form-label" for="v2_host">{{ trans('model.node.v2_host') }}</label>
- <div class="col-md-4 pl-0">
- <input class="form-control" id="v2_host" name="v2_host" type="text">
- </div>
- <div class="text-help offset-md-3">
- {{ trans('admin.node.info.v2_host_hint') }}
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_path">{{ trans('model.node.v2_path') }}</label>
- <input class="form-control col-md-4" id="v2_path" name="v2_path" type="text">
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_sni">{{ trans('model.node.v2_sni') }}</label>
- <input class="form-control col-md-4" id="v2_sni" name="v2_sni" type="text">
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="v2_tls">{{ trans('model.node.v2_tls') }}</label>
- <div class="col-md-9">
- <input id="v2_tls" name="v2_tls" data-plugin="switchery" type="checkbox"
- onchange="switchSetting('v2_tls')">
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="tls_provider">{{ trans('model.node.v2_tls_provider') }}</label>
- <input class="form-control col-md-9" id="tls_provider" name="tls_provider" type="text" />
- <div class="text-help offset-md-3"> {{ trans('admin.node.info.v2_tls_provider_hint') }}
- <a href="https://proxypanel.gitbook.io/wiki/webapi/webapi-basic-setting#vnet-v2-ray-hou-duan"
- target="_blank">VNET-V2Ray</a>、
- <a href="https://proxypanel.gitbook.io/wiki/webapi/webapi-basic-setting#v-2-ray-poseidon-hou-duan"
- target="_blank">V2Ray-Poseidon</a>
- </div>
- </div>
- </div>
- <!-- Trojan 设置部分 -->
- <div class="trojan-setting">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="trojan_port">{{ trans('model.node.service_port') }}</label>
- <input class="form-control col-md-4" id="trojan_port" name="port" type="number" value="443" hidden />
- </div>
- </div>
- </div>
- <div class="relay-config">
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="relay_port">{{ trans('model.node.relay_port') }}</label>
- <input class="form-control col-md-4" id="relay_port" name="port" type="number" value="443" hidden />
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="is_udp">{{ trans('model.node.udp') }}</label>
- <div class="col-md-9">
- <input id="is_udp" name="is_udp" data-plugin="switchery" type="checkbox">
- </div>
- </div>
- <div class="form-group row">
- <label class="col-md-3 col-form-label" for="status">{{ trans('common.status.attribute') }}</label>
- <div class="col-md-9">
- <input id="status" name="status" data-plugin="switchery" type="checkbox">
- </div>
- </div>
- </div>
- </div>
- <div class="col-md-12 form-actions">
- <div class="float-right">
- <a class="btn btn-danger" href="{{ route('admin.node.index') }}">{{ trans('common.back') }}</a>
- <button class="btn btn-success" type="submit">{{ trans('common.submit') }}</button>
- </div>
- </div>
- </div>
- </div>
- </form>
- </div>
- </div>
- </div>
- @endsection
- @section('javascript')
- <script src="/assets/global/vendor/bootstrap-select/bootstrap-select.min.js"></script>
- <script src="/assets/global/vendor/bootstrap-datepicker/bootstrap-datepicker.min.js"></script>
- <script src="/assets/global/js/Plugin/bootstrap-select.js"></script>
- <script src="/assets/global/js/Plugin/bootstrap-datepicker.js"></script>
- <script src="/assets/global/vendor/switchery/switchery.min.js"></script>
- <script src="/assets/global/js/Plugin/switchery.js"></script>
- <script>
- const string = "{{ strtolower(Str::random()) }}";
- $("[name='next_renewal_date']").datepicker({
- format: "yyyy-mm-dd"
- });
- function calculateNextNextRenewalDate() {
- const nextRenewalDate = $("#next_renewal_date").val();
- const termValue = parseInt($("#subscription_term_value").val() || 0);
- const termUnit = $("#subscription_term_unit").val();
- const nextNextRenewalDate = $("#next_next_renewal_date");
- if (!nextRenewalDate || termValue <= 0) {
- nextNextRenewalDate.val("");
- return;
- }
- const currentDate = new Date(nextRenewalDate);
- const originalDay = currentDate.getDate();
- if (termUnit === "months") {
- // 获取当前月份和年份
- let targetMonth = currentDate.getMonth() + termValue;
- let targetYear = currentDate.getFullYear() + Math.floor(targetMonth / 12);
- targetMonth = targetMonth % 12;
- // 先将日期设置为目标月的同一天
- currentDate.setFullYear(targetYear, targetMonth, originalDay);
- // 检查是否因月份天数不同而被自动调整
- if (currentDate.getMonth() !== targetMonth) {
- // 如果被调整,说明目标月份的天数比原始日期少
- // 将日期设置为目标月份的最后一天
- currentDate.setFullYear(targetYear, targetMonth + 1, 0);
- }
- } else {
- // 处理天数和年份的情况
- const adjustments = {
- days: "Date",
- years: "FullYear"
- };
- currentDate[`set${adjustments[termUnit]}`](
- currentDate[`get${adjustments[termUnit]}`]() + termValue
- );
- }
- nextNextRenewalDate.val(currentDate.toISOString().split("T")[0]);
- }
- $(document).ready(function() {
- initializeUI(); // 初始化
- bindEvents(); // 事件绑定
- @isset($node)
- setupNodeData(@json($node));
- @else
- setupDefaultValues();
- @endisset
- });
- function initializeUI() {
- $(".single-setting").hide();
- $("#v2_path").val("/" + string);
- }
- function bindEvents() {
- $("input:radio[name='type']").on("change", updateServiceType);
- $("#obfs").on("changed.bs.select", toggleObfsParam);
- $("#relay_node_id").on("changed.bs.select", toggleRelayConfig);
- $("#v2_net").on("changed.bs.select", updateV2RaySettings);
- $("#nodeForm").on("submit", formSubmit);
- $(document).on("change", "#next_renewal_date, #subscription_term_value, #subscription_term_unit", calculateNextNextRenewalDate);
- }
- function setupNodeData(nodeData) {
- const {
- type,
- labels,
- relay_node_id,
- port,
- profile,
- tls_provider,
- details
- } = nodeData;
- // 设置选项和输入值
- ["is_ddns", "is_udp", "status"].forEach(prop => nodeData[prop] && $(`#${prop}`).click());
- ["is_display", "detection_type", "type"].forEach(prop => $(`input[name="${prop}"][value="${nodeData[prop]}"]`).click());
- ["name", "server", "ip", "ipv6", "push_port", "traffic_rate", "speed_limit", "client_limit", "description", "sort"]
- .forEach(prop => $(`#${prop}`).val(nodeData[prop]));
- ["level", "rule_group_id", "country_code", "relay_node_id"].forEach(prop => $(`#${prop}`).selectpicker("val", nodeData[prop]));
- $("#labels").selectpicker("val", labels.map(label => label.id));
- if (details?.next_renewal_date) $("#next_renewal_date").datepicker("update", details.next_renewal_date);
- if (details?.subscription_term) setSubscriptionTerm(details.subscription_term);
- if (details?.renewal_cost) $("#renewal_cost").val(details.renewal_cost);
- calculateNextNextRenewalDate(); // 手动触发计算下下次续费日期
- if (relay_node_id) {
- $("#relay_port").val(port);
- } else {
- setupNodeTypeHandlers(type, profile, port, tls_provider);
- }
- function setSubscriptionTerm(term) {
- const [value, unit] = term.split(" ");
- $("#subscription_term_value").val(value || "");
- $("#subscription_term_unit").selectpicker("val", unit || "day"); // 默认选择 day
- }
- }
- function setupDefaultValues() {
- switchSetting("single");
- switchSetting("is_ddns");
- $("input[name='type'][value='0']").click();
- $("#status, #is_udp").click();
- }
- function setupNodeTypeHandlers(type, profile, port, tls_provider) {
- const typeHandlers = {
- 0: () => $("#method").selectpicker("val", profile?.method || null),
- 1: setSSRValues,
- 2: setV2RayValues,
- 3: () => $("#trojan_port").val(port),
- 4: setSSRValues
- };
- typeHandlers[type] && typeHandlers[type]();
- $("input[name='port']").val(port);
- function setSSRValues() {
- ["protocol", "obfs"].forEach(prop => $(`#${prop}`).selectpicker("val", profile[prop] || null));
- ["protocol_param", "obfs_param"].forEach(prop => $(`#${prop}`).val(profile[prop] || null));
- if (profile.passwd && port) {
- $("#single").click();
- $("#passwd").val(profile.passwd);
- }
- }
- function setV2RayValues() {
- ["v2_alter_id", "v2_host", "v2_sni", "v2_path"].forEach(prop => $(`#${prop}`).val(profile[prop] || null));
- ["v2_net", "v2_type"].forEach(prop => $(`#${prop}`).selectpicker("val", profile[prop] || null));
- $("#v2_method").selectpicker("val", profile["method"] || null);
- $("#v2_port").val(port);
- profile.v2_tls && $("#v2_tls").click();
- $("#tls_provider").val(tls_provider);
- }
- }
- function formSubmit(event) {
- event.preventDefault(); // 阻止表单的默认提交行为
- const $form = $(event.target); // 获取触发事件的表单
- // 获取所有非 hidden 的表单数据
- const data = Object.fromEntries(
- $form.find("input:not([hidden]), select, textarea")
- .serializeArray()
- .map(item => [item.name, item.value])
- );
- // 拼接 subscription_term
- const termValue = $("#subscription_term_value").val();
- const termUnit = $("#subscription_term_unit").val();
- data["subscription_term"] = termValue ? `${termValue} ${termUnit}` : null;
- // 将序列化的表单数据转换为 JSON 对象
- $form.find("input[type='checkbox']").each(function() {
- data[this.name] = this.checked ? 1 : 0;
- });
- // 处理多选 select
- $form.find("select[multiple]").each(function() {
- data[this.name] = $(this).val();
- });
- $.ajax({
- url: '{{ isset($node) ? route('admin.node.update', $node) : route('admin.node.store') }}',
- method: '{{ isset($node) ? 'PUT' : 'POST' }}',
- contentType: "application/json",
- data: JSON.stringify(data),
- success: function(ret) {
- if (ret.status === "success") {
- swal.fire({
- title: ret.message,
- icon: "success",
- timer: 1000,
- showConfirmButton: false
- }).then(() => window.location.href =
- '{{ route('admin.node.index') . (Request::getQueryString() ? '?' . Request::getQueryString() : '') }}');
- } else {
- swal.fire({
- title: '{{ trans('common.error') }}',
- text: ret.message,
- icon: "error"
- });
- }
- },
- error: function(data) {
- const errors = data.responseJSON?.errors;
- if (errors) {
- const errorList = Object.values(errors).map(error => `<li>${error}</li>`).join("");
- swal.fire({
- title: '{{ trans('admin.hint') }}',
- html: `<ul>${errorList}</ul>`,
- icon: "error",
- confirmButtonText: '{{ trans('common.confirm') }}'
- });
- }
- }
- });
- }
- function switchSetting(id) {
- const check = document.getElementById(id).checked;
- if (id === "single") {
- $(".single-setting").toggle(check);
- $("#single_port").attr({
- "hidden": !check,
- "required": check
- });
- if (!check) $("#passwd").val("");
- } else if (id === "is_ddns") {
- $("#ip, #ipv6").attr("readonly", check).val("");
- $("#server").attr("required", check);
- }
- }
- // 设置服务类型
- function updateServiceType() {
- const type = parseInt($(this).val());
- const settingsMap = {
- 0: [".ss-setting"],
- 1: [".ss-setting", ".ssr-setting"],
- 2: [".v2ray-setting", "#v2_port"],
- 3: [".trojan-setting", "#trojan_port"],
- 4: [".ss-setting", ".ssr-setting"]
- };
- $(".ss-setting, .ssr-setting, .v2ray-setting, .trojan-setting").hide();
- Object.keys(settingsMap).forEach(key => $(settingsMap[key].join(",")).hide());
- (settingsMap[type] || []).forEach(selector => $(selector).show());
- }
- function toggleObfsParam() {
- const $obfsParam = $(".obfs_param");
- $obfsParam.toggle($("#obfs").val() !== "plain");
- if ($("#obfs").val() === "plain") $("#obfs_param").val("");
- }
- function toggleRelayConfig() {
- const hasRelay = $("#relay_node_id").val() !== "";
- $(".relay-config").toggle(hasRelay);
- $(".proxy-config").toggle(!hasRelay);
- $("#relay_port").attr({
- hidden: !hasRelay,
- required: hasRelay
- });
- }
- // 设置V2Ray详细设置
- function updateV2RaySettings() {
- const net = $(this).val();
- const $type = $(".v2_type");
- const $typeOption = $("#type_option");
- const $host = $(".v2_host");
- const $path = $("#v2_path");
- $type.show();
- $host.show();
- if (!$path.val()) {
- $path.val("/" + string);
- }
- switch (net) {
- case "ws":
- case "http":
- $type.hide();
- break;
- case "domainsocket":
- $type.hide();
- $host.hide();
- break;
- case "quic":
- $typeOption.attr("disabled", false);
- if (!$path.val()) {
- $path.val(string);
- }
- break;
- case "kcp":
- case "tcp":
- default:
- $typeOption.attr("disabled", true);
- break;
- }
- $("#v2_type").selectpicker("refresh");
- }
- // 服务条款
- window.showTnc = function() {
- const jsonConfig = {
- "additional_ports": {
- "443": {
- "passwd": "ProxyPanel",
- "method": "none",
- "protocol": "auth_chain_a",
- "protocol_param": "#",
- "obfs": "plain",
- "obfs_param": "fe2.update.microsoft.com"
- }
- }
- };
- swal.fire({
- title: "[节点 user-config.json 配置示例]",
- width: "36em",
- html: `
- <div class="text-left">
- <ol>
- <li>请勿直接复制黏贴以下配置,SSR(R)会报错的</li>
- <li>确保服务器时间为CST</li>
- </ol>
- <pre class="bg-grey-800 text-white">${JSON.stringify(jsonConfig, null, 2)}</pre>
- </div>
- `,
- icon: "info"
- });
- };
- // 模式提示
- window.showPortsOnlyConfig = function() {
- swal.fire({
- title: "[节点 user-config.json 配置示例]",
- width: "36em",
- html: `
- <ul class="bg-grey-800 text-white text-left">
- <li>严格模式:"additional_ports_only": "true"</li>
- <li>兼容模式:"additional_ports_only": "false"</li>
- </ul>
- `,
- icon: "info"
- });
- };
- </script>
- @endsection
|