|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 |
-
- #import "Apollo.h"
- #import "assets/TFHelper.h"
-
- static BOOL isEnabled;
- static BOOL isApolloEnabled;
- static BOOL isTFDeletedOnly;
- static CGFloat pushshiftRequestTimeoutValue;
- static BOOL shouldApolloHaveButton;
-
- %group Apollo
-
- NSDictionary* apolloBodyAttributes = nil;
- BOOL shouldAddUndeleteCell = NO;
- id apolloCommentCell;
- id apolloCommentController;
-
- %hook ApolloButtonNode
- %end
-
- %hook IconActionTableViewCell
- %end
-
-
- %hook RKComment
-
- - (BOOL)isDeleted {
- return NO;
- }
-
- - (BOOL)isModeratorRemoved {
- return NO;
- }
-
- %end
-
- //1.8+, unsure why this is all I needed to add for 1.8 support, the rest of this still works even w/o changing RKComment and RKLink references
- %hook RDKComment
-
- - (BOOL)isDeleted {
- return NO;
- }
-
- - (BOOL)isModeratorRemoved {
- return NO;
- }
-
- %end
-
-
- %hook RKLink
- %property(strong, nonatomic) NSString *undeleteAuthor;
-
- - (id)author {
- return [self undeleteAuthor] ? [self undeleteAuthor] : %orig;
- }
-
- %end
-
-
- %hook RDKLink
- %property(strong, nonatomic) NSString *undeleteAuthor;
-
- - (id)author {
- return [self undeleteAuthor] ? [self undeleteAuthor] : %orig;
- }
-
- %end
-
-
- %hook MarkdownRenderer
-
- + (id)attributedStringFromHTML:(id)arg1 attributes:(id)arg2 compact:(BOOL)arg3 {
-
- apolloBodyAttributes = [arg2 copy];
-
- return %orig;
- }
-
- %end
-
-
- %hook ActionController
-
- - (id) tableView:(id)arg1 cellForRowAtIndexPath:(NSIndexPath *)arg2 {
-
- if (shouldAddUndeleteCell) {
- if ([arg2 row] == [self tableView:arg1 numberOfRowsInSection:0] - 1){
-
- id undeleteCell = [arg1 dequeueReusableCellWithIdentifier:@"IconActionCell" forIndexPath:arg2];
- id prevCell = [arg1 dequeueReusableCellWithIdentifier:@"IconActionCell"];
-
- UIImageView *prevCellImageView = MSHookIvar<UIImageView *>(prevCell, "iconImageView");
- CGSize prevImageSize = [[prevCellImageView image] size];
-
- UIImage *undeleteImage;
-
- //if (@available(iOS 13.0, *)){
- if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"13.0")) {
-
- undeleteImage = [UIImage systemImageNamed:@"eye"];
-
- if (!undeleteImage){
- undeleteImage = [UIImage imageWithContentsOfFile:@"/var/mobile/Library/Application Support/TFDidThatSay/eye160dark.png"];
- } else {
- CGSize squareSize = CGSizeMake(undeleteImage.size.width, undeleteImage.size.width);
-
- UIGraphicsBeginImageContextWithOptions(squareSize, NO, 0);
- [undeleteImage drawInRect:CGRectMake(0, (squareSize.height - undeleteImage.size.height) / 2, squareSize.width, undeleteImage.size.height)];
- undeleteImage = UIGraphicsGetImageFromCurrentImageContext();
- UIGraphicsEndImageContext();
- }
- } else {
- undeleteImage = [UIImage imageWithContentsOfFile:@"/var/mobile/Library/Application Support/TFDidThatSay/eye160dark.png"];
- }
-
- CGFloat undeleteImageSizeValue = prevImageSize.width > prevImageSize.height ? prevImageSize.width : prevImageSize.height;
-
- if (undeleteImageSizeValue == 0) {
- undeleteImageSizeValue = 25;
- }
-
- UIGraphicsBeginImageContextWithOptions(CGSizeMake(undeleteImageSizeValue, undeleteImageSizeValue), NO, 0);
- [undeleteImage drawInRect:CGRectMake(0, 0, undeleteImageSizeValue, undeleteImageSizeValue)];
- undeleteImage = [UIGraphicsGetImageFromCurrentImageContext() imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
- UIGraphicsEndImageContext();
-
- UILabel *undeleteLabel = MSHookIvar<UILabel *>(undeleteCell, "actionTitleLabel");
- UIImageView *undeleteImageView = MSHookIvar<UIImageView *>(undeleteCell, "iconImageView");
-
- undeleteLabel.text = @"TF Did That Say?";
- undeleteImageView.image = undeleteImage;
-
- return undeleteCell;
- }
- }
-
- return %orig;
- }
-
- - (void)tableView:(id)arg1 didSelectRowAtIndexPath:(NSIndexPath *)arg2 {
-
- if (shouldAddUndeleteCell) {
- if ([arg2 row] == [self tableView:arg1 numberOfRowsInSection:0] - 1){
-
- if (apolloCommentCell) {
- [apolloCommentCell undeleteCellWasSelected];
- } else {
- [apolloCommentController undeleteCellWasSelected];
- }
- }
- }
-
- %orig;
- }
-
- - (NSInteger) tableView:(id)arg1 numberOfRowsInSection:(NSInteger)arg2 {
-
- if (shouldAddUndeleteCell){
- return %orig + 1;
- } else {
- return %orig;
- }
- }
-
- - (id) animationControllerForDismissedController:(id)arg1 {
-
- shouldAddUndeleteCell = NO;
-
- return %orig;
- }
-
- %end
-
-
- %hook CommentCellNode
- %property(strong,nonatomic) UIButton *undeleteButton;
-
- - (void)moreOptionsTappedWithSender:(id)arg1 {
-
- if (!shouldApolloHaveButton){
-
- NSString *commentBody = [MSHookIvar<RKComment *>(self, "comment") body];
-
- if ([%c(TFHelper) shouldShowUndeleteButtonWithInfo:commentBody isDeletedOnly:isTFDeletedOnly]){
- shouldAddUndeleteCell = YES;
- apolloCommentCell = self;
- apolloCommentController = nil;
- }
- }
-
- %orig;
- }
-
- - (void)longPressedWithGestureRecognizer:(id)arg1 {
-
- if (!shouldApolloHaveButton){
-
- NSString *commentBody = [MSHookIvar<RKComment *>(self, "comment") body];
-
- if ([%c(TFHelper) shouldShowUndeleteButtonWithInfo:commentBody isDeletedOnly:isTFDeletedOnly]){
- shouldAddUndeleteCell = YES;
- apolloCommentCell = self;
- apolloCommentController = nil;
- }
- }
-
- %orig;
- }
-
- - (void)didLoad {
- %orig;
-
- if (shouldApolloHaveButton){
-
- NSString *commentBody = [MSHookIvar<RKComment *>(self, "comment") body];
-
- if ([%c(TFHelper) shouldShowUndeleteButtonWithInfo:commentBody isDeletedOnly:isTFDeletedOnly]){
-
- CGFloat imageSize = 20.0f;
-
- UIButton *undeleteButton = [UIButton buttonWithType:UIButtonTypeCustom];
- [undeleteButton addTarget:self action:@selector(didTapUndeleteButton:) forControlEvents:UIControlEventTouchUpInside];
- undeleteButton.frame = CGRectMake(0, 0, imageSize, imageSize);
-
- UIImage* undeleteImage = [UIImage imageWithContentsOfFile:@"/var/mobile/Library/Application Support/TFDidThatSay/eye160dark.png"];
- [undeleteButton setImage:undeleteImage forState:UIControlStateNormal];
-
- [[self view] addSubview:undeleteButton];
- [self setUndeleteButton:undeleteButton];
- }
- }
- }
-
- - (void)_layoutSublayouts {
- %orig;
-
- if (shouldApolloHaveButton){
- if ([self undeleteButton]){
-
- CGFloat imageSize = 20.0f;
-
- id moreNode = MSHookIvar<id>(self, "moreOptionsNode");
- id ageNode = MSHookIvar<id>(self, "ageNode");
-
- CGRect nodeFrame = [moreNode frame];
- CGFloat centerHeight = (nodeFrame.size.height + nodeFrame.origin.y * 2) / 2.0f;
- CGFloat nodeSpacing = [ageNode frame].origin.x - nodeFrame.origin.x - nodeFrame.size.width;
-
- [[self undeleteButton] setFrame:CGRectMake(nodeFrame.origin.x - imageSize - nodeSpacing, centerHeight - (imageSize / 2), imageSize, imageSize)];
- }
- }
- }
-
- %new
- - (void)didTapUndeleteButton:(id)sender {
-
- [sender setEnabled:NO];
-
- id comment = MSHookIvar<id>(self, "comment");
-
- [%c(TFHelper) getUndeleteDataWithID:[[comment fullName] componentsSeparatedByString:@"_"][1] isComment:YES timeout:pushshiftRequestTimeoutValue extraData:@{@"sender" : sender} completionTarget:self completionSelector:@selector(completeUndeleteCommentAction:)];
- }
-
- %new
- - (void)undeleteCellWasSelected {
-
- RKComment *comment = MSHookIvar<RKComment *>(self, "comment");
-
- [%c(TFHelper) getUndeleteDataWithID:[[comment fullName] componentsSeparatedByString:@"_"][1] isComment:YES timeout:pushshiftRequestTimeoutValue extraData:nil completionTarget:self completionSelector:@selector(completeUndeleteCommentAction:)];
- }
-
- %new
- - (void)completeUndeleteCommentAction:(NSDictionary *)data {
-
- RKComment *comment = MSHookIvar<RKComment *>(self, "comment");
- id bodyNode = MSHookIvar<id>(self, "bodyNode");
- id authorNode = MSHookIvar<id>(self, "authorNode");
-
- NSString *author = data[@"author"];
- NSString *body = data[@"body"];
-
- [comment setAuthor:author];
- [comment setBody:body];
-
- NSAttributedString *prevAuthorAttributedString = [authorNode attributedTitleForState:UIControlStateNormal];
- NSDictionary *authorStringAttributes = [prevAuthorAttributedString attributesAtIndex:0 longestEffectiveRange:nil inRange:NSMakeRange(0, [prevAuthorAttributedString length])];
- NSAttributedString *newAuthorAttributedString = [[NSAttributedString alloc] initWithString:author attributes:authorStringAttributes];
-
- [authorNode setAttributedTitle:newAuthorAttributedString forState:UIControlStateNormal];
- [bodyNode setAttributedString:[%c(MarkdownRenderer) attributedStringFromMarkdown:body withAttributes:apolloBodyAttributes]];
-
- if ([data objectForKey:@"sender"]) {
- [data[@"sender"] setEnabled:YES];
- }
- }
-
- %end
-
-
- %hook CommentsViewController
- %property(strong, nonatomic) id headerCellNode;
-
- - (void)moreOptionsBarButtonItemTappedWithSender:(id)arg1 {
-
- RKLink *post = MSHookIvar<RKLink *>(self, "link");
- NSString *postBody = [post selfText];
-
- if ([post isSelfPost]){
- if ([%c(TFHelper) shouldShowUndeleteButtonWithInfo:postBody isDeletedOnly:isTFDeletedOnly]){
- shouldAddUndeleteCell = YES;
- apolloCommentCell = nil;
- apolloCommentController = self;
- }
- }
-
- %orig;
- }
-
- %new
- - (void)undeleteCellWasSelected {
-
- RKLink *post = MSHookIvar<RKLink *>(self, "link");
-
- [%c(TFHelper) getUndeleteDataWithID:[[post fullName] componentsSeparatedByString:@"_"][1] isComment:NO timeout:pushshiftRequestTimeoutValue extraData:nil completionTarget:self completionSelector:@selector(completeUndeletePostAction:)];
- }
-
- %new
- - (void)completeUndeletePostAction:(NSDictionary *)data {
-
- RKLink *post = MSHookIvar<RKLink *>(self, "link");
-
- id headerCellNode = [self headerCellNode];
- id bodyNode = MSHookIvar<id>(headerCellNode, "bodyNode");
- id postInfoNode = MSHookIvar<id>(headerCellNode, "postInfoNode");
- id authorNode = MSHookIvar<id>(postInfoNode, "authorButtonNode");
-
- NSString *author = data[@"author"];
- NSString *authorTextString = [NSString stringWithFormat:@"by %@", author];
- NSString *body = data[@"body"];
-
- [post setUndeleteAuthor:author];
- [post setSelfText:body];
-
- NSAttributedString *prevAuthorAttributedString = [authorNode attributedTitleForState:UIControlStateNormal];
- NSDictionary *authorStringAttributes = [prevAuthorAttributedString attributesAtIndex:0 longestEffectiveRange:nil inRange:NSMakeRange(0, [prevAuthorAttributedString length])];
- NSAttributedString* newAuthorAttributedString = [[NSAttributedString alloc] initWithString:authorTextString attributes:authorStringAttributes];
-
- [authorNode setAttributedTitle:newAuthorAttributedString forState:UIControlStateNormal];
- [bodyNode setAttributedString:[%c(MarkdownRenderer) attributedStringFromMarkdown:body withAttributes:apolloBodyAttributes]];
- }
-
- %end
-
-
- %hook CommentsHeaderCellNode
-
- - (void)didLoad {
- %orig;
-
- [[self closestViewController] setHeaderCellNode:self];
- }
-
- - (void)_layoutSublayouts {
- %orig;
-
- [[self closestViewController] setHeaderCellNode:self];
- }
-
- - (void)longPressedWithGestureRecognizer:(id)arg1 {
-
- RKLink *post = MSHookIvar<RKLink *>(self, "link");
- NSString *postBody = [post selfText];
-
- if ([post isSelfPost]) {
- if ([%c(TFHelper) shouldShowUndeleteButtonWithInfo:postBody isDeletedOnly:isTFDeletedOnly]){
- shouldAddUndeleteCell = YES;
- apolloCommentCell = nil;
- apolloCommentController = self;
- }
- }
-
- %orig;
- }
-
- %end
-
- %end
-
-
- static void loadPrefs(){
- NSMutableDictionary *prefs = [[NSMutableDictionary alloc] initWithContentsOfFile:@"/User/Library/Preferences/com.lint.undelete.prefs.plist"];
-
- if (prefs){
- isEnabled = [prefs objectForKey:@"isEnabled"] ? [[prefs objectForKey:@"isEnabled"] boolValue] : YES;
- isApolloEnabled = [prefs objectForKey:@"isApolloEnabled"] ? [[prefs objectForKey:@"isApolloEnabled"] boolValue] : YES;
- isTFDeletedOnly = [prefs objectForKey:@"isTFDeletedOnly"] ? [[prefs objectForKey:@"isTFDeletedOnly"] boolValue] : YES;
- pushshiftRequestTimeoutValue = [prefs objectForKey:@"requestTimeoutValue"] ? [[prefs objectForKey:@"requestTimeoutValue"] doubleValue] : 10;
- shouldApolloHaveButton = [prefs objectForKey:@"shouldApolloHaveButton"] ? [[prefs objectForKey:@"shouldApolloHaveButton"] boolValue] : NO;
- } else {
- isEnabled = YES;
- isApolloEnabled = YES;
- isTFDeletedOnly = YES;
- pushshiftRequestTimeoutValue = 10;
- shouldApolloHaveButton = NO;
- }
- }
-
- static void prefsChanged(CFNotificationCenterRef center, void *observer, CFStringRef name, const void *object, CFDictionaryRef userInfo) {
- loadPrefs();
- }
-
-
- %ctor {
- loadPrefs();
-
- NSString* processName = [[NSProcessInfo processInfo] processName];
-
- if ([processName isEqualToString:@"Apollo"]){
- if (isApolloEnabled && isEnabled){
-
- CFNotificationCenterAddObserver(CFNotificationCenterGetDarwinNotifyCenter(), NULL, (CFNotificationCallback)prefsChanged, CFSTR("com.lint.undelete.prefs.changed"), NULL, CFNotificationSuspensionBehaviorCoalesce);
-
- %init(Apollo, CommentsHeaderCellNode = objc_getClass("Apollo.CommentsHeaderCellNode"), CommentCellNode = objc_getClass("Apollo.CommentCellNode"), ApolloButtonNode = objc_getClass("Apollo.ApolloButtonNode"), ActionController = objc_getClass("Apollo.ActionController"), IconActionTableViewCell = objc_getClass("Apollo.IconActionTableViewCell"), CommentsViewController = objc_getClass("Apollo.CommentsViewController"));
- }
- }
- }
|