mirror of
https://github.com/faroukbmiled/RyukGram.git
synced 2026-06-08 00:13: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
214 lines
9.5 KiB
Objective-C
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
|