Compare commits
21 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
74c3d0ae25 | ||
|
|
ecd1bcafd5 | ||
|
|
184d2d117a | ||
|
|
89e6f17794 | ||
|
|
aef50fe0e3 | ||
|
|
fc0767ed25 | ||
|
|
dbf1724cca | ||
|
|
909aa4038e | ||
|
|
2d0a7d8d46 | ||
|
|
ca96cd1d82 | ||
|
|
91ab1e5dac | ||
|
|
b3a5f74df8 | ||
|
|
1f98be8ad8 | ||
|
|
453c7c98d0 | ||
|
|
91faed35c0 | ||
|
|
07bbaf6b6f | ||
|
|
e8feb7c431 | ||
|
|
4d16820526 | ||
|
|
92294b49c6 | ||
|
|
8a188a37c9 | ||
|
|
48af16c265 |
8
.github/workflows/build.yml
vendored
8
.github/workflows/build.yml
vendored
@@ -89,7 +89,7 @@ jobs:
|
|||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
submodules: recursive
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Download
|
- name: Download
|
||||||
uses: actions/download-artifact@v4
|
uses: actions/download-artifact@v4
|
||||||
@@ -101,13 +101,13 @@ jobs:
|
|||||||
- name: Pre Release
|
- name: Pre Release
|
||||||
run: |
|
run: |
|
||||||
pip install gitchangelog pystache mustache markdown
|
pip install gitchangelog pystache mustache markdown
|
||||||
prelease=$(curl --silent "https://api.github.com/repos/chen08209/FlClash/releases/latest" | grep -Po '"tag_name": "\K.*?(?=")' || echo "")
|
pre=$(curl --silent "https://api.github.com/repos/chen08209/FlClash/releases/latest" | grep -Po '"tag_name": "\K.*?(?=")' || echo "")
|
||||||
if [ -z "$prelease" ]; then
|
if [ -z "pre" ]; then
|
||||||
echo "init" > release.md
|
echo "init" > release.md
|
||||||
else
|
else
|
||||||
current="${{ github.ref_name }}"
|
current="${{ github.ref_name }}"
|
||||||
echo -e "\n\n<details markdown=1><summary>All changes from $current to the latest commit:</summary>\n\n" >> release.md
|
echo -e "\n\n<details markdown=1><summary>All changes from $current to the latest commit:</summary>\n\n" >> release.md
|
||||||
gitchangelog "${prelease}" >> release.md 2>&1 || echo "Error in gitchangelog"
|
gitchangelog "${pre}.." >> release.md 2>&1 || echo "Error in gitchangelog"
|
||||||
echo -e "\n\n</details>" >> release.md
|
echo -e "\n\n</details>" >> release.md
|
||||||
fi
|
fi
|
||||||
- name: Release
|
- name: Release
|
||||||
|
|||||||
@@ -164,10 +164,7 @@ func patchConfig(general *config.General) {
|
|||||||
resolver.DisableIPv6 = !general.IPv6
|
resolver.DisableIPv6 = !general.IPv6
|
||||||
}
|
}
|
||||||
|
|
||||||
func applyConfig(isPatch bool) bool {
|
func applyConfig(isPatch bool) {
|
||||||
if currentConfig == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
cfg, err := config.ParseRawConfig(currentConfig)
|
cfg, err := config.ParseRawConfig(currentConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
cfg, _ = config.ParseRawConfig(config.DefaultRawConfig())
|
cfg, _ = config.ParseRawConfig(config.DefaultRawConfig())
|
||||||
@@ -176,7 +173,5 @@ func applyConfig(isPatch bool) bool {
|
|||||||
patchConfig(cfg.General)
|
patchConfig(cfg.General)
|
||||||
} else {
|
} else {
|
||||||
executor.ApplyConfig(cfg, true)
|
executor.ApplyConfig(cfg, true)
|
||||||
|
|
||||||
}
|
}
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ func InitDartApi(api unsafe.Pointer) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func sendToPort(port int64, msg string) {
|
func SendToPort(port int64, msg string) {
|
||||||
var obj C.Dart_CObject
|
var obj C.Dart_CObject
|
||||||
obj._type = C.Dart_CObject_kString
|
obj._type = C.Dart_CObject_kString
|
||||||
msgString := C.CString(msg)
|
msgString := C.CString(msg)
|
||||||
|
|||||||
@@ -24,5 +24,5 @@ func (message *Message) toJson() string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func SendMessage(message Message) {
|
func SendMessage(message Message) {
|
||||||
sendToPort(*Port, message.toJson())
|
SendToPort(*Port, message.toJson())
|
||||||
}
|
}
|
||||||
|
|||||||
41
core/hub.go
41
core/hub.go
@@ -63,26 +63,30 @@ func validateConfig(s *C.char) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//export updateConfig
|
//export updateConfig
|
||||||
func updateConfig(s *C.char) bool {
|
func updateConfig(s *C.char, port C.longlong) {
|
||||||
paramsString := C.GoString(s)
|
i := int64(port)
|
||||||
var params = &GenerateConfigParams{}
|
go func() {
|
||||||
err := json.Unmarshal([]byte(paramsString), params)
|
paramsString := C.GoString(s)
|
||||||
if err != nil {
|
var params = &GenerateConfigParams{}
|
||||||
log.Errorln("generateConfig Unmarshal error %v", err)
|
err := json.Unmarshal([]byte(paramsString), params)
|
||||||
return false
|
if err != nil {
|
||||||
}
|
bridge.SendToPort(i, err.Error())
|
||||||
prof := decorationConfig(params.ProfilePath, *params.Config)
|
return
|
||||||
currentConfig = prof
|
}
|
||||||
if *params.IsPatch {
|
prof := decorationConfig(params.ProfilePath, *params.Config)
|
||||||
return applyConfig(true)
|
currentConfig = prof
|
||||||
} else {
|
if *params.IsPatch {
|
||||||
return applyConfig(false)
|
applyConfig(true)
|
||||||
}
|
} else {
|
||||||
|
applyConfig(false)
|
||||||
|
}
|
||||||
|
bridge.SendToPort(i, "")
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
//export getProxies
|
//export getProxies
|
||||||
func getProxies() *C.char {
|
func getProxies() *C.char {
|
||||||
data, err := json.Marshal(tunnel.Proxies())
|
data, err := json.Marshal(tunnel.ProxiesWithProviders())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return C.CString("")
|
return C.CString("")
|
||||||
}
|
}
|
||||||
@@ -98,7 +102,7 @@ func changeProxy(s *C.char) bool {
|
|||||||
log.Infoln("Unmarshal ChangeProxyParams %v", err)
|
log.Infoln("Unmarshal ChangeProxyParams %v", err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
proxies := tunnel.Proxies()
|
proxies := tunnel.ProxiesWithProviders()
|
||||||
proxy := proxies[*params.GroupName]
|
proxy := proxies[*params.GroupName]
|
||||||
if proxy == nil {
|
if proxy == nil {
|
||||||
return false
|
return false
|
||||||
@@ -148,7 +152,7 @@ func asyncTestDelay(s *C.char) {
|
|||||||
ctx, cancel := context.WithTimeout(context.Background(), time.Millisecond*time.Duration(params.Timeout))
|
ctx, cancel := context.WithTimeout(context.Background(), time.Millisecond*time.Duration(params.Timeout))
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
proxies := tunnel.Proxies()
|
proxies := tunnel.ProxiesWithProviders()
|
||||||
proxy := proxies[params.ProxyName]
|
proxy := proxies[params.ProxyName]
|
||||||
|
|
||||||
delayData := &Delay{
|
delayData := &Delay{
|
||||||
@@ -235,7 +239,6 @@ func getProviders() *C.char {
|
|||||||
return C.CString(string(data))
|
return C.CString(string(data))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//export getProvider
|
//export getProvider
|
||||||
func getProvider(name *C.char) *C.char {
|
func getProvider(name *C.char) *C.char {
|
||||||
providerName := C.GoString(name)
|
providerName := C.GoString(name)
|
||||||
|
|||||||
@@ -112,7 +112,6 @@ class ApplicationState extends State<Application> {
|
|||||||
primaryColor: config.primaryColor,
|
primaryColor: config.primaryColor,
|
||||||
),
|
),
|
||||||
builder: (_, state, child) {
|
builder: (_, state, child) {
|
||||||
debugPrint("[Application] update===>");
|
|
||||||
return DynamicColorBuilder(
|
return DynamicColorBuilder(
|
||||||
builder: (lightDynamic, darkDynamic) {
|
builder: (lightDynamic, darkDynamic) {
|
||||||
_updateSystemColorSchemes(lightDynamic, darkDynamic);
|
_updateSystemColorSchemes(lightDynamic, darkDynamic);
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
|
import 'dart:async';
|
||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
import 'dart:ffi';
|
import 'dart:ffi';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
import 'dart:isolate';
|
import 'dart:isolate';
|
||||||
|
|
||||||
import 'package:ffi/ffi.dart';
|
import 'package:ffi/ffi.dart';
|
||||||
|
import 'package:flutter/cupertino.dart';
|
||||||
import '../enum/enum.dart';
|
import '../enum/enum.dart';
|
||||||
import '../models/models.dart';
|
import '../models/models.dart';
|
||||||
import '../common/common.dart';
|
import '../common/common.dart';
|
||||||
@@ -16,19 +18,22 @@ class ClashCore {
|
|||||||
late final ClashFFI clashFFI;
|
late final ClashFFI clashFFI;
|
||||||
late final DynamicLibrary lib;
|
late final DynamicLibrary lib;
|
||||||
|
|
||||||
ClashCore._internal() {
|
DynamicLibrary _getClashLib() {
|
||||||
if (Platform.isWindows) {
|
if (Platform.isWindows) {
|
||||||
lib = DynamicLibrary.open("libclash.dll");
|
return DynamicLibrary.open("libclash.dll");
|
||||||
clashFFI = ClashFFI(lib);
|
|
||||||
}
|
}
|
||||||
if (Platform.isMacOS) {
|
if (Platform.isMacOS) {
|
||||||
lib = DynamicLibrary.open("libclash.dylib");
|
return DynamicLibrary.open("libclash.dylib");
|
||||||
clashFFI = ClashFFI(lib);
|
|
||||||
}
|
}
|
||||||
if (Platform.isAndroid || Platform.isLinux) {
|
if (Platform.isAndroid || Platform.isLinux) {
|
||||||
lib = DynamicLibrary.open("libclash.so");
|
return DynamicLibrary.open("libclash.so");
|
||||||
clashFFI = ClashFFI(lib);
|
|
||||||
}
|
}
|
||||||
|
throw "Platform is not supported";
|
||||||
|
}
|
||||||
|
|
||||||
|
ClashCore._internal() {
|
||||||
|
lib = _getClashLib();
|
||||||
|
clashFFI = ClashFFI(lib);
|
||||||
clashFFI.initNativeApiBridge(
|
clashFFI.initNativeApiBridge(
|
||||||
NativeApi.initializeApiDLData,
|
NativeApi.initializeApiDLData,
|
||||||
receiver.sendPort.nativePort,
|
receiver.sendPort.nativePort,
|
||||||
@@ -61,12 +66,21 @@ class ClashCore {
|
|||||||
1;
|
1;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool updateConfig(UpdateConfigParams updateConfigParams) {
|
Future<String> updateConfig(UpdateConfigParams updateConfigParams) {
|
||||||
|
final completer = Completer<String>();
|
||||||
|
final receiver = ReceivePort();
|
||||||
|
receiver.listen((message) {
|
||||||
|
if(!completer.isCompleted){
|
||||||
|
completer.complete(message);
|
||||||
|
receiver.close();
|
||||||
|
}
|
||||||
|
});
|
||||||
final params = json.encode(updateConfigParams);
|
final params = json.encode(updateConfigParams);
|
||||||
return clashFFI.updateConfig(
|
clashFFI.updateConfig(
|
||||||
params.toNativeUtf8().cast(),
|
params.toNativeUtf8().cast(),
|
||||||
) ==
|
receiver.sendPort.nativePort,
|
||||||
1;
|
);
|
||||||
|
return completer.future;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<List<Group>> getProxiesGroups() {
|
Future<List<Group>> getProxiesGroups() {
|
||||||
@@ -74,17 +88,19 @@ class ClashCore {
|
|||||||
final proxiesRawString = proxiesRaw.cast<Utf8>().toDartString();
|
final proxiesRawString = proxiesRaw.cast<Utf8>().toDartString();
|
||||||
return Isolate.run<List<Group>>(() {
|
return Isolate.run<List<Group>>(() {
|
||||||
final proxies = json.decode(proxiesRawString);
|
final proxies = json.decode(proxiesRawString);
|
||||||
final groupsRaw = (proxies[UsedProxy.GLOBAL.name]["all"] as List)
|
final groupNames = [
|
||||||
.where((e) {
|
UsedProxy.GLOBAL.name,
|
||||||
final proxy = proxies[e];
|
...(proxies[UsedProxy.GLOBAL.name]["all"] as List).where((e) {
|
||||||
final excludeName = !UsedProxyExtension.valueList
|
final proxy = proxies[e];
|
||||||
.where((element) => element != UsedProxy.GLOBAL.name)
|
return GroupTypeExtension.valueList.contains(proxy['type']);
|
||||||
.contains(proxy['name']);
|
})
|
||||||
final validType = GroupTypeExtension.valueList.contains(proxy['type']);
|
];
|
||||||
return excludeName && validType;
|
final groupsRaw = groupNames.map((groupName) {
|
||||||
}).map((groupName) {
|
|
||||||
final group = proxies[groupName];
|
final group = proxies[groupName];
|
||||||
group["all"] = ((group["all"] ?? []) as List)
|
group["all"] = ((group["all"] ?? []) as List)
|
||||||
|
.where(
|
||||||
|
(name) => !groupNames.contains(groupNames),
|
||||||
|
)
|
||||||
.map(
|
.map(
|
||||||
(name) => proxies[name],
|
(name) => proxies[name],
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -907,19 +907,22 @@ class ClashFFI {
|
|||||||
late final _validateConfig =
|
late final _validateConfig =
|
||||||
_validateConfigPtr.asFunction<int Function(ffi.Pointer<ffi.Char>)>();
|
_validateConfigPtr.asFunction<int Function(ffi.Pointer<ffi.Char>)>();
|
||||||
|
|
||||||
int updateConfig(
|
void updateConfig(
|
||||||
ffi.Pointer<ffi.Char> s,
|
ffi.Pointer<ffi.Char> s,
|
||||||
|
int port,
|
||||||
) {
|
) {
|
||||||
return _updateConfig(
|
return _updateConfig(
|
||||||
s,
|
s,
|
||||||
|
port,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
late final _updateConfigPtr =
|
late final _updateConfigPtr = _lookup<
|
||||||
_lookup<ffi.NativeFunction<GoUint8 Function(ffi.Pointer<ffi.Char>)>>(
|
ffi.NativeFunction<
|
||||||
'updateConfig');
|
ffi.Void Function(
|
||||||
|
ffi.Pointer<ffi.Char>, ffi.LongLong)>>('updateConfig');
|
||||||
late final _updateConfig =
|
late final _updateConfig =
|
||||||
_updateConfigPtr.asFunction<int Function(ffi.Pointer<ffi.Char>)>();
|
_updateConfigPtr.asFunction<void Function(ffi.Pointer<ffi.Char>, int)>();
|
||||||
|
|
||||||
ffi.Pointer<ffi.Char> getProxies() {
|
ffi.Pointer<ffi.Char> getProxies() {
|
||||||
return _getProxies();
|
return _getProxies();
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ Function debounce<F extends Function>(F func,{int milliseconds = 600}) {
|
|||||||
if (timer != null) {
|
if (timer != null) {
|
||||||
timer!.cancel();
|
timer!.cancel();
|
||||||
}
|
}
|
||||||
timer = Timer(Duration(milliseconds: milliseconds), () {
|
timer = Timer(Duration(milliseconds: milliseconds), () async {
|
||||||
Function.apply(func, args ?? [], namedArgs);
|
await Function.apply(func, args ?? [], namedArgs);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -56,7 +56,9 @@ class AppController {
|
|||||||
updateRunTime() {
|
updateRunTime() {
|
||||||
if (proxyManager.startTime != null) {
|
if (proxyManager.startTime != null) {
|
||||||
final startTimeStamp = proxyManager.startTime!.millisecondsSinceEpoch;
|
final startTimeStamp = proxyManager.startTime!.millisecondsSinceEpoch;
|
||||||
final nowTimeStamp = DateTime.now().millisecondsSinceEpoch;
|
final nowTimeStamp = DateTime
|
||||||
|
.now()
|
||||||
|
.millisecondsSinceEpoch;
|
||||||
appState.runTime = nowTimeStamp - startTimeStamp;
|
appState.runTime = nowTimeStamp - startTimeStamp;
|
||||||
} else {
|
} else {
|
||||||
appState.runTime = null;
|
appState.runTime = null;
|
||||||
@@ -115,7 +117,7 @@ class AppController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> updateClashConfig({bool isPatch = true}) async {
|
Future<String> updateClashConfig({bool isPatch = true}) async {
|
||||||
return await globalState.updateClashConfig(
|
return await globalState.updateClashConfig(
|
||||||
clashConfig: clashConfig,
|
clashConfig: clashConfig,
|
||||||
config: config,
|
config: config,
|
||||||
@@ -123,12 +125,31 @@ class AppController {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
applyProfile() async {
|
||||||
|
await globalState.applyProfile(
|
||||||
|
appState: appState,
|
||||||
|
config: config,
|
||||||
|
clashConfig: clashConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Function? _changeProfileDebounce;
|
||||||
|
|
||||||
|
changeProfileDebounce(String? profileId) {
|
||||||
|
if (profileId == config.currentProfileId) return;
|
||||||
|
config.currentProfileId = profileId;
|
||||||
|
_changeProfileDebounce ??= debounce<Function(String?)>((profileId) async {
|
||||||
|
await applyProfile();
|
||||||
|
appState.delayMap = {};
|
||||||
|
saveConfigPreferences();
|
||||||
|
});
|
||||||
|
_changeProfileDebounce!([profileId]);
|
||||||
|
}
|
||||||
|
|
||||||
changeProfile(String? value) async {
|
changeProfile(String? value) async {
|
||||||
if (value == config.currentProfileId) return;
|
if (value == config.currentProfileId) return;
|
||||||
config.currentProfileId = value;
|
config.currentProfileId = value;
|
||||||
await updateClashConfig(isPatch: false);
|
await applyProfile();
|
||||||
await updateGroups();
|
|
||||||
changeProxy();
|
|
||||||
appState.delayMap = {};
|
appState.delayMap = {};
|
||||||
saveConfigPreferences();
|
saveConfigPreferences();
|
||||||
}
|
}
|
||||||
@@ -138,14 +159,11 @@ class AppController {
|
|||||||
if (!profile.autoUpdate) return;
|
if (!profile.autoUpdate) return;
|
||||||
final isNotNeedUpdate = profile.lastUpdateDate
|
final isNotNeedUpdate = profile.lastUpdateDate
|
||||||
?.add(
|
?.add(
|
||||||
profile.autoUpdateDuration,
|
profile.autoUpdateDuration,
|
||||||
)
|
)
|
||||||
.isBeforeNow();
|
.isBeforeNow();
|
||||||
if (isNotNeedUpdate == false) continue;
|
if (isNotNeedUpdate == false) continue;
|
||||||
final result = await profile.update();
|
await profile.update();
|
||||||
if (result.type == ResultType.error) continue;
|
|
||||||
await updateGroups();
|
|
||||||
changeProxy();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -159,7 +177,7 @@ class AppController {
|
|||||||
|
|
||||||
clearCurrentDelay() {
|
clearCurrentDelay() {
|
||||||
final currentProxyName =
|
final currentProxyName =
|
||||||
appState.getCurrentProxyName(config.currentProxyName, clashConfig.mode);
|
appState.getCurrentProxyName(config.currentProxyName, clashConfig.mode);
|
||||||
if (currentProxyName == null) return;
|
if (currentProxyName == null) return;
|
||||||
appState.setDelay(Delay(name: currentProxyName, value: null));
|
appState.setDelay(Delay(name: currentProxyName, value: null));
|
||||||
}
|
}
|
||||||
@@ -246,7 +264,7 @@ class AppController {
|
|||||||
|
|
||||||
toProfiles() {
|
toProfiles() {
|
||||||
final index = globalState.currentNavigationItems.indexWhere(
|
final index = globalState.currentNavigationItems.indexWhere(
|
||||||
(element) => element.label == "profiles",
|
(element) => element.label == "profiles",
|
||||||
);
|
);
|
||||||
if (index != -1) {
|
if (index != -1) {
|
||||||
toPage(index);
|
toPage(index);
|
||||||
@@ -259,7 +277,7 @@ class AppController {
|
|||||||
final commonScaffoldState = globalState.homeScaffoldKey.currentState;
|
final commonScaffoldState = globalState.homeScaffoldKey.currentState;
|
||||||
if (commonScaffoldState?.mounted != true) return;
|
if (commonScaffoldState?.mounted != true) return;
|
||||||
commonScaffoldState?.loadingRun(
|
commonScaffoldState?.loadingRun(
|
||||||
() async {
|
() async {
|
||||||
await Future.delayed(const Duration(milliseconds: 300));
|
await Future.delayed(const Duration(milliseconds: 300));
|
||||||
final profile = Profile(
|
final profile = Profile(
|
||||||
url: url,
|
url: url,
|
||||||
@@ -280,7 +298,7 @@ class AppController {
|
|||||||
|
|
||||||
initLink() {
|
initLink() {
|
||||||
linkManager.initAppLinksListen(
|
linkManager.initAppLinksListen(
|
||||||
(url) {
|
(url) {
|
||||||
globalState.showMessage(
|
globalState.showMessage(
|
||||||
title: "${appLocalizations.add}${appLocalizations.profile}",
|
title: "${appLocalizations.add}${appLocalizations.profile}",
|
||||||
message: TextSpan(
|
message: TextSpan(
|
||||||
@@ -289,14 +307,20 @@ class AppController {
|
|||||||
TextSpan(
|
TextSpan(
|
||||||
text: " $url ",
|
text: " $url ",
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme
|
||||||
|
.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.primary,
|
||||||
decoration: TextDecoration.underline,
|
decoration: TextDecoration.underline,
|
||||||
decorationColor: Theme.of(context).colorScheme.primary,
|
decorationColor: Theme
|
||||||
|
.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.primary,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
TextSpan(
|
TextSpan(
|
||||||
text:
|
text:
|
||||||
"${appLocalizations.create}${appLocalizations.profile}"),
|
"${appLocalizations.create}${appLocalizations.profile}"),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
onTab: () {
|
onTab: () {
|
||||||
@@ -316,7 +340,7 @@ class AppController {
|
|||||||
final commonScaffoldState = globalState.homeScaffoldKey.currentState;
|
final commonScaffoldState = globalState.homeScaffoldKey.currentState;
|
||||||
if (commonScaffoldState?.mounted != true) return;
|
if (commonScaffoldState?.mounted != true) return;
|
||||||
commonScaffoldState?.loadingRun(
|
commonScaffoldState?.loadingRun(
|
||||||
() async {
|
() async {
|
||||||
await Future.delayed(const Duration(milliseconds: 300));
|
await Future.delayed(const Duration(milliseconds: 300));
|
||||||
final bytes = result.data?.bytes;
|
final bytes = result.data?.bytes;
|
||||||
if (bytes == null) {
|
if (bytes == null) {
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ class CoreInfo extends StatelessWidget {
|
|||||||
return Selector<AppState, VersionInfo?>(
|
return Selector<AppState, VersionInfo?>(
|
||||||
selector: (_, appState) => appState.versionInfo,
|
selector: (_, appState) => appState.versionInfo,
|
||||||
builder: (_, versionInfo, __) {
|
builder: (_, versionInfo, __) {
|
||||||
debugPrint("[CoreInfo] update===>");
|
|
||||||
return CommonCard(
|
return CommonCard(
|
||||||
info: Info(
|
info: Info(
|
||||||
label: appLocalizations.coreInfo,
|
label: appLocalizations.coreInfo,
|
||||||
|
|||||||
@@ -111,7 +111,6 @@ class _NetworkDetectionState extends State<NetworkDetection> {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
builder: (_, state, __) {
|
builder: (_, state, __) {
|
||||||
debugPrint("[UpdateCurrentDelay] update===>");
|
|
||||||
_updateCurrentDelay(
|
_updateCurrentDelay(
|
||||||
state.currentProxyName,
|
state.currentProxyName,
|
||||||
state.delay,
|
state.delay,
|
||||||
@@ -145,7 +144,6 @@ class _NetworkDetectionState extends State<NetworkDetection> {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
builder: (_, state, __) {
|
builder: (_, state, __) {
|
||||||
debugPrint("[NetworkDetection] update===>");
|
|
||||||
return Container(
|
return Container(
|
||||||
padding: const EdgeInsets.all(16).copyWith(top: 0),
|
padding: const EdgeInsets.all(16).copyWith(top: 0),
|
||||||
child: Column(
|
child: Column(
|
||||||
|
|||||||
@@ -63,7 +63,6 @@ class _StartButtonState extends State<StartButton>
|
|||||||
hasProfile: config.profiles.isNotEmpty,
|
hasProfile: config.profiles.isNotEmpty,
|
||||||
),
|
),
|
||||||
builder: (_, state, child) {
|
builder: (_, state, child) {
|
||||||
debugPrint("[StartButton] update===>");
|
|
||||||
if (!state.isInit || !state.hasProfile) {
|
if (!state.isInit || !state.hasProfile) {
|
||||||
return Container();
|
return Container();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ class ProfilesFragment extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _ProfilesFragmentState extends State<ProfilesFragment> {
|
class _ProfilesFragmentState extends State<ProfilesFragment> {
|
||||||
|
|
||||||
String _getLastUpdateTimeDifference(DateTime lastDateTime) {
|
String _getLastUpdateTimeDifference(DateTime lastDateTime) {
|
||||||
final currentDateTime = DateTime.now();
|
final currentDateTime = DateTime.now();
|
||||||
final difference = currentDateTime.difference(lastDateTime);
|
final difference = currentDateTime.difference(lastDateTime);
|
||||||
@@ -209,7 +210,7 @@ class _ProfilesFragmentState extends State<ProfilesFragment> {
|
|||||||
child: _profileItem(
|
child: _profileItem(
|
||||||
profile: profile,
|
profile: profile,
|
||||||
groupValue: state.currentProfileId,
|
groupValue: state.currentProfileId,
|
||||||
onChanged: context.appController.changeProfile,
|
onChanged: context.appController.changeProfileDebounce,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -234,7 +235,6 @@ class _ProfilesFragmentState extends State<ProfilesFragment> {
|
|||||||
currentProfileId: config.currentProfileId,
|
currentProfileId: config.currentProfileId,
|
||||||
),
|
),
|
||||||
builder: (context, state, child) {
|
builder: (context, state, child) {
|
||||||
debugPrint("[Profiles] update===>");
|
|
||||||
if (state.profiles.isEmpty) {
|
if (state.profiles.isEmpty) {
|
||||||
return NullStatus(
|
return NullStatus(
|
||||||
label: appLocalizations.nullProfileDesc,
|
label: appLocalizations.nullProfileDesc,
|
||||||
|
|||||||
@@ -91,7 +91,6 @@ class _ProxiesFragmentState extends State<ProxiesFragment>
|
|||||||
vsync: this,
|
vsync: this,
|
||||||
initialIndex: state.currentIndex,
|
initialIndex: state.currentIndex,
|
||||||
);
|
);
|
||||||
debugPrint("[Proxies] update===>");
|
|
||||||
return Column(
|
return Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
@@ -194,7 +193,7 @@ class _ProxiesTabViewState extends State<ProxiesTabView>
|
|||||||
_controller.dispose();
|
_controller.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
get group => widget.group;
|
Group get group => widget.group;
|
||||||
|
|
||||||
get measure => context.appController.measure;
|
get measure => context.appController.measure;
|
||||||
|
|
||||||
@@ -366,25 +365,33 @@ class _ProxiesTabViewState extends State<ProxiesTabView>
|
|||||||
}) {
|
}) {
|
||||||
return SingleChildScrollView(
|
return SingleChildScrollView(
|
||||||
padding: const EdgeInsets.all(16),
|
padding: const EdgeInsets.all(16),
|
||||||
child: AnimateGrid<Proxy>(
|
child: Selector3<AppState, Config, ClashConfig, ProxiesCardSelectorState>(
|
||||||
items: _getProxies(group.all, proxiesSortType),
|
selector: (_, appState, config, clashConfig) =>
|
||||||
columns: columns,
|
ProxiesCardSelectorState(
|
||||||
itemHeight: _getItemHeight(),
|
currentGroupName: appState.getCurrentGroupName(
|
||||||
keyBuilder: (item) {
|
config.currentGroupName,
|
||||||
return ObjectKey(item);
|
clashConfig.mode,
|
||||||
},
|
),
|
||||||
builder: (_, proxy) {
|
currentProxyName: appState.getCurrentProxyName(
|
||||||
return Selector3<AppState, Config, ClashConfig, String?>(
|
config.currentProxyName,
|
||||||
selector: (_, appState, config, clashConfig) =>
|
clashConfig.mode,
|
||||||
appState.getCurrentProxyName(
|
),
|
||||||
config.currentProxyName,
|
),
|
||||||
clashConfig.mode,
|
builder: (_, state, __) {
|
||||||
),
|
return AnimateGrid<Proxy>(
|
||||||
builder: (_, value, __) {
|
items: _getProxies(group.all, proxiesSortType),
|
||||||
final currentProxyName =
|
columns: columns,
|
||||||
group.type == GroupType.Selector ? value : group.now;
|
itemHeight: _getItemHeight(),
|
||||||
|
keyBuilder: (item) {
|
||||||
|
return ObjectKey(item);
|
||||||
|
},
|
||||||
|
builder: (_, proxy) {
|
||||||
|
final isSelected = group.type == GroupType.Selector
|
||||||
|
? group.name == state.currentGroupName &&
|
||||||
|
proxy.name == state.currentProxyName
|
||||||
|
: group.now == proxy.name;
|
||||||
return _card(
|
return _card(
|
||||||
isSelected: proxy.name == currentProxyName,
|
isSelected: isSelected,
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
if (group.type == GroupType.Selector) {
|
if (group.type == GroupType.Selector) {
|
||||||
final config = context.read<Config>();
|
final config = context.read<Config>();
|
||||||
|
|||||||
@@ -1887,3 +1887,148 @@ abstract class _ProxiesSelectorState implements ProxiesSelectorState {
|
|||||||
_$$ProxiesSelectorStateImplCopyWith<_$ProxiesSelectorStateImpl>
|
_$$ProxiesSelectorStateImplCopyWith<_$ProxiesSelectorStateImpl>
|
||||||
get copyWith => throw _privateConstructorUsedError;
|
get copyWith => throw _privateConstructorUsedError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
mixin _$ProxiesCardSelectorState {
|
||||||
|
String? get currentGroupName => throw _privateConstructorUsedError;
|
||||||
|
String? get currentProxyName => throw _privateConstructorUsedError;
|
||||||
|
|
||||||
|
@JsonKey(ignore: true)
|
||||||
|
$ProxiesCardSelectorStateCopyWith<ProxiesCardSelectorState> get copyWith =>
|
||||||
|
throw _privateConstructorUsedError;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
abstract class $ProxiesCardSelectorStateCopyWith<$Res> {
|
||||||
|
factory $ProxiesCardSelectorStateCopyWith(ProxiesCardSelectorState value,
|
||||||
|
$Res Function(ProxiesCardSelectorState) then) =
|
||||||
|
_$ProxiesCardSelectorStateCopyWithImpl<$Res, ProxiesCardSelectorState>;
|
||||||
|
@useResult
|
||||||
|
$Res call({String? currentGroupName, String? currentProxyName});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
class _$ProxiesCardSelectorStateCopyWithImpl<$Res,
|
||||||
|
$Val extends ProxiesCardSelectorState>
|
||||||
|
implements $ProxiesCardSelectorStateCopyWith<$Res> {
|
||||||
|
_$ProxiesCardSelectorStateCopyWithImpl(this._value, this._then);
|
||||||
|
|
||||||
|
// ignore: unused_field
|
||||||
|
final $Val _value;
|
||||||
|
// ignore: unused_field
|
||||||
|
final $Res Function($Val) _then;
|
||||||
|
|
||||||
|
@pragma('vm:prefer-inline')
|
||||||
|
@override
|
||||||
|
$Res call({
|
||||||
|
Object? currentGroupName = freezed,
|
||||||
|
Object? currentProxyName = freezed,
|
||||||
|
}) {
|
||||||
|
return _then(_value.copyWith(
|
||||||
|
currentGroupName: freezed == currentGroupName
|
||||||
|
? _value.currentGroupName
|
||||||
|
: currentGroupName // ignore: cast_nullable_to_non_nullable
|
||||||
|
as String?,
|
||||||
|
currentProxyName: freezed == currentProxyName
|
||||||
|
? _value.currentProxyName
|
||||||
|
: currentProxyName // ignore: cast_nullable_to_non_nullable
|
||||||
|
as String?,
|
||||||
|
) as $Val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
abstract class _$$ProxiesCardSelectorStateImplCopyWith<$Res>
|
||||||
|
implements $ProxiesCardSelectorStateCopyWith<$Res> {
|
||||||
|
factory _$$ProxiesCardSelectorStateImplCopyWith(
|
||||||
|
_$ProxiesCardSelectorStateImpl value,
|
||||||
|
$Res Function(_$ProxiesCardSelectorStateImpl) then) =
|
||||||
|
__$$ProxiesCardSelectorStateImplCopyWithImpl<$Res>;
|
||||||
|
@override
|
||||||
|
@useResult
|
||||||
|
$Res call({String? currentGroupName, String? currentProxyName});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
class __$$ProxiesCardSelectorStateImplCopyWithImpl<$Res>
|
||||||
|
extends _$ProxiesCardSelectorStateCopyWithImpl<$Res,
|
||||||
|
_$ProxiesCardSelectorStateImpl>
|
||||||
|
implements _$$ProxiesCardSelectorStateImplCopyWith<$Res> {
|
||||||
|
__$$ProxiesCardSelectorStateImplCopyWithImpl(
|
||||||
|
_$ProxiesCardSelectorStateImpl _value,
|
||||||
|
$Res Function(_$ProxiesCardSelectorStateImpl) _then)
|
||||||
|
: super(_value, _then);
|
||||||
|
|
||||||
|
@pragma('vm:prefer-inline')
|
||||||
|
@override
|
||||||
|
$Res call({
|
||||||
|
Object? currentGroupName = freezed,
|
||||||
|
Object? currentProxyName = freezed,
|
||||||
|
}) {
|
||||||
|
return _then(_$ProxiesCardSelectorStateImpl(
|
||||||
|
currentGroupName: freezed == currentGroupName
|
||||||
|
? _value.currentGroupName
|
||||||
|
: currentGroupName // ignore: cast_nullable_to_non_nullable
|
||||||
|
as String?,
|
||||||
|
currentProxyName: freezed == currentProxyName
|
||||||
|
? _value.currentProxyName
|
||||||
|
: currentProxyName // ignore: cast_nullable_to_non_nullable
|
||||||
|
as String?,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @nodoc
|
||||||
|
|
||||||
|
class _$ProxiesCardSelectorStateImpl implements _ProxiesCardSelectorState {
|
||||||
|
const _$ProxiesCardSelectorStateImpl(
|
||||||
|
{required this.currentGroupName, required this.currentProxyName});
|
||||||
|
|
||||||
|
@override
|
||||||
|
final String? currentGroupName;
|
||||||
|
@override
|
||||||
|
final String? currentProxyName;
|
||||||
|
|
||||||
|
@override
|
||||||
|
String toString() {
|
||||||
|
return 'ProxiesCardSelectorState(currentGroupName: $currentGroupName, currentProxyName: $currentProxyName)';
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
bool operator ==(Object other) {
|
||||||
|
return identical(this, other) ||
|
||||||
|
(other.runtimeType == runtimeType &&
|
||||||
|
other is _$ProxiesCardSelectorStateImpl &&
|
||||||
|
(identical(other.currentGroupName, currentGroupName) ||
|
||||||
|
other.currentGroupName == currentGroupName) &&
|
||||||
|
(identical(other.currentProxyName, currentProxyName) ||
|
||||||
|
other.currentProxyName == currentProxyName));
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
int get hashCode =>
|
||||||
|
Object.hash(runtimeType, currentGroupName, currentProxyName);
|
||||||
|
|
||||||
|
@JsonKey(ignore: true)
|
||||||
|
@override
|
||||||
|
@pragma('vm:prefer-inline')
|
||||||
|
_$$ProxiesCardSelectorStateImplCopyWith<_$ProxiesCardSelectorStateImpl>
|
||||||
|
get copyWith => __$$ProxiesCardSelectorStateImplCopyWithImpl<
|
||||||
|
_$ProxiesCardSelectorStateImpl>(this, _$identity);
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract class _ProxiesCardSelectorState implements ProxiesCardSelectorState {
|
||||||
|
const factory _ProxiesCardSelectorState(
|
||||||
|
{required final String? currentGroupName,
|
||||||
|
required final String? currentProxyName}) =
|
||||||
|
_$ProxiesCardSelectorStateImpl;
|
||||||
|
|
||||||
|
@override
|
||||||
|
String? get currentGroupName;
|
||||||
|
@override
|
||||||
|
String? get currentProxyName;
|
||||||
|
@override
|
||||||
|
@JsonKey(ignore: true)
|
||||||
|
_$$ProxiesCardSelectorStateImplCopyWith<_$ProxiesCardSelectorStateImpl>
|
||||||
|
get copyWith => throw _privateConstructorUsedError;
|
||||||
|
}
|
||||||
|
|||||||
@@ -111,3 +111,10 @@ class ProxiesSelectorState with _$ProxiesSelectorState{
|
|||||||
}) = _ProxiesSelectorState;
|
}) = _ProxiesSelectorState;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@freezed
|
||||||
|
class ProxiesCardSelectorState with _$ProxiesCardSelectorState{
|
||||||
|
const factory ProxiesCardSelectorState({
|
||||||
|
required String? currentGroupName,
|
||||||
|
required String? currentProxyName,
|
||||||
|
}) = _ProxiesCardSelectorState;
|
||||||
|
}
|
||||||
|
|||||||
@@ -140,7 +140,6 @@ class HomePage extends StatelessWidget {
|
|||||||
child: Selector<AppState, List<NavigationItem>>(
|
child: Selector<AppState, List<NavigationItem>>(
|
||||||
selector: (_, appState) => appState.navigationItems,
|
selector: (_, appState) => appState.navigationItems,
|
||||||
builder: (_, navigationItems, __) {
|
builder: (_, navigationItems, __) {
|
||||||
debugPrint("[Home] update===>");
|
|
||||||
final desktopNavigationItems = navigationItems
|
final desktopNavigationItems = navigationItems
|
||||||
.where(
|
.where(
|
||||||
(element) =>
|
(element) =>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import 'dart:io';
|
|||||||
import 'package:animations/animations.dart';
|
import 'package:animations/animations.dart';
|
||||||
import 'package:fl_clash/clash/clash.dart';
|
import 'package:fl_clash/clash/clash.dart';
|
||||||
import 'package:fl_clash/enum/enum.dart';
|
import 'package:fl_clash/enum/enum.dart';
|
||||||
import 'package:fl_clash/models/clash_config.dart';
|
|
||||||
import 'package:fl_clash/plugins/app.dart';
|
import 'package:fl_clash/plugins/app.dart';
|
||||||
import 'package:fl_clash/widgets/scaffold.dart';
|
import 'package:fl_clash/widgets/scaffold.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@@ -39,7 +38,7 @@ class GlobalState {
|
|||||||
timer?.cancel();
|
timer?.cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> updateClashConfig({
|
Future<String> updateClashConfig({
|
||||||
required ClashConfig clashConfig,
|
required ClashConfig clashConfig,
|
||||||
required Config config,
|
required Config config,
|
||||||
bool isPatch = true,
|
bool isPatch = true,
|
||||||
@@ -88,6 +87,25 @@ class GlobalState {
|
|||||||
updateCurrentDelayDebounce!([proxyName]);
|
updateCurrentDelayDebounce!([proxyName]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
applyProfile({
|
||||||
|
required AppState appState,
|
||||||
|
required Config config,
|
||||||
|
required ClashConfig clashConfig,
|
||||||
|
}) async {
|
||||||
|
final res = await updateClashConfig(
|
||||||
|
clashConfig: clashConfig,
|
||||||
|
config: config,
|
||||||
|
isPatch: false,
|
||||||
|
);
|
||||||
|
if (res.isNotEmpty) return Result.error(message: res);
|
||||||
|
await updateGroups(appState);
|
||||||
|
changeProxy(
|
||||||
|
appState: appState,
|
||||||
|
config: config,
|
||||||
|
clashConfig: clashConfig,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
init({
|
init({
|
||||||
required AppState appState,
|
required AppState appState,
|
||||||
required Config config,
|
required Config config,
|
||||||
@@ -101,18 +119,12 @@ class GlobalState {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
if (!appState.isInit) return;
|
if (!appState.isInit) return;
|
||||||
await updateClashConfig(
|
await applyProfile(
|
||||||
clashConfig: clashConfig,
|
|
||||||
config: config,
|
|
||||||
isPatch: false,
|
|
||||||
);
|
|
||||||
updateGroups(appState);
|
|
||||||
updateCoreVersionInfo(appState);
|
|
||||||
changeProxy(
|
|
||||||
appState: appState,
|
appState: appState,
|
||||||
config: config,
|
config: config,
|
||||||
clashConfig: clashConfig,
|
clashConfig: clashConfig,
|
||||||
);
|
);
|
||||||
|
updateCoreVersionInfo(appState);
|
||||||
}
|
}
|
||||||
|
|
||||||
changeProxy({
|
changeProxy({
|
||||||
@@ -167,7 +179,7 @@ class GlobalState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<void> updateGroups(AppState appState) async {
|
Future<void> updateGroups(AppState appState) async {
|
||||||
appState.groups = await clashCore.getProxiesGroups();
|
appState.groups = await clashCore.getProxiesGroups();
|
||||||
}
|
}
|
||||||
|
|
||||||
showMessage({
|
showMessage({
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ class AppStateContainer extends StatelessWidget {
|
|||||||
return Selector<Config, bool>(
|
return Selector<Config, bool>(
|
||||||
selector: (_, config) => config.autoLaunch,
|
selector: (_, config) => config.autoLaunch,
|
||||||
builder: (_, isAutoLaunch, child) {
|
builder: (_, isAutoLaunch, child) {
|
||||||
debugPrint("[autoLaunchContainer] update===>");
|
|
||||||
autoLaunch?.updateStatus(isAutoLaunch);
|
autoLaunch?.updateStatus(isAutoLaunch);
|
||||||
return child!;
|
return child!;
|
||||||
},
|
},
|
||||||
@@ -35,7 +34,6 @@ class AppStateContainer extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
builder: (context, state, child) {
|
builder: (context, state, child) {
|
||||||
debugPrint("[NavigationsContainer] update===>");
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback(
|
WidgetsBinding.instance.addPostFrameCallback(
|
||||||
(_) {
|
(_) {
|
||||||
context.appController.appState.navigationItems =
|
context.appController.appState.navigationItems =
|
||||||
|
|||||||
@@ -446,7 +446,6 @@ class _OpenContainerRoute<T> extends ModalRoute<T> {
|
|||||||
return Selector<Config, ThemeMode>(
|
return Selector<Config, ThemeMode>(
|
||||||
selector: (_, config) => config.themeMode,
|
selector: (_, config) => config.themeMode,
|
||||||
builder: (_, __, ___) {
|
builder: (_, __, ___) {
|
||||||
debugPrint("[OpenContainerTheme] update===>");
|
|
||||||
_colorTween = _getColorTween(
|
_colorTween = _getColorTween(
|
||||||
transitionType: transitionType,
|
transitionType: transitionType,
|
||||||
closedColor: Theme.of(context).colorScheme.background,
|
closedColor: Theme.of(context).colorScheme.background,
|
||||||
|
|||||||
@@ -144,7 +144,6 @@ class _TrayContainerState extends State<TrayContainer> with TrayListener {
|
|||||||
locale: config.locale,
|
locale: config.locale,
|
||||||
),
|
),
|
||||||
builder: (_, state, child) {
|
builder: (_, state, child) {
|
||||||
debugPrint("[TrayContainer] update===>");
|
|
||||||
WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
|
WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
|
||||||
updateMenu(state);
|
updateMenu(state);
|
||||||
});
|
});
|
||||||
|
|||||||
12
pubspec.lock
12
pubspec.lock
@@ -279,10 +279,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: flutter_plugin_android_lifecycle
|
name: flutter_plugin_android_lifecycle
|
||||||
sha256: "592dc01a18961a51c24ae5d963b724b2b7fa4a95c100fe8eb6ca8a5a4732cadf"
|
sha256: "8cf40eebf5dec866a6d1956ad7b4f7016e6c0cc69847ab946833b7d43743809f"
|
||||||
url: "https://pub.flutter-io.cn"
|
url: "https://pub.flutter-io.cn"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.0.18"
|
version: "2.0.19"
|
||||||
flutter_test:
|
flutter_test:
|
||||||
dependency: "direct dev"
|
dependency: "direct dev"
|
||||||
description: flutter
|
description: flutter
|
||||||
@@ -957,10 +957,10 @@ packages:
|
|||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: win32_registry
|
name: win32_registry
|
||||||
sha256: "10589e0d7f4e053f2c61023a31c9ce01146656a70b7b7f0828c0b46d7da2a9bb"
|
sha256: "41fd8a189940d8696b1b810efb9abcf60827b6cbfab90b0c43e8439e3a39d85a"
|
||||||
url: "https://pub.flutter-io.cn"
|
url: "https://pub.flutter-io.cn"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.1.3"
|
version: "1.1.2"
|
||||||
window_manager:
|
window_manager:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -997,10 +997,10 @@ packages:
|
|||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: yaml_edit
|
name: yaml_edit
|
||||||
sha256: c566f4f804215d84a7a2c377667f546c6033d5b34b4f9e60dfb09d17c4e97826
|
sha256: e9c1a3543d2da0db3e90270dbb1e4eebc985ee5e3ffe468d83224472b2194a5f
|
||||||
url: "https://pub.flutter-io.cn"
|
url: "https://pub.flutter-io.cn"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.2.0"
|
version: "2.2.1"
|
||||||
sdks:
|
sdks:
|
||||||
dart: ">=3.3.0 <4.0.0"
|
dart: ">=3.3.0 <4.0.0"
|
||||||
flutter: ">=3.19.0"
|
flutter: ">=3.19.0"
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
name: fl_clash
|
name: fl_clash
|
||||||
description: A multi-platform proxy client based on ClashMeta, simple and easy to use, open-source and ad-free.
|
description: A multi-platform proxy client based on ClashMeta, simple and easy to use, open-source and ad-free.
|
||||||
publish_to: 'none'
|
publish_to: 'none'
|
||||||
version: 0.7.2
|
version: 0.7.8
|
||||||
environment:
|
environment:
|
||||||
sdk: '>=3.1.0 <4.0.0'
|
sdk: '>=3.1.0 <4.0.0'
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user