Files
faroukbmiled 2977873932 [release] RyukGram v1.2.2
- 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
2026-04-24 02:50:30 +01:00

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;
}