From d0ffa8ea32b9ef79dcca70f76d58439e014d1485 Mon Sep 17 00:00:00 2001 From: Norbel AMBANUMBEN Date: Wed, 19 Jun 2024 11:39:06 +0100 Subject: [PATCH 1/4] feat: internal refactoring to enable use of descriptor format in application core. --- ooniprobe.xcodeproj/project.pbxproj | 18 ++ ooniprobe/OONIDescriptor.swift | 244 ++++++++++++++++++ ooniprobe/View/DashboardTableViewController.m | 18 +- .../View/RunTest/TestOverviewViewController.h | 2 +- .../View/RunTest/TestOverviewViewController.m | 35 ++- .../TestResults/Rows/DashboardTableViewCell.h | 3 +- .../TestResults/Rows/DashboardTableViewCell.m | 21 +- ooniprobe/ooniprobe-Bridging-Header.h | 9 + 8 files changed, 309 insertions(+), 41 deletions(-) create mode 100644 ooniprobe/OONIDescriptor.swift create mode 100644 ooniprobe/ooniprobe-Bridging-Header.h diff --git a/ooniprobe.xcodeproj/project.pbxproj b/ooniprobe.xcodeproj/project.pbxproj index bce38e71d..4815377fe 100644 --- a/ooniprobe.xcodeproj/project.pbxproj +++ b/ooniprobe.xcodeproj/project.pbxproj @@ -16,6 +16,7 @@ 7940AA8B28117E9000C0EB5D /* ShareViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7940AA8A28117E9000C0EB5D /* ShareViewController.swift */; }; 7940AA8E28117E9000C0EB5D /* MainInterface.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 7940AA8C28117E9000C0EB5D /* MainInterface.storyboard */; }; 7940AA9228117E9000C0EB5D /* share.appex in Embed App Extensions */ = {isa = PBXBuildFile; fileRef = 7940AA8828117E9000C0EB5D /* share.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; + 7945903D2C21BFB1008116BF /* OONIDescriptor.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7945903C2C21BFB1008116BF /* OONIDescriptor.swift */; }; 79780FCF27E9F18E002A38B1 /* Languages.plist in Resources */ = {isa = PBXBuildFile; fileRef = 79780FCE27E9F18E002A38B1 /* Languages.plist */; }; 7AED19812A6EC9A2003B265A /* libresolv.tbd in Frameworks */ = {isa = PBXBuildFile; fileRef = 7AED19802A6EC9A2003B265A /* libresolv.tbd */; }; 7AED19832A6EC9C7003B265A /* libresolv.tbd in Frameworks */ = {isa = PBXBuildFile; fileRef = 7AED19822A6EC9C7003B265A /* libresolv.tbd */; }; @@ -225,6 +226,8 @@ 7940AA8D28117E9000C0EB5D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/MainInterface.storyboard; sourceTree = ""; }; 7940AA8F28117E9000C0EB5D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 7940AA972811840900C0EB5D /* share.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = share.entitlements; sourceTree = ""; }; + 7945903B2C21BFB1008116BF /* ooniprobe-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "ooniprobe-Bridging-Header.h"; sourceTree = ""; }; + 7945903C2C21BFB1008116BF /* OONIDescriptor.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OONIDescriptor.swift; sourceTree = ""; }; 79780FCE27E9F18E002A38B1 /* Languages.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; path = Languages.plist; sourceTree = ""; }; 79AA093C2A86E44400C23E27 /* vi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = vi; path = vi.lproj/Localizable.strings; sourceTree = ""; }; 79AA093D2A86E47600C23E27 /* my */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = my; path = my.lproj/Localizable.strings; sourceTree = ""; }; @@ -654,6 +657,7 @@ 526C702925C99AB100C7A164 /* Colors.xcassets */, D4A2F5EB1A6C3244001B8460 /* LaunchScreen.xib */, D4A2F5DC1A6C3244001B8460 /* Supporting Files */, + 7945903B2C21BFB1008116BF /* ooniprobe-Bridging-Header.h */, ); path = ooniprobe; sourceTree = ""; @@ -720,6 +724,7 @@ ED25280F1CEA34DA0073A29B /* Model */ = { isa = PBXGroup; children = ( + 7945903C2C21BFB1008116BF /* OONIDescriptor.swift */, ED90A91A2198DE5000204B46 /* Database */, ED90A9232198DE5000204B46 /* Settings */, EDB252A120FA46DA00B4EDE4 /* JsonResult */, @@ -1276,6 +1281,7 @@ D4A2F5D81A6C3244001B8460 = { CreatedOnToolsVersion = 6.1.1; DevelopmentTeam = MADPSAYN6T; + LastSwiftMigration = 1540; ProvisioningStyle = Automatic; SystemCapabilities = { com.apple.BackgroundModes = { @@ -1539,6 +1545,7 @@ buildActionMask = 2147483647; files = ( EDB3D64F26204C9300724ECF /* Experimental.m in Sources */, + 7945903D2C21BFB1008116BF /* OONIDescriptor.swift in Sources */, ED4DF7B52607970C00521C5B /* Signal.m in Sources */, ED365217223A40480093180B /* FailedTestDetailsViewController.m in Sources */, ED1CC80220159D970041089A /* TestRunningViewController.m in Sources */, @@ -1745,6 +1752,7 @@ 17E7EDBC21BFEDD1001961C7 /* Debug */ = { isa = XCBuildConfiguration; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; CLANG_ANALYZER_NONNULL = YES; CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; @@ -1777,6 +1785,7 @@ 17E7EDBD21BFEDD1001961C7 /* Release */ = { isa = XCBuildConfiguration; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; CLANG_ANALYZER_NONNULL = YES; CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; @@ -2012,6 +2021,7 @@ ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; + CLANG_ENABLE_MODULES = YES; CODE_SIGN_ENTITLEMENTS = ooniprobe/ooniprobe.entitlements; CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; @@ -2045,6 +2055,9 @@ PROVISIONING_PROFILE = ""; PROVISIONING_PROFILE_SPECIFIER = ""; STRIP_INSTALLED_PRODUCT = NO; + SWIFT_OBJC_BRIDGING_HEADER = "ooniprobe/ooniprobe-Bridging-Header.h"; + SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; USER_HEADER_SEARCH_PATHS = ""; VALID_ARCHS = "arm64 x86_64"; @@ -2060,6 +2073,7 @@ ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_LAUNCHIMAGE_NAME = LaunchImage; CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; + CLANG_ENABLE_MODULES = YES; CODE_SIGN_ENTITLEMENTS = ooniprobe/ooniprobe.entitlements; CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; @@ -2095,6 +2109,8 @@ PROVISIONING_PROFILE = ""; PROVISIONING_PROFILE_SPECIFIER = ""; STRIP_INSTALLED_PRODUCT = NO; + SWIFT_OBJC_BRIDGING_HEADER = "ooniprobe/ooniprobe-Bridging-Header.h"; + SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; USER_HEADER_SEARCH_PATHS = ""; VALID_ARCHS = arm64; @@ -2105,6 +2121,7 @@ isa = XCBuildConfiguration; baseConfigurationReference = AC948DA31A71FB931BBB1BB3 /* Pods-OONIProbeUnitTests.debug.xcconfig */; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; BUNDLE_LOADER = "$(TEST_HOST)"; CLANG_ANALYZER_NONNULL = YES; CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; @@ -2136,6 +2153,7 @@ isa = XCBuildConfiguration; baseConfigurationReference = CAFF8CF760913F0065D0615D /* Pods-OONIProbeUnitTests.release.xcconfig */; buildSettings = { + ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; BUNDLE_LOADER = "$(TEST_HOST)"; CLANG_ANALYZER_NONNULL = YES; CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; diff --git a/ooniprobe/OONIDescriptor.swift b/ooniprobe/OONIDescriptor.swift new file mode 100644 index 000000000..f411a16a6 --- /dev/null +++ b/ooniprobe/OONIDescriptor.swift @@ -0,0 +1,244 @@ +import Foundation +import UIKit + +/// `NettestName` is an enumeration that defines a set of named constants for different nettests. +/// Each case in the enumeration represents a different nettest. +enum NettestName: String { + case webConnectivity = "web_connectivity" // Represents the Web Connectivity test + case whatsapp = "whatsapp" // Represents the WhatsApp test + case telegram = "telegram" // Represents the Telegram test + case facebookMessenger = "facebook_messenger" // Represents the Facebook Messenger test + case signal = "signal" // Represents the Signal test + case psiphon = "psiphon" // Represents the Psiphon test + case tor = "tor" // Represents the Tor test + case ndt = "ndt" // Represents the NDT test + case dash = "dash" // Represents the DASH test + case httpHeaderFieldManipulation = "http_header_field_manipulation" // Represents the HTTP Header Field Manipulation test + case httpInvalidRequestLine = "http_invalid_request_line" // Represents the HTTP Invalid Request Line test + case stunReachability = "stunreachability" // Represents the STUN Reachability test + case dnsCheck = "dnscheck" // Represents the DNS Check test + case riseupVPN = "riseupvpn" // Represents the RiseupVPN test + case echCheck = "echcheck" // Represents the ECH Check test + case torsf = "torsf" // Represents the TorSF test + case vanillaTor = "vanilla_tor" // Represents the Vanilla Tor test +} + +/// `Nettest` is a class that represents a nettest. +/// It contains information about the nettest such as the name, inputs, etc. +/// The class also provides a method to get the test object for the nettest. +@objc(Nettest) +public class Nettest: NSObject { + + // MARK: Initializers + init(name: String, inputs: [String]?) { + self.name = name + self.inputs = inputs + } + + // MARK: Properties + @objc dynamic var name: String + @objc dynamic var inputs: [String]? + + // MARK: Methods + @objc public func getTest() -> AbstractTest { + switch NettestName(rawValue: self.name) { + case .webConnectivity: + return WebConnectivity() + case .whatsapp: + return Whatsapp() + case .telegram: + return Telegram() + case .facebookMessenger: + return FacebookMessenger() + case .signal: + return Signal() + case .psiphon: + return Psiphon() + case .tor: + return Tor() + case .ndt: + return NdtTest() + case .dash: + return Dash() + case .httpHeaderFieldManipulation: + return HttpHeaderFieldManipulation() + case .httpInvalidRequestLine: + return HttpInvalidRequestLine() + case .riseupVPN: + return RiseupVPN() + default: + return Experimental(name: name) + } + } +} + +/// `OONIDescriptor` is a class that represents an OONI descriptor. +/// It contains information about the descriptor such as the name, title, icon, color, etc. +/// It also contains information about the nettests that are part of the descriptor. +/// The class also provides a method to get the OONI descriptors for the OONI dashboard. +/// The class also provides a method to get the test suite for the current descriptor. +@objc(OONIDescriptor) +public class OONIDescriptor: NSObject { + + // MARK: Initializers + init(name: String, + title: String, + shortDescription: String, + longDescription: String, + icon: String, + color: UIColor, + animation: String?, + dataUsage: String, + nettest: [Nettest], + longRunningTests: [Nettest]?) { + self.name = name + self.title = title + self.shortDescription = shortDescription + self.longDescription = longDescription + self.icon = icon + self.color = color + self.animation = animation + self.dataUsage = dataUsage + self.nettest = nettest + self.longRunningTests = longRunningTests + } + + // MARK: Properties + @objc dynamic var name: String + @objc dynamic var title: String + @objc dynamic var shortDescription: String + @objc dynamic var longDescription: String + @objc dynamic var icon: String + @objc dynamic var color: UIColor + @objc dynamic var animation: String? + @objc dynamic var dataUsage: String + @objc dynamic var nettest: [Nettest] + @objc dynamic var longRunningTests: [Nettest]? + + // MARK: Methods + + // Get the OONI descriptors for the OONI dashboard. + @objc public static func getOONIDescriptors() -> [OONIDescriptor] { + + [ + OONIDescriptor( + name: "websites", + title: NSLocalizedString("Test.Websites.Fullname", comment: ""), + shortDescription: NSLocalizedString("Dashboard.Websites.Card.Description", comment: ""), + longDescription: NSLocalizedString("Dashboard.Websites.Overview.Paragraph", comment: ""), + icon: "websites", + color: UIColor(named: "color_indigo6")!, + animation: "websites", + dataUsage: "~ 8 MB", + nettest: [ + Nettest(name: NettestName.webConnectivity.rawValue, inputs: []) + ], + longRunningTests: [] + ), + + OONIDescriptor( + name: "instant_messaging", + title: NSLocalizedString("Test.InstantMessaging.Fullname", comment: ""), + shortDescription: NSLocalizedString("Dashboard.InstantMessaging.Card.Description", comment: ""), + longDescription: NSLocalizedString("Dashboard.InstantMessaging.Overview.Paragraph", comment: ""), + icon: "instant_messaging", + color: UIColor(named: "color_indigo5")!, + animation: "instant_messaging", + dataUsage: NSLocalizedString("small.datausage", comment: ""), + nettest: [ + Nettest(name: NettestName.whatsapp.rawValue, inputs: []), + Nettest(name: NettestName.telegram.rawValue, inputs: []), + Nettest(name: NettestName.facebookMessenger.rawValue, inputs: []), + Nettest(name: NettestName.signal.rawValue, inputs: []) + ], + longRunningTests: [] + ), + + OONIDescriptor( + name: "circumvention", + title: NSLocalizedString("Test.Circumvention.Fullname", comment: ""), + shortDescription: NSLocalizedString("Dashboard.Circumvention.Card.Description", comment: ""), + longDescription: NSLocalizedString("Dashboard.Circumvention.Overview.Paragraph", comment: ""), + icon: "circumvention", + color: UIColor(named: "color_indigo2")!, + animation: "circumvention", + dataUsage: NSLocalizedString("small.datausage", comment: ""), + nettest: [ + Nettest(name: NettestName.psiphon.rawValue, inputs: []), + Nettest(name: NettestName.tor.rawValue, inputs: []) + ], + longRunningTests: [] + ), + + OONIDescriptor( + name: "performance", + title: NSLocalizedString("Test.Performance.Fullname", comment: ""), + shortDescription: NSLocalizedString("Dashboard.Performance.Card.Description", comment: ""), + longDescription: NSLocalizedString("Dashboard.Performance.Overview.Paragraph", comment: ""), + icon: "performance", + color: UIColor(named: "color_indigo4")!, + animation: "performance", + dataUsage: NSLocalizedString("performance.datausage", comment: ""), + nettest: [ + Nettest(name: NettestName.ndt.rawValue, inputs: []), + Nettest(name: NettestName.dash.rawValue, inputs: []), + Nettest(name: NettestName.httpHeaderFieldManipulation.rawValue, inputs: []), + Nettest(name: NettestName.httpInvalidRequestLine.rawValue, inputs: []) + ], + longRunningTests: [] + ), + + OONIDescriptor( + name: "experimental", + title: NSLocalizedString("Test.Experimental.Fullname", comment: ""), + shortDescription: NSLocalizedString("Dashboard.Experimental.Card.Description", comment: ""), + longDescription: NSLocalizedString("Dashboard.Experimental.Overview.Paragraph", comment: ""), + icon: "experimental", + color: UIColor(named: "color_indigo1")!, + animation: "experimental", + dataUsage: NSLocalizedString("TestResults.NotAvailable", comment: ""), + nettest: [ + Nettest(name: "stunreachability", inputs: []), + Nettest(name: "dnscheck", inputs: []), + Nettest(name: "riseupvpn", inputs: []), + Nettest(name: "echcheck", inputs: []), + ], + longRunningTests: [ + Nettest(name: "torsf", inputs: []), + Nettest(name: "vanilla_tor", inputs: []), + ] + ) + ] + } + + + /// Returns the test suite for the current descriptor. + /// + /// @return DynamicTestSuite representing the test suite for the current descriptor. + @objc public func getTestSuites() -> Any { + DynamicTestSuite(descriptor: self) + } +} + + +/// This class is used to create [AbstractTest] dynamically for all instances where a Test Suite is required. +/// It is used to create a test suite from a Descriptor. +/// It acts as a bridge between the Descriptor format and the [AbstractSuite]. +@objc(DynamicTestSuite) +class DynamicTestSuite: AbstractSuite { + // MARK: Initializers + @objc init(descriptor: OONIDescriptor) { + self.descriptor = descriptor + super.init() + self.name = descriptor.name + self.dataUsage = descriptor.dataUsage + let tests = NSMutableArray() + tests.addObjects(from: descriptor.nettest.map { nettest in + nettest.getTest() + }) + self.testList = tests + } + + // MARK: Properties + @objc dynamic var descriptor: OONIDescriptor +} diff --git a/ooniprobe/View/DashboardTableViewController.m b/ooniprobe/View/DashboardTableViewController.m index cca72aa1e..529054a9d 100644 --- a/ooniprobe/View/DashboardTableViewController.m +++ b/ooniprobe/View/DashboardTableViewController.m @@ -3,6 +3,7 @@ #import "ThirdPartyServices.h" #import "Suite.h" #import "RunningTest.h" +#import "ooniprobe-Swift.h" @interface DashboardTableViewController () @@ -95,7 +96,7 @@ -(void)setShadowRunButton{ } -(void)loadTests{ - items = [TestUtility getTestObjects]; + items = [[OONIDescriptor getOONIDescriptors] mutableCopy]; [self.tableView reloadData]; } @@ -127,11 +128,11 @@ - (NSInteger) tableView:(UITableView *)tableView numberOfRowsInSection:(NSIntege - (UITableViewCell *) tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { DashboardTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath]; - AbstractSuite *test = [items objectAtIndex:indexPath.row]; + OONIDescriptor *test = [items objectAtIndex:indexPath.row]; if (cell == nil) { cell = [[DashboardTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"Cell"]; } - [cell setTestSuite:test]; + [cell setDescriptor:test]; return cell; } @@ -152,7 +153,12 @@ -(IBAction)run:(id)sender{ -(IBAction)runAll{ if ([TestUtility checkConnectivity:self] && [TestUtility checkTestRunning:self]){ - [[RunningTest currentTest] setAndRun:[NSMutableArray arrayWithArray:items] inView: self]; + // convert items to DynamicTestSuite + NSMutableArray *testSuites = [[NSMutableArray alloc] init]; + for (OONIDescriptor *decriptor in self.items){ + [testSuites addObject:[[DynamicTestSuite alloc] initWithDescriptor:decriptor]]; + } + [[RunningTest currentTest] setAndRun:testSuites inView: self]; [self reloadConstraints]; } } @@ -161,8 +167,8 @@ - (void) prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender { if ([[segue identifier] isEqualToString:@"toTestOverview"]){ NSIndexPath* indexPath = [self.tableView indexPathForSelectedRow]; TestOverviewViewController *vc = (TestOverviewViewController * )segue.destinationViewController; - AbstractSuite *testSuite = [items objectAtIndex:indexPath.row]; - [vc setTestSuite:testSuite]; + OONIDescriptor *test = [items objectAtIndex:indexPath.row]; + [vc setDescriptor:test]; [self.tableView deselectRowAtIndexPath:indexPath animated:YES]; } } diff --git a/ooniprobe/View/RunTest/TestOverviewViewController.h b/ooniprobe/View/RunTest/TestOverviewViewController.h index deea5a82f..adbaa7930 100644 --- a/ooniprobe/View/RunTest/TestOverviewViewController.h +++ b/ooniprobe/View/RunTest/TestOverviewViewController.h @@ -12,7 +12,7 @@ UIColor *defaultColor; } -@property (nonatomic, strong) AbstractSuite *testSuite; +@property (nonatomic, strong) id descriptor; @property (strong, nonatomic) IBOutlet UIImageView *testImage; @property (strong, nonatomic) IBOutlet UILabel *testNameLabel; diff --git a/ooniprobe/View/RunTest/TestOverviewViewController.m b/ooniprobe/View/RunTest/TestOverviewViewController.m index d9fbea26c..8d5df5165 100644 --- a/ooniprobe/View/RunTest/TestOverviewViewController.m +++ b/ooniprobe/View/RunTest/TestOverviewViewController.m @@ -1,13 +1,14 @@ #import "TestOverviewViewController.h" #import "ThirdPartyServices.h" #import "RunningTest.h" +#import "ooniprobe-Swift.h" @interface TestOverviewViewController () @end @implementation TestOverviewViewController -@synthesize testSuite; +@synthesize descriptor; - (void)viewDidLoad { [super viewDidLoad]; @@ -15,8 +16,8 @@ - (void)viewDidLoad { [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(settingsChanged) name:@"settingsChanged" object:nil]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(changeConstraints) name:@"networkTestEndedUI" object:nil]; - [self.testNameLabel setText:[LocalizationUtility getNameForTest:testSuite.name]]; - NSString *testLongDesc = [LocalizationUtility getLongDescriptionForTest:testSuite.name]; + [self.testNameLabel setText:[LocalizationUtility getNameForTest:[descriptor performSelector:@selector(name)]]]; + NSString *testLongDesc = [LocalizationUtility getLongDescriptionForTest:[descriptor performSelector:@selector(name)]]; [self.testDescriptionLabel setFont:[UIFont fontWithName:@"FiraSans-Regular" size:14]]; [self.testDescriptionLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; NSMutableDictionary *linkAttributes = [NSMutableDictionary dictionary]; @@ -37,19 +38,15 @@ - (void)viewDidLoad { [[UIApplication sharedApplication] openURL:url]; }]; [self.runButton setTitle:[NSString stringWithFormat:@"%@", NSLocalizedString(@"Dashboard.Overview.Run", nil)] forState:UIControlStateNormal]; - if ([testSuite.name isEqualToString:@"websites"]) + if ([[descriptor performSelector:@selector(name)] isEqualToString:@"websites"]) [self.websitesButton setTitle:[NSString stringWithFormat:@"%@", NSLocalizedString(@"Dashboard.Overview.ChooseWebsites", nil)] forState:UIControlStateNormal]; else [self.websitesButton setHidden:YES]; [self reloadLastMeasurement]; - [self.testImage setImage:[UIImage imageNamed:[NSString stringWithFormat:@"%@", testSuite.name]]]; - defaultColor = [TestUtility getBackgroundColorForTest:testSuite.name]; + [self.testImage setImage:[UIImage imageNamed:[NSString stringWithFormat:@"%@", [descriptor performSelector:@selector(name)]]]]; + defaultColor = [TestUtility getBackgroundColorForTest:[descriptor performSelector:@selector(name)]]; [self.runButton setTitleColor:defaultColor forState:UIControlStateNormal]; - if (testSuite.getTestList.count <= 0) { - self.runButton.userInteractionEnabled = NO; - [self.runButton setTitleColor:[UIColor colorNamed:@"color_gray3"] forState:UIControlStateNormal]; - } [self.backgroundView setBackgroundColor:defaultColor]; [NavigationBarUtility setNavigationBar:self.navigationController.navigationBar color:defaultColor]; self.navigationController.navigationBar.topItem.title = @""; @@ -81,14 +78,14 @@ -(void)changeConstraints{ -(void)reloadLastMeasurement{ dispatch_async(dispatch_get_main_queue(), ^{ [self.estimatedLabel setText:NSLocalizedString(@"Dashboard.Overview.Estimated", nil)]; - [self.estimatedDetailLabel setText: - [NSString stringWithFormat:@"%@ %@", - testSuite.dataUsage, - [LocalizationUtility getReadableRuntime:[testSuite getRuntime]]]]; +// [self.estimatedDetailLabel setText: +// [NSString stringWithFormat:@"%@ %@", +// testSuite.dataUsage, +// [LocalizationUtility getReadableRuntime:[testSuite getRuntime]]]]; [self.lastrunLabel setText:NSLocalizedString(@"Dashboard.Overview.LatestTest", nil)]; NSString *ago; - SRKResultSet *results = [[[[[Result query] limit:1] where:[NSString stringWithFormat:@"test_group_name = '%@'", testSuite.name]] orderByDescending:@"start_time"] fetch]; + SRKResultSet *results = [[[[[Result query] limit:1] where:[NSString stringWithFormat:@"test_group_name = '%@'", [descriptor performSelector:@selector(name)]]] orderByDescending:@"start_time"] fetch]; if ([results count] > 0){ ago = [[[results objectAtIndex:0] start_time] timeAgoSinceNow]; } @@ -99,8 +96,8 @@ -(void)reloadLastMeasurement{ } -(void)settingsChanged{ - [testSuite.testList removeAllObjects]; - [testSuite getTestList]; +// [testSuite.testList removeAllObjects]; +// [testSuite getTestList]; [self reloadLastMeasurement]; } @@ -124,7 +121,7 @@ - (void)willMoveToParentViewController:(UIViewController *)parent { -(IBAction)run:(id)sender{ if ([TestUtility checkConnectivity:self] && [TestUtility checkTestRunning:self]){ - [[RunningTest currentTest] setAndRun:[NSMutableArray arrayWithObject:testSuite] inView: self]; + [[RunningTest currentTest] setAndRun:[NSMutableArray arrayWithObject:[[DynamicTestSuite alloc] initWithDescriptor:descriptor]] inView: self]; [self changeConstraints]; } } @@ -133,7 +130,7 @@ -(IBAction)run:(id)sender{ - (void) prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender { if ([[segue identifier] isEqualToString:@"toTestSettings"]){ SettingsTableViewController *vc = (SettingsTableViewController * )segue.destinationViewController; - [vc setTestSuite:testSuite]; + //[vc setTestSuite:testSuite]; } } diff --git a/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.h b/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.h index 5745bd6db..e0aea8b6a 100644 --- a/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.h +++ b/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.h @@ -2,6 +2,7 @@ #import "TestUtility.h" #import "RunButton.h" #import "Suite.h" +#import "ooniprobe-Swift.h" @interface DashboardTableViewCell : UITableViewCell @property (strong, nonatomic) IBOutlet UIView *cardbackgroundView; @@ -9,5 +10,5 @@ @property (strong, nonatomic) IBOutlet UILabel *descLabel; @property (strong, nonatomic) IBOutlet UIImageView *testLogo; --(void)setTestSuite:(AbstractSuite*)testSuite; +-(void)setDescriptor:(OONIDescriptor*)descriptor; @end diff --git a/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.m b/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.m index be70d9ef1..3846af273 100644 --- a/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.m +++ b/ooniprobe/View/TestResults/Rows/DashboardTableViewCell.m @@ -40,20 +40,13 @@ - (void)setHighlighted:(BOOL)highlighted animated:(BOOL)animated{ } } --(void)setTestSuite:(AbstractSuite*)testSuite{ - [self.titleLabel setText:[LocalizationUtility getNameForTest:testSuite.name]]; - [self.descLabel setText:[LocalizationUtility getDescriptionForTest:testSuite.name]]; - if (testSuite.getTestList.count <= 0) { - [self.titleLabel setTextColor:[UIColor colorNamed:@"disabled_test_text"]]; - [self.descLabel setTextColor:[UIColor colorNamed:@"disabled_test_text"]]; - [self.testLogo setImage:[self imageWithGradient:[UIImage imageNamed:[NSString stringWithFormat:@"%@", testSuite.name]] startColor:[UIColor colorNamed:@"disabled_test_text"] endColor:[UIColor colorNamed:@"disabled_test_text"]]]; - [self.cardbackgroundView setBackgroundColor:[UIColor colorNamed:@"disabled_test_background"]]; - } else { - [self.titleLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; - [self.descLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; - [self.testLogo setImage:[self imageWithGradient:[UIImage imageNamed:[NSString stringWithFormat:@"%@", testSuite.name]] startColor:[TestUtility getGradientColorForTest:testSuite.name] endColor:[TestUtility getColorForTest:testSuite.name]]]; - [self.cardbackgroundView setBackgroundColor:[UIColor colorNamed:@"color_gray0"]]; - } +-(void)setDescriptor:(OONIDescriptor*)descriptor{ + [self.titleLabel setText:descriptor.title]; + [self.descLabel setText:descriptor.shortDescription]; + [self.titleLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; + [self.descLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; + [self.cardbackgroundView setBackgroundColor:[UIColor colorNamed:@"color_gray0"]]; + [self.testLogo setImage:[self imageWithGradient:[UIImage imageNamed:descriptor.icon] startColor:[TestUtility getGradientColorForTest:descriptor.name] endColor:[TestUtility getColorForTest:descriptor.name]]]; } -(void)setRoundedView{ diff --git a/ooniprobe/ooniprobe-Bridging-Header.h b/ooniprobe/ooniprobe-Bridging-Header.h new file mode 100644 index 000000000..99a557856 --- /dev/null +++ b/ooniprobe/ooniprobe-Bridging-Header.h @@ -0,0 +1,9 @@ +// +// Use this file to import your target's public headers that you would like to expose to Swift. +// + +#import "DashboardTableViewController.h" +#import "TestOverviewViewController.h" +#import "AbstractSuite.h" +#import "Tests.h" +#import "UIView+Toast.h" From b78e75007259f44023e957d8b937d629aa6832cd Mon Sep 17 00:00:00 2001 From: Norbel AMBANUMBEN Date: Mon, 24 Jun 2024 14:00:19 +0100 Subject: [PATCH 2/4] fix: update based on PR review --- ooniprobe/OONIDescriptor.swift | 6 +++--- .../View/RunTest/TestOverviewViewController.m | 20 +++++++++---------- 2 files changed, 13 insertions(+), 13 deletions(-) diff --git a/ooniprobe/OONIDescriptor.swift b/ooniprobe/OONIDescriptor.swift index f411a16a6..a96065cf5 100644 --- a/ooniprobe/OONIDescriptor.swift +++ b/ooniprobe/OONIDescriptor.swift @@ -144,7 +144,7 @@ public class OONIDescriptor: NSObject { icon: "instant_messaging", color: UIColor(named: "color_indigo5")!, animation: "instant_messaging", - dataUsage: NSLocalizedString("small.datausage", comment: ""), + dataUsage: "< 1 MB", nettest: [ Nettest(name: NettestName.whatsapp.rawValue, inputs: []), Nettest(name: NettestName.telegram.rawValue, inputs: []), @@ -162,7 +162,7 @@ public class OONIDescriptor: NSObject { icon: "circumvention", color: UIColor(named: "color_indigo2")!, animation: "circumvention", - dataUsage: NSLocalizedString("small.datausage", comment: ""), + dataUsage: "< 1 MB", nettest: [ Nettest(name: NettestName.psiphon.rawValue, inputs: []), Nettest(name: NettestName.tor.rawValue, inputs: []) @@ -178,7 +178,7 @@ public class OONIDescriptor: NSObject { icon: "performance", color: UIColor(named: "color_indigo4")!, animation: "performance", - dataUsage: NSLocalizedString("performance.datausage", comment: ""), + dataUsage: "5 - 200 MB", nettest: [ Nettest(name: NettestName.ndt.rawValue, inputs: []), Nettest(name: NettestName.dash.rawValue, inputs: []), diff --git a/ooniprobe/View/RunTest/TestOverviewViewController.m b/ooniprobe/View/RunTest/TestOverviewViewController.m index 8d5df5165..36f788133 100644 --- a/ooniprobe/View/RunTest/TestOverviewViewController.m +++ b/ooniprobe/View/RunTest/TestOverviewViewController.m @@ -21,8 +21,8 @@ - (void)viewDidLoad { [self.testDescriptionLabel setFont:[UIFont fontWithName:@"FiraSans-Regular" size:14]]; [self.testDescriptionLabel setTextColor:[UIColor colorNamed:@"color_gray9"]]; NSMutableDictionary *linkAttributes = [NSMutableDictionary dictionary]; - [linkAttributes setObject:[NSNumber numberWithBool:YES] forKey:(NSString *)kCTUnderlineStyleAttributeName]; - [linkAttributes setObject:[UIColor colorNamed:@"color_base"] forKey:(NSString *)kCTForegroundColorAttributeName]; + linkAttributes[(NSString *) kCTUnderlineStyleAttributeName] = @YES; + linkAttributes[(NSString *) kCTForegroundColorAttributeName] = [UIColor colorNamed:@"color_base"]; self.testDescriptionLabel.linkAttributes = [NSDictionary dictionaryWithDictionary:linkAttributes]; [self.testDescriptionLabel setMarkdown:testLongDesc]; if ([[UIDevice currentDevice].systemVersion floatValue] >= 9.0) { @@ -78,14 +78,13 @@ -(void)changeConstraints{ -(void)reloadLastMeasurement{ dispatch_async(dispatch_get_main_queue(), ^{ [self.estimatedLabel setText:NSLocalizedString(@"Dashboard.Overview.Estimated", nil)]; -// [self.estimatedDetailLabel setText: -// [NSString stringWithFormat:@"%@ %@", -// testSuite.dataUsage, -// [LocalizationUtility getReadableRuntime:[testSuite getRuntime]]]]; + [self.estimatedDetailLabel setText: + [NSString stringWithFormat:@"%@ %@", [self->descriptor performSelector:@selector(dataUsage)], + [LocalizationUtility getReadableRuntime:[[[DynamicTestSuite alloc] initWithDescriptor:self->descriptor] getRuntime]]]]; [self.lastrunLabel setText:NSLocalizedString(@"Dashboard.Overview.LatestTest", nil)]; NSString *ago; - SRKResultSet *results = [[[[[Result query] limit:1] where:[NSString stringWithFormat:@"test_group_name = '%@'", [descriptor performSelector:@selector(name)]]] orderByDescending:@"start_time"] fetch]; + SRKResultSet *results = [[[[[Result query] limit:1] where:[NSString stringWithFormat:@"test_group_name = '%@'", [self->descriptor performSelector:@selector(name)]]] orderByDescending:@"start_time"] fetch]; if ([results count] > 0){ ago = [[[results objectAtIndex:0] start_time] timeAgoSinceNow]; } @@ -96,8 +95,7 @@ -(void)reloadLastMeasurement{ } -(void)settingsChanged{ -// [testSuite.testList removeAllObjects]; -// [testSuite getTestList]; + // NOTE(aanorbel): Reload the descriptor when the settings change. [self reloadLastMeasurement]; } @@ -127,10 +125,12 @@ -(IBAction)run:(id)sender{ } +/// NOTE(aanorbel): `performSegueWithIdentifier:@"toTestSettings"` is never called in the codebase thus, this method is not used. +/// I would like to remove it but needs to be sure of the impact. - (void) prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender { if ([[segue identifier] isEqualToString:@"toTestSettings"]){ SettingsTableViewController *vc = (SettingsTableViewController * )segue.destinationViewController; - //[vc setTestSuite:testSuite]; + [vc setTestSuite:[[DynamicTestSuite alloc] initWithDescriptor:descriptor]]; } } From 0ef1aea383ccc6e6f84eb4e649a0a66162382a62 Mon Sep 17 00:00:00 2001 From: Norbel AMBANUMBEN Date: Mon, 24 Jun 2024 14:43:03 +0100 Subject: [PATCH 3/4] feat (OONI Run v2): Update Dashboard view to match new design (#556) Fixes https://github.com/ooni/ooni.org/issues/1512 ## Proposed Changes - Update storyboard to match designs - Update `DashboardTableViewCell` definition to fine-tune views |.|.| |-|-| | ![Simulator Screenshot - iPhone 15 Pro Max - 2024-02-27 at 21 09 40](https://github.com/ooni/probe-ios/assets/17911892/8fab41df-703e-49c5-90c6-7921173460be) | ![Simulator Screenshot - iPhone 15 Pro Max - 2024-02-27 at 21 09 52](https://github.com/ooni/probe-ios/assets/17911892/334fa805-37af-4027-9ea7-fbecf033d944) | --- ooniprobe.xcodeproj/project.pbxproj | 4 + .../color_background.colorset/Contents.json | 56 ++++++++++++++ .../ooni_probe_logo.pdf | Bin 10996 -> 16347 bytes .../timer.imageset/Contents.json | 21 ++++++ .../Images.xcassets/timer.imageset/timer.png | Bin 0 -> 11125 bytes ooniprobe/Storyboards/Dashboard.storyboard | 69 ++++++++++-------- ooniprobe/Utils.swift | 43 +++++++++++ ooniprobe/View/DashboardTableViewController.m | 24 ++++-- .../TestResults/Rows/DashboardTableViewCell.m | 43 +---------- 9 files changed, 183 insertions(+), 77 deletions(-) create mode 100644 ooniprobe/Colors.xcassets/color_background.colorset/Contents.json create mode 100644 ooniprobe/Images.xcassets/timer.imageset/Contents.json create mode 100644 ooniprobe/Images.xcassets/timer.imageset/timer.png create mode 100644 ooniprobe/Utils.swift diff --git a/ooniprobe.xcodeproj/project.pbxproj b/ooniprobe.xcodeproj/project.pbxproj index 4815377fe..b8cc63cc6 100644 --- a/ooniprobe.xcodeproj/project.pbxproj +++ b/ooniprobe.xcodeproj/project.pbxproj @@ -12,6 +12,7 @@ 17E7EDC021BFEE0C001961C7 /* SnapshotHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 17E7EDBF21BFEE0C001961C7 /* SnapshotHelper.swift */; }; 526C702A25C99AB200C7A164 /* Colors.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 526C702925C99AB100C7A164 /* Colors.xcassets */; }; 58E18F9EBF4FAD4EFCE020F3 /* Pods_ooniprobe.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = BB32DF3D6FC174AA1AF76009 /* Pods_ooniprobe.framework */; }; + 793587D32B8E081600038F88 /* Utils.swift in Sources */ = {isa = PBXBuildFile; fileRef = 793587D22B8E081600038F88 /* Utils.swift */; }; 793587BA2B852EDD00038F88 /* OoniRunViewUITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 793587B92B852EDD00038F88 /* OoniRunViewUITests.swift */; }; 7940AA8B28117E9000C0EB5D /* ShareViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7940AA8A28117E9000C0EB5D /* ShareViewController.swift */; }; 7940AA8E28117E9000C0EB5D /* MainInterface.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 7940AA8C28117E9000C0EB5D /* MainInterface.storyboard */; }; @@ -220,6 +221,7 @@ 17E7EDBF21BFEE0C001961C7 /* SnapshotHelper.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = SnapshotHelper.swift; path = fastlane/SnapshotHelper.swift; sourceTree = SOURCE_ROOT; }; 526C702925C99AB100C7A164 /* Colors.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Colors.xcassets; sourceTree = ""; }; 588219FACC9F793A15BDEA33 /* Pods_OONIProbeUnitTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_OONIProbeUnitTests.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + 793587D22B8E081600038F88 /* Utils.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Utils.swift; sourceTree = ""; }; 793587B92B852EDD00038F88 /* OoniRunViewUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OoniRunViewUITests.swift; sourceTree = ""; }; 7940AA8828117E9000C0EB5D /* share.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = share.appex; sourceTree = BUILT_PRODUCTS_DIR; }; 7940AA8A28117E9000C0EB5D /* ShareViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShareViewController.swift; sourceTree = ""; }; @@ -657,6 +659,7 @@ 526C702925C99AB100C7A164 /* Colors.xcassets */, D4A2F5EB1A6C3244001B8460 /* LaunchScreen.xib */, D4A2F5DC1A6C3244001B8460 /* Supporting Files */, + 793587D22B8E081600038F88 /* Utils.swift */, 7945903B2C21BFB1008116BF /* ooniprobe-Bridging-Header.h */, ); path = ooniprobe; @@ -1628,6 +1631,7 @@ EDE47C41241FC6BC0013A1CB /* NetworkSession.m in Sources */, EDF4ED26248A9A64001A5406 /* Advanced.m in Sources */, EDA17AFB208A30E100D46D0F /* MiddleBoxesDetailsViewController.m in Sources */, + 793587D32B8E081600038F88 /* Utils.swift in Sources */, ED1CC7FF20158D630041089A /* ConfigureButton.m in Sources */, ED3932C41E3114740064CD11 /* AboutViewController.m in Sources */, ED2E066F21D4867B00E9B9EE /* HttpInvalidRequestLine.m in Sources */, diff --git a/ooniprobe/Colors.xcassets/color_background.colorset/Contents.json b/ooniprobe/Colors.xcassets/color_background.colorset/Contents.json new file mode 100644 index 000000000..7bed699c7 --- /dev/null +++ b/ooniprobe/Colors.xcassets/color_background.colorset/Contents.json @@ -0,0 +1,56 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF5", + "green" : "0xF3", + "red" : "0xF1" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "light" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0xF5", + "green" : "0xF3", + "red" : "0xF1" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0x40", + "green" : "0x3A", + "red" : "0x34" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ooniprobe/Images.xcassets/ooni_probe_logo.imageset/ooni_probe_logo.pdf b/ooniprobe/Images.xcassets/ooni_probe_logo.imageset/ooni_probe_logo.pdf index cb12903bd40067f943e58c0282b5170b6c001bc0..879b3c65883ca0e283f7b632127e8bf6aee69c62 100644 GIT binary patch literal 16347 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%0Df?*UbCD}y)y~&-TIC4RzJZ)(M~_lZ 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 JLQxX^e*jw+NeKV| diff --git a/ooniprobe/Images.xcassets/timer.imageset/Contents.json b/ooniprobe/Images.xcassets/timer.imageset/Contents.json new file mode 100644 index 000000000..33b93aa4e --- /dev/null +++ b/ooniprobe/Images.xcassets/timer.imageset/Contents.json @@ -0,0 +1,21 @@ +{ + "images" : [ + { + "filename" : "timer.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "idiom" : "universal", + "scale" : "2x" + }, + { + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/ooniprobe/Images.xcassets/timer.imageset/timer.png b/ooniprobe/Images.xcassets/timer.imageset/timer.png new file mode 100644 index 0000000000000000000000000000000000000000..3277fb324a71d989bc7dd548a2cdc2acfc5b9fd3 GIT binary patch literal 11125 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 literal 0 HcmV?d00001 diff --git a/ooniprobe/Storyboards/Dashboard.storyboard b/ooniprobe/Storyboards/Dashboard.storyboard index 2e3bfc852..741d4dc64 100644 --- a/ooniprobe/Storyboards/Dashboard.storyboard +++ b/ooniprobe/Storyboards/Dashboard.storyboard @@ -28,10 +28,10 @@ - + - + @@ -57,7 +57,7 @@ - - + @@ -511,7 +511,7 @@ measurement-kit: x.y.z - + @@ -522,7 +522,7 @@ measurement-kit: x.y.z - + diff --git a/ooniprobe/Utility/SettingsUtility.m b/ooniprobe/Utility/SettingsUtility.m index b24257cdb..bab78e54f 100644 --- a/ooniprobe/Utility/SettingsUtility.m +++ b/ooniprobe/Utility/SettingsUtility.m @@ -6,7 +6,7 @@ @implementation SettingsUtility + (NSArray*)getSettingsCategories{ - return @[@"notifications", @"automated_testing", @"test_options", @"privacy", @"advanced", @"ooni_backend_proxy", @"send_email", @"about_ooni"]; + return @[@"notifications", @"test_options", @"privacy", @"advanced", @"ooni_backend_proxy", @"send_email", @"about_ooni"]; } + (NSArray*)getSettingsForCategory:(NSString*)categoryName{ @@ -15,10 +15,11 @@ + (NSArray*)getSettingsForCategory:(NSString*)categoryName{ return @[@"notifications_enabled"]; } if ([categoryName isEqualToString:@"automated_testing"]) { - if ([SettingsUtility isAutomatedTestEnabled]) + if ([SettingsUtility isAutomatedTestEnabled]) { return @[@"automated_testing_enabled", @"automated_testing_wifionly", @"automated_testing_charging"]; - else + } else { return @[@"automated_testing_enabled"]; + } } else if ([categoryName isEqualToString:@"privacy"]) { return @[@"upload_results", @"send_crash"]; @@ -28,25 +29,25 @@ + (NSArray*)getSettingsForCategory:(NSString*)categoryName{ return @[@"AppleLanguages", @"debug_logs", @"storage_usage" , @"warn_vpn_in_use"]; } else if ([categoryName isEqualToString:@"test_options"]) { - return [TestUtility getTestOptionTypes]; - } - else if ([[TestUtility getTestOptionTypes] containsObject:categoryName]) - return [SettingsUtility getSettingsForTest:categoryName :YES]; - else + return @[@"automated_testing_enabled", @"automated_testing_wifionly", @"automated_testing_charging", @"website_categories", @"max_runtime_enabled", @"max_runtime"]; + } else { return nil; + } } + (NSString*)getTypeForSetting:(NSString*)setting{ - if ([setting isEqualToString:@"experimental"] || [setting isEqualToString:@"long_running_tests_in_foreground"]) + if ([setting isEqualToString:@"experimental"] || [setting isEqualToString:@"long_running_tests_in_foreground"]) { return @"bool"; - else if ([setting isEqualToString:@"website_categories"] || [[TestUtility getTestOptionTypes] containsObject:setting]) + } else if ([setting isEqualToString:@"website_categories"]) { return @"segue"; - else if ([setting isEqualToString:@"monthly_mobile_allowance"] || - [setting isEqualToString:@"monthly_wifi_allowance"] || - [setting isEqualToString:@"max_runtime"]) + } else if ([setting isEqualToString:@"monthly_mobile_allowance"] || + [setting isEqualToString:@"monthly_wifi_allowance"] || + [setting isEqualToString:@"max_runtime"]) { return @"int"; - if ([setting isEqualToString:@"storage_usage"] || [setting isEqualToString:@"AppleLanguages"]) + } + if ([setting isEqualToString:@"storage_usage"] || [setting isEqualToString:@"AppleLanguages"]) { return @"button"; + } return @"bool"; } diff --git a/ooniprobe/View/Settings/SettingsCategoriesTableViewController.m b/ooniprobe/View/Settings/SettingsCategoriesTableViewController.m index c99795301..b4858e2fd 100644 --- a/ooniprobe/View/Settings/SettingsCategoriesTableViewController.m +++ b/ooniprobe/View/Settings/SettingsCategoriesTableViewController.m @@ -14,6 +14,8 @@ - (void)awakeFromNib{ - (void)viewDidLoad { [super viewDidLoad]; + self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone; // remove separator + self.tableView.backgroundColor = [UIColor colorNamed:@"color_gray1"]; [NavigationBarUtility setNavigationBar:self.navigationController.navigationBar]; categories = [SettingsUtility getSettingsCategories]; } @@ -45,6 +47,7 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N cell.textLabel.text = [LocalizationUtility getNameForSetting:current]; cell.textLabel.textColor = [UIColor colorNamed:@"color_gray9"]; cell.imageView.image = [UIImage imageNamed:current]; + cell.backgroundColor = [UIColor colorNamed:@"color_gray1"]; return cell; } diff --git a/ooniprobe/View/Settings/SettingsTableViewController.m b/ooniprobe/View/Settings/SettingsTableViewController.m index d53c24a8c..83d2a8470 100644 --- a/ooniprobe/View/Settings/SettingsTableViewController.m +++ b/ooniprobe/View/Settings/SettingsTableViewController.m @@ -54,6 +54,8 @@ -(void)reloadSettings { //hide rows smooth dispatch_async(dispatch_get_main_queue(), ^{ [self.tableView reloadSections:[NSIndexSet indexSetWithIndex:0] withRowAnimation:UITableViewRowAnimationFade]; + self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone; // remove separator + self.tableView.backgroundColor = [UIColor colorNamed:@"color_gray1"]; }); } @@ -63,8 +65,9 @@ -(void)viewWillDisappear:(BOOL)animated{ [testSuite.testList removeAllObjects]; [testSuite getTestList]; } - if (testSuite != nil || [[TestUtility getTestOptionTypes] containsObject:category]) + if (testSuite != nil) { [[NSNotificationCenter defaultCenter] postNotificationName:@"settingsChanged" object:nil]; + } } #pragma mark - Table view data source @@ -113,9 +116,6 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N else { cell = [tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath]; } - if ([[TestUtility getTestOptionTypes] containsObject:current]){ - cell.imageView.image = [UIImage imageNamed:current]; - } cell.textLabel.text = [LocalizationUtility getNameForSetting:current]; cell.textLabel.textColor = [UIColor colorNamed:@"color_gray9"]; UISwitch *switchview = [[UISwitch alloc] initWithFrame:CGRectZero]; @@ -132,13 +132,7 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N } else cell = [tableView dequeueReusableCellWithIdentifier:@"Cell" forIndexPath:indexPath]; - if ([[TestUtility getTestOptionTypes] containsObject:current]){ - if ([NSLocale characterDirectionForLanguage:[NSLocale preferredLanguages][0]] == NSLocaleLanguageDirectionRightToLeft) { - cell.imageView.image = [self imageWithImage:[UIImage imageNamed:current] convertToSize:CGSizeMake(32, 32)]; - } else { - cell.imageView.image = [UIImage imageNamed:current]; - } - } + cell.textLabel.text = [LocalizationUtility getNameForSetting:current]; cell.textLabel.textColor = [UIColor colorNamed:@"color_gray9"]; cell.accessoryView = nil; @@ -188,6 +182,7 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N cell.accessoryView = cleanButton; } } + cell.backgroundColor = [UIColor colorNamed:@"color_gray1"]; return cell; } diff --git a/ooniprobe/View/Settings/WebsiteCategoriesTableViewController.m b/ooniprobe/View/Settings/WebsiteCategoriesTableViewController.m index a63099fc4..bacfcf0c1 100644 --- a/ooniprobe/View/Settings/WebsiteCategoriesTableViewController.m +++ b/ooniprobe/View/Settings/WebsiteCategoriesTableViewController.m @@ -12,6 +12,8 @@ - (void)viewDidLoad { self.navigationController.navigationBar.topItem.title = @""; self.tableView.estimatedRowHeight = 60.0; self.tableView.rowHeight = UITableViewAutomaticDimension; + self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone; // remove separator + self.tableView.backgroundColor = [UIColor colorNamed:@"color_gray1"]; } -(void)viewWillAppear:(BOOL)animated{ @@ -53,6 +55,8 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N else switchview.on = YES; cell.accessoryView = switchview; + cell.backgroundColor = [UIColor colorNamed:@"color_gray1"]; + return cell; }