mirror of
https://github.com/faroukbmiled/RyukGram.git
synced 2026-04-29 15:47:54 +02:00
2977873932
- Profile Analyzer (beta) — follower/following scans with mutuals, non-followbacks, new/lost trackers, and profile change history; searchable lists with batch follow/unfollow - Theme settings — force dark mode, Full OLED, OLED chat theme, and keyboard theme picker - Confirm story like - Confirm story emoji reaction - Swipe down to dismiss media viewer - Manually add users to story/chat exclusion lists by username - Keep stories visually seen locally - Auto-scroll reels mode - Quality picker: audio-only and raw photo download rows - Clear cache button with optional auto-clear interval - Spanish, Russian, Korean, Arabic, and Chinese (Traditional) translations - About page with version, credits, and links - Release notes popup on first launch of a new version - Anonymous live viewing - Toggle live comments - Disappearing DM media overlay — action button, mark-as-viewed eye, and audio toggle - Hide RyukGram UI on screenshots, screen recordings, and mirroring - Open link from clipboard — long-press the search tab - Messages-only mode: optional "Hide tab bar" sub-toggle - Fake profile stats — verified badge and follower/following/post counts on your own profile - Language switcher + import/export localization from Debug - Reveal poll/slider vote counts and quiz answers on stories and reels before interacting - Force legacy Quiz sticker back into the story composer tray - Advanced experimental features menu — toggle hidden IG experiments (QuickSnap, Homecoming, Prism, Direct Notes reply types) with apply-on-restart batching and a crash-loop auto-reset - Shortcut to Advanced experimental features from the General experimental features section - Push notifications render with rich previews on sideload again - IG 426 compatibility across story audio toggle, like confirmation, seen-on-like, live comments, notes audio download - Call confirm split into separate voice-call and video-call toggles - Messages-only mode: tab swiping disabled - Settings quick-access broken in non-English languages - Story seen-receipt block restored on IG v426 - Block selected mode no longer marks listed stories as seen - Hide explore posts grid works again on recent IG versions - Hide suggested stories no longer breaks profile highlights - Hide trending searches now also hides the category chip bar - Story eye long-press menu opens next to the button - Disable video autoplay: tap-to-play now works on videos inside carousels - Disable vanish mode swipe fixed on IG 426 - "Confirm shh mode" renamed to "Confirm vanish mode" across all languages - Confirm sticker interaction split into separate story and highlight toggles - Shared link embed presets: added eeinstagram.com and vxinstagram.com - Downloaded media filenames follow `@username_context_timestamp` - Reels pause mode: optional tap-to-mute on photo reels - Backup & Restore — scope picker with live preview for Settings / Excluded lists / Analyzer data - Profile Analyzer: filter by Not verified - Settings header: tap to open a sheet with GitHub and Telegram channel links - Thanks to Furamako for the Spanish translation - Thanks to [ZomkaDEV](https://github.com/ZomkaDEV) for the Russian translation - Thanks to [@ch1tmdgus](https://github.com/ch1tmdgus) (N4C) for the Korean translation - Thanks to [@bruuhim](https://github.com/bruuhim) for the Arabic translation - Thanks to [@jaydenjcpy](https://github.com/jaydenjcpy) for the Chinese (Traditional) translation - Thanks to [@darthplagueiswise](https://github.com/darthplagueiswise) (Radan) for the experimental flag feature set - Thanks to [@asdfzxcvbn](https://github.com/asdfzxcvbn) for [zxPluginsInject](https://github.com/asdfzxcvbn/zxPluginsInject) and [ipapatch](https://github.com/asdfzxcvbn/ipapatch) - Preserved unsent messages can't be removed via "Delete for you"; pull-to-refresh clears them (warning available in settings) - "Delete for you" detection uses a ~2s window after the local action — a real unsend landing in that window may be missed (rare) - With Liquid Glass buttons + Hide UI on capture both on, the DM eye leaves an empty glass bubble in captures
182 lines
7.2 KiB
Objective-C
182 lines
7.2 KiB
Objective-C
#import "SCIActionButton.h"
|
|
#import "SCIActionMenu.h"
|
|
#import "SCIRepostSheet.h"
|
|
#import "../Utils.h"
|
|
#import <objc/runtime.h>
|
|
|
|
// Associated-object keys for per-button config.
|
|
static const void *kSCICtxKey = &kSCICtxKey;
|
|
static const void *kSCIProviderKey = &kSCIProviderKey;
|
|
static const void *kSCIPrefKey = &kSCIPrefKey;
|
|
const void *kSCIDismissKey = &kSCIDismissKey;
|
|
|
|
|
|
@interface SCIActionButton () <UIContextMenuInteractionDelegate>
|
|
@end
|
|
|
|
@implementation SCIActionButton
|
|
|
|
// Singleton delegate for UIContextMenuInteraction.
|
|
+ (instancetype)shared {
|
|
static SCIActionButton *s;
|
|
static dispatch_once_t once;
|
|
dispatch_once(&once, ^{ s = [SCIActionButton new]; });
|
|
return s;
|
|
}
|
|
|
|
+ (UIMenu *)deferredMenuForContext:(SCIActionContext)ctx
|
|
fromView:(UIView *)sourceView
|
|
mediaProvider:(SCIActionMediaProvider)provider {
|
|
__weak UIView *weakSource = sourceView;
|
|
SCIActionMediaProvider capturedProvider = [provider copy];
|
|
|
|
UIDeferredMenuElement *deferred = [UIDeferredMenuElement
|
|
elementWithUncachedProvider:^(void (^completion)(NSArray<UIMenuElement *> * _Nonnull)) {
|
|
UIView *view = weakSource;
|
|
id media = (view && capturedProvider) ? capturedProvider(view) : nil;
|
|
NSArray *actions = [SCIMediaActions actionsForContext:ctx
|
|
media:media
|
|
fromView:view];
|
|
UIMenu *built = [SCIActionMenu buildMenuWithActions:actions];
|
|
completion(built.children);
|
|
}];
|
|
|
|
return [UIMenu menuWithTitle:@""
|
|
image:nil
|
|
identifier:nil
|
|
options:0
|
|
children:@[deferred]];
|
|
}
|
|
|
|
+ (void)configureButton:(UIButton *)button
|
|
context:(SCIActionContext)ctx
|
|
prefKey:(NSString *)prefKey
|
|
mediaProvider:(SCIActionMediaProvider)provider {
|
|
if (!button) return;
|
|
|
|
// Stash config on the button.
|
|
objc_setAssociatedObject(button, kSCICtxKey, @(ctx), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
|
|
objc_setAssociatedObject(button, kSCIProviderKey, [provider copy], OBJC_ASSOCIATION_COPY_NONATOMIC);
|
|
objc_setAssociatedObject(button, kSCIPrefKey, [prefKey copy], OBJC_ASSOCIATION_COPY_NONATOMIC);
|
|
|
|
// Read default tap mode fresh.
|
|
NSString *defaultTap = [SCIUtils getStringPref:prefKey];
|
|
if (!defaultTap.length) defaultTap = @"menu";
|
|
|
|
// Remove previous wiring to stay idempotent.
|
|
[button removeTarget:nil action:NULL forControlEvents:UIControlEventTouchUpInside];
|
|
for (id<UIInteraction> it in [button.interactions copy]) {
|
|
if ([(id)it isKindOfClass:[UIContextMenuInteraction class]]) {
|
|
[button removeInteraction:it];
|
|
}
|
|
}
|
|
|
|
if ([defaultTap isEqualToString:@"menu"]) {
|
|
// Tap opens menu natively.
|
|
button.menu = [self deferredMenuForContext:ctx fromView:button mediaProvider:provider];
|
|
button.showsMenuAsPrimaryAction = YES;
|
|
return;
|
|
}
|
|
|
|
// Tap fires dedicated action; long-press opens menu.
|
|
button.showsMenuAsPrimaryAction = NO;
|
|
button.menu = nil;
|
|
[button addTarget:[self shared]
|
|
action:@selector(sciTapHandler:)
|
|
forControlEvents:UIControlEventTouchUpInside];
|
|
|
|
UIContextMenuInteraction *interaction =
|
|
[[UIContextMenuInteraction alloc] initWithDelegate:[self shared]];
|
|
[button addInteraction:interaction];
|
|
}
|
|
|
|
// Haptic + scale-bounce feedback.
|
|
+ (void)bounceButton:(UIView *)view {
|
|
UIImpactFeedbackGenerator *haptic =
|
|
[[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleMedium];
|
|
[haptic impactOccurred];
|
|
[UIView animateWithDuration:0.1
|
|
animations:^{ view.transform = CGAffineTransformMakeScale(0.82, 0.82); }
|
|
completion:^(BOOL _) {
|
|
[UIView animateWithDuration:0.1 animations:^{
|
|
view.transform = CGAffineTransformIdentity;
|
|
}];
|
|
}];
|
|
}
|
|
|
|
// Default-tap handler.
|
|
- (void)sciTapHandler:(UIButton *)sender {
|
|
[SCIActionButton bounceButton:sender];
|
|
|
|
NSNumber *ctxNum = objc_getAssociatedObject(sender, kSCICtxKey);
|
|
SCIActionMediaProvider provider = objc_getAssociatedObject(sender, kSCIProviderKey);
|
|
NSString *prefKey = objc_getAssociatedObject(sender, kSCIPrefKey);
|
|
if (!ctxNum || !provider) return;
|
|
|
|
NSString *tap = [SCIUtils getStringPref:prefKey];
|
|
if (!tap.length) tap = @"menu";
|
|
id media = provider(sender);
|
|
if (media == (id)kCFNull) return;
|
|
|
|
SCIActionContext tapCtx = (SCIActionContext)ctxNum.integerValue;
|
|
NSString *tapCtxLabel = [SCIMediaActions contextLabelForContext:tapCtx];
|
|
|
|
if ([tap isEqualToString:@"expand"]) {
|
|
[SCIMediaActions expandMedia:media fromView:sender caption:nil];
|
|
} else if ([tap isEqualToString:@"download_share"]) {
|
|
[SCIMediaActions setCurrentFilenameStem:[SCIMediaActions filenameStemForMedia:media contextLabel:tapCtxLabel]];
|
|
[SCIMediaActions downloadAndShareMedia:media];
|
|
} else if ([tap isEqualToString:@"download_photos"]) {
|
|
[SCIMediaActions setCurrentFilenameStem:[SCIMediaActions filenameStemForMedia:media contextLabel:tapCtxLabel]];
|
|
[SCIMediaActions downloadAndSaveMedia:media];
|
|
} else if ([tap isEqualToString:@"copy_link"]) {
|
|
[SCIMediaActions copyURLForMedia:media];
|
|
} else if ([tap isEqualToString:@"repost"]) {
|
|
NSURL *vidURL = [SCIUtils getVideoUrlForMedia:(id)media];
|
|
NSURL *imgURL = [SCIUtils getPhotoUrlForMedia:(id)media];
|
|
[SCIRepostSheet repostWithVideoURL:vidURL photoURL:imgURL];
|
|
} else if ([tap isEqualToString:@"view_mentions"]) {
|
|
UIViewController *host = [SCIUtils nearestViewControllerForView:sender];
|
|
if (host) {
|
|
extern void sciShowStoryMentions(UIViewController *, UIView *);
|
|
sciShowStoryMentions(host, sender);
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - UIContextMenuInteractionDelegate
|
|
|
|
- (UIContextMenuConfiguration *)contextMenuInteraction:(UIContextMenuInteraction *)interaction
|
|
configurationForMenuAtLocation:(CGPoint)location {
|
|
UIView *view = interaction.view;
|
|
NSNumber *ctxNum = objc_getAssociatedObject(view, kSCICtxKey);
|
|
SCIActionMediaProvider provider = objc_getAssociatedObject(view, kSCIProviderKey);
|
|
if (!ctxNum || !provider) return nil;
|
|
SCIActionContext ctx = (SCIActionContext)ctxNum.integerValue;
|
|
|
|
return [UIContextMenuConfiguration
|
|
configurationWithIdentifier:nil
|
|
previewProvider:nil
|
|
actionProvider:^UIMenu * _Nullable(NSArray<UIMenuElement *> * _Nonnull suggested) {
|
|
return [SCIActionButton deferredMenuForContext:ctx
|
|
fromView:view
|
|
mediaProvider:provider];
|
|
}];
|
|
}
|
|
|
|
- (void)contextMenuInteraction:(UIContextMenuInteraction *)interaction
|
|
willEndForConfiguration:(UIContextMenuConfiguration *)configuration
|
|
animator:(id<UIContextMenuInteractionAnimating>)animator {
|
|
UIView *view = interaction.view;
|
|
void (^dismiss)(void) = objc_getAssociatedObject(view, kSCIDismissKey);
|
|
if (dismiss) {
|
|
if (animator) {
|
|
[animator addCompletion:^{ dismiss(); }];
|
|
} else {
|
|
dismiss();
|
|
}
|
|
}
|
|
}
|
|
|
|
@end
|