mirror of
https://github.com/faroukbmiled/RyukGram.git
synced 2026-06-06 07:23:53 +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
280 lines
10 KiB
Objective-C
280 lines
10 KiB
Objective-C
#import "SCIChrome.h"
|
|
#import "Utils.h"
|
|
#import "SCIPrefObserver.h"
|
|
|
|
// MARK: - Canvas discovery
|
|
|
|
static UIView *sciFindCanvasDeep(UIView *root, int depth) {
|
|
if (depth > 4) return nil;
|
|
for (UIView *sub in root.subviews) {
|
|
if ([NSStringFromClass([sub class]) containsString:@"CanvasView"]) return sub;
|
|
UIView *found = sciFindCanvasDeep(sub, depth + 1);
|
|
if (found) return found;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
// MARK: - SCIChromeCanvas
|
|
|
|
@interface SCIChromeCanvas ()
|
|
@property (nonatomic, strong) UITextField *secureField;
|
|
@property (nonatomic, strong, nullable) UIView *canvas;
|
|
@end
|
|
|
|
@implementation SCIChromeCanvas
|
|
|
|
+ (NSHashTable<SCIChromeCanvas *> *)instances {
|
|
static NSHashTable *t;
|
|
static dispatch_once_t once;
|
|
dispatch_once(&once, ^{ t = [NSHashTable weakObjectsHashTable]; });
|
|
return t;
|
|
}
|
|
|
|
+ (void)ensureObserverInstalled {
|
|
static dispatch_once_t once;
|
|
dispatch_once(&once, ^{
|
|
[SCIPrefObserver observeKey:@"hide_ui_on_capture" handler:^{
|
|
for (SCIChromeCanvas *v in [SCIChromeCanvas instances]) [v applyPref];
|
|
}];
|
|
});
|
|
}
|
|
|
|
- (instancetype)initWithFrame:(CGRect)frame {
|
|
self = [super initWithFrame:frame];
|
|
if (self) {
|
|
[SCIChromeCanvas ensureObserverInstalled];
|
|
self.translatesAutoresizingMaskIntoConstraints = NO;
|
|
_secureField = [UITextField new];
|
|
_secureField.userInteractionEnabled = NO;
|
|
_secureField.autocorrectionType = UITextAutocorrectionTypeNo;
|
|
_secureField.spellCheckingType = UITextSpellCheckingTypeNo;
|
|
_secureField.smartDashesType = UITextSmartDashesTypeNo;
|
|
_secureField.smartQuotesType = UITextSmartQuotesTypeNo;
|
|
_secureField.smartInsertDeleteType = UITextSmartInsertDeleteTypeNo;
|
|
_secureField.autocapitalizationType = UITextAutocapitalizationTypeNone;
|
|
[self applyPref];
|
|
[[SCIChromeCanvas instances] addObject:self];
|
|
[self attachCanvasIfPossible];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (UIView *)contentContainer { return self.canvas ?: self; }
|
|
|
|
- (void)applyPref {
|
|
BOOL on = [SCIUtils getBoolPref:@"hide_ui_on_capture"];
|
|
if (self.secureField.secureTextEntry != on) self.secureField.secureTextEntry = on;
|
|
}
|
|
|
|
- (void)didMoveToWindow { [super didMoveToWindow]; [self attachCanvasIfPossible]; }
|
|
- (void)layoutSubviews { [super layoutSubviews]; [self attachCanvasIfPossible]; }
|
|
|
|
- (void)attachCanvasIfPossible {
|
|
if (self.canvas && self.canvas.superview == self) return;
|
|
|
|
[self.secureField layoutIfNeeded];
|
|
UIView *c = sciFindCanvasDeep(self.secureField, 0);
|
|
if (!c) return;
|
|
|
|
// Migrate anything that landed on self (contentContainer fallback) into
|
|
// the canvas so redaction covers it.
|
|
NSMutableArray<UIView *> *stashed = [NSMutableArray array];
|
|
for (UIView *sub in self.subviews) {
|
|
if (sub != c) [stashed addObject:sub];
|
|
}
|
|
|
|
[c removeFromSuperview];
|
|
[self insertSubview:c atIndex:0];
|
|
c.translatesAutoresizingMaskIntoConstraints = NO;
|
|
[NSLayoutConstraint activateConstraints:@[
|
|
[c.leadingAnchor constraintEqualToAnchor:self.leadingAnchor],
|
|
[c.trailingAnchor constraintEqualToAnchor:self.trailingAnchor],
|
|
[c.topAnchor constraintEqualToAnchor:self.topAnchor],
|
|
[c.bottomAnchor constraintEqualToAnchor:self.bottomAnchor],
|
|
]];
|
|
self.canvas = c;
|
|
|
|
for (UIView *v in stashed) {
|
|
[v removeFromSuperview];
|
|
[c addSubview:v];
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
// MARK: - SCIChromeButton
|
|
|
|
@interface SCIChromeButton ()
|
|
@property (nonatomic, strong) SCIChromeCanvas *chromeCanvas;
|
|
@property (nonatomic, strong) UIView *bubbleView;
|
|
@property (nonatomic, strong, readwrite) UIImageView *iconView;
|
|
@end
|
|
|
|
@implementation SCIChromeButton
|
|
|
|
- (instancetype)initWithSymbol:(NSString *)symbol
|
|
pointSize:(CGFloat)pointSize
|
|
diameter:(CGFloat)diameter {
|
|
self = [super initWithFrame:CGRectMake(0, 0, diameter, diameter)];
|
|
if (self) {
|
|
_diameter = diameter;
|
|
_symbolName = [symbol copy];
|
|
_symbolPointSize = pointSize;
|
|
_iconTint = [UIColor whiteColor];
|
|
_bubbleColor = [UIColor colorWithWhite:0.0 alpha:0.4];
|
|
[self buildChrome];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)buildChrome {
|
|
self.adjustsImageWhenHighlighted = NO;
|
|
self.translatesAutoresizingMaskIntoConstraints = NO;
|
|
|
|
_chromeCanvas = [SCIChromeCanvas new];
|
|
_chromeCanvas.userInteractionEnabled = NO;
|
|
[self addSubview:_chromeCanvas];
|
|
[NSLayoutConstraint activateConstraints:@[
|
|
[_chromeCanvas.leadingAnchor constraintEqualToAnchor:self.leadingAnchor],
|
|
[_chromeCanvas.trailingAnchor constraintEqualToAnchor:self.trailingAnchor],
|
|
[_chromeCanvas.topAnchor constraintEqualToAnchor:self.topAnchor],
|
|
[_chromeCanvas.bottomAnchor constraintEqualToAnchor:self.bottomAnchor],
|
|
]];
|
|
|
|
_bubbleView = [UIView new];
|
|
_bubbleView.userInteractionEnabled = NO;
|
|
_bubbleView.translatesAutoresizingMaskIntoConstraints = NO;
|
|
_bubbleView.backgroundColor = _bubbleColor;
|
|
_bubbleView.layer.cornerRadius = _diameter / 2;
|
|
_bubbleView.clipsToBounds = YES;
|
|
|
|
_iconView = [UIImageView new];
|
|
_iconView.userInteractionEnabled = NO;
|
|
_iconView.contentMode = UIViewContentModeCenter;
|
|
_iconView.translatesAutoresizingMaskIntoConstraints = NO;
|
|
_iconView.tintColor = _iconTint;
|
|
[self reloadIcon];
|
|
|
|
UIView *host = _chromeCanvas.contentContainer;
|
|
[host addSubview:_bubbleView];
|
|
[host addSubview:_iconView];
|
|
[NSLayoutConstraint activateConstraints:@[
|
|
[_bubbleView.leadingAnchor constraintEqualToAnchor:host.leadingAnchor],
|
|
[_bubbleView.trailingAnchor constraintEqualToAnchor:host.trailingAnchor],
|
|
[_bubbleView.topAnchor constraintEqualToAnchor:host.topAnchor],
|
|
[_bubbleView.bottomAnchor constraintEqualToAnchor:host.bottomAnchor],
|
|
[_iconView.centerXAnchor constraintEqualToAnchor:host.centerXAnchor],
|
|
[_iconView.centerYAnchor constraintEqualToAnchor:host.centerYAnchor],
|
|
]];
|
|
}
|
|
|
|
- (CGSize)intrinsicContentSize { return CGSizeMake(_diameter, _diameter); }
|
|
|
|
- (void)setSymbolName:(NSString *)symbolName {
|
|
_symbolName = [symbolName copy];
|
|
[self reloadIcon];
|
|
}
|
|
|
|
- (void)setSymbolPointSize:(CGFloat)symbolPointSize {
|
|
_symbolPointSize = symbolPointSize;
|
|
[self reloadIcon];
|
|
}
|
|
|
|
- (void)setIconTint:(UIColor *)iconTint {
|
|
_iconTint = [iconTint copy];
|
|
_iconView.tintColor = iconTint;
|
|
}
|
|
|
|
- (void)setBubbleColor:(UIColor *)bubbleColor {
|
|
_bubbleColor = [bubbleColor copy];
|
|
_bubbleView.backgroundColor = bubbleColor;
|
|
}
|
|
|
|
- (void)reloadIcon {
|
|
if (!_symbolName.length) { _iconView.image = nil; return; }
|
|
UIImageSymbolConfiguration *cfg = [UIImageSymbolConfiguration configurationWithPointSize:_symbolPointSize
|
|
weight:UIImageSymbolWeightSemibold];
|
|
_iconView.image = [UIImage systemImageNamed:_symbolName withConfiguration:cfg];
|
|
}
|
|
|
|
- (void)layoutSubviews {
|
|
[super layoutSubviews];
|
|
// Keep the bubble circular when the caller resizes via constraints.
|
|
CGFloat r = MIN(self.bounds.size.width, self.bounds.size.height) / 2;
|
|
_bubbleView.layer.cornerRadius = r;
|
|
}
|
|
|
|
@end
|
|
|
|
// MARK: - SCIChromeLabel
|
|
|
|
@interface SCIChromeLabel ()
|
|
@property (nonatomic, strong) SCIChromeCanvas *chromeCanvas;
|
|
@property (nonatomic, strong) UILabel *label;
|
|
@end
|
|
|
|
@implementation SCIChromeLabel
|
|
|
|
- (instancetype)initWithText:(NSString *)text {
|
|
self = [super initWithFrame:CGRectZero];
|
|
if (self) {
|
|
self.translatesAutoresizingMaskIntoConstraints = NO;
|
|
|
|
_chromeCanvas = [SCIChromeCanvas new];
|
|
_chromeCanvas.userInteractionEnabled = NO;
|
|
[self addSubview:_chromeCanvas];
|
|
[NSLayoutConstraint activateConstraints:@[
|
|
[_chromeCanvas.leadingAnchor constraintEqualToAnchor:self.leadingAnchor],
|
|
[_chromeCanvas.trailingAnchor constraintEqualToAnchor:self.trailingAnchor],
|
|
[_chromeCanvas.topAnchor constraintEqualToAnchor:self.topAnchor],
|
|
[_chromeCanvas.bottomAnchor constraintEqualToAnchor:self.bottomAnchor],
|
|
]];
|
|
|
|
_label = [UILabel new];
|
|
_label.translatesAutoresizingMaskIntoConstraints = NO;
|
|
_label.text = text;
|
|
|
|
UIView *host = _chromeCanvas.contentContainer;
|
|
[host addSubview:_label];
|
|
[NSLayoutConstraint activateConstraints:@[
|
|
[_label.leadingAnchor constraintEqualToAnchor:host.leadingAnchor],
|
|
[_label.trailingAnchor constraintEqualToAnchor:host.trailingAnchor],
|
|
[_label.topAnchor constraintEqualToAnchor:host.topAnchor],
|
|
[_label.bottomAnchor constraintEqualToAnchor:host.bottomAnchor],
|
|
]];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (NSString *)text { return _label.text; }
|
|
- (void)setText:(NSString *)t { _label.text = t; }
|
|
- (UIFont *)font { return _label.font; }
|
|
- (void)setFont:(UIFont *)f { _label.font = f; }
|
|
- (UIColor *)textColor { return _label.textColor; }
|
|
- (void)setTextColor:(UIColor *)c { _label.textColor = c; }
|
|
- (NSTextAlignment)textAlignment { return _label.textAlignment; }
|
|
- (void)setTextAlignment:(NSTextAlignment)a { _label.textAlignment = a; }
|
|
|
|
@end
|
|
|
|
// MARK: - Bar button helpers
|
|
|
|
UIBarButtonItem *SCIChromeBarButtonItem(NSString *symbol,
|
|
CGFloat pointSize,
|
|
id target,
|
|
SEL action,
|
|
SCIChromeButton **outButton) {
|
|
SCIChromeButton *btn = [[SCIChromeButton alloc] initWithSymbol:symbol
|
|
pointSize:pointSize
|
|
diameter:28];
|
|
btn.bubbleColor = [UIColor clearColor];
|
|
if (target && action) [btn addTarget:target action:action forControlEvents:UIControlEventTouchUpInside];
|
|
if (outButton) *outButton = btn;
|
|
return [[UIBarButtonItem alloc] initWithCustomView:btn];
|
|
}
|
|
|
|
SCIChromeButton *SCIChromeButtonForBarItem(UIBarButtonItem *item) {
|
|
UIView *v = item.customView;
|
|
return [v isKindOfClass:[SCIChromeButton class]] ? (SCIChromeButton *)v : nil;
|
|
}
|