Integrate stuff from old UIKit project

This commit is contained in:
Justin Mazzocchi 2020-08-20 19:29:01 -07:00
parent 7863866f68
commit c309b94ad0
No known key found for this signature in database
GPG key ID: E223E6937AAFB01C
30 changed files with 1812 additions and 86 deletions

View file

@ -52,10 +52,22 @@
D020F51224ECA309005AB084 /* MastodonContext.swift in Sources */ = {isa = PBXBuildFile; fileRef = D020F51024ECA309005AB084 /* MastodonContext.swift */; };
D020F51424ECBA60005AB084 /* LazyView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D020F51324ECBA60005AB084 /* LazyView.swift */; };
D020F51524ECBA60005AB084 /* LazyView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D020F51324ECBA60005AB084 /* LazyView.swift */; };
D02D86D924EF61E4004583CC /* StatusTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86D724EF61E4004583CC /* StatusTableViewCell.swift */; };
D02D86DA24EF61E4004583CC /* StatusTableViewCell.xib in Resources */ = {isa = PBXBuildFile; fileRef = D02D86D824EF61E4004583CC /* StatusTableViewCell.xib */; };
D02D86E424EF9848004583CC /* TouchFallthroughTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86E324EF9848004583CC /* TouchFallthroughTextView.swift */; };
D02D86E624EF998B004583CC /* HTML.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86E524EF998B004583CC /* HTML.swift */; };
D02D86E724EF998B004583CC /* HTML.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86E524EF998B004583CC /* HTML.swift */; };
D02D86EC24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86EB24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift */; };
D02D86ED24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86EB24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift */; };
D02D86EF24EFB13A004583CC /* Date+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86EE24EFB13A004583CC /* Date+Extensions.swift */; };
D02D86F024EFB13A004583CC /* Date+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D86EE24EFB13A004583CC /* Date+Extensions.swift */; };
D02D870524EFBB79004583CC /* String+UIKitExtensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D02D870424EFBB79004583CC /* String+UIKitExtensions.swift */; };
D03658D124EDD80900AC17EC /* ContextEndpoint+Stubbing.swift in Sources */ = {isa = PBXBuildFile; fileRef = D03658D024EDD80900AC17EC /* ContextEndpoint+Stubbing.swift */; };
D03658D224EDD80900AC17EC /* ContextEndpoint+Stubbing.swift in Sources */ = {isa = PBXBuildFile; fileRef = D03658D024EDD80900AC17EC /* ContextEndpoint+Stubbing.swift */; };
D03DF45B24E62A68007A8CD5 /* DeletionEndpoint.swift in Sources */ = {isa = PBXBuildFile; fileRef = D03DF45A24E62A68007A8CD5 /* DeletionEndpoint.swift */; };
D03DF45C24E62A68007A8CD5 /* DeletionEndpoint.swift in Sources */ = {isa = PBXBuildFile; fileRef = D03DF45A24E62A68007A8CD5 /* DeletionEndpoint.swift */; };
D042650824F058280096ED10 /* Localizable.stringsdict in Resources */ = {isa = PBXBuildFile; fileRef = D042650724F058280096ED10 /* Localizable.stringsdict */; };
D042650924F058280096ED10 /* Localizable.stringsdict in Resources */ = {isa = PBXBuildFile; fileRef = D042650724F058280096ED10 /* Localizable.stringsdict */; };
D047FAAE24C3E21200AF17C5 /* MetatextApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = D047FA8524C3E21000AF17C5 /* MetatextApp.swift */; };
D047FAAF24C3E21200AF17C5 /* MetatextApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = D047FA8524C3E21000AF17C5 /* MetatextApp.swift */; };
D047FAB224C3E21200AF17C5 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = D047FA8724C3E21200AF17C5 /* Assets.xcassets */; };
@ -132,6 +144,11 @@
D081A40624D0F1A8001B016E /* String+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D081A40424D0F1A8001B016E /* String+Extensions.swift */; };
D0A1CA7424DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A1CA7324DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift */; };
D0A1CA7524DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A1CA7324DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift */; };
D0A2453724EF346800B07068 /* StatusListViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A2453624EF346800B07068 /* StatusListViewController.swift */; };
D0A2453924EF364100B07068 /* StatusListView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A2453824EF364100B07068 /* StatusListView.swift */; };
D0A2453F24EF55D000B07068 /* StatusViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A2453E24EF55D000B07068 /* StatusViewModel.swift */; };
D0A2454124EF563000B07068 /* StatusService.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A2454024EF563000B07068 /* StatusService.swift */; };
D0A2454224EF563000B07068 /* StatusService.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A2454024EF563000B07068 /* StatusService.swift */; };
D0A652AD24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A652AC24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift */; };
D0A652AE24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0A652AC24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift */; };
D0B23F0D24D210E90066F411 /* NSError+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0B23F0C24D210E90066F411 /* NSError+Extensions.swift */; };
@ -142,8 +159,6 @@
D0BEC93C24C96FD500E864C4 /* RootView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0BEC93A24C96FD500E864C4 /* RootView.swift */; };
D0BEC94724CA22C400E864C4 /* StatusesViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0BEC94624CA22C400E864C4 /* StatusesViewModel.swift */; };
D0BEC94824CA22C400E864C4 /* StatusesViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0BEC94624CA22C400E864C4 /* StatusesViewModel.swift */; };
D0BEC94A24CA231200E864C4 /* StatusesView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0BEC94924CA231200E864C4 /* StatusesView.swift */; };
D0BEC94B24CA231200E864C4 /* StatusesView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0BEC94924CA231200E864C4 /* StatusesView.swift */; };
D0C963FB24CC359D003BD330 /* AlertItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0C963FA24CC359D003BD330 /* AlertItem.swift */; };
D0C963FC24CC359D003BD330 /* AlertItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0C963FA24CC359D003BD330 /* AlertItem.swift */; };
D0C963FE24CC3812003BD330 /* Publisher+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0C963FD24CC3812003BD330 /* Publisher+Extensions.swift */; };
@ -187,6 +202,7 @@
D0E5363024E5436C00FB1CE1 /* PushNotification.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0E5362F24E5436C00FB1CE1 /* PushNotification.swift */; };
D0E5363124E5453E00FB1CE1 /* PushNotification.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0E5362F24E5436C00FB1CE1 /* PushNotification.swift */; };
D0E5363224E5453F00FB1CE1 /* PushNotification.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0E5362F24E5436C00FB1CE1 /* PushNotification.swift */; };
D0E900CE24F1F28A00B55F5A /* UIColor+Extensions.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0E900CD24F1F28A00B55F5A /* UIColor+Extensions.swift */; };
D0EC8DC224DF7D9C00A08489 /* IdentityService.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0EC8DC124DF7D9C00A08489 /* IdentityService.swift */; };
D0EC8DC324DF7D9C00A08489 /* IdentityService.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0EC8DC124DF7D9C00A08489 /* IdentityService.swift */; };
D0EC8DC524DF842700A08489 /* KeychainService.swift in Sources */ = {isa = PBXBuildFile; fileRef = D0EC8DC424DF842700A08489 /* KeychainService.swift */; };
@ -286,8 +302,16 @@
D020F50D24ECA25F005AB084 /* ContextEndpoint.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContextEndpoint.swift; sourceTree = "<group>"; };
D020F51024ECA309005AB084 /* MastodonContext.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MastodonContext.swift; sourceTree = "<group>"; };
D020F51324ECBA60005AB084 /* LazyView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LazyView.swift; sourceTree = "<group>"; };
D02D86D724EF61E4004583CC /* StatusTableViewCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusTableViewCell.swift; sourceTree = "<group>"; };
D02D86D824EF61E4004583CC /* StatusTableViewCell.xib */ = {isa = PBXFileReference; lastKnownFileType = file.xib; path = StatusTableViewCell.xib; sourceTree = "<group>"; };
D02D86E324EF9848004583CC /* TouchFallthroughTextView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = TouchFallthroughTextView.swift; sourceTree = "<group>"; };
D02D86E524EF998B004583CC /* HTML.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HTML.swift; sourceTree = "<group>"; };
D02D86EB24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "CodingUserInfoKey+Extensions.swift"; sourceTree = "<group>"; };
D02D86EE24EFB13A004583CC /* Date+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Date+Extensions.swift"; sourceTree = "<group>"; };
D02D870424EFBB79004583CC /* String+UIKitExtensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+UIKitExtensions.swift"; sourceTree = "<group>"; };
D03658D024EDD80900AC17EC /* ContextEndpoint+Stubbing.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ContextEndpoint+Stubbing.swift"; sourceTree = "<group>"; };
D03DF45A24E62A68007A8CD5 /* DeletionEndpoint.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeletionEndpoint.swift; sourceTree = "<group>"; };
D042650724F058280096ED10 /* Localizable.stringsdict */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.stringsdict; path = Localizable.stringsdict; sourceTree = "<group>"; };
D047FA8524C3E21000AF17C5 /* MetatextApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MetatextApp.swift; sourceTree = "<group>"; };
D047FA8724C3E21200AF17C5 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
D047FA8C24C3E21200AF17C5 /* Metatext.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Metatext.app; sourceTree = BUILT_PRODUCTS_DIR; };
@ -331,12 +355,15 @@
D075817B24E6659A0081F6A3 /* NotificationTypesPreferencesView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NotificationTypesPreferencesView.swift; sourceTree = "<group>"; };
D081A40424D0F1A8001B016E /* String+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "String+Extensions.swift"; sourceTree = "<group>"; };
D0A1CA7324DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "KingfisherOptionsInfo+Extensions.swift"; sourceTree = "<group>"; };
D0A2453624EF346800B07068 /* StatusListViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusListViewController.swift; sourceTree = "<group>"; };
D0A2453824EF364100B07068 /* StatusListView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusListView.swift; sourceTree = "<group>"; };
D0A2453E24EF55D000B07068 /* StatusViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusViewModel.swift; sourceTree = "<group>"; };
D0A2454024EF563000B07068 /* StatusService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusService.swift; sourceTree = "<group>"; };
D0A652AC24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "PreferencesEndpoint+Stubbing.swift"; sourceTree = "<group>"; };
D0B23F0C24D210E90066F411 /* NSError+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "NSError+Extensions.swift"; sourceTree = "<group>"; };
D0BEC93724C9632800E864C4 /* RootViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RootViewModel.swift; sourceTree = "<group>"; };
D0BEC93A24C96FD500E864C4 /* RootView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RootView.swift; sourceTree = "<group>"; };
D0BEC94624CA22C400E864C4 /* StatusesViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusesViewModel.swift; sourceTree = "<group>"; };
D0BEC94924CA231200E864C4 /* StatusesView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = StatusesView.swift; sourceTree = "<group>"; };
D0C963FA24CC359D003BD330 /* AlertItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AlertItem.swift; sourceTree = "<group>"; };
D0C963FD24CC3812003BD330 /* Publisher+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "Publisher+Extensions.swift"; sourceTree = "<group>"; };
D0CD847224DBDEC700CF380C /* MastodonPreferences.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MastodonPreferences.swift; sourceTree = "<group>"; };
@ -357,6 +384,7 @@
D0E5361D24E3EB4D00FB1CE1 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
D0E5362824E4A06B00FB1CE1 /* Notification Service Extension.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = "Notification Service Extension.entitlements"; sourceTree = "<group>"; };
D0E5362F24E5436C00FB1CE1 /* PushNotification.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PushNotification.swift; sourceTree = "<group>"; };
D0E900CD24F1F28A00B55F5A /* UIColor+Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIColor+Extensions.swift"; sourceTree = "<group>"; };
D0EC8DC124DF7D9C00A08489 /* IdentityService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = IdentityService.swift; sourceTree = "<group>"; };
D0EC8DC424DF842700A08489 /* KeychainService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = KeychainService.swift; sourceTree = "<group>"; };
D0EC8DC724DF8B3C00A08489 /* SecretsService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SecretsService.swift; sourceTree = "<group>"; };
@ -425,7 +453,11 @@
D0159FA224DE955900E78478 /* CustomEmojiText.swift */,
D0159F8C24DE743700E78478 /* IdentitiesView.swift */,
D0159F8E24DE743700E78478 /* SecondaryNavigationView.swift */,
D0A2453824EF364100B07068 /* StatusListView.swift */,
D02D86D724EF61E4004583CC /* StatusTableViewCell.swift */,
D02D86D824EF61E4004583CC /* StatusTableViewCell.xib */,
D0159F8D24DE743700E78478 /* TabNavigationView.swift */,
D02D86E324EF9848004583CC /* TouchFallthroughTextView.swift */,
);
path = Views;
sourceTree = "<group>";
@ -485,17 +517,27 @@
D019E6F224DF7C9E00697C7D /* Services */ = {
isa = PBXGroup;
children = (
D054951024EB101F008B00A5 /* Status List Services */,
D0EC8DCD24DFB64200A08489 /* AuthenticationService.swift */,
D0EC8DCA24DFA06700A08489 /* IdentitiesService.swift */,
D0EC8DC124DF7D9C00A08489 /* IdentityService.swift */,
D0EC8DC424DF842700A08489 /* KeychainService.swift */,
D0EC8DC724DF8B3C00A08489 /* SecretsService.swift */,
D054951024EB101F008B00A5 /* Status List Services */,
D0A2454024EF563000B07068 /* StatusService.swift */,
D0EC8DD724E096C900A08489 /* UserNotificationService.swift */,
);
path = Services;
sourceTree = "<group>";
};
D02D870024EFBAD5004583CC /* Extensions */ = {
isa = PBXGroup;
children = (
D02D870424EFBB79004583CC /* String+UIKitExtensions.swift */,
D0E900CD24F1F28A00B55F5A /* UIColor+Extensions.swift */,
);
path = Extensions;
sourceTree = "<group>";
};
D047FA7F24C3E21000AF17C5 = {
isa = PBXGroup;
children = (
@ -543,7 +585,9 @@
D047FA8E24C3E21200AF17C5 /* iOS */ = {
isa = PBXGroup;
children = (
D02D870024EFBAD5004583CC /* Extensions */,
D047FA8F24C3E21200AF17C5 /* Info.plist */,
D0A2453524EF344600B07068 /* View Controllers */,
D0159F8024DE739500E78478 /* View Models */,
D0159F7F24DE739000E78478 /* Views */,
);
@ -592,11 +636,12 @@
D0ED1BD624CF94B200B4899C /* Application.swift */,
D05494E924EA3F54008B00A5 /* Attachment.swift */,
D05494EF24EA3FE5008B00A5 /* Card.swift */,
D020F51024ECA309005AB084 /* MastodonContext.swift */,
D0666A5324C6C3E500F3F04B /* Emoji.swift */,
D02D86E524EF998B004583CC /* HTML.swift */,
D0666A4A24C6C37700F3F04B /* Identity.swift */,
D0666A4D24C6C39600F3F04B /* Instance.swift */,
D057426C24EA339300839EBA /* ListTimeline.swift */,
D020F51024ECA309005AB084 /* MastodonContext.swift */,
D0ED1BE224CFA84400B4899C /* MastodonError.swift */,
D0CD847224DBDEC700CF380C /* MastodonPreferences.swift */,
D05494E624EA3F1A008B00A5 /* Mention.swift */,
@ -623,10 +668,19 @@
isa = PBXGroup;
children = (
D06B491E24D3F7FE00642749 /* Localizable.strings */,
D042650724F058280096ED10 /* Localizable.stringsdict */,
);
path = Localizations;
sourceTree = "<group>";
};
D0A2453524EF344600B07068 /* View Controllers */ = {
isa = PBXGroup;
children = (
D0A2453624EF346800B07068 /* StatusListViewController.swift */,
);
path = "View Controllers";
sourceTree = "<group>";
};
D0DB6EF024C5224F00D965FE /* Views */ = {
isa = PBXGroup;
children = (
@ -636,7 +690,6 @@
D0091B6724DC10B30040E8D2 /* PostingReadingPreferencesView.swift */,
D0091B6D24DD68090040E8D2 /* PreferencesView.swift */,
D0BEC93A24C96FD500E864C4 /* RootView.swift */,
D0BEC94924CA231200E864C4 /* StatusesView.swift */,
);
path = Views;
sourceTree = "<group>";
@ -661,6 +714,7 @@
D0091B7024DD68220040E8D2 /* PreferencesViewModel.swift */,
D0BEC93724C9632800E864C4 /* RootViewModel.swift */,
D0BEC94624CA22C400E864C4 /* StatusesViewModel.swift */,
D0A2453E24EF55D000B07068 /* StatusViewModel.swift */,
);
path = "View Models";
sourceTree = "<group>";
@ -668,13 +722,15 @@
D0DB6F1624C665B400D965FE /* Extensions */ = {
isa = PBXGroup;
children = (
D02D86EB24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift */,
D0EC8DE724E21FEC00A08489 /* Data+Extensions.swift */,
D02D86EE24EFB13A004583CC /* Date+Extensions.swift */,
D0A1CA7324DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift */,
D0B23F0C24D210E90066F411 /* NSError+Extensions.swift */,
D0159FA424DE989700E78478 /* NSMutableAttributedString+Extensions.swift */,
D0C963FD24CC3812003BD330 /* Publisher+Extensions.swift */,
D081A40424D0F1A8001B016E /* String+Extensions.swift */,
D065F53A24D3B33A00741304 /* View+Extensions.swift */,
D0EC8DE724E21FEC00A08489 /* Data+Extensions.swift */,
);
path = Extensions;
sourceTree = "<group>";
@ -904,6 +960,8 @@
D054950124EA4FFE008B00A5 /* DevelopmentAssets.xcassets in Resources */,
D047FAB224C3E21200AF17C5 /* Assets.xcassets in Resources */,
D06B491F24D3F7FE00642749 /* Localizable.strings in Resources */,
D02D86DA24EF61E4004583CC /* StatusTableViewCell.xib in Resources */,
D042650824F058280096ED10 /* Localizable.stringsdict in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
@ -912,6 +970,7 @@
buildActionMask = 2147483647;
files = (
D054950224EA4FFE008B00A5 /* DevelopmentAssets.xcassets in Resources */,
D042650924F058280096ED10 /* Localizable.stringsdict in Resources */,
D047FAB324C3E21200AF17C5 /* Assets.xcassets in Resources */,
D06B492024D3FB8000642749 /* Localizable.strings in Resources */,
);
@ -975,6 +1034,7 @@
isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647;
files = (
D02D86E624EF998B004583CC /* HTML.swift in Sources */,
D04FD73924D4A7B4007D572D /* AccountEndpoint+Stubbing.swift in Sources */,
D05494F724EA49F7008B00A5 /* TimelinesEndpoint.swift in Sources */,
D054951B24EB2825008B00A5 /* TransientStatusCollection.swift in Sources */,
@ -987,6 +1047,7 @@
D0BEC94724CA22C400E864C4 /* StatusesViewModel.swift in Sources */,
D0666A4E24C6C39600F3F04B /* Instance.swift in Sources */,
D057426724E9FE1D00839EBA /* ContentDatabase.swift in Sources */,
D02D86E424EF9848004583CC /* TouchFallthroughTextView.swift in Sources */,
D054951524EB1053008B00A5 /* TimelineService.swift in Sources */,
D019E6E924DF72E700697C7D /* InstanceEndpoint.swift in Sources */,
D0ED1BE324CFA84400B4899C /* MastodonError.swift in Sources */,
@ -995,6 +1056,7 @@
D0666A6324C6DC6C00F3F04B /* AppAuthorization.swift in Sources */,
D05494F024EA3FE5008B00A5 /* Card.swift in Sources */,
D019E6E524DF72E700697C7D /* AccountEndpoint.swift in Sources */,
D0A2453924EF364100B07068 /* StatusListView.swift in Sources */,
D075817C24E6659A0081F6A3 /* NotificationTypesPreferencesView.swift in Sources */,
D020F51424ECBA60005AB084 /* LazyView.swift in Sources */,
D065F53B24D3B33A00741304 /* View+Extensions.swift in Sources */,
@ -1008,8 +1070,12 @@
D019E6ED24DF7BF300697C7D /* IdentityDatabase.swift in Sources */,
D081A40524D0F1A8001B016E /* String+Extensions.swift in Sources */,
D019E6E724DF72E700697C7D /* AccessTokenEndpoint.swift in Sources */,
D02D870524EFBB79004583CC /* String+UIKitExtensions.swift in Sources */,
D0BEC93824C9632800E864C4 /* RootViewModel.swift in Sources */,
D02D86EC24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift in Sources */,
D0A2454124EF563000B07068 /* StatusService.swift in Sources */,
D0ED1BC124CED48800B4899C /* HTTPClient.swift in Sources */,
D0E900CE24F1F28A00B55F5A /* UIColor+Extensions.swift in Sources */,
D0EC8DEE24E2704D00A08489 /* PushSubscriptionEndpoint.swift in Sources */,
D0159F9324DE743700E78478 /* SecondaryNavigationView.swift in Sources */,
D019E6E324DF72E700697C7D /* PreferencesEndpoint.swift in Sources */,
@ -1017,13 +1083,15 @@
D0666A5424C6C3E500F3F04B /* Emoji.swift in Sources */,
D05494FA24EA4E5E008B00A5 /* TimelinesEndpoint+Stubbing.swift in Sources */,
D0A652AD24DE3EB6002EA33F /* PreferencesEndpoint+Stubbing.swift in Sources */,
D02D86D924EF61E4004583CC /* StatusTableViewCell.swift in Sources */,
D0DC175524D00F0A00A75C65 /* AccessTokenEndpoint+Stubbing.swift in Sources */,
D0B23F0D24D210E90066F411 /* NSError+Extensions.swift in Sources */,
D05494ED24EA3FA9008B00A5 /* Poll.swift in Sources */,
D019E6D924DF728400697C7D /* MastodonDecoder.swift in Sources */,
D052BBCA24D74C9200A80A7A /* MockUserDefaults.swift in Sources */,
D0A2453F24EF55D000B07068 /* StatusViewModel.swift in Sources */,
D02D86EF24EFB13A004583CC /* Date+Extensions.swift in Sources */,
D0DC175224D008E300A75C65 /* MastodonTarget+Stubbing.swift in Sources */,
D0BEC94A24CA231200E864C4 /* StatusesView.swift in Sources */,
D0BEC93B24C96FD500E864C4 /* RootView.swift in Sources */,
D04FD74224D4AA34007D572D /* DevelopmentModels.swift in Sources */,
D0EC8DC524DF842700A08489 /* KeychainService.swift in Sources */,
@ -1039,6 +1107,7 @@
D0EC8DE524E0B44500A08489 /* UserNotificationService.swift in Sources */,
D0EC8DCB24DFA06700A08489 /* IdentitiesService.swift in Sources */,
D0091B7124DD68220040E8D2 /* PreferencesViewModel.swift in Sources */,
D0A2453724EF346800B07068 /* StatusListViewController.swift in Sources */,
D057426A24EA32AC00839EBA /* Timeline.swift in Sources */,
D0DC174D24CFF1F100A75C65 /* Stubbing.swift in Sources */,
D0091B6B24DC10CE0040E8D2 /* PostingReadingPreferencesViewModel.swift in Sources */,
@ -1088,6 +1157,7 @@
D0ED1BD824CF94B200B4899C /* Application.swift in Sources */,
D047FAAF24C3E21200AF17C5 /* MetatextApp.swift in Sources */,
D0BEC94824CA22C400E864C4 /* StatusesViewModel.swift in Sources */,
D02D86ED24EF9CA3004583CC /* CodingUserInfoKey+Extensions.swift in Sources */,
D0159FA624DE98F600E78478 /* NSMutableAttributedString+Extensions.swift in Sources */,
D0EC8DC324DF7D9C00A08489 /* IdentityService.swift in Sources */,
D0666A4F24C6C39600F3F04B /* Instance.swift in Sources */,
@ -1114,9 +1184,9 @@
D052BBCB24D74C9300A80A7A /* MockUserDefaults.swift in Sources */,
D0DC175324D008E300A75C65 /* MastodonTarget+Stubbing.swift in Sources */,
D0EC8DE924E21FEC00A08489 /* Data+Extensions.swift in Sources */,
D0BEC94B24CA231200E864C4 /* StatusesView.swift in Sources */,
D05494E524EA3EF7008B00A5 /* Tag.swift in Sources */,
D0BEC93C24C96FD500E864C4 /* RootView.swift in Sources */,
D02D86E724EF998B004583CC /* HTML.swift in Sources */,
D0159F9B24DE748900E78478 /* SidebarNavigationViewModel.swift in Sources */,
D04FD74324D4AA34007D572D /* DevelopmentModels.swift in Sources */,
D0DC175924D0130800A75C65 /* HTTPStubs.swift in Sources */,
@ -1138,6 +1208,7 @@
D0091B6C24DC10CE0040E8D2 /* PostingReadingPreferencesViewModel.swift in Sources */,
D0091B6F24DD68090040E8D2 /* PreferencesView.swift in Sources */,
D0EC8DC924DF8B3C00A08489 /* SecretsService.swift in Sources */,
D02D86F024EFB13A004583CC /* Date+Extensions.swift in Sources */,
D0EC8DE024E09D7000A08489 /* AppDelegate.swift in Sources */,
D0DB6EF524C5233E00D965FE /* AddIdentityView.swift in Sources */,
D019E6EA24DF72E700697C7D /* InstanceEndpoint.swift in Sources */,
@ -1154,6 +1225,7 @@
D0EC8DEC24E26F1100A08489 /* PushSubscription.swift in Sources */,
D03DF45C24E62A68007A8CD5 /* DeletionEndpoint.swift in Sources */,
D0A1CA7524DAC2F1003063E9 /* KingfisherOptionsInfo+Extensions.swift in Sources */,
D0A2454224EF563000B07068 /* StatusService.swift in Sources */,
D0ED1BC524CED54D00B4899C /* HTTPTarget.swift in Sources */,
D0C963FF24CC3812003BD330 /* Publisher+Extensions.swift in Sources */,
D075817D24E6659A0081F6A3 /* NotificationTypesPreferencesView.swift in Sources */,

View file

@ -8,7 +8,7 @@ import GRDB
struct ContentDatabase {
private let databaseQueue: DatabaseQueue
init(identityID: UUID, inMemory: Bool = false) throws {
init(identityID: UUID, environment: AppEnvironment) throws {
guard
let documentsDirectory = NSSearchPathForDirectoriesInDomains(
.documentDirectory,
@ -16,7 +16,7 @@ struct ContentDatabase {
.first
else { throw DatabaseError.documentsDirectoryNotFound }
if inMemory {
if environment.inMemoryContent {
databaseQueue = DatabaseQueue()
} else {
databaseQueue = try DatabaseQueue(path: "\(documentsDirectory)/\(identityID.uuidString).sqlite3")
@ -24,6 +24,7 @@ struct ContentDatabase {
try Self.migrate(databaseQueue)
try Self.createTemporaryTables(databaseQueue)
Self.attributedStringCache = environment.attributedStringCache
}
}
@ -79,6 +80,8 @@ extension ContentDatabase {
}
private extension ContentDatabase {
static var attributedStringCache: AttributedStringCache?
// swiftlint:disable function_body_length
static func migrate(_ writer: DatabaseWriter) throws {
var migrator = DatabaseMigrator()
@ -181,6 +184,16 @@ private extension ContentDatabase {
}
extension Account: TableRecord, FetchableRecord, PersistableRecord {
static var databaseDecodingUserInfo: [CodingUserInfoKey: Any] {
var userInfo = [CodingUserInfoKey: Any]()
if let attributedStringCache = ContentDatabase.attributedStringCache {
userInfo[.attributedStringCache] = attributedStringCache
}
return userInfo
}
static func databaseJSONDecoder(for column: String) -> JSONDecoder {
MastodonDecoder()
}
@ -276,7 +289,7 @@ private struct StoredStatus: Codable, Hashable {
let uri: String
let createdAt: Date
let accountId: String
let content: String
let content: HTML
let visibility: Status.Visibility
let sensitive: Bool
let spoilerText: String
@ -354,6 +367,16 @@ private extension StoredStatus {
}
extension StoredStatus: TableRecord, FetchableRecord, PersistableRecord {
static var databaseDecodingUserInfo: [CodingUserInfoKey: Any] {
var userInfo = [CodingUserInfoKey: Any]()
if let attributedStringCache = ContentDatabase.attributedStringCache {
userInfo[.attributedStringCache] = attributedStringCache
}
return userInfo
}
static func databaseJSONDecoder(for column: String) -> JSONDecoder {
MastodonDecoder()
}

View file

@ -0,0 +1,9 @@
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
typealias AttributedStringCache = NSCache<NSString, NSAttributedString>
extension CodingUserInfoKey {
static let attributedStringCache = CodingUserInfoKey(rawValue: "com.metabolist.metatext.attributed-string-cache")!
}

View file

@ -0,0 +1,81 @@
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
extension Date {
var timeAgo: String? {
let calendar = Calendar.autoupdatingCurrent
let now = Date()
if
let oneMinuteAgo = calendar.date(byAdding: DateComponents(minute: -1), to: now),
oneMinuteAgo < self {
Self.abbreviatedDateComponentsFormatter.allowedUnits = [.second]
} else if
let oneHourAgo = calendar.date(byAdding: DateComponents(hour: -1), to: now),
oneHourAgo < self {
Self.abbreviatedDateComponentsFormatter.allowedUnits = [.minute]
} else if
let oneDayAgo = calendar.date(byAdding: DateComponents(day: -1), to: now),
oneDayAgo < self {
Self.abbreviatedDateComponentsFormatter.allowedUnits = [.hour]
} else if
let oneWeekAgo = calendar.date(byAdding: DateComponents(weekOfMonth: -1), to: now),
oneWeekAgo < self {
Self.abbreviatedDateComponentsFormatter.allowedUnits = [.day]
} else {
return Date.shortDateStyleRelativeDateFormatter.string(from: self)
}
return Self.abbreviatedDateComponentsFormatter.string(from: self, to: now)
}
var fullUnitTimeUntil: String? {
let calendar = Calendar.autoupdatingCurrent
let now = Date()
if
let oneDayFromNow = calendar.date(byAdding: DateComponents(day: 1), to: now),
self > oneDayFromNow {
Self.fullDateComponentsFormatter.allowedUnits = [.day]
} else if
let oneHourFromNow = calendar.date(byAdding: DateComponents(hour: 1), to: now),
self > oneHourFromNow {
Self.fullDateComponentsFormatter.allowedUnits = [.hour]
} else if
let oneMinuteFromNow = calendar.date(byAdding: DateComponents(minute: 1), to: now),
self > oneMinuteFromNow {
Self.fullDateComponentsFormatter.allowedUnits = [.minute]
} else {
Self.fullDateComponentsFormatter.allowedUnits = [.second]
}
return Self.fullDateComponentsFormatter.string(from: now, to: self)
}
}
private extension Date {
private static let abbreviatedDateComponentsFormatter: DateComponentsFormatter = {
let dateComponentsFormatter = DateComponentsFormatter()
dateComponentsFormatter.unitsStyle = .abbreviated
return dateComponentsFormatter
}()
private static let fullDateComponentsFormatter: DateComponentsFormatter = {
let dateComponentsFormatter = DateComponentsFormatter()
dateComponentsFormatter.unitsStyle = .full
return dateComponentsFormatter
}()
private static let shortDateStyleRelativeDateFormatter: DateFormatter = {
let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .short
return dateFormatter
}()
}

View file

@ -26,6 +26,15 @@
"preferences.notification-types.reblog" = "Reblog";
"preferences.notification-types.mention" = "Mention";
"preferences.notification-types.poll" = "Poll";
"status.reblogged-by" = "%@ boosted";
"status.pinned-post" = "Pinned post";
"status.show-more" = "Show More";
"status.show-less" = "Show Less";
"status.poll.vote" = "Vote";
"status.poll.participation-count" = "%ld people";
"status.poll.time-left" = "%@ left";
"status.poll.refresh" = "Refresh";
"status.poll.closed" = "Closed";
"status.visibility.public" = "Public";
"status.visibility.unlisted" = "Unlisted";
"status.visibility.private" = "Private";

View file

@ -0,0 +1,54 @@
<?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>status.reblogs-count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@reblogs@</string>
<key>reblogs</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>ld</string>
<key>one</key>
<string>%ld Boost</string>
<key>other</key>
<string>%ld Boosts</string>
</dict>
</dict>
<key>status.favorites-count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@favorites@</string>
<key>favorites</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>ld</string>
<key>one</key>
<string>%ld Favorite</string>
<key>other</key>
<string>%ld Favorites</string>
</dict>
</dict>
<key>account.followers-count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@followers@</string>
<key>followers</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>ld</string>
<key>one</key>
<string>%ld Follower</string>
<key>other</key>
<string>%ld Followers</string>
</dict>
</dict>
</dict>
</plist>

View file

@ -5,7 +5,7 @@ import Foundation
struct Account: Codable, Hashable {
struct Field: Codable, Hashable {
let name: String
let value: String
let value: HTML
let verifiedAt: Date?
}
@ -18,7 +18,7 @@ struct Account: Codable, Hashable {
let followersCount: Int
let followingCount: Int
let statusesCount: Int
let note: String
let note: HTML
let url: URL
let avatar: URL
let avatarStatic: URL

View file

@ -8,6 +8,7 @@ struct AppEnvironment {
let keychainServiceType: KeychainService.Type
let userDefaults: UserDefaults
let inMemoryContent: Bool
let attributedStringCache = AttributedStringCache()
}
extension AppEnvironment {

117
Shared/Model/HTML.swift Normal file
View file

@ -0,0 +1,117 @@
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
struct HTML: Hashable {
let raw: String
let attributed: NSAttributedString
}
extension HTML: Codable {
init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
let attributedStringCache = decoder.userInfo[.attributedStringCache] as? AttributedStringCache
raw = try container.decode(String.self)
if let attributed = attributedStringCache?.object(forKey: raw as NSString) {
self.attributed = attributed
return
}
attributed = HTMLParser(string: raw).parse()
attributedStringCache?.setObject(attributed, forKey: raw as NSString)
}
func encode(to encoder: Encoder) throws {
var container = encoder.singleValueContainer()
try container.encode(raw)
}
}
// https://docs.joinmastodon.org/spec/activitypub/#sanitization
private class HTMLParser: NSObject {
private struct Link: Hashable {
let href: URL
let location: Int
var length = 0
}
private let rawString: String
private let parser: XMLParser
private let parseStopColumn: Int
private var constructedString = ""
private var attributesStack = [[String: String]]()
private var currentLink: Link?
private var links = Set<Link>()
private static let containerTag = "com.metabolist.metatext.container-tag"
private static let openingContainerTag = "<\(containerTag)>"
private static let closingContainerTag = "</\(containerTag)>"
init(string: String) {
rawString = Self.openingContainerTag + string + Self.closingContainerTag
parser = XMLParser(data: Data(rawString.utf8))
parseStopColumn = rawString.count - Self.closingContainerTag.count
super.init()
parser.delegate = self
}
func parse() -> NSAttributedString {
parser.parse()
let attributedString = NSMutableAttributedString(string: constructedString)
for link in links {
attributedString.addAttribute(.link,
value: link.href,
range: .init(location: link.location, length: link.length))
}
return attributedString
}
}
extension HTMLParser: XMLParserDelegate {
func parser(_ parser: XMLParser,
didStartElement elementName: String,
namespaceURI: String?,
qualifiedName qName: String?,
attributes attributeDict: [String: String] = [:]) {
attributesStack.append(attributeDict)
if elementName == "a", let hrefString = attributeDict["href"], let href = URL(string: hrefString) {
currentLink = Link(href: href, location: constructedString.count)
} else if elementName == "br" {
constructedString.append("\n")
}
}
func parser(_ parser: XMLParser,
didEndElement elementName: String,
namespaceURI: String?,
qualifiedName qName: String?) {
let attributes = attributesStack.removeLast()
if attributes["class"] == "ellipsis" {
constructedString.append("")
}
if elementName == "a", var link = currentLink {
link.length = constructedString.count - link.location
links.insert(link)
} else if elementName == "p", parser.columnNumber < parseStopColumn {
constructedString.append("\n\n")
}
}
func parser(_ parser: XMLParser, foundCharacters string: String) {
if attributesStack.last?["class"] != "invisible" {
constructedString.append(string)
}
}
}

View file

@ -17,7 +17,7 @@ class Status: Codable, Identifiable {
let uri: String
let createdAt: Date
let account: Account
let content: String
let content: HTML
let visibility: Visibility
let sensitive: Bool
let spoilerText: String
@ -49,7 +49,7 @@ class Status: Codable, Identifiable {
uri: String,
createdAt: Date,
account: Account,
content: String,
content: HTML,
visibility: Status.Visibility,
sensitive: Bool,
spoilerText: String,
@ -106,6 +106,12 @@ class Status: Codable, Identifiable {
}
}
extension Status {
var displayStatus: Status {
reblog ?? self
}
}
extension Status: Hashable {
static func == (lhs: Status, rhs: Status) -> Bool {
lhs.id == rhs.id

View file

@ -10,7 +10,7 @@ class HTTPClient {
private let session: Session
private let decoder: DataDecoder
init(session: Session, decoder: DataDecoder = JSONDecoder()) {
init(session: Session, decoder: DataDecoder) {
self.session = session
self.decoder = decoder
}

View file

@ -7,8 +7,12 @@ class MastodonClient: HTTPClient {
var instanceURL: URL?
var accessToken: String?
init(session: Session) {
super.init(session: session, decoder: MastodonDecoder())
required init(environment: AppEnvironment) {
let decoder = MastodonDecoder()
decoder.userInfo[.attributedStringCache] = environment.attributedStringCache
super.init(session: environment.session, decoder: decoder)
}
override func request<T: DecodableTarget>(_ target: T) -> AnyPublisher<T.ResultType, Error> {

View file

@ -9,7 +9,7 @@ struct AuthenticationService {
private let webAuthSessionContextProvider = WebAuthSessionContextProvider()
init(environment: AppEnvironment) {
networkClient = MastodonClient(session: environment.session)
networkClient = MastodonClient(environment: environment)
webAuthSessionType = environment.webAuthSessionType
}
}

View file

@ -52,7 +52,7 @@ extension IdentitiesService {
func deleteIdentity(_ identity: Identity) -> AnyPublisher<Void, Error> {
let secretsService = SecretsService(identityID: identity.id, keychainService: environment.keychainServiceType)
let networkClient = MastodonClient(session: environment.session)
let networkClient = MastodonClient(environment: environment)
networkClient.instanceURL = identity.url

View file

@ -34,11 +34,11 @@ class IdentityService {
secretsService = SecretsService(
identityID: identityID,
keychainService: environment.keychainServiceType)
networkClient = MastodonClient(session: environment.session)
networkClient = MastodonClient(environment: environment)
networkClient.instanceURL = identity.url
networkClient.accessToken = try? secretsService.item(.accessToken)
contentDatabase = try ContentDatabase(identityID: identityID, inMemory: environment.inMemoryContent)
contentDatabase = try ContentDatabase(identityID: identityID, environment: environment)
observation.catch { [weak self] error -> Empty<Identity, Never> in
self?.observationErrorsInput.send(error)

View file

@ -34,6 +34,32 @@ struct ContextService {
extension ContextService: StatusListService {
var contextParent: Status? { status }
func isReplyInContext(status: Status) -> Bool {
let flatContext = flattenedContext()
guard
let index = flatContext.firstIndex(where: { $0.id == status.id }),
index > 0
else { return false }
let previousStatus = flatContext[index - 1]
return previousStatus.id != contextParent?.id && status.inReplyToId == previousStatus.id
}
func hasReplyFollowing(status: Status) -> Bool {
let flatContext = flattenedContext()
guard
let index = flatContext.firstIndex(where: { $0.id == status.id }),
flatContext.count > index + 1
else { return false }
let nextStatus = flatContext[index + 1]
return status.id != contextParent?.id && nextStatus.inReplyToId == status.id
}
func request(maxID: String?, minID: String?) -> AnyPublisher<Void, Error> {
networkClient.request(ContextEndpoint.context(id: status.id))
.handleEvents(receiveOutput: context.send)
@ -42,7 +68,17 @@ extension ContextService: StatusListService {
.eraseToAnyPublisher()
}
func statusService(status: Status) -> StatusService {
StatusService(status: status, networkClient: networkClient, contentDatabase: contentDatabase)
}
func contextService(status: Status) -> ContextService {
ContextService(status: status, networkClient: networkClient, contentDatabase: contentDatabase)
ContextService(status: status.displayStatus, networkClient: networkClient, contentDatabase: contentDatabase)
}
}
private extension ContextService {
func flattenedContext() -> [Status] {
context.value.ancestors + [status] + context.value.descendants
}
}

View file

@ -6,10 +6,20 @@ import Combine
protocol StatusListService {
var statusSections: AnyPublisher<[[Status]], Error> { get }
var contextParent: Status? { get }
func isPinned(status: Status) -> Bool
func isReplyInContext(status: Status) -> Bool
func hasReplyFollowing(status: Status) -> Bool
func request(maxID: String?, minID: String?) -> AnyPublisher<Void, Error>
func statusService(status: Status) -> StatusService
func contextService(status: Status) -> ContextService
}
extension StatusListService {
var contextParent: Status? { nil }
func isPinned(status: Status) -> Bool { false }
func isReplyInContext(status: Status) -> Bool { false }
func hasReplyFollowing(status: Status) -> Bool { false }
}

View file

@ -28,7 +28,11 @@ extension TimelineService: StatusListService {
.eraseToAnyPublisher()
}
func statusService(status: Status) -> StatusService {
StatusService(status: status, networkClient: networkClient, contentDatabase: contentDatabase)
}
func contextService(status: Status) -> ContextService {
ContextService(status: status, networkClient: networkClient, contentDatabase: contentDatabase)
ContextService(status: status.displayStatus, networkClient: networkClient, contentDatabase: contentDatabase)
}
}

View file

@ -0,0 +1,16 @@
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
import Combine
struct StatusService {
let status: Status
private let networkClient: MastodonClient
private let contentDatabase: ContentDatabase
init(status: Status, networkClient: MastodonClient, contentDatabase: ContentDatabase) {
self.status = status
self.networkClient = networkClient
self.contentDatabase = contentDatabase
}
}

View file

@ -0,0 +1,112 @@
// Copyright © 2020 Metabolist. All rights reserved.
import Foundation
import Combine
struct StatusViewModel {
let content: NSAttributedString
let contentEmoji: [Emoji]
let displayName: String
let displayNameEmoji: [Emoji]
let spoilerText: String
let isReblog: Bool
let rebloggedByDisplayName: String
let rebloggedByDisplayNameEmoji: [Emoji]
let pollOptionTitles: [String]
let pollEmoji: [Emoji]
var isPinned = false
var isContextParent = false
var isReplyInContext = false
var hasReplyFollowing = false
var sensitiveContentToggled = false
private let statusService: StatusService
init(statusService: StatusService) {
self.statusService = statusService
content = statusService.status.displayStatus.content.attributed
contentEmoji = statusService.status.displayStatus.emojis
displayName = statusService.status.displayStatus.account.displayName == ""
? statusService.status.displayStatus.account.username
: statusService.status.displayStatus.account.displayName
displayNameEmoji = statusService.status.displayStatus.account.emojis
spoilerText = statusService.status.displayStatus.spoilerText
isReblog = statusService.status.reblog != nil
rebloggedByDisplayName = statusService.status.account.displayName == ""
? statusService.status.account.username
: statusService.status.account.displayName
rebloggedByDisplayNameEmoji = statusService.status.account.emojis
pollOptionTitles = statusService.status.displayStatus.poll?.options.map { $0.title } ?? []
pollEmoji = statusService.status.displayStatus.poll?.emojis ?? []
}
}
extension StatusViewModel {
var shouldDisplaySensitiveContent: Bool {
if statusService.status.displayStatus.sensitive {
return sensitiveContentToggled
} else {
return true
}
}
var accountName: String { "@" + statusService.status.displayStatus.account.acct }
var avatarURL: URL { statusService.status.displayStatus.account.avatar }
var time: String? { statusService.status.displayStatus.createdAt.timeAgo }
var contextParentTime: String {
Self.contextParentDateFormatter.string(from: statusService.status.displayStatus.createdAt)
}
var applicationName: String? { statusService.status.displayStatus.application?.name }
var applicationURL: URL? {
guard let website = statusService.status.displayStatus.application?.website else { return nil }
return URL(string: website)
}
var repliesCount: Int { statusService.status.displayStatus.repliesCount }
var reblogsCount: Int { statusService.status.displayStatus.reblogsCount }
var favoritesCount: Int { statusService.status.displayStatus.favouritesCount }
var reblogged: Bool { statusService.status.displayStatus.reblogged ?? false }
var favorited: Bool { statusService.status.displayStatus.favourited ?? false }
var sensitive: Bool { statusService.status.displayStatus.sensitive }
var sharingURL: URL? { statusService.status.displayStatus.url }
var cardURL: URL? { statusService.status.displayStatus.card?.url }
var cardTitle: String? { statusService.status.displayStatus.card?.title }
var cardDescription: String? { statusService.status.displayStatus.card?.description }
var cardImageURL: URL? { statusService.status.displayStatus.card?.image }
var canBeReblogged: Bool {
switch statusService.status.displayStatus.visibility {
case .direct, .private:
return false
default:
return true
}
}
}
private extension StatusViewModel {
private static let contextParentDateFormatter: DateFormatter = {
let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .short
dateFormatter.timeStyle = .short
return dateFormatter
}()
}

View file

@ -7,15 +7,15 @@ class StatusesViewModel: ObservableObject {
@Published private(set) var statusSections = [[Status]]()
@Published var alertItem: AlertItem?
@Published private(set) var loading = false
let scrollToStatusID: AnyPublisher<String, Never>
let scrollToStatus: AnyPublisher<Status, Never>
private let statusListService: StatusListService
private let scrollToStatusIDInput = PassthroughSubject<String, Never>()
private let scrollToStatusInput = PassthroughSubject<Status, Never>()
private var hasScrolledToParentAfterContextLoad = false
private var cancellables = Set<AnyCancellable>()
init(statusListService: StatusListService) {
self.statusListService = statusListService
scrollToStatusID = scrollToStatusIDInput.eraseToAnyPublisher()
scrollToStatus = scrollToStatusInput.eraseToAnyPublisher()
statusListService.statusSections
.assignErrorsToAlertItem(to: \.alertItem, on: self)
@ -30,7 +30,7 @@ class StatusesViewModel: ObservableObject {
!($0.first ?? []).isEmpty || !(($0.last ?? []).isEmpty),
!self.hasScrolledToParentAfterContextLoad {
self.hasScrolledToParentAfterContextLoad = true
self.scrollToStatusIDInput.send(contextParent.id)
self.scrollToStatusInput.send(contextParent)
}
}
.store(in: &cancellables)
@ -50,7 +50,25 @@ extension StatusesViewModel {
.store(in: &cancellables)
}
func statusViewModel(status: Status) -> StatusViewModel {
var statusViewModel = Self.viewModelCache[status]
?? StatusViewModel(statusService: statusListService.statusService(status: status))
statusViewModel.isContextParent = status == contextParent
statusViewModel.isPinned = statusListService.isPinned(status: status)
statusViewModel.isReplyInContext = statusListService.isReplyInContext(status: status)
statusViewModel.hasReplyFollowing = statusListService.hasReplyFollowing(status: status)
Self.viewModelCache[status] = statusViewModel
return statusViewModel
}
func contextViewModel(status: Status) -> StatusesViewModel {
StatusesViewModel(statusListService: statusListService.contextService(status: status))
}
}
private extension StatusesViewModel {
static var viewModelCache = [Status: StatusViewModel]()
}

View file

@ -1,56 +0,0 @@
// Copyright © 2020 Metabolist. All rights reserved.
import SwiftUI
struct StatusesView: View {
@StateObject var viewModel: StatusesViewModel
var body: some View {
ScrollViewReader { scrollViewProxy in
ScrollView {
LazyVStack {
ForEach(Array(zip(viewModel.statusSections.indices, viewModel.statusSections)),
id: \.0) { _, statuses in
ForEach(statuses) { status in
if status == viewModel.contextParent {
statusView(status: status)
} else {
NavigationLink(destination:
LazyView(StatusesView(viewModel:
viewModel.contextViewModel(status: status)))) {
statusView(status: status)
}
.buttonStyle(PlainButtonStyle())
}
Divider()
}
}
if viewModel.loading {
ProgressView()
}
}
}
.onReceive(viewModel.scrollToStatusID.receive(on: DispatchQueue.main)) { id in
withAnimation {
scrollViewProxy.scrollTo(id)
}
}
}
.onAppear { viewModel.request() }
.alertItem($viewModel.alertItem)
}
}
private extension StatusesView {
func statusView(status: Status) -> some View {
Text(status.content)
}
}
#if DEBUG
struct StatusesView_Previews: PreviewProvider {
static var previews: some View {
StatusesView(viewModel: .development)
}
}
#endif

View file

@ -0,0 +1,24 @@
// Copyright © 2020 Metabolist. All rights reserved.
import UIKit
extension String {
func countEmphasizedAttributedString(count: Int, highlighted: Bool = false) -> NSAttributedString {
let countRange = (self as NSString).range(of: String.localizedStringWithFormat("%ld", count))
let attributed = NSMutableAttributedString(
string: self,
attributes: [
.font: UIFont.preferredFont(forTextStyle: .body),
.foregroundColor: highlighted ? UIColor.tertiaryLabel : UIColor.secondaryLabel
])
attributed.addAttributes(
[
.font: UIFont.preferredFont(forTextStyle: .headline),
.foregroundColor: highlighted ? UIColor.secondaryLabel : UIColor.label
],
range: countRange)
return attributed
}
}

View file

@ -0,0 +1,12 @@
// Copyright © 2020 Metabolist. All rights reserved.
import UIKit
extension UIColor {
func image(_ size: CGSize = CGSize(width: 1, height: 1)) -> UIImage {
UIGraphicsImageRenderer(size: size).image { context in
self.setFill()
context.fill(CGRect(origin: .zero, size: size))
}
}
}

View file

@ -0,0 +1,136 @@
// Copyright © 2020 Metabolist. All rights reserved.
import SwiftUI
import Combine
class StatusListViewController: UITableViewController {
private let viewModel: StatusesViewModel
private var cancellables = Set<AnyCancellable>()
private var cellHeightCaches = [CGFloat: [Status: CGFloat]]()
private lazy var dataSource: UITableViewDiffableDataSource<Int, Status> = {
UITableViewDiffableDataSource(tableView: tableView) { [weak self] tableView, indexPath, status in
guard
let self = self,
let cell = tableView.dequeueReusableCell(
withIdentifier: String(describing: StatusTableViewCell.self),
for: indexPath) as? StatusTableViewCell
else { return nil }
let statusViewModel = self.viewModel.statusViewModel(status: status)
cell.viewModel = statusViewModel
cell.delegate = self
return cell
}
}()
init(viewModel: StatusesViewModel) {
self.viewModel = viewModel
super.init(style: .plain)
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewDidLoad() {
super.viewDidLoad()
for cellClass in [StatusTableViewCell.self] {
let classString = String(describing: cellClass)
tableView.register(
UINib(nibName: classString, bundle: nil),
forCellReuseIdentifier: classString)
}
tableView.dataSource = dataSource
tableView.cellLayoutMarginsFollowReadableWidth = true
tableView.separatorInset = .zero
viewModel.$statusSections.map { $0.snapshot() }
.sink { [weak self] in self?.dataSource.apply($0, animatingDifferences: false) }
.store(in: &cancellables)
viewModel.scrollToStatus
.receive(on: DispatchQueue.main)
.sink { [weak self] in
guard
let self = self,
let indexPath = self.dataSource.indexPath(for: $0)
else { return }
self.tableView.scrollToRow(at: indexPath, at: .none, animated: true)
}
.store(in: &cancellables)
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
viewModel.request()
}
override func tableView(_ tableView: UITableView,
willDisplay cell: UITableViewCell,
forRowAt indexPath: IndexPath) {
guard let item = dataSource.itemIdentifier(for: indexPath) else { return }
var heightCache = cellHeightCaches[tableView.frame.width] ?? [Status: CGFloat]()
heightCache[item] = cell.frame.height
cellHeightCaches[tableView.frame.width] = heightCache
}
override func tableView(_ tableView: UITableView, estimatedHeightForRowAt indexPath: IndexPath) -> CGFloat {
guard let item = dataSource.itemIdentifier(for: indexPath) else { return UITableView.automaticDimension }
return cellHeightCaches[tableView.frame.width]?[item] ?? UITableView.automaticDimension
}
override func tableView(_ tableView: UITableView, shouldHighlightRowAt indexPath: IndexPath) -> Bool {
viewModel.statusSections[indexPath.section][indexPath.row] != viewModel.contextParent
}
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
let status = viewModel.statusSections[indexPath.section][indexPath.row]
navigationController?.pushViewController(
StatusListViewController(viewModel: viewModel.contextViewModel(status: status)),
animated: true)
}
}
extension StatusListViewController: StatusTableViewCellDelegate {
func statusTableViewCellDidHaveShareButtonTapped(_ cell: StatusTableViewCell) {
guard let url = cell.viewModel.sharingURL else { return }
share(url: url)
}
}
private extension StatusListViewController {
func share(url: URL) {
let activityViewController = UIActivityViewController(activityItems: [url], applicationActivities: nil)
present(activityViewController, animated: true, completion: nil)
}
}
private extension Array where Element == [Status] {
func snapshot() -> NSDiffableDataSourceSnapshot<Int, Status> {
var snapshot = NSDiffableDataSourceSnapshot<Int, Status>()
let sections = [Int](0..<count)
snapshot.appendSections(sections)
for section in sections {
snapshot.appendItems(self[section], toSection: section)
}
return snapshot
}
}

View file

@ -0,0 +1,15 @@
// Copyright © 2020 Metabolist. All rights reserved.
import SwiftUI
struct StatusListView: UIViewControllerRepresentable {
let viewModel: StatusesViewModel
func makeUIViewController(context: Context) -> StatusListViewController {
StatusListViewController(viewModel: viewModel)
}
func updateUIViewController(_ uiViewController: StatusListViewController, context: Context) {
}
}

View file

@ -0,0 +1,367 @@
// Copyright © 2020 Metabolist. All rights reserved.
import AVKit
import Kingfisher
import UIKit
protocol StatusTableViewCellDelegate: class {
func statusTableViewCellDidHaveShareButtonTapped(_ cell: StatusTableViewCell)
}
class StatusTableViewCell: UITableViewCell {
@IBOutlet weak var metaIcon: UIImageView!
@IBOutlet weak var metaLabel: UILabel!
@IBOutlet weak var contentTextView: TouchFallthroughTextView!
@IBOutlet weak var avatarButton: UIButton!
@IBOutlet weak var avatarImageView: AnimatedImageView!
@IBOutlet weak var displayNameLabel: UILabel!
@IBOutlet weak var accountLabel: UILabel!
@IBOutlet weak var timeLabel: UILabel!
@IBOutlet weak var spoilerTextLabel: UILabel!
@IBOutlet weak var toggleSensitiveContentButton: UIButton!
@IBOutlet weak var replyButton: UIButton!
@IBOutlet weak var reblogButton: UIButton!
@IBOutlet weak var favoriteButton: UIButton!
@IBOutlet weak var shareButton: UIButton!
@IBOutlet weak var cardView: UIView!
@IBOutlet weak var cardImageView: UIImageView!
@IBOutlet weak var cardTitleLabel: UILabel!
@IBOutlet weak var cardDescriptionLabel: UILabel!
@IBOutlet weak var cardURLLabel: UILabel!
@IBOutlet weak var cardButton: UIButton!
@IBOutlet weak var sensitiveContentView: UIStackView!
@IBOutlet weak var hasReplyFollowingView: UIView!
@IBOutlet weak var inReplyToView: UIView!
@IBOutlet weak var avatarReplyContextView: UIView!
@IBOutlet weak var nameDateView: UIStackView!
@IBOutlet weak var contextParentAvatarNameView: UIStackView!
@IBOutlet weak var contextParentAvatarImageView: AnimatedImageView!
@IBOutlet weak var contextParentAvatarButton: UIButton!
@IBOutlet weak var contextParentDisplayNameLabel: UILabel!
@IBOutlet weak var contextParentAccountLabel: UILabel!
@IBOutlet weak var actionButtonsView: UIStackView!
@IBOutlet weak var contextParentReplyButton: UIButton!
@IBOutlet weak var contextParentReblogButton: UIButton!
@IBOutlet weak var contextParentFavoriteButton: UIButton!
@IBOutlet weak var contextParentShareButton: UIButton!
@IBOutlet weak var contextParentActionsButton: UIButton!
@IBOutlet weak var contextParentTimeLabel: UILabel!
@IBOutlet weak var timeApplicationDividerView: UILabel!
@IBOutlet weak var applicationButton: UIButton!
@IBOutlet weak var contextParentRebloggedByButton: UIButton!
@IBOutlet weak var contextParentFavoritedByButton: UIButton!
@IBOutlet weak var contextParentItems: UIStackView!
@IBOutlet weak var contextParentRebloggedByFavoritedByView: UIStackView!
@IBOutlet weak var contextParentRebloggedByFavoritedBySeparator: UIView!
weak var delegate: StatusTableViewCellDelegate?
@IBOutlet private var separatorConstraints: [NSLayoutConstraint]!
var viewModel: StatusViewModel! {
didSet {
let mutableContent = NSMutableAttributedString(attributedString: viewModel.content)
let mutableDisplayName = NSMutableAttributedString(string: viewModel.displayName)
let mutableSpoilerText = NSMutableAttributedString(string: viewModel.spoilerText)
let contentFont = UIFont.preferredFont(forTextStyle: viewModel.isContextParent ? .title3 : .callout)
contentTextView.shouldFallthrough = !viewModel.isContextParent
avatarReplyContextView.isHidden = viewModel.isContextParent
nameDateView.isHidden = viewModel.isContextParent
contextParentAvatarNameView.isHidden = !viewModel.isContextParent
actionButtonsView.isHidden = viewModel.isContextParent
contextParentItems.isHidden = !viewModel.isContextParent
let avatarImageView: UIImageView
let displayNameLabel: UILabel
let accountLabel: UILabel
if viewModel.isContextParent {
avatarImageView = contextParentAvatarImageView
displayNameLabel = contextParentDisplayNameLabel
accountLabel = contextParentAccountLabel
} else {
avatarImageView = self.avatarImageView
displayNameLabel = self.displayNameLabel
accountLabel = self.accountLabel
}
let contentRange = NSRange(location: 0, length: mutableContent.length)
mutableContent.removeAttribute(.font, range: contentRange)
mutableContent.addAttributes(
[.font: contentFont as Any,
.foregroundColor: UIColor.label],
range: contentRange)
mutableContent.insert(emojis: viewModel.contentEmoji) { [weak self] in
self?.contentTextView.setNeedsDisplay()
}
mutableContent.resizeAttachments(toLineHeight: contentFont.lineHeight)
contentTextView.attributedText = mutableContent
contentTextView.isHidden = contentTextView.text == ""
mutableDisplayName.insert(emojis: viewModel.displayNameEmoji) { displayNameLabel.setNeedsDisplay() }
mutableDisplayName.resizeAttachments(toLineHeight: displayNameLabel.font.lineHeight)
displayNameLabel.attributedText = mutableDisplayName
mutableSpoilerText.insert(emojis: viewModel.contentEmoji) { [weak self] in
self?.spoilerTextLabel.setNeedsDisplay()
}
mutableSpoilerText.resizeAttachments(toLineHeight: spoilerTextLabel.font.lineHeight)
spoilerTextLabel.attributedText = mutableSpoilerText
spoilerTextLabel.isHidden = !viewModel.sensitive || spoilerTextLabel.text == ""
toggleSensitiveContentButton.setTitle(
viewModel.shouldDisplaySensitiveContent
? NSLocalizedString("status.show-less", comment: "")
: NSLocalizedString("status.show-more", comment: ""),
for: .normal)
accountLabel.text = viewModel.accountName
timeLabel.text = viewModel.time
contextParentTimeLabel.text = viewModel.contextParentTime
timeApplicationDividerView.isHidden = viewModel.applicationName == nil
applicationButton.isHidden = viewModel.applicationName == nil
applicationButton.setTitle(viewModel.applicationName, for: .normal)
applicationButton.isEnabled = viewModel.applicationURL != nil
avatarImageView.kf.setImage(with: viewModel.avatarURL)
toggleSensitiveContentButton.isHidden = !viewModel.sensitive
replyButton.setTitle(viewModel.repliesCount == 0 ? "" : String(viewModel.repliesCount), for: .normal)
reblogButton.setTitle(viewModel.reblogsCount == 0 ? "" : String(viewModel.reblogsCount), for: .normal)
setReblogButtonColor(reblogged: viewModel.reblogged)
favoriteButton.setTitle(viewModel.favoritesCount == 0 ? "" : String(viewModel.favoritesCount), for: .normal)
setFavoriteButtonColor(favorited: viewModel.favorited)
reblogButton.isEnabled = viewModel.canBeReblogged
contextParentReblogButton.isEnabled = viewModel.canBeReblogged
let noReblogs = viewModel.reblogsCount == 0
let noFavorites = viewModel.favoritesCount == 0
let noInteractions = noReblogs && noFavorites
setAttributedLocalizedTitle(
button: contextParentRebloggedByButton,
localizationKey: "status.reblogs-count",
count: viewModel.reblogsCount)
contextParentRebloggedByButton.isHidden = noReblogs
setAttributedLocalizedTitle(
button: contextParentFavoritedByButton,
localizationKey: "status.favorites-count",
count: viewModel.favoritesCount)
contextParentFavoritedByButton.isHidden = noFavorites
contextParentRebloggedByFavoritedByView.isHidden = noInteractions
contextParentRebloggedByFavoritedBySeparator.isHidden = noInteractions
if
viewModel.isReblog {
let metaText = String.localizedStringWithFormat(
NSLocalizedString("status.reblogged-by", comment: ""),
viewModel.rebloggedByDisplayName)
let mutableMetaText = NSMutableAttributedString(string: metaText)
mutableMetaText.insert(emojis: viewModel.rebloggedByDisplayNameEmoji) { [weak self] in
self?.metaLabel.setNeedsDisplay()
}
mutableMetaText.resizeAttachments(toLineHeight: metaLabel.font.lineHeight)
metaLabel.attributedText = mutableMetaText
metaIcon.image = UIImage(
systemName: "arrow.2.squarepath",
withConfiguration: UIImage.SymbolConfiguration(scale: .small))
metaLabel.isHidden = false
metaIcon.isHidden = false
} else if viewModel.isPinned {
metaLabel.text = NSLocalizedString("status.pinned-post", comment: "")
metaIcon.image = UIImage(
systemName: "pin",
withConfiguration: UIImage.SymbolConfiguration(scale: .small))
metaLabel.isHidden = false
metaIcon.isHidden = false
} else {
metaLabel.isHidden = true
metaIcon.isHidden = true
}
if let cardURL = viewModel.cardURL {
cardTitleLabel.text = viewModel.cardTitle
cardDescriptionLabel.text = viewModel.cardDescription
cardDescriptionLabel.isHidden = cardDescriptionLabel.text == ""
|| cardDescriptionLabel.text == cardTitleLabel.text
if
let host = cardURL.host, host.hasPrefix("www."),
let withoutWww = cardURL.host?.components(separatedBy: "www.").last {
cardURLLabel.text = withoutWww
} else {
cardURLLabel.text = cardURL.host
}
if let cardImageURL = viewModel.cardImageURL {
cardImageView.isHidden = false
cardImageView.kf.setImage(with: cardImageURL)
} else {
cardImageView.isHidden = true
}
cardView.isHidden = false
} else {
cardView.isHidden = true
}
sensitiveContentView.isHidden = !viewModel.shouldDisplaySensitiveContent
inReplyToView.isHidden = !viewModel.isReplyInContext
hasReplyFollowingView.isHidden = !viewModel.hasReplyFollowing
}
}
override func awakeFromNib() {
super.awakeFromNib()
for constraint in separatorConstraints {
constraint.constant = 1 / UIScreen.main.scale
}
avatarImageView.kf.indicatorType = .activity
contextParentAvatarImageView.kf.indicatorType = .activity
cardImageView.kf.indicatorType = .activity
contentTextView.delegate = self
let highlightedButtonBackgroundImage = UIColor(white: 0, alpha: 0.5).image()
cardButton.setBackgroundImage(highlightedButtonBackgroundImage, for: .highlighted)
avatarButton.setBackgroundImage(highlightedButtonBackgroundImage, for: .highlighted)
contextParentAvatarButton.setBackgroundImage(highlightedButtonBackgroundImage, for: .highlighted)
}
override func prepareForReuse() {
super.prepareForReuse()
avatarImageView.kf.cancelDownloadTask()
cardImageView.kf.cancelDownloadTask()
}
override func layoutSubviews() {
super.layoutSubviews()
for button: UIButton in [toggleSensitiveContentButton] where button.frame.height != 0 {
button.layer.cornerRadius = button.frame.height / 2
}
if hasReplyFollowingView.isHidden {
separatorInset.right = UIDevice.current.userInterfaceIdiom == .phone ? 0 : layoutMargins.right
} else {
separatorInset.right = .greatestFiniteMagnitude
}
separatorInset.left = UIDevice.current.userInterfaceIdiom == .phone ? 0 : layoutMargins.left
}
}
extension StatusTableViewCell {
@IBAction func avatarButtonTapped(_ sender: Any) {
}
@IBAction func toggleSensitiveContentButtonTapped(_ sender: Any) {
}
@IBAction func cardButtonTapped(_ sender: UIButton) {
}
@IBAction func replyButtonTapped(_ sender: UIButton) {
}
@IBAction func reblogButtonTapped(_ sender: UIButton) {
}
@IBAction func favoriteButtonTapped(_ sender: UIButton) {
}
@IBAction func actionsButtonTapped(_ sender: Any) {
}
@IBAction func shareButtonTapped(_ sender: Any) {
delegate?.statusTableViewCellDidHaveShareButtonTapped(self)
}
@IBAction func applicationButtonTapped(_ sender: Any) {
}
@IBAction func contextParentRebloggedByButtonTapped(_ sender: Any) {
}
@IBAction func contextParentFavoritedByButtonTapped(_ sender: Any) {
}
}
extension StatusTableViewCell: UITextViewDelegate {
func textView(
_ textView: UITextView,
shouldInteractWith URL: URL,
in characterRange: NSRange,
interaction: UITextItemInteraction) -> Bool {
switch interaction {
case .invokeDefaultAction: print(URL); return false
case .preview: return false
case .presentActions: return false
@unknown default: return false
}
}
}
private extension StatusTableViewCell {
private static let defaultAspectRatioConstraintMultiplier: CGFloat = 4.0 / 3.0
private static let hasReplyFollowingSeparatorInsets = UIEdgeInsets(
top: 0,
left: 0,
bottom: 0,
right: .greatestFiniteMagnitude)
private func setReblogButtonColor(reblogged: Bool) {
let reblogColor: UIColor = reblogged ? .systemGreen : .secondaryLabel
let reblogButton: UIButton
if viewModel.isContextParent {
reblogButton = contextParentReblogButton
} else {
reblogButton = self.reblogButton
}
reblogButton.tintColor = reblogColor
reblogButton.setTitleColor(reblogColor, for: .normal)
}
private func setFavoriteButtonColor(favorited: Bool) {
let favoriteColor: UIColor = favorited ? .systemYellow : .secondaryLabel
let favoriteButton: UIButton
let scale: UIImage.SymbolScale
if viewModel.isContextParent {
favoriteButton = contextParentFavoriteButton
scale = .medium
} else {
favoriteButton = self.favoriteButton
scale = .small
}
favoriteButton.tintColor = favoriteColor
favoriteButton.setTitleColor(favoriteColor, for: .normal)
favoriteButton.setImage(UIImage(
systemName: favorited ? "star.fill" : "star",
withConfiguration: UIImage.SymbolConfiguration(scale: scale)),
for: .normal)
}
private func setAttributedLocalizedTitle(button: UIButton, localizationKey: String, count: Int) {
let localizedTitle = String.localizedStringWithFormat(NSLocalizedString(localizationKey, comment: ""), count)
button.setAttributedTitle(localizedTitle.countEmphasizedAttributedString(count: count), for: .normal)
button.setAttributedTitle(
localizedTitle.countEmphasizedAttributedString(count: count, highlighted: true),
for: .highlighted)
}
}

View file

@ -0,0 +1,556 @@
<?xml version="1.0" encoding="UTF-8"?>
<document type="com.apple.InterfaceBuilder3.CocoaTouch.XIB" version="3.0" toolsVersion="17154" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES">
<device id="retina4_7" orientation="portrait" appearance="light"/>
<dependencies>
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="17124"/>
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
<capability name="System colors in document resources" minToolsVersion="11.0"/>
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
</dependencies>
<objects>
<placeholder placeholderIdentifier="IBFilesOwner" id="-1" userLabel="File's Owner"/>
<placeholder placeholderIdentifier="IBFirstResponder" id="-2" customClass="UIResponder"/>
<tableViewCell contentMode="scaleToFill" selectionStyle="default" indentationWidth="10" rowHeight="1003" id="KGk-i7-Jjw" customClass="StatusTableViewCell" customModule="Metatext" customModuleProvider="target">
<rect key="frame" x="0.0" y="0.0" width="377" height="1003"/>
<autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMaxY="YES"/>
<tableViewCellContentView key="contentView" opaque="NO" clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="center" tableViewCell="KGk-i7-Jjw" id="H2p-sc-9uM">
<rect key="frame" x="0.0" y="0.0" width="377" height="1003"/>
<autoresizingMask key="autoresizingMask"/>
<subviews>
<stackView opaque="NO" contentMode="scaleToFill" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="PuM-4R-mah">
<rect key="frame" x="16" y="11" width="345" height="981"/>
<subviews>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="NHl-tq-tPi">
<rect key="frame" x="0.0" y="0.0" width="50" height="981"/>
<subviews>
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFit" horizontalHuggingPriority="251" verticalHuggingPriority="251" image="arrow.2.squarepath" catalog="system" translatesAutoresizingMaskIntoConstraints="NO" id="uhT-HE-MkE">
<rect key="frame" x="29.5" y="0.0" width="20.5" height="14.5"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="small"/>
</imageView>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="tqr-8D-x5Z">
<rect key="frame" x="24" y="-11" width="2" height="37.5"/>
<color key="backgroundColor" systemColor="quaternaryLabelColor"/>
<constraints>
<constraint firstAttribute="width" constant="2" id="Zpn-G6-ZXB"/>
</constraints>
</view>
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleToFill" horizontalHuggingPriority="251" verticalHuggingPriority="251" translatesAutoresizingMaskIntoConstraints="NO" id="Rag-3x-Vbq" customClass="AnimatedImageView" customModule="Kingfisher">
<rect key="frame" x="0.0" y="26.5" width="50" height="50"/>
<constraints>
<constraint firstAttribute="height" constant="50" id="OSq-XT-4Ch"/>
<constraint firstAttribute="width" constant="50" id="jO1-At-ia1"/>
</constraints>
<userDefinedRuntimeAttributes>
<userDefinedRuntimeAttribute type="number" keyPath="layer.cornerRadius">
<integer key="value" value="25"/>
</userDefinedRuntimeAttribute>
</userDefinedRuntimeAttributes>
</imageView>
<button opaque="NO" clipsSubviews="YES" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="dRj-hF-Ov8">
<rect key="frame" x="0.0" y="26.5" width="50" height="50"/>
<userDefinedRuntimeAttributes>
<userDefinedRuntimeAttribute type="number" keyPath="layer.cornerRadius">
<integer key="value" value="25"/>
</userDefinedRuntimeAttribute>
</userDefinedRuntimeAttributes>
<connections>
<action selector="avatarButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="CZ5-0T-jMR"/>
</connections>
</button>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="H9F-5F-FkJ">
<rect key="frame" x="24" y="76.5" width="2" height="915.5"/>
<color key="backgroundColor" systemColor="quaternaryLabelColor"/>
<constraints>
<constraint firstAttribute="width" constant="2" id="zZf-bJ-KBC"/>
</constraints>
</view>
</subviews>
<color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/>
<constraints>
<constraint firstItem="tqr-8D-x5Z" firstAttribute="centerX" secondItem="Rag-3x-Vbq" secondAttribute="centerX" id="2CR-TM-SV1"/>
<constraint firstItem="H9F-5F-FkJ" firstAttribute="centerX" secondItem="Rag-3x-Vbq" secondAttribute="centerX" id="9GK-oD-aKw"/>
<constraint firstItem="dRj-hF-Ov8" firstAttribute="leading" secondItem="Rag-3x-Vbq" secondAttribute="leading" id="9VS-vb-y46"/>
<constraint firstAttribute="trailing" secondItem="Rag-3x-Vbq" secondAttribute="trailing" id="Cau-hc-kV9"/>
<constraint firstItem="dRj-hF-Ov8" firstAttribute="bottom" secondItem="Rag-3x-Vbq" secondAttribute="bottom" id="M6f-Zs-pEW"/>
<constraint firstItem="dRj-hF-Ov8" firstAttribute="trailing" secondItem="Rag-3x-Vbq" secondAttribute="trailing" id="NIM-Pj-Jrj"/>
<constraint firstItem="dRj-hF-Ov8" firstAttribute="top" secondItem="Rag-3x-Vbq" secondAttribute="top" id="NXO-iK-dvp"/>
<constraint firstItem="Rag-3x-Vbq" firstAttribute="leading" secondItem="NHl-tq-tPi" secondAttribute="leading" id="R0T-s6-Kon"/>
<constraint firstItem="tqr-8D-x5Z" firstAttribute="bottom" secondItem="Rag-3x-Vbq" secondAttribute="top" id="a20-V8-0xV"/>
<constraint firstItem="H9F-5F-FkJ" firstAttribute="top" secondItem="Rag-3x-Vbq" secondAttribute="bottom" id="gby-3L-E2T"/>
<constraint firstAttribute="trailing" secondItem="uhT-HE-MkE" secondAttribute="trailing" id="osH-xC-Krv"/>
</constraints>
</view>
<stackView opaque="NO" contentMode="scaleToFill" axis="vertical" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="qtT-hh-JlW">
<rect key="frame" x="58" y="0.0" width="287" height="981"/>
<subviews>
<stackView opaque="NO" contentMode="top" axis="vertical" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="Vfx-Uf-SCV">
<rect key="frame" x="0.0" y="0.0" width="287" height="167"/>
<subviews>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="ful-we-pCE">
<rect key="frame" x="0.0" y="0.0" width="287" height="14.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCaption1"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
<stackView opaque="NO" contentMode="scaleToFill" spacing="4" translatesAutoresizingMaskIntoConstraints="NO" id="Cyx-R2-XUe">
<rect key="frame" x="0.0" y="22.5" width="287" height="18"/>
<subviews>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" horizontalCompressionResistancePriority="751" verticalCompressionResistancePriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="aHQ-z9-LWz">
<rect key="frame" x="0.0" y="0.0" width="43.5" height="18"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleHeadline"/>
<nil key="textColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="hrV-68-5hn">
<rect key="frame" x="47.5" y="0.0" width="198" height="18"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleSubhead"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" horizontalCompressionResistancePriority="752" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="lGC-HZ-ZT3">
<rect key="frame" x="249.5" y="0.0" width="37.5" height="18"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleSubhead"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
</subviews>
</stackView>
<stackView opaque="NO" contentMode="scaleToFill" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="P6a-zL-9B9">
<rect key="frame" x="0.0" y="48.5" width="287" height="50"/>
<subviews>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="oXX-vk-ESS">
<rect key="frame" x="0.0" y="0.0" width="50" height="50"/>
<subviews>
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFit" horizontalHuggingPriority="251" verticalHuggingPriority="251" translatesAutoresizingMaskIntoConstraints="NO" id="BLe-zN-EsU" customClass="AnimatedImageView" customModule="Kingfisher">
<rect key="frame" x="0.0" y="0.0" width="50" height="50"/>
<constraints>
<constraint firstAttribute="width" constant="50" id="1bA-D3-I4a"/>
<constraint firstAttribute="height" constant="50" id="G3i-Y5-C3e"/>
</constraints>
<userDefinedRuntimeAttributes>
<userDefinedRuntimeAttribute type="number" keyPath="layer.cornerRadius">
<integer key="value" value="25"/>
</userDefinedRuntimeAttribute>
</userDefinedRuntimeAttributes>
</imageView>
<button opaque="NO" clipsSubviews="YES" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="aNl-ag-2fc">
<rect key="frame" x="0.0" y="0.0" width="50" height="50"/>
<userDefinedRuntimeAttributes>
<userDefinedRuntimeAttribute type="number" keyPath="layer.cornerRadius">
<integer key="value" value="25"/>
</userDefinedRuntimeAttribute>
</userDefinedRuntimeAttributes>
<connections>
<action selector="avatarButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="S3W-K2-aSY"/>
</connections>
</button>
</subviews>
<color key="backgroundColor" white="0.0" alpha="0.0" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/>
<constraints>
<constraint firstAttribute="bottom" secondItem="BLe-zN-EsU" secondAttribute="bottom" id="0vg-93-dFw"/>
<constraint firstItem="aNl-ag-2fc" firstAttribute="leading" secondItem="BLe-zN-EsU" secondAttribute="leading" id="1Mr-NZ-A1g"/>
<constraint firstItem="BLe-zN-EsU" firstAttribute="top" secondItem="oXX-vk-ESS" secondAttribute="top" id="5vR-d6-1ne"/>
<constraint firstItem="BLe-zN-EsU" firstAttribute="leading" secondItem="oXX-vk-ESS" secondAttribute="leading" id="G4T-oU-MdP"/>
<constraint firstItem="aNl-ag-2fc" firstAttribute="bottom" secondItem="BLe-zN-EsU" secondAttribute="bottom" id="HPR-zH-Wia"/>
<constraint firstItem="aNl-ag-2fc" firstAttribute="trailing" secondItem="BLe-zN-EsU" secondAttribute="trailing" id="Xx3-36-S6g"/>
<constraint firstAttribute="trailing" secondItem="BLe-zN-EsU" secondAttribute="trailing" id="baJ-by-gCl"/>
<constraint firstItem="aNl-ag-2fc" firstAttribute="top" secondItem="BLe-zN-EsU" secondAttribute="top" id="zdp-yv-BHJ"/>
</constraints>
</view>
<stackView opaque="NO" contentMode="scaleToFill" axis="vertical" translatesAutoresizingMaskIntoConstraints="NO" id="mYD-9U-oFJ">
<rect key="frame" x="58" y="0.0" width="229" height="50"/>
<subviews>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="ya9-zV-dMO">
<rect key="frame" x="0.0" y="0.0" width="229" height="32"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleHeadline"/>
<nil key="textColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="252" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="aa2-u6-YBQ">
<rect key="frame" x="0.0" y="32" width="229" height="18"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleSubhead"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
</subviews>
</stackView>
</subviews>
</stackView>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" numberOfLines="0" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="7Mz-X2-wmw">
<rect key="frame" x="0.0" y="106.5" width="287" height="19.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCallout"/>
<nil key="textColor"/>
<nil key="highlightedColor"/>
</label>
<button opaque="NO" contentMode="scaleToFill" verticalHuggingPriority="251" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="Occ-DT-kxh">
<rect key="frame" x="0.0" y="134" width="287" height="33"/>
<color key="backgroundColor" systemColor="linkColor"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleHeadline"/>
<state key="normal" title="Show More">
<color key="titleColor" white="1" alpha="1" colorSpace="custom" customColorSpace="genericGamma22GrayColorSpace"/>
</state>
<connections>
<action selector="toggleSensitiveContentButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="m7M-MA-f02"/>
</connections>
</button>
</subviews>
</stackView>
<stackView opaque="NO" contentMode="scaleToFill" verticalCompressionResistancePriority="749" axis="vertical" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="Rdc-ZD-zdc">
<rect key="frame" x="0.0" y="175" width="287" height="654.5"/>
<subviews>
<textView clipsSubviews="YES" multipleTouchEnabled="YES" contentMode="scaleToFill" verticalCompressionResistancePriority="751" scrollEnabled="NO" editable="NO" text="Content" textAlignment="natural" adjustsFontForContentSizeCategory="YES" translatesAutoresizingMaskIntoConstraints="NO" id="GvW-gH-ofw" customClass="TouchFallthroughTextView" customModule="Metatext" customModuleProvider="target">
<rect key="frame" x="0.0" y="0.0" width="287" height="345"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCallout"/>
<textInputTraits key="textInputTraits" autocapitalizationType="sentences"/>
</textView>
<view clipsSubviews="YES" contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="gCY-Qy-hbS">
<rect key="frame" x="0.0" y="353" width="287" height="301.5"/>
<subviews>
<stackView opaque="NO" contentMode="scaleToFill" axis="vertical" translatesAutoresizingMaskIntoConstraints="NO" id="3C9-uk-wu2">
<rect key="frame" x="0.0" y="0.0" width="287" height="301.5"/>
<subviews>
<imageView clipsSubviews="YES" userInteractionEnabled="NO" contentMode="scaleAspectFill" horizontalHuggingPriority="251" verticalHuggingPriority="251" translatesAutoresizingMaskIntoConstraints="NO" id="9jB-zW-tJB">
<rect key="frame" x="0.0" y="0.0" width="287" height="215"/>
<constraints>
<constraint firstAttribute="width" secondItem="9jB-zW-tJB" secondAttribute="height" multiplier="4:3" priority="999" id="ioh-Zj-QGD"/>
</constraints>
</imageView>
<stackView opaque="NO" contentMode="scaleToFill" axis="vertical" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="V94-ed-vFr">
<rect key="frame" x="0.0" y="215" width="287" height="86.5"/>
<subviews>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="252" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" numberOfLines="0" baselineAdjustment="alignBaselines" minimumScaleFactor="0.5" adjustsFontForContentSizeCategory="YES" translatesAutoresizingMaskIntoConstraints="NO" id="dS1-ny-kA1">
<rect key="frame" x="8" y="8" width="271" height="20.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleHeadline"/>
<nil key="textColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="RAj-xv-7Sr">
<rect key="frame" x="8" y="36.5" width="271" height="18"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleSubhead"/>
<nil key="textColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="IhY-1O-KAR">
<rect key="frame" x="8" y="62.5" width="271" height="16"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleFootnote"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
</subviews>
<directionalEdgeInsets key="directionalLayoutMargins" top="8" leading="8" bottom="8" trailing="8"/>
</stackView>
</subviews>
</stackView>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="6pk-EO-aa0">
<rect key="frame" x="0.0" y="0.0" width="287" height="301.5"/>
<connections>
<action selector="cardButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="0L8-TA-GxD"/>
</connections>
</button>
</subviews>
<color key="backgroundColor" systemColor="secondarySystemBackgroundColor"/>
<constraints>
<constraint firstItem="6pk-EO-aa0" firstAttribute="leading" secondItem="gCY-Qy-hbS" secondAttribute="leading" id="4oF-J2-goz"/>
<constraint firstAttribute="trailing" secondItem="3C9-uk-wu2" secondAttribute="trailing" id="6in-BD-bgA"/>
<constraint firstItem="3C9-uk-wu2" firstAttribute="leading" secondItem="gCY-Qy-hbS" secondAttribute="leading" id="BRL-4d-akw"/>
<constraint firstAttribute="trailing" secondItem="6pk-EO-aa0" secondAttribute="trailing" id="LCI-Ey-TJ0"/>
<constraint firstItem="6pk-EO-aa0" firstAttribute="top" secondItem="gCY-Qy-hbS" secondAttribute="top" id="XYJ-Zq-6iT"/>
<constraint firstItem="3C9-uk-wu2" firstAttribute="top" secondItem="gCY-Qy-hbS" secondAttribute="top" id="dAK-CM-V0l"/>
<constraint firstAttribute="bottom" secondItem="6pk-EO-aa0" secondAttribute="bottom" id="e4U-OB-8hi"/>
<constraint firstAttribute="bottom" secondItem="3C9-uk-wu2" secondAttribute="bottom" id="uvq-Pb-NhR"/>
</constraints>
<userDefinedRuntimeAttributes>
<userDefinedRuntimeAttribute type="number" keyPath="layer.cornerRadius">
<integer key="value" value="8"/>
</userDefinedRuntimeAttribute>
</userDefinedRuntimeAttributes>
</view>
</subviews>
</stackView>
<stackView opaque="NO" contentMode="scaleToFill" distribution="fillEqually" translatesAutoresizingMaskIntoConstraints="NO" id="wFL-zS-Rev">
<rect key="frame" x="0.0" y="837.5" width="287" height="18.5"/>
<subviews>
<button opaque="NO" contentMode="scaleToFill" verticalHuggingPriority="251" contentHorizontalAlignment="leading" contentVerticalAlignment="center" adjustsImageSizeForAccessibilityContentSizeCategory="YES" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="1mc-Dy-v2f">
<rect key="frame" x="0.0" y="0.0" width="72" height="18.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCaption2"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<inset key="titleEdgeInsets" minX="4" minY="0.0" maxX="0.0" maxY="0.0"/>
<state key="normal" title="69" image="bubble.right" catalog="system">
<color key="titleColor" systemColor="secondaryLabelColor"/>
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="small"/>
</state>
<connections>
<action selector="replyButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="fIh-jy-zyy"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" verticalHuggingPriority="251" contentHorizontalAlignment="leading" contentVerticalAlignment="center" adjustsImageSizeForAccessibilityContentSizeCategory="YES" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="25b-IE-HZK">
<rect key="frame" x="72" y="0.0" width="71.5" height="18.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCaption2"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<inset key="titleEdgeInsets" minX="4" minY="0.0" maxX="0.0" maxY="0.0"/>
<state key="normal" title="69" image="arrow.2.squarepath" catalog="system">
<color key="titleColor" systemColor="secondaryLabelColor"/>
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="small"/>
</state>
<connections>
<action selector="reblogButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="2od-kC-4K5"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" verticalHuggingPriority="251" contentHorizontalAlignment="leading" contentVerticalAlignment="center" adjustsImageSizeForAccessibilityContentSizeCategory="YES" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="7f5-WO-gpi">
<rect key="frame" x="143.5" y="0.0" width="72" height="18.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCaption2"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<inset key="titleEdgeInsets" minX="4" minY="0.0" maxX="0.0" maxY="0.0"/>
<state key="normal" title="69" image="star" catalog="system">
<color key="titleColor" systemColor="secondaryLabelColor"/>
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="small"/>
</state>
<connections>
<action selector="favoriteButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="hyh-wI-9WM"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" verticalHuggingPriority="251" contentHorizontalAlignment="center" contentVerticalAlignment="center" adjustsImageSizeForAccessibilityContentSizeCategory="YES" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="nrc-xp-Llw">
<rect key="frame" x="215.5" y="0.0" width="71.5" height="18.5"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleCaption2"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="square.and.arrow.up" catalog="system">
<color key="titleColor" systemColor="secondaryLabelColor"/>
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="small"/>
</state>
<connections>
<action selector="shareButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="Owv-iB-VxJ"/>
</connections>
</button>
</subviews>
</stackView>
<stackView opaque="NO" contentMode="scaleToFill" axis="vertical" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="Fhw-OW-ixv">
<rect key="frame" x="0.0" y="864" width="287" height="117"/>
<subviews>
<stackView opaque="NO" contentMode="scaleToFill" spacing="4" translatesAutoresizingMaskIntoConstraints="NO" id="xvi-yh-xAX">
<rect key="frame" x="0.0" y="0.0" width="287" height="28"/>
<subviews>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Label" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="hRM-1y-2pU">
<rect key="frame" x="0.0" y="0.0" width="33" height="28"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleFootnote"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
<label opaque="NO" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="•" textAlignment="natural" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" adjustsFontForContentSizeCategory="YES" adjustsFontSizeToFit="NO" translatesAutoresizingMaskIntoConstraints="NO" id="NkD-O9-e9D">
<rect key="frame" x="37" y="0.0" width="6.5" height="28"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleFootnote"/>
<color key="textColor" systemColor="secondaryLabelColor"/>
<nil key="highlightedColor"/>
</label>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="leading" contentVerticalAlignment="center" buttonType="system" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="hkk-pX-8oP">
<rect key="frame" x="47.5" y="0.0" width="239.5" height="28"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleFootnote"/>
<state key="normal" title="Button"/>
<state key="disabled">
<color key="titleColor" systemColor="secondaryLabelColor"/>
</state>
<connections>
<action selector="applicationButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="yAU-hM-cUP"/>
</connections>
</button>
</subviews>
</stackView>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="4ZR-31-F0G">
<rect key="frame" x="0.0" y="36" width="287" height="1"/>
<color key="backgroundColor" systemColor="separatorColor"/>
<constraints>
<constraint firstAttribute="height" constant="1" id="jaw-Uw-Xby"/>
</constraints>
</view>
<stackView opaque="NO" contentMode="scaleToFill" distribution="fillEqually" translatesAutoresizingMaskIntoConstraints="NO" id="VJg-8P-ax9">
<rect key="frame" x="0.0" y="45" width="287" height="33"/>
<subviews>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="leading" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="pqY-NQ-5Fz">
<rect key="frame" x="0.0" y="0.0" width="143.5" height="33"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleBody"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" title="666">
<color key="titleColor" systemColor="secondaryLabelColor"/>
</state>
<state key="highlighted">
<color key="titleColor" systemColor="tertiaryLabelColor"/>
</state>
<connections>
<action selector="contextParentRebloggedByButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="X5h-c5-5Yq"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="leading" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="Olw-ZF-pYB">
<rect key="frame" x="143.5" y="0.0" width="143.5" height="33"/>
<fontDescription key="fontDescription" style="UICTFontTextStyleBody"/>
<state key="normal" title="666">
<color key="titleColor" systemColor="secondaryLabelColor"/>
</state>
<state key="highlighted">
<color key="titleColor" systemColor="tertiaryLabelColor"/>
</state>
<connections>
<action selector="contextParentFavoritedByButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="eFA-jK-7pJ"/>
</connections>
</button>
</subviews>
</stackView>
<view contentMode="scaleToFill" translatesAutoresizingMaskIntoConstraints="NO" id="SlW-Ii-ozK">
<rect key="frame" x="0.0" y="86" width="287" height="1"/>
<color key="backgroundColor" systemColor="separatorColor"/>
<constraints>
<constraint firstAttribute="height" constant="1" id="DoC-rC-wYG"/>
</constraints>
</view>
<stackView opaque="NO" contentMode="scaleToFill" distribution="fillEqually" spacing="8" translatesAutoresizingMaskIntoConstraints="NO" id="XOe-fg-wO6">
<rect key="frame" x="0.0" y="95" width="287" height="22"/>
<subviews>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="38F-f5-8Ha">
<rect key="frame" x="0.0" y="0.0" width="51" height="22"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="bubble.right" catalog="system">
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="medium"/>
</state>
<connections>
<action selector="replyButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="jkz-ad-n2Y"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="Rjo-bC-85y">
<rect key="frame" x="59" y="0.0" width="51" height="22"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="arrow.2.squarepath" catalog="system">
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="medium"/>
</state>
<connections>
<action selector="reblogButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="hYp-sm-i4K"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="Q79-pC-6ru">
<rect key="frame" x="118" y="0.0" width="51" height="22"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="star" catalog="system">
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="medium"/>
</state>
<connections>
<action selector="favoriteButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="NdC-VQ-9ZD"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="5Ft-M2-BMM">
<rect key="frame" x="177" y="0.0" width="51" height="22"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="square.and.arrow.up" catalog="system">
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="medium"/>
</state>
<connections>
<action selector="shareButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="hUE-mZ-QST"/>
</connections>
</button>
<button opaque="NO" contentMode="scaleToFill" contentHorizontalAlignment="center" contentVerticalAlignment="center" lineBreakMode="middleTruncation" translatesAutoresizingMaskIntoConstraints="NO" id="Z1Y-Ch-xB7">
<rect key="frame" x="236" y="0.0" width="51" height="22"/>
<color key="tintColor" systemColor="secondaryLabelColor"/>
<state key="normal" image="ellipsis" catalog="system">
<preferredSymbolConfiguration key="preferredSymbolConfiguration" scale="medium"/>
</state>
<connections>
<action selector="actionsButtonTapped:" destination="KGk-i7-Jjw" eventType="touchUpInside" id="7qm-km-Ydt"/>
</connections>
</button>
</subviews>
</stackView>
</subviews>
</stackView>
</subviews>
</stackView>
</subviews>
<constraints>
<constraint firstItem="uhT-HE-MkE" firstAttribute="centerY" secondItem="ful-we-pCE" secondAttribute="centerY" id="5rP-PQ-irT"/>
<constraint firstItem="Rag-3x-Vbq" firstAttribute="top" secondItem="Cyx-R2-XUe" secondAttribute="top" constant="4" id="8h6-s3-iOk"/>
</constraints>
</stackView>
</subviews>
<constraints>
<constraint firstAttribute="trailingMargin" secondItem="PuM-4R-mah" secondAttribute="trailing" id="7TB-kA-Oo4"/>
<constraint firstItem="PuM-4R-mah" firstAttribute="top" secondItem="H2p-sc-9uM" secondAttribute="topMargin" id="81w-ID-ymP"/>
<constraint firstAttribute="bottomMargin" secondItem="PuM-4R-mah" secondAttribute="bottom" id="DB9-xL-qNI"/>
<constraint firstItem="PuM-4R-mah" firstAttribute="leading" secondItem="H2p-sc-9uM" secondAttribute="leadingMargin" id="VY2-YX-92L"/>
<constraint firstItem="tqr-8D-x5Z" firstAttribute="top" secondItem="H2p-sc-9uM" secondAttribute="top" id="YwS-sf-Z70"/>
<constraint firstAttribute="bottom" secondItem="H9F-5F-FkJ" secondAttribute="bottom" id="bWC-zo-sgU"/>
</constraints>
</tableViewCellContentView>
<viewLayoutGuide key="safeArea" id="njF-e1-oar"/>
<connections>
<outlet property="accountLabel" destination="hrV-68-5hn" id="Kn4-pt-nQa"/>
<outlet property="actionButtonsView" destination="wFL-zS-Rev" id="h7S-r0-vMa"/>
<outlet property="applicationButton" destination="hkk-pX-8oP" id="tHC-cy-B25"/>
<outlet property="avatarButton" destination="dRj-hF-Ov8" id="1aD-hK-1iz"/>
<outlet property="avatarImageView" destination="Rag-3x-Vbq" id="BaZ-ad-Qgs"/>
<outlet property="avatarReplyContextView" destination="NHl-tq-tPi" id="CCJ-SE-Egv"/>
<outlet property="cardButton" destination="6pk-EO-aa0" id="5SC-dt-7P7"/>
<outlet property="cardDescriptionLabel" destination="RAj-xv-7Sr" id="Ph4-KM-qED"/>
<outlet property="cardImageView" destination="9jB-zW-tJB" id="0Vr-xC-rIa"/>
<outlet property="cardTitleLabel" destination="dS1-ny-kA1" id="hYw-f7-WrH"/>
<outlet property="cardURLLabel" destination="IhY-1O-KAR" id="9Cl-2P-Z7x"/>
<outlet property="cardView" destination="gCY-Qy-hbS" id="u7g-iK-qhw"/>
<outlet property="contentTextView" destination="GvW-gH-ofw" id="sVs-dt-0Zt"/>
<outlet property="contextParentAccountLabel" destination="aa2-u6-YBQ" id="NeO-aF-DPZ"/>
<outlet property="contextParentActionsButton" destination="Z1Y-Ch-xB7" id="7S4-vc-gKe"/>
<outlet property="contextParentAvatarButton" destination="aNl-ag-2fc" id="Tah-tr-ckM"/>
<outlet property="contextParentAvatarImageView" destination="BLe-zN-EsU" id="99c-tt-zvI"/>
<outlet property="contextParentAvatarNameView" destination="P6a-zL-9B9" id="eGx-i7-i1a"/>
<outlet property="contextParentDisplayNameLabel" destination="ya9-zV-dMO" id="5QN-x6-V5E"/>
<outlet property="contextParentFavoriteButton" destination="Q79-pC-6ru" id="pZI-Yh-Tbh"/>
<outlet property="contextParentFavoritedByButton" destination="Olw-ZF-pYB" id="AwZ-wP-X55"/>
<outlet property="contextParentItems" destination="Fhw-OW-ixv" id="wws-EA-hjY"/>
<outlet property="contextParentReblogButton" destination="Rjo-bC-85y" id="hGu-h5-nbh"/>
<outlet property="contextParentRebloggedByButton" destination="pqY-NQ-5Fz" id="Cd9-Lf-lTr"/>
<outlet property="contextParentRebloggedByFavoritedBySeparator" destination="SlW-Ii-ozK" id="bKO-mz-IIW"/>
<outlet property="contextParentRebloggedByFavoritedByView" destination="VJg-8P-ax9" id="fzs-Fe-m0t"/>
<outlet property="contextParentReplyButton" destination="38F-f5-8Ha" id="bwI-S3-ZUM"/>
<outlet property="contextParentShareButton" destination="5Ft-M2-BMM" id="EC4-Xj-qcK"/>
<outlet property="contextParentTimeLabel" destination="hRM-1y-2pU" id="nwb-rf-gXn"/>
<outlet property="displayNameLabel" destination="aHQ-z9-LWz" id="v3P-4w-7mg"/>
<outlet property="favoriteButton" destination="7f5-WO-gpi" id="VvN-cW-anv"/>
<outlet property="hasReplyFollowingView" destination="H9F-5F-FkJ" id="abM-Ye-pzJ"/>
<outlet property="inReplyToView" destination="tqr-8D-x5Z" id="l7j-I4-vLJ"/>
<outlet property="metaIcon" destination="uhT-HE-MkE" id="1z9-PN-DvJ"/>
<outlet property="metaLabel" destination="ful-we-pCE" id="FtP-We-cCj"/>
<outlet property="nameDateView" destination="Cyx-R2-XUe" id="FGc-TV-Ufc"/>
<outlet property="reblogButton" destination="25b-IE-HZK" id="iZD-V3-d3T"/>
<outlet property="replyButton" destination="1mc-Dy-v2f" id="p07-xE-ZUI"/>
<outlet property="sensitiveContentView" destination="Rdc-ZD-zdc" id="yAX-uL-QH0"/>
<outlet property="shareButton" destination="nrc-xp-Llw" id="VeC-Mh-brV"/>
<outlet property="spoilerTextLabel" destination="7Mz-X2-wmw" id="7OY-Oa-Lhk"/>
<outlet property="timeApplicationDividerView" destination="NkD-O9-e9D" id="ygX-on-Xqu"/>
<outlet property="timeLabel" destination="lGC-HZ-ZT3" id="4Xu-ia-eDI"/>
<outlet property="toggleSensitiveContentButton" destination="Occ-DT-kxh" id="yd4-xI-94j"/>
<outletCollection property="separatorConstraints" destination="jaw-Uw-Xby" collectionClass="NSMutableArray" id="OvM-ag-v1s"/>
<outletCollection property="separatorConstraints" destination="DoC-rC-wYG" collectionClass="NSMutableArray" id="6NT-uI-Td4"/>
</connections>
<point key="canvasLocation" x="340" y="-525"/>
</tableViewCell>
</objects>
<resources>
<image name="arrow.2.squarepath" catalog="system" width="128" height="89"/>
<image name="bubble.right" catalog="system" width="128" height="110"/>
<image name="ellipsis" catalog="system" width="128" height="37"/>
<image name="square.and.arrow.up" catalog="system" width="115" height="128"/>
<image name="star" catalog="system" width="128" height="116"/>
<systemColor name="linkColor">
<color red="0.0" green="0.47843137254901963" blue="1" alpha="1" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
<systemColor name="quaternaryLabelColor">
<color red="0.23529411764705882" green="0.23529411764705882" blue="0.2627450980392157" alpha="0.17999999999999999" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
<systemColor name="secondaryLabelColor">
<color red="0.23529411764705882" green="0.23529411764705882" blue="0.2627450980392157" alpha="0.59999999999999998" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
<systemColor name="secondarySystemBackgroundColor">
<color red="0.94901960784313721" green="0.94901960784313721" blue="0.96862745098039216" alpha="1" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
<systemColor name="separatorColor">
<color red="0.23529411764705882" green="0.23529411764705882" blue="0.2627450980392157" alpha="0.28999999999999998" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
<systemColor name="tertiaryLabelColor">
<color red="0.23529411764705882" green="0.23529411764705882" blue="0.2627450980392157" alpha="0.29999999999999999" colorSpace="custom" customColorSpace="sRGB"/>
</systemColor>
</resources>
</document>

View file

@ -40,7 +40,7 @@ private extension TabNavigationView {
func view(tab: TabNavigationViewModel.Tab) -> some View {
switch tab {
case .timelines:
StatusesView(viewModel: viewModel.timelineViewModel)
StatusListView(viewModel: viewModel.timelineViewModel)
.navigationBarTitle(viewModel.identity.handle, displayMode: .inline)
.navigationBarItems(
leading: Button {

View file

@ -0,0 +1,100 @@
// Copyright © 2020 Metabolist. All rights reserved.
import UIKit
class TouchFallthroughTextView: UITextView {
var shouldFallthrough: Bool = true
override init(frame: CGRect, textContainer: NSTextContainer?) {
super.init(frame: frame, textContainer: textContainer)
initializationActions()
}
required init?(coder: NSCoder) {
super.init(coder: coder)
initializationActions()
}
override func point(inside point: CGPoint, with event: UIEvent?) -> Bool {
shouldFallthrough ? urlAndRect(at: point) != nil : super.point(inside: point, with: event)
}
override var selectedTextRange: UITextRange? {
get { shouldFallthrough ? nil : super.selectedTextRange }
set {
if !shouldFallthrough {
super.selectedTextRange = newValue
}
}
}
override var intrinsicContentSize: CGSize {
return text == "" ? .zero : super.intrinsicContentSize
}
func urlAndRect(at point: CGPoint) -> (URL, CGRect)? {
guard
let pos = closestPosition(to: point),
let range = tokenizer.rangeEnclosingPosition(
pos, with: .character,
inDirection: UITextDirection.layout(.left))
else { return nil }
let urlAtPointIndex = offset(from: beginningOfDocument, to: range.start)
guard let url = attributedText.attribute(
.link, at: offset(from: beginningOfDocument, to: range.start),
effectiveRange: nil) as? URL
else { return nil }
let maxLength = attributedText.length
var min = urlAtPointIndex
var max = urlAtPointIndex
attributedText.enumerateAttribute(
.link,
in: NSRange(location: 0, length: urlAtPointIndex),
options: .reverse) { attribute, range, stop in
if let attributeURL = attribute as? URL, attributeURL == url, min > 0 {
min = range.location
} else {
stop.pointee = true
}
}
attributedText.enumerateAttribute(
.link,
in: NSRange(location: urlAtPointIndex, length: maxLength - urlAtPointIndex),
options: []) { attribute, range, stop in
if let attributeURL = attribute as? URL, attributeURL == url, max < maxLength {
max = range.location + range.length
} else {
stop.pointee = true
}
}
var urlRect = CGRect.zero
layoutManager.enumerateEnclosingRects(
forGlyphRange: NSRange(location: min, length: max - min),
withinSelectedGlyphRange: NSRange(location: NSNotFound, length: 0),
in: textContainer) { rect, _ in
if urlRect.origin == .zero {
urlRect.origin = rect.origin
}
urlRect = urlRect.union(rect)
}
return (url, urlRect)
}
}
private extension TouchFallthroughTextView {
private func initializationActions() {
textDragInteraction?.isEnabled = false
textContainerInset = .zero
textContainer.lineFragmentPadding = 0
linkTextAttributes = [.foregroundColor: tintColor as Any, .underlineColor: UIColor.clear]
}
}