Files
RyukGram/src/Features/ProfileAnalyzer/SCIProfileAnalyzerService.m
T
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

214 lines
9.5 KiB
Objective-C

#import "SCIProfileAnalyzerService.h"
#import "../../Networking/SCIInstagramAPI.h"
#import "../../Utils.h"
const NSInteger SCIProfileAnalyzerMaxFollowerCount = 13000;
#define SCI_PA_PAGE_DELAY_S 0.25 // small pause between pages — lightweight rate cushion
@interface SCIProfileAnalyzerService () {
@public
NSInteger _expectedFollowers;
NSInteger _expectedFollowing;
}
@property (nonatomic, assign) BOOL cancelled;
@property (nonatomic, assign) BOOL isRunning;
@end
@implementation SCIProfileAnalyzerService
+ (instancetype)sharedService {
static SCIProfileAnalyzerService *s;
static dispatch_once_t once;
dispatch_once(&once, ^{ s = [self new]; });
return s;
}
- (void)cancel { self.cancelled = YES; }
- (void)finishWithSnapshot:(SCIProfileAnalyzerSnapshot *)s error:(NSError *)e completion:(SCIPACompletion)completion {
self.isRunning = NO;
self.cancelled = NO;
if (completion) dispatch_async(dispatch_get_main_queue(), ^{ completion(s, e); });
}
- (NSError *)errorWithCode:(SCIProfileAnalyzerError)code message:(NSString *)msg {
return [NSError errorWithDomain:@"SCIProfileAnalyzer" code:code
userInfo:@{ NSLocalizedDescriptionKey: msg ?: @"" }];
}
- (void)runForSelfWithHeaderInfo:(SCIPAHeaderInfo)headerInfo
progress:(SCIPAProgress)progress
completion:(SCIPACompletion)completion {
if (self.isRunning) {
if (completion) completion(nil, [self errorWithCode:SCIProfileAnalyzerErrorCancelled
message:SCILocalized(@"Another analysis is already running")]);
return;
}
self.isRunning = YES;
self.cancelled = NO;
NSString *selfPK = [SCIUtils currentUserPK];
if (!selfPK.length) {
[self finishWithSnapshot:nil
error:[self errorWithCode:SCIProfileAnalyzerErrorNoSession message:SCILocalized(@"No active Instagram session found")]
completion:completion];
return;
}
__weak typeof(self) weakSelf = self;
[self reportProgress:progress status:SCILocalized(@"Fetching profile info…") fraction:0.02];
[SCIInstagramAPI sendRequestWithMethod:@"GET"
path:[NSString stringWithFormat:@"users/%@/info/", selfPK]
body:nil
completion:^(NSDictionary *resp, NSError *error) {
typeof(self) strongSelf = weakSelf;
if (!strongSelf) return;
if (strongSelf.cancelled) {
[strongSelf finishWithSnapshot:nil error:[strongSelf errorWithCode:SCIProfileAnalyzerErrorCancelled message:SCILocalized(@"Cancelled")]
completion:completion];
return;
}
NSDictionary *user = [resp[@"user"] isKindOfClass:[NSDictionary class]] ? resp[@"user"] : nil;
if (!user) {
[strongSelf finishWithSnapshot:nil
error:[strongSelf errorWithCode:SCIProfileAnalyzerErrorNetwork message:SCILocalized(@"Couldn't fetch profile information")]
completion:completion];
return;
}
NSInteger followerCount = [user[@"follower_count"] integerValue];
if (followerCount > SCIProfileAnalyzerMaxFollowerCount) {
[strongSelf finishWithSnapshot:nil
error:[strongSelf errorWithCode:SCIProfileAnalyzerErrorTooManyFollowers
message:SCILocalized(@"Too many followers to analyze")]
completion:completion];
return;
}
SCIProfileAnalyzerSnapshot *snap = [SCIProfileAnalyzerSnapshot new];
snap.selfPK = selfPK;
snap.selfUsername = user[@"username"];
snap.selfFullName = user[@"full_name"];
snap.selfProfilePicURL = user[@"profile_pic_url"];
snap.followerCount = followerCount;
snap.followingCount = [user[@"following_count"] integerValue];
snap.mediaCount = [user[@"media_count"] integerValue];
snap.scanDate = [NSDate date];
strongSelf->_expectedFollowers = followerCount;
strongSelf->_expectedFollowing = snap.followingCount;
if (headerInfo) dispatch_async(dispatch_get_main_queue(), ^{ headerInfo(user); });
[strongSelf fetchFollowersForPK:selfPK snapshot:snap progress:progress completion:completion];
}];
}
- (void)reportProgress:(SCIPAProgress)p status:(NSString *)s fraction:(double)f {
if (!p) return;
dispatch_async(dispatch_get_main_queue(), ^{ p(s, f); });
}
#pragma mark - Paginated fetchers
- (void)fetchFollowersForPK:(NSString *)pk
snapshot:(SCIProfileAnalyzerSnapshot *)snap
progress:(SCIPAProgress)progress
completion:(SCIPACompletion)completion {
NSMutableArray *acc = [NSMutableArray array];
[self pagePath:[NSString stringWithFormat:@"friendships/%@/followers/", pk]
acc:acc
maxId:nil
total:snap.followerCount
stage:@"followers"
progress:progress
completion:^(NSArray *users, NSError *error) {
if (error || self.cancelled) {
[self finishWithSnapshot:nil error:error ?: [self errorWithCode:SCIProfileAnalyzerErrorCancelled message:SCILocalized(@"Cancelled")]
completion:completion];
return;
}
snap.followers = users;
[self fetchFollowingForPK:pk snapshot:snap progress:progress completion:completion];
}];
}
- (void)fetchFollowingForPK:(NSString *)pk
snapshot:(SCIProfileAnalyzerSnapshot *)snap
progress:(SCIPAProgress)progress
completion:(SCIPACompletion)completion {
NSMutableArray *acc = [NSMutableArray array];
[self pagePath:[NSString stringWithFormat:@"friendships/%@/following/", pk]
acc:acc
maxId:nil
total:snap.followingCount
stage:@"following"
progress:progress
completion:^(NSArray *users, NSError *error) {
if (error || self.cancelled) {
[self finishWithSnapshot:nil error:error ?: [self errorWithCode:SCIProfileAnalyzerErrorCancelled message:SCILocalized(@"Cancelled")]
completion:completion];
return;
}
snap.following = users;
[self finishWithSnapshot:snap error:nil completion:completion];
}];
}
- (void)pagePath:(NSString *)basePath
acc:(NSMutableArray *)acc
maxId:(NSString *)maxId
total:(NSInteger)total
stage:(NSString *)stage
progress:(SCIPAProgress)progress
completion:(void(^)(NSArray *users, NSError *error))completion {
if (self.cancelled) {
completion(nil, [self errorWithCode:SCIProfileAnalyzerErrorCancelled message:SCILocalized(@"Cancelled")]);
return;
}
NSString *path = maxId.length ? [NSString stringWithFormat:@"%@?max_id=%@", basePath, maxId] : basePath;
__weak typeof(self) weakSelf = self;
[SCIInstagramAPI sendRequestWithMethod:@"GET" path:path body:nil completion:^(NSDictionary *resp, NSError *error) {
typeof(self) strongSelf = weakSelf;
if (!strongSelf) return;
if (error) { completion(nil, [strongSelf errorWithCode:SCIProfileAnalyzerErrorNetwork message:error.localizedDescription]); return; }
NSArray *users = resp[@"users"];
if ([users isKindOfClass:[NSArray class]]) {
for (NSDictionary *d in users) {
SCIProfileAnalyzerUser *u = [SCIProfileAnalyzerUser userFromAPIDict:d];
if (u) [acc addObject:u];
}
}
// Weight each stage by its share of expected work so the ring moves
// proportionally regardless of follower/following ratio. 3% reserved
// up front for the initial user-info call.
NSInteger followerTarget = strongSelf->_expectedFollowers;
NSInteger followingTarget = strongSelf->_expectedFollowing;
double total0 = MAX(1, followerTarget + followingTarget);
double stageWeight = ([stage isEqualToString:@"followers"] ? followerTarget : followingTarget) / total0;
double stageOffset = ([stage isEqualToString:@"followers"] ? 0.0 : (double)followerTarget / total0);
double stageLocal = total > 0 ? MIN(1.0, (double)acc.count / (double)total) : 0;
double frac = 0.03 + (stageOffset + stageLocal * stageWeight) * 0.97;
NSString *fmt = [stage isEqualToString:@"followers"]
? SCILocalized(@"Fetching followers (%lu/%ld)…")
: SCILocalized(@"Fetching following (%lu/%ld)…");
NSString *label = [NSString stringWithFormat:fmt, (unsigned long)acc.count, (long)total];
[strongSelf reportProgress:progress status:label fraction:frac];
id next = resp[@"next_max_id"];
NSString *nextMax = [next isKindOfClass:[NSString class]] ? next : ([next respondsToSelector:@selector(stringValue)] ? [next stringValue] : nil);
if (!nextMax.length || strongSelf.cancelled) {
completion(acc, strongSelf.cancelled ? [strongSelf errorWithCode:SCIProfileAnalyzerErrorCancelled message:SCILocalized(@"Cancelled")] : nil);
return;
}
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(SCI_PA_PAGE_DELAY_S * NSEC_PER_SEC)),
dispatch_get_main_queue(), ^{
[strongSelf pagePath:basePath acc:acc maxId:nextMax total:total stage:stage progress:progress completion:completion];
});
}];
}
@end