initial: push
BIN
.DS_Store
vendored
Normal file
604
MeowRelay.xcodeproj/project.pbxproj
Normal file
|
|
@ -0,0 +1,604 @@
|
||||||
|
// !$*UTF8*$!
|
||||||
|
{
|
||||||
|
archiveVersion = 1;
|
||||||
|
classes = {
|
||||||
|
};
|
||||||
|
objectVersion = 77;
|
||||||
|
objects = {
|
||||||
|
|
||||||
|
/* Begin PBXContainerItemProxy section */
|
||||||
|
36E7064A2F0ED8500091D2B2 /* PBXContainerItemProxy */ = {
|
||||||
|
isa = PBXContainerItemProxy;
|
||||||
|
containerPortal = 36E706322F0ED84F0091D2B2 /* Project object */;
|
||||||
|
proxyType = 1;
|
||||||
|
remoteGlobalIDString = 36E706392F0ED84F0091D2B2;
|
||||||
|
remoteInfo = MeowRelay;
|
||||||
|
};
|
||||||
|
36E706542F0ED8500091D2B2 /* PBXContainerItemProxy */ = {
|
||||||
|
isa = PBXContainerItemProxy;
|
||||||
|
containerPortal = 36E706322F0ED84F0091D2B2 /* Project object */;
|
||||||
|
proxyType = 1;
|
||||||
|
remoteGlobalIDString = 36E706392F0ED84F0091D2B2;
|
||||||
|
remoteInfo = MeowRelay;
|
||||||
|
};
|
||||||
|
/* End PBXContainerItemProxy section */
|
||||||
|
|
||||||
|
/* Begin PBXFileReference section */
|
||||||
|
36E7063A2F0ED84F0091D2B2 /* MeowRelay.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = MeowRelay.app; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
|
36E706492F0ED8500091D2B2 /* MeowRelayTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = MeowRelayTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
|
36E706532F0ED8500091D2B2 /* MeowRelayUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = MeowRelayUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
|
/* End PBXFileReference section */
|
||||||
|
|
||||||
|
/* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */
|
||||||
|
36E706672F0ED8E60091D2B2 /* Exceptions for "MeowRelay" folder in "MeowRelay" target */ = {
|
||||||
|
isa = PBXFileSystemSynchronizedBuildFileExceptionSet;
|
||||||
|
membershipExceptions = (
|
||||||
|
Info.plist,
|
||||||
|
);
|
||||||
|
target = 36E706392F0ED84F0091D2B2 /* MeowRelay */;
|
||||||
|
};
|
||||||
|
/* End PBXFileSystemSynchronizedBuildFileExceptionSet section */
|
||||||
|
|
||||||
|
/* Begin PBXFileSystemSynchronizedRootGroup section */
|
||||||
|
36E7063C2F0ED84F0091D2B2 /* MeowRelay */ = {
|
||||||
|
isa = PBXFileSystemSynchronizedRootGroup;
|
||||||
|
exceptions = (
|
||||||
|
36E706672F0ED8E60091D2B2 /* Exceptions for "MeowRelay" folder in "MeowRelay" target */,
|
||||||
|
);
|
||||||
|
path = MeowRelay;
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
|
/* End PBXFileSystemSynchronizedRootGroup section */
|
||||||
|
|
||||||
|
/* Begin PBXFrameworksBuildPhase section */
|
||||||
|
36E706372F0ED84F0091D2B2 /* Frameworks */ = {
|
||||||
|
isa = PBXFrameworksBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E706462F0ED8500091D2B2 /* Frameworks */ = {
|
||||||
|
isa = PBXFrameworksBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E706502F0ED8500091D2B2 /* Frameworks */ = {
|
||||||
|
isa = PBXFrameworksBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXFrameworksBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin PBXGroup section */
|
||||||
|
36E706312F0ED84F0091D2B2 = {
|
||||||
|
isa = PBXGroup;
|
||||||
|
children = (
|
||||||
|
36E7063C2F0ED84F0091D2B2 /* MeowRelay */,
|
||||||
|
36E7063B2F0ED84F0091D2B2 /* Products */,
|
||||||
|
);
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
|
36E7063B2F0ED84F0091D2B2 /* Products */ = {
|
||||||
|
isa = PBXGroup;
|
||||||
|
children = (
|
||||||
|
36E7063A2F0ED84F0091D2B2 /* MeowRelay.app */,
|
||||||
|
36E706492F0ED8500091D2B2 /* MeowRelayTests.xctest */,
|
||||||
|
36E706532F0ED8500091D2B2 /* MeowRelayUITests.xctest */,
|
||||||
|
);
|
||||||
|
name = Products;
|
||||||
|
sourceTree = "<group>";
|
||||||
|
};
|
||||||
|
/* End PBXGroup section */
|
||||||
|
|
||||||
|
/* Begin PBXNativeTarget section */
|
||||||
|
36E706392F0ED84F0091D2B2 /* MeowRelay */ = {
|
||||||
|
isa = PBXNativeTarget;
|
||||||
|
buildConfigurationList = 36E7065D2F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelay" */;
|
||||||
|
buildPhases = (
|
||||||
|
36E706362F0ED84F0091D2B2 /* Sources */,
|
||||||
|
36E706372F0ED84F0091D2B2 /* Frameworks */,
|
||||||
|
36E706382F0ED84F0091D2B2 /* Resources */,
|
||||||
|
);
|
||||||
|
buildRules = (
|
||||||
|
);
|
||||||
|
dependencies = (
|
||||||
|
);
|
||||||
|
fileSystemSynchronizedGroups = (
|
||||||
|
36E7063C2F0ED84F0091D2B2 /* MeowRelay */,
|
||||||
|
);
|
||||||
|
name = MeowRelay;
|
||||||
|
packageProductDependencies = (
|
||||||
|
);
|
||||||
|
productName = MeowRelay;
|
||||||
|
productReference = 36E7063A2F0ED84F0091D2B2 /* MeowRelay.app */;
|
||||||
|
productType = "com.apple.product-type.application";
|
||||||
|
};
|
||||||
|
36E706482F0ED8500091D2B2 /* MeowRelayTests */ = {
|
||||||
|
isa = PBXNativeTarget;
|
||||||
|
buildConfigurationList = 36E706602F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelayTests" */;
|
||||||
|
buildPhases = (
|
||||||
|
36E706452F0ED8500091D2B2 /* Sources */,
|
||||||
|
36E706462F0ED8500091D2B2 /* Frameworks */,
|
||||||
|
36E706472F0ED8500091D2B2 /* Resources */,
|
||||||
|
);
|
||||||
|
buildRules = (
|
||||||
|
);
|
||||||
|
dependencies = (
|
||||||
|
36E7064B2F0ED8500091D2B2 /* PBXTargetDependency */,
|
||||||
|
);
|
||||||
|
name = MeowRelayTests;
|
||||||
|
packageProductDependencies = (
|
||||||
|
);
|
||||||
|
productName = MeowRelayTests;
|
||||||
|
productReference = 36E706492F0ED8500091D2B2 /* MeowRelayTests.xctest */;
|
||||||
|
productType = "com.apple.product-type.bundle.unit-test";
|
||||||
|
};
|
||||||
|
36E706522F0ED8500091D2B2 /* MeowRelayUITests */ = {
|
||||||
|
isa = PBXNativeTarget;
|
||||||
|
buildConfigurationList = 36E706632F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelayUITests" */;
|
||||||
|
buildPhases = (
|
||||||
|
36E7064F2F0ED8500091D2B2 /* Sources */,
|
||||||
|
36E706502F0ED8500091D2B2 /* Frameworks */,
|
||||||
|
36E706512F0ED8500091D2B2 /* Resources */,
|
||||||
|
);
|
||||||
|
buildRules = (
|
||||||
|
);
|
||||||
|
dependencies = (
|
||||||
|
36E706552F0ED8500091D2B2 /* PBXTargetDependency */,
|
||||||
|
);
|
||||||
|
name = MeowRelayUITests;
|
||||||
|
packageProductDependencies = (
|
||||||
|
);
|
||||||
|
productName = MeowRelayUITests;
|
||||||
|
productReference = 36E706532F0ED8500091D2B2 /* MeowRelayUITests.xctest */;
|
||||||
|
productType = "com.apple.product-type.bundle.ui-testing";
|
||||||
|
};
|
||||||
|
/* End PBXNativeTarget section */
|
||||||
|
|
||||||
|
/* Begin PBXProject section */
|
||||||
|
36E706322F0ED84F0091D2B2 /* Project object */ = {
|
||||||
|
isa = PBXProject;
|
||||||
|
attributes = {
|
||||||
|
BuildIndependentTargetsInParallel = 1;
|
||||||
|
LastSwiftUpdateCheck = 2620;
|
||||||
|
LastUpgradeCheck = 2620;
|
||||||
|
TargetAttributes = {
|
||||||
|
36E706392F0ED84F0091D2B2 = {
|
||||||
|
CreatedOnToolsVersion = 26.2;
|
||||||
|
};
|
||||||
|
36E706482F0ED8500091D2B2 = {
|
||||||
|
CreatedOnToolsVersion = 26.2;
|
||||||
|
TestTargetID = 36E706392F0ED84F0091D2B2;
|
||||||
|
};
|
||||||
|
36E706522F0ED8500091D2B2 = {
|
||||||
|
CreatedOnToolsVersion = 26.2;
|
||||||
|
TestTargetID = 36E706392F0ED84F0091D2B2;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
buildConfigurationList = 36E706352F0ED84F0091D2B2 /* Build configuration list for PBXProject "MeowRelay" */;
|
||||||
|
developmentRegion = en;
|
||||||
|
hasScannedForEncodings = 0;
|
||||||
|
knownRegions = (
|
||||||
|
en,
|
||||||
|
Base,
|
||||||
|
);
|
||||||
|
mainGroup = 36E706312F0ED84F0091D2B2;
|
||||||
|
minimizedProjectReferenceProxies = 1;
|
||||||
|
preferredProjectObjectVersion = 77;
|
||||||
|
productRefGroup = 36E7063B2F0ED84F0091D2B2 /* Products */;
|
||||||
|
projectDirPath = "";
|
||||||
|
projectRoot = "";
|
||||||
|
targets = (
|
||||||
|
36E706392F0ED84F0091D2B2 /* MeowRelay */,
|
||||||
|
36E706482F0ED8500091D2B2 /* MeowRelayTests */,
|
||||||
|
36E706522F0ED8500091D2B2 /* MeowRelayUITests */,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
/* End PBXProject section */
|
||||||
|
|
||||||
|
/* Begin PBXResourcesBuildPhase section */
|
||||||
|
36E706382F0ED84F0091D2B2 /* Resources */ = {
|
||||||
|
isa = PBXResourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E706472F0ED8500091D2B2 /* Resources */ = {
|
||||||
|
isa = PBXResourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E706512F0ED8500091D2B2 /* Resources */ = {
|
||||||
|
isa = PBXResourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXResourcesBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin PBXSourcesBuildPhase section */
|
||||||
|
36E706362F0ED84F0091D2B2 /* Sources */ = {
|
||||||
|
isa = PBXSourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E706452F0ED8500091D2B2 /* Sources */ = {
|
||||||
|
isa = PBXSourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
36E7064F2F0ED8500091D2B2 /* Sources */ = {
|
||||||
|
isa = PBXSourcesBuildPhase;
|
||||||
|
buildActionMask = 2147483647;
|
||||||
|
files = (
|
||||||
|
);
|
||||||
|
runOnlyForDeploymentPostprocessing = 0;
|
||||||
|
};
|
||||||
|
/* End PBXSourcesBuildPhase section */
|
||||||
|
|
||||||
|
/* Begin PBXTargetDependency section */
|
||||||
|
36E7064B2F0ED8500091D2B2 /* PBXTargetDependency */ = {
|
||||||
|
isa = PBXTargetDependency;
|
||||||
|
target = 36E706392F0ED84F0091D2B2 /* MeowRelay */;
|
||||||
|
targetProxy = 36E7064A2F0ED8500091D2B2 /* PBXContainerItemProxy */;
|
||||||
|
};
|
||||||
|
36E706552F0ED8500091D2B2 /* PBXTargetDependency */ = {
|
||||||
|
isa = PBXTargetDependency;
|
||||||
|
target = 36E706392F0ED84F0091D2B2 /* MeowRelay */;
|
||||||
|
targetProxy = 36E706542F0ED8500091D2B2 /* PBXContainerItemProxy */;
|
||||||
|
};
|
||||||
|
/* End PBXTargetDependency section */
|
||||||
|
|
||||||
|
/* Begin XCBuildConfiguration section */
|
||||||
|
36E7065B2F0ED8500091D2B2 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||||
|
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
|
||||||
|
CLANG_ANALYZER_NONNULL = YES;
|
||||||
|
CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
|
||||||
|
CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
|
||||||
|
CLANG_ENABLE_MODULES = YES;
|
||||||
|
CLANG_ENABLE_OBJC_ARC = YES;
|
||||||
|
CLANG_ENABLE_OBJC_WEAK = YES;
|
||||||
|
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||||
|
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_COMMA = YES;
|
||||||
|
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||||
|
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||||
|
CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
|
||||||
|
CLANG_WARN_EMPTY_BODY = YES;
|
||||||
|
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||||
|
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||||
|
CLANG_WARN_INT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||||
|
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||||
|
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
|
||||||
|
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||||
|
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||||
|
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||||
|
CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
|
||||||
|
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||||
|
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||||
|
COPY_PHASE_STRIP = NO;
|
||||||
|
DEBUG_INFORMATION_FORMAT = dwarf;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||||
|
ENABLE_TESTABILITY = YES;
|
||||||
|
ENABLE_USER_SCRIPT_SANDBOXING = YES;
|
||||||
|
GCC_C_LANGUAGE_STANDARD = gnu17;
|
||||||
|
GCC_DYNAMIC_NO_PIC = NO;
|
||||||
|
GCC_NO_COMMON_BLOCKS = YES;
|
||||||
|
GCC_OPTIMIZATION_LEVEL = 0;
|
||||||
|
GCC_PREPROCESSOR_DEFINITIONS = (
|
||||||
|
"DEBUG=1",
|
||||||
|
"$(inherited)",
|
||||||
|
);
|
||||||
|
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||||
|
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||||
|
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||||
|
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||||
|
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||||
|
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||||
|
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 26.2;
|
||||||
|
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
|
||||||
|
MTL_FAST_MATH = YES;
|
||||||
|
ONLY_ACTIVE_ARCH = YES;
|
||||||
|
SDKROOT = macosx;
|
||||||
|
SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)";
|
||||||
|
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
36E7065C2F0ED8500091D2B2 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||||
|
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
|
||||||
|
CLANG_ANALYZER_NONNULL = YES;
|
||||||
|
CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
|
||||||
|
CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
|
||||||
|
CLANG_ENABLE_MODULES = YES;
|
||||||
|
CLANG_ENABLE_OBJC_ARC = YES;
|
||||||
|
CLANG_ENABLE_OBJC_WEAK = YES;
|
||||||
|
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||||
|
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_COMMA = YES;
|
||||||
|
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||||
|
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||||
|
CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
|
||||||
|
CLANG_WARN_EMPTY_BODY = YES;
|
||||||
|
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||||
|
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||||
|
CLANG_WARN_INT_CONVERSION = YES;
|
||||||
|
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||||
|
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||||
|
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||||
|
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
|
||||||
|
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||||
|
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||||
|
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||||
|
CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
|
||||||
|
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||||
|
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||||
|
COPY_PHASE_STRIP = NO;
|
||||||
|
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
ENABLE_NS_ASSERTIONS = NO;
|
||||||
|
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||||
|
ENABLE_USER_SCRIPT_SANDBOXING = YES;
|
||||||
|
GCC_C_LANGUAGE_STANDARD = gnu17;
|
||||||
|
GCC_NO_COMMON_BLOCKS = YES;
|
||||||
|
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||||
|
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||||
|
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||||
|
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||||
|
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||||
|
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||||
|
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 26.2;
|
||||||
|
MTL_ENABLE_DEBUG_INFO = NO;
|
||||||
|
MTL_FAST_MATH = YES;
|
||||||
|
SDKROOT = macosx;
|
||||||
|
SWIFT_COMPILATION_MODE = wholemodule;
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
36E7065E2F0ED8500091D2B2 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
|
ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
COMBINE_HIDPI_IMAGES = YES;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
ENABLE_APP_SANDBOX = YES;
|
||||||
|
ENABLE_HARDENED_RUNTIME = YES;
|
||||||
|
ENABLE_INCOMING_NETWORK_CONNECTIONS = NO;
|
||||||
|
ENABLE_OUTGOING_NETWORK_CONNECTIONS = YES;
|
||||||
|
ENABLE_PREVIEWS = YES;
|
||||||
|
ENABLE_RESOURCE_ACCESS_AUDIO_INPUT = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_BLUETOOTH = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CALENDARS = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CAMERA = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CONTACTS = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_LOCATION = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_PRINTING = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_USB = NO;
|
||||||
|
ENABLE_USER_SELECTED_FILES = readonly;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
INFOPLIST_FILE = MeowRelay/Info.plist;
|
||||||
|
INFOPLIST_KEY_CFBundleDisplayName = MeowRelay;
|
||||||
|
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking";
|
||||||
|
INFOPLIST_KEY_LSUIElement = YES;
|
||||||
|
INFOPLIST_KEY_NSHumanReadableCopyright = "";
|
||||||
|
LD_RUNPATH_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"@executable_path/../Frameworks",
|
||||||
|
);
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 14.6;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelay;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
REGISTER_APP_GROUPS = YES;
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = YES;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = YES;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
36E7065F2F0ED8500091D2B2 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
|
ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
|
||||||
|
"CODE_SIGN_IDENTITY[sdk=macosx*]" = "-";
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
COMBINE_HIDPI_IMAGES = YES;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
ENABLE_APP_SANDBOX = YES;
|
||||||
|
ENABLE_HARDENED_RUNTIME = YES;
|
||||||
|
ENABLE_INCOMING_NETWORK_CONNECTIONS = NO;
|
||||||
|
ENABLE_OUTGOING_NETWORK_CONNECTIONS = YES;
|
||||||
|
ENABLE_PREVIEWS = YES;
|
||||||
|
ENABLE_RESOURCE_ACCESS_AUDIO_INPUT = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_BLUETOOTH = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CALENDARS = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CAMERA = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_CONTACTS = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_LOCATION = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_PRINTING = NO;
|
||||||
|
ENABLE_RESOURCE_ACCESS_USB = NO;
|
||||||
|
ENABLE_USER_SELECTED_FILES = readonly;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
INFOPLIST_FILE = MeowRelay/Info.plist;
|
||||||
|
INFOPLIST_KEY_CFBundleDisplayName = MeowRelay;
|
||||||
|
INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.social-networking";
|
||||||
|
INFOPLIST_KEY_LSUIElement = YES;
|
||||||
|
INFOPLIST_KEY_NSHumanReadableCopyright = "";
|
||||||
|
LD_RUNPATH_SEARCH_PATHS = (
|
||||||
|
"$(inherited)",
|
||||||
|
"@executable_path/../Frameworks",
|
||||||
|
);
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 14.6;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelay;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
REGISTER_APP_GROUPS = YES;
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = YES;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_DEFAULT_ACTOR_ISOLATION = MainActor;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = YES;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
36E706612F0ED8500091D2B2 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 26.2;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelayTests;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = NO;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = NO;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
TEST_HOST = "$(BUILT_PRODUCTS_DIR)/MeowRelay.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/MeowRelay";
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
36E706622F0ED8500091D2B2 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
BUNDLE_LOADER = "$(TEST_HOST)";
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
MACOSX_DEPLOYMENT_TARGET = 26.2;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelayTests;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = NO;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = NO;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
TEST_HOST = "$(BUILT_PRODUCTS_DIR)/MeowRelay.app/$(BUNDLE_EXECUTABLE_FOLDER_PATH)/MeowRelay";
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
36E706642F0ED8500091D2B2 /* Debug */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelayUITests;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = NO;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = NO;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
TEST_TARGET_NAME = MeowRelay;
|
||||||
|
};
|
||||||
|
name = Debug;
|
||||||
|
};
|
||||||
|
36E706652F0ED8500091D2B2 /* Release */ = {
|
||||||
|
isa = XCBuildConfiguration;
|
||||||
|
buildSettings = {
|
||||||
|
CODE_SIGN_STYLE = Automatic;
|
||||||
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
|
DEVELOPMENT_TEAM = L46RPVUB7F;
|
||||||
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
MARKETING_VERSION = 1.0;
|
||||||
|
PRODUCT_BUNDLE_IDENTIFIER = com.meowfox.MeowRelayUITests;
|
||||||
|
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||||
|
STRING_CATALOG_GENERATE_SYMBOLS = NO;
|
||||||
|
SWIFT_APPROACHABLE_CONCURRENCY = YES;
|
||||||
|
SWIFT_EMIT_LOC_STRINGS = NO;
|
||||||
|
SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES;
|
||||||
|
SWIFT_VERSION = 5.0;
|
||||||
|
TEST_TARGET_NAME = MeowRelay;
|
||||||
|
};
|
||||||
|
name = Release;
|
||||||
|
};
|
||||||
|
/* End XCBuildConfiguration section */
|
||||||
|
|
||||||
|
/* Begin XCConfigurationList section */
|
||||||
|
36E706352F0ED84F0091D2B2 /* Build configuration list for PBXProject "MeowRelay" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
36E7065B2F0ED8500091D2B2 /* Debug */,
|
||||||
|
36E7065C2F0ED8500091D2B2 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
36E7065D2F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelay" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
36E7065E2F0ED8500091D2B2 /* Debug */,
|
||||||
|
36E7065F2F0ED8500091D2B2 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
36E706602F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelayTests" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
36E706612F0ED8500091D2B2 /* Debug */,
|
||||||
|
36E706622F0ED8500091D2B2 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
36E706632F0ED8500091D2B2 /* Build configuration list for PBXNativeTarget "MeowRelayUITests" */ = {
|
||||||
|
isa = XCConfigurationList;
|
||||||
|
buildConfigurations = (
|
||||||
|
36E706642F0ED8500091D2B2 /* Debug */,
|
||||||
|
36E706652F0ED8500091D2B2 /* Release */,
|
||||||
|
);
|
||||||
|
defaultConfigurationIsVisible = 0;
|
||||||
|
defaultConfigurationName = Release;
|
||||||
|
};
|
||||||
|
/* End XCConfigurationList section */
|
||||||
|
};
|
||||||
|
rootObject = 36E706322F0ED84F0091D2B2 /* Project object */;
|
||||||
|
}
|
||||||
7
MeowRelay.xcodeproj/project.xcworkspace/contents.xcworkspacedata
generated
Normal file
|
|
@ -0,0 +1,7 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<Workspace
|
||||||
|
version = "1.0">
|
||||||
|
<FileRef
|
||||||
|
location = "self:">
|
||||||
|
</FileRef>
|
||||||
|
</Workspace>
|
||||||
BIN
MeowRelay.xcodeproj/project.xcworkspace/xcuserdata/meowfox.xcuserdatad/UserInterfaceState.xcuserstate
generated
Normal file
|
|
@ -0,0 +1,14 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>SchemeUserState</key>
|
||||||
|
<dict>
|
||||||
|
<key>MeowRelay.xcscheme_^#shared#^_</key>
|
||||||
|
<dict>
|
||||||
|
<key>orderHint</key>
|
||||||
|
<integer>0</integer>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
BIN
MeowRelay/.DS_Store
vendored
Normal file
BIN
MeowRelay/Assets.xcassets/.DS_Store
vendored
Normal file
11
MeowRelay/Assets.xcassets/AccentColor.colorset/Contents.json
Normal file
|
|
@ -0,0 +1,11 @@
|
||||||
|
{
|
||||||
|
"colors" : [
|
||||||
|
{
|
||||||
|
"idiom" : "universal"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
||||||
68
MeowRelay/Assets.xcassets/AppIcon.appiconset/Contents.json
Normal file
|
|
@ -0,0 +1,68 @@
|
||||||
|
{
|
||||||
|
"images" : [
|
||||||
|
{
|
||||||
|
"filename" : "icon_16x16.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "16x16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_16x16@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "16x16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_32x32.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "32x32"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_32x32@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "32x32"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_128x128.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "128x128"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_128x128@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "128x128"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_256x256.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "256x256"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_256x256@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "256x256"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_512x512.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "512x512"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_512x512@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "512x512"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
||||||
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_128x128.png
Normal file
|
After Width: | Height: | Size: 8.5 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_128x128@2x.png
Normal file
|
After Width: | Height: | Size: 18 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_16x16.png
Normal file
|
After Width: | Height: | Size: 919 B |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_16x16@2x.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_256x256.png
Normal file
|
After Width: | Height: | Size: 18 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_256x256@2x.png
Normal file
|
After Width: | Height: | Size: 38 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_32x32.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_32x32@2x.png
Normal file
|
After Width: | Height: | Size: 4.1 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_512x512.png
Normal file
|
After Width: | Height: | Size: 38 KiB |
BIN
MeowRelay/Assets.xcassets/AppIcon.appiconset/icon_512x512@2x.png
Normal file
|
After Width: | Height: | Size: 44 KiB |
6
MeowRelay/Assets.xcassets/Contents.json
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
{
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
||||||
68
MeowRelay/Assets.xcassets/trayIcon.appiconset/Contents.json
Normal file
|
|
@ -0,0 +1,68 @@
|
||||||
|
{
|
||||||
|
"images" : [
|
||||||
|
{
|
||||||
|
"filename" : "icon_16x16.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "16x16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_16x16@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "16x16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_32x32.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "32x32"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_32x32@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "32x32"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_128x128.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "128x128"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_128x128@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "128x128"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_256x256.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "256x256"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_256x256@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "256x256"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_512x512.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "512x512"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon_512x512@2x.png",
|
||||||
|
"idiom" : "mac",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "512x512"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
||||||
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_128x128.png
Normal file
|
After Width: | Height: | Size: 8.5 KiB |
|
After Width: | Height: | Size: 18 KiB |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_16x16.png
Normal file
|
After Width: | Height: | Size: 919 B |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_16x16@2x.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_256x256.png
Normal file
|
After Width: | Height: | Size: 18 KiB |
|
After Width: | Height: | Size: 38 KiB |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_32x32.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_32x32@2x.png
Normal file
|
After Width: | Height: | Size: 4.1 KiB |
BIN
MeowRelay/Assets.xcassets/trayIcon.appiconset/icon_512x512.png
Normal file
|
After Width: | Height: | Size: 38 KiB |
|
After Width: | Height: | Size: 44 KiB |
11
MeowRelay/Info.plist
Normal file
|
|
@ -0,0 +1,11 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>NSAppTransportSecurity</key>
|
||||||
|
<dict>
|
||||||
|
<key>NSAllowsArbitraryLoads</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
482
MeowRelay/MeowRelayApp.swift
Normal file
|
|
@ -0,0 +1,482 @@
|
||||||
|
import SwiftUI
|
||||||
|
import Combine
|
||||||
|
|
||||||
|
struct Route: Identifiable, Hashable, Equatable {
|
||||||
|
let id: String
|
||||||
|
let code: String
|
||||||
|
let provider: String
|
||||||
|
let name: String
|
||||||
|
let flag: String
|
||||||
|
|
||||||
|
init(code: String, provider: String, name: String) {
|
||||||
|
self.code = code
|
||||||
|
self.provider = provider
|
||||||
|
self.name = name
|
||||||
|
self.id = code + provider + name
|
||||||
|
|
||||||
|
let base: UInt32 = 127397
|
||||||
|
var s = ""
|
||||||
|
for v in code.uppercased().unicodeScalars {
|
||||||
|
if let scal = UnicodeScalar(base + v.value) {
|
||||||
|
s.unicodeScalars.append(scal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.flag = s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Feature: Identifiable, Codable, Equatable {
|
||||||
|
var id: String { name }
|
||||||
|
let name: String
|
||||||
|
let desc: String
|
||||||
|
var value: Bool
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ServerItem: Identifiable, Equatable {
|
||||||
|
let id: String
|
||||||
|
let detail: ServerDetail
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ServerDetail: Decodable, Equatable {
|
||||||
|
let connected: Int?
|
||||||
|
let traffic: [Double]?
|
||||||
|
let uptime: Double?
|
||||||
|
let ping: [String]?
|
||||||
|
let speed: [Double]?
|
||||||
|
let country: String?
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UserProfile: Decodable {
|
||||||
|
let user: String?
|
||||||
|
let current: String?
|
||||||
|
let ebalka: Bool?
|
||||||
|
let userData: UserData?
|
||||||
|
let features: [Feature]?
|
||||||
|
let provider: String?
|
||||||
|
enum CodingKeys: String, CodingKey {
|
||||||
|
case user, current, ebalka, features, provider
|
||||||
|
case userData = "user_data"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct UserData: Decodable {
|
||||||
|
let share: Bool?
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ServerStats: Decodable {
|
||||||
|
let uptime: Double?
|
||||||
|
let servers: [String: ServerDetail]?
|
||||||
|
}
|
||||||
|
|
||||||
|
class RelayViewModel: ObservableObject {
|
||||||
|
@Published var activeTab: Int = 0
|
||||||
|
@Published var routes: [Route] = []
|
||||||
|
@Published var serverList: [ServerItem] = []
|
||||||
|
@Published var features: [Feature] = []
|
||||||
|
@Published var userLabel: String = "..."
|
||||||
|
@Published var currentCode: String = ""
|
||||||
|
@Published var currentProvider: String = ""
|
||||||
|
@Published var isShared: Bool = false
|
||||||
|
@Published var isEbalka: Bool = false
|
||||||
|
@Published var statsLabel: String = "--"
|
||||||
|
@Published var totalServers: Int = 0
|
||||||
|
|
||||||
|
private var cancellables = Set<AnyCancellable>()
|
||||||
|
private let baseURL = "http://10.7.0.1:7777"
|
||||||
|
|
||||||
|
init() {
|
||||||
|
refreshAll()
|
||||||
|
Timer.publish(every: 3, on: .main, in: .common)
|
||||||
|
.autoconnect()
|
||||||
|
.sink { [weak self] _ in
|
||||||
|
guard let self = self else { return }
|
||||||
|
if self.activeTab == 1 { self.fetchStats() }
|
||||||
|
}
|
||||||
|
.store(in: &cancellables)
|
||||||
|
}
|
||||||
|
|
||||||
|
func refreshAll() {
|
||||||
|
fetchProfile()
|
||||||
|
fetchRoutes()
|
||||||
|
fetchStats()
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchProfile() {
|
||||||
|
get(url: "/me/profile", type: UserProfile.self) { [weak self] p in
|
||||||
|
guard let self = self else { return }
|
||||||
|
let newUser = p.user ?? "Unknown"
|
||||||
|
if self.userLabel != newUser { self.userLabel = newUser }
|
||||||
|
|
||||||
|
let parts = p.current?.components(separatedBy: "-")
|
||||||
|
let newCode = parts?.first?.uppercased() ?? ""
|
||||||
|
if self.currentCode != newCode { self.currentCode = newCode }
|
||||||
|
|
||||||
|
let newProv = p.provider ?? ""
|
||||||
|
if self.currentProvider != newProv { self.currentProvider = newProv }
|
||||||
|
|
||||||
|
let newShare = p.userData?.share ?? false
|
||||||
|
if self.isShared != newShare { self.isShared = newShare }
|
||||||
|
|
||||||
|
let newEbalka = p.ebalka ?? false
|
||||||
|
if self.isEbalka != newEbalka { self.isEbalka = newEbalka }
|
||||||
|
|
||||||
|
let newFeat = p.features ?? []
|
||||||
|
if self.features != newFeat { self.features = newFeat }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchRoutes() {
|
||||||
|
guard let url = URL(string: "\(baseURL)/routes") else { return }
|
||||||
|
URLSession.shared.dataTask(with: url) { [weak self] data, _, _ in
|
||||||
|
guard let self = self, let data = data else { return }
|
||||||
|
var newRoutes: [Route] = []
|
||||||
|
if let dict = try? JSONSerialization.jsonObject(with: data) as? [String: Any] {
|
||||||
|
for (k, v) in dict { self.parseRoute(k, v, &newRoutes) }
|
||||||
|
} else if let arr = try? JSONSerialization.jsonObject(with: data) as? [Any] {
|
||||||
|
for item in arr { self.parseRoute(nil, item, &newRoutes) }
|
||||||
|
}
|
||||||
|
let sorted = newRoutes.sorted { $0.name < $1.name }
|
||||||
|
DispatchQueue.main.async {
|
||||||
|
if self.routes != sorted { self.routes = sorted }
|
||||||
|
}
|
||||||
|
}.resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func parseRoute(_ key: String?, _ val: Any, _ list: inout [Route]) {
|
||||||
|
let code = key?.uppercased() ?? ""
|
||||||
|
if let name = val as? String {
|
||||||
|
let c = code.isEmpty ? name.uppercased() : code
|
||||||
|
list.append(Route(code: c, provider: "", name: name))
|
||||||
|
} else if let obj = val as? [String: Any] {
|
||||||
|
let c = (obj["code"] as? String ?? code).uppercased()
|
||||||
|
let name = obj["name"] as? String ?? c
|
||||||
|
let provider = obj["provider"] as? String ?? c
|
||||||
|
list.append(Route(code: c, provider: provider, name: name))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchStats() {
|
||||||
|
get(url: "/stats", type: ServerStats.self) { [weak self] s in
|
||||||
|
guard let self = self else { return }
|
||||||
|
let sec = Int(s.uptime ?? 0)
|
||||||
|
let newStatsLabel = "\(sec / 3600)h \((sec % 3600) / 60)m"
|
||||||
|
if self.statsLabel != newStatsLabel { self.statsLabel = newStatsLabel }
|
||||||
|
|
||||||
|
let dict = s.servers ?? [:]
|
||||||
|
if self.totalServers != dict.count { self.totalServers = dict.count }
|
||||||
|
|
||||||
|
let list = dict.map { ServerItem(id: $0.key, detail: $0.value) }
|
||||||
|
let sorted = list.sorted { ($0.detail.connected ?? 0) > ($1.detail.connected ?? 0) }
|
||||||
|
|
||||||
|
if self.serverList != sorted { self.serverList = sorted }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func selectRoute(_ r: Route) {
|
||||||
|
currentCode = r.code
|
||||||
|
currentProvider = r.provider
|
||||||
|
post("/me/country", ["countryCode": r.code, "provider": r.provider])
|
||||||
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { self.fetchProfile() }
|
||||||
|
}
|
||||||
|
|
||||||
|
func toggleShare(_ val: Bool) {
|
||||||
|
isShared = val
|
||||||
|
post("/me/profile", ["shareTraffic": val])
|
||||||
|
}
|
||||||
|
|
||||||
|
func toggleEbalka(_ val: Bool) {
|
||||||
|
isEbalka = val
|
||||||
|
post("/me/ebalka", ["ip": NSNull(), "ebalka": val])
|
||||||
|
}
|
||||||
|
|
||||||
|
func toggleFeature(_ name: String, _ val: Bool) {
|
||||||
|
post("/me/feature", ["name": name, "value": val])
|
||||||
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) { self.fetchProfile() }
|
||||||
|
}
|
||||||
|
|
||||||
|
func srvAction(_ endpoint: String, _ id: String) {
|
||||||
|
post("/admin/\(endpoint)", ["iface": id])
|
||||||
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { self.fetchStats() }
|
||||||
|
}
|
||||||
|
|
||||||
|
private func get<T: Decodable>(url: String, type: T.Type, completion: @escaping (T) -> Void) {
|
||||||
|
guard let u = URL(string: baseURL + url) else { return }
|
||||||
|
URLSession.shared.dataTask(with: u) { d, _, _ in
|
||||||
|
guard let d = d, let obj = try? JSONDecoder().decode(T.self, from: d) else { return }
|
||||||
|
DispatchQueue.main.async { completion(obj) }
|
||||||
|
}.resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
private func post(_ url: String, _ body: [String: Any]) {
|
||||||
|
guard let u = URL(string: baseURL + url) else { return }
|
||||||
|
var req = URLRequest(url: u)
|
||||||
|
req.httpMethod = "POST"
|
||||||
|
req.addValue("application/json", forHTTPHeaderField: "Content-Type")
|
||||||
|
req.httpBody = try? JSONSerialization.data(withJSONObject: body)
|
||||||
|
URLSession.shared.dataTask(with: req).resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
func flag(_ c: String?) -> String {
|
||||||
|
guard let code = c else { return "🏳️" }
|
||||||
|
let base: UInt32 = 127397
|
||||||
|
var s = ""
|
||||||
|
for v in code.uppercased().unicodeScalars {
|
||||||
|
if let scal = UnicodeScalar(base + v.value) {
|
||||||
|
s.unicodeScalars.append(scal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func fmtTime(_ sec: Double) -> String {
|
||||||
|
let s = Int(sec)
|
||||||
|
let d = s / 86400; let h = (s % 86400) / 3600; let m = (s % 3600) / 60
|
||||||
|
if d > 0 { return "\(d)d \(h)h" }
|
||||||
|
return "\(h)h \(m)m"
|
||||||
|
}
|
||||||
|
|
||||||
|
func avgPing(_ arr: [String]?) -> Double {
|
||||||
|
guard let arr = arr, !arr.isEmpty else { return 0 }
|
||||||
|
let sum = arr.compactMap { Double($0) }.reduce(0, +)
|
||||||
|
return sum / Double(arr.count)
|
||||||
|
}
|
||||||
|
|
||||||
|
func fmtBytes(_ v: Double?) -> String {
|
||||||
|
guard let v = v else { return "0B" }
|
||||||
|
let units = ["B", "KB", "MB", "GB"]
|
||||||
|
var s = v; var i = 0
|
||||||
|
while s > 1024 && i < 3 { s /= 1024; i += 1 }
|
||||||
|
return String(format: "%.1f%@", s, units[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
func fmtSpeed(_ v: Double?) -> String {
|
||||||
|
guard let v = v else { return "0B/s" }
|
||||||
|
return fmtBytes(v) + "/s"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@main
|
||||||
|
struct MeowRelayApp: App {
|
||||||
|
@StateObject var vm = RelayViewModel()
|
||||||
|
|
||||||
|
var body: some Scene {
|
||||||
|
MenuBarExtra("MeowRelay", systemImage: "pawprint") {
|
||||||
|
VStack(spacing: 0) {
|
||||||
|
HStack {
|
||||||
|
Text("MeowRelay").font(.headline).foregroundColor(.primary)
|
||||||
|
Spacer()
|
||||||
|
Text(vm.userLabel).font(.caption).foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
.padding(12)
|
||||||
|
|
||||||
|
Picker("", selection: $vm.activeTab) {
|
||||||
|
Text("Routes").tag(0)
|
||||||
|
Text("Stats").tag(1)
|
||||||
|
Text("Features").tag(2)
|
||||||
|
}
|
||||||
|
.pickerStyle(.segmented)
|
||||||
|
.padding(.horizontal)
|
||||||
|
.padding(.bottom, 8)
|
||||||
|
|
||||||
|
Divider()
|
||||||
|
|
||||||
|
Group {
|
||||||
|
if vm.activeTab == 0 {
|
||||||
|
RoutesView(vm: vm)
|
||||||
|
} else if vm.activeTab == 1 {
|
||||||
|
StatsView(vm: vm)
|
||||||
|
} else {
|
||||||
|
FeaturesView(vm: vm)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.frame(height: 320)
|
||||||
|
|
||||||
|
Divider()
|
||||||
|
|
||||||
|
HStack {
|
||||||
|
Text("Up: \(vm.statsLabel)").font(.caption).foregroundColor(.secondary)
|
||||||
|
Spacer()
|
||||||
|
Button("Quit") { NSApplication.shared.terminate(nil) }
|
||||||
|
.buttonStyle(.plain)
|
||||||
|
.font(.caption)
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
.padding(10)
|
||||||
|
}
|
||||||
|
.frame(width: 320)
|
||||||
|
.background(.ultraThinMaterial)
|
||||||
|
}
|
||||||
|
.menuBarExtraStyle(.window)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct RoutesView: View {
|
||||||
|
@ObservedObject var vm: RelayViewModel
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack(spacing: 0) {
|
||||||
|
HStack {
|
||||||
|
Text(vm.currentCode).bold()
|
||||||
|
if !vm.currentProvider.isEmpty {
|
||||||
|
Text("(\(vm.currentProvider))").font(.caption).foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
Button(action: { vm.refreshAll() }) {
|
||||||
|
Image(systemName: "arrow.clockwise")
|
||||||
|
}.buttonStyle(.plain)
|
||||||
|
}
|
||||||
|
.padding(8)
|
||||||
|
.background(Color.primary.opacity(0.05))
|
||||||
|
|
||||||
|
ScrollView {
|
||||||
|
LazyVStack(spacing: 1) {
|
||||||
|
ForEach(vm.routes) { r in
|
||||||
|
let isActive = (vm.currentCode == r.code && vm.currentProvider == r.provider)
|
||||||
|
|
||||||
|
Button(action: { vm.selectRoute(r) }) {
|
||||||
|
HStack {
|
||||||
|
Text(r.flag)
|
||||||
|
Text(r.name).foregroundColor(isActive ? .green : .primary)
|
||||||
|
if !r.provider.isEmpty {
|
||||||
|
Text(r.provider).font(.caption).foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
if isActive {
|
||||||
|
Image(systemName: "checkmark").foregroundColor(.green)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.padding(.vertical, 6)
|
||||||
|
.padding(.horizontal, 12)
|
||||||
|
.contentShape(Rectangle())
|
||||||
|
}
|
||||||
|
.buttonStyle(.plain)
|
||||||
|
.background(isActive ? Color.primary.opacity(0.1) : Color.clear)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HStack(spacing: 12) {
|
||||||
|
Toggle("Share", isOn: Binding(
|
||||||
|
get: { vm.isShared },
|
||||||
|
set: { vm.toggleShare($0) }
|
||||||
|
))
|
||||||
|
.toggleStyle(.switch)
|
||||||
|
.controlSize(.small)
|
||||||
|
|
||||||
|
Divider().frame(height: 16)
|
||||||
|
|
||||||
|
Toggle("80% Loss", isOn: Binding(
|
||||||
|
get: { vm.isEbalka },
|
||||||
|
set: { vm.toggleEbalka($0) }
|
||||||
|
))
|
||||||
|
.toggleStyle(.switch)
|
||||||
|
.controlSize(.small)
|
||||||
|
}
|
||||||
|
.padding(8)
|
||||||
|
.background(Color.primary.opacity(0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct StatsView: View {
|
||||||
|
@ObservedObject var vm: RelayViewModel
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
ScrollView {
|
||||||
|
LazyVStack(spacing: 4) {
|
||||||
|
if vm.serverList.isEmpty {
|
||||||
|
Text("No active servers").foregroundColor(.secondary).padding()
|
||||||
|
}
|
||||||
|
ForEach(vm.serverList) { item in
|
||||||
|
let d = item.detail
|
||||||
|
let ping = vm.avgPing(d.ping)
|
||||||
|
let pingText = String(format: "%.3f", ping)
|
||||||
|
let pColor: Color = ping < 50 ? .green : (ping < 150 ? .yellow : .red)
|
||||||
|
|
||||||
|
VStack(alignment: .leading, spacing: 6) {
|
||||||
|
HStack {
|
||||||
|
Text(vm.flag(d.country) + " " + item.id).bold().foregroundColor(.primary)
|
||||||
|
Spacer()
|
||||||
|
Text("\(d.connected ?? 0) users").font(.caption).foregroundColor(.green)
|
||||||
|
}
|
||||||
|
HStack {
|
||||||
|
VStack(alignment: .leading, spacing: 2) {
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("↓").foregroundColor(.green)
|
||||||
|
Text("\(vm.fmtBytes(d.traffic?[0]))")
|
||||||
|
Text("(\(vm.fmtSpeed(d.speed?[0])))").foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
HStack(spacing: 4) {
|
||||||
|
Text("↑").foregroundColor(.blue)
|
||||||
|
Text("\(vm.fmtBytes(d.traffic?[1]))")
|
||||||
|
Text("(\(vm.fmtSpeed(d.speed?[1])))").foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.font(.system(size: 10))
|
||||||
|
|
||||||
|
Spacer()
|
||||||
|
|
||||||
|
VStack(alignment: .trailing, spacing: 2) {
|
||||||
|
Text("\(pingText) ms").font(.caption).bold().foregroundColor(pColor)
|
||||||
|
Text(vm.fmtTime(d.uptime ?? 0)).font(.system(size: 10)).foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
HStack(spacing: 8) {
|
||||||
|
TinyBtn("Connect") { vm.srvAction("connect_by_id", item.id) }
|
||||||
|
TinyBtn("Kill") { vm.srvAction("kill_by_id", item.id) }
|
||||||
|
TinyBtn("DPI") { vm.srvAction("dpi_test_by_id", item.id) }
|
||||||
|
}.padding(.top, 2)
|
||||||
|
}
|
||||||
|
.padding(10)
|
||||||
|
.background(Color.primary.opacity(0.05))
|
||||||
|
.cornerRadius(6)
|
||||||
|
.padding(.horizontal, 8)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.padding(.top, 4)
|
||||||
|
}
|
||||||
|
.onAppear { vm.fetchStats() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct FeaturesView: View {
|
||||||
|
@ObservedObject var vm: RelayViewModel
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
ScrollView {
|
||||||
|
LazyVStack {
|
||||||
|
ForEach(vm.features) { f in
|
||||||
|
HStack {
|
||||||
|
VStack(alignment: .leading) {
|
||||||
|
Text(f.name).font(.subheadline)
|
||||||
|
Text(f.desc).font(.caption2).foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
Toggle("", isOn: Binding(
|
||||||
|
get: { f.value },
|
||||||
|
set: { vm.toggleFeature(f.name, $0) }
|
||||||
|
)).toggleStyle(.switch).labelsHidden()
|
||||||
|
}
|
||||||
|
.padding(8)
|
||||||
|
.background(Color.primary.opacity(0.05))
|
||||||
|
.cornerRadius(6)
|
||||||
|
.padding(.horizontal, 8)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.padding(.top, 4)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct TinyBtn: View {
|
||||||
|
let t: String
|
||||||
|
let a: () -> Void
|
||||||
|
init(_ t: String, _ a: @escaping () -> Void) { self.t = t; self.a = a }
|
||||||
|
var body: some View {
|
||||||
|
Button(action: a) {
|
||||||
|
Text(t).font(.system(size: 10)).padding(.horizontal, 6).padding(.vertical, 3)
|
||||||
|
.background(Color.primary.opacity(0.1))
|
||||||
|
.cornerRadius(4)
|
||||||
|
}.buttonStyle(.plain)
|
||||||
|
}
|
||||||
|
}
|
||||||