mirror of https://github.com/procxx/kepka.git
new open with menu for os x
This commit is contained in:
parent
13887d91d9
commit
471d83fb97
|
@ -491,7 +491,10 @@ void VideoSaveLink::doSave(bool forceSavingAs) const {
|
||||||
|
|
||||||
QString already = data->already(true);
|
QString already = data->already(true);
|
||||||
if (!already.isEmpty() && !forceSavingAs) {
|
if (!already.isEmpty() && !forceSavingAs) {
|
||||||
psOpenFile(already, true);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (!psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
||||||
QString name = already.isEmpty() ? QString(".mov") : already;
|
QString name = already.isEmpty() ? QString(".mov") : already;
|
||||||
|
@ -577,7 +580,10 @@ void AudioSaveLink::doSave(bool forceSavingAs) const {
|
||||||
|
|
||||||
QString already = data->already(true);
|
QString already = data->already(true);
|
||||||
if (!already.isEmpty() && !forceSavingAs) {
|
if (!already.isEmpty() && !forceSavingAs) {
|
||||||
psOpenFile(already, true);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (!psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
||||||
QString name = already.isEmpty() ? QString(".ogg") : already;
|
QString name = already.isEmpty() ? QString(".ogg") : already;
|
||||||
|
@ -678,7 +684,10 @@ void DocumentSaveLink::doSave(bool forceSavingAs) const {
|
||||||
|
|
||||||
QString already = data->already(true);
|
QString already = data->already(true);
|
||||||
if (!already.isEmpty() && !forceSavingAs) {
|
if (!already.isEmpty() && !forceSavingAs) {
|
||||||
psOpenFile(already, true);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (!psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
QDir alreadyDir(already.isEmpty() ? QDir() : QFileInfo(already).dir());
|
||||||
QString name = already.isEmpty() ? data->name : already, filter;
|
QString name = already.isEmpty() ? data->name : already, filter;
|
||||||
|
|
|
@ -1050,7 +1050,12 @@ void MainWidget::videoLoadProgress(mtpFileLoader *loader) {
|
||||||
video->finish();
|
video->finish();
|
||||||
QString already = video->already();
|
QString already = video->already();
|
||||||
if (!already.isEmpty() && video->openOnSave) {
|
if (!already.isEmpty() && video->openOnSave) {
|
||||||
psOpenFile(already, video->openOnSave < 0);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (video->openOnSave < 0 && !psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
} else {
|
||||||
|
psOpenFile(already, video->openOnSave < 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1105,7 +1110,12 @@ void MainWidget::audioLoadProgress(mtpFileLoader *loader) {
|
||||||
audioVoice()->play(audio);
|
audioVoice()->play(audio);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
psOpenFile(already, audio->openOnSave < 0);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (audio->openOnSave < 0 && !psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
} else {
|
||||||
|
psOpenFile(already, audio->openOnSave < 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1164,7 +1174,12 @@ void MainWidget::documentLoadProgress(mtpFileLoader *loader) {
|
||||||
psOpenFile(already);
|
psOpenFile(already);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
psOpenFile(already, document->openOnSave < 0);
|
QPoint pos(QCursor::pos());
|
||||||
|
if (document->openOnSave < 0 && !psShowOpenWithMenu(pos.x(), pos.y(), already)) {
|
||||||
|
psOpenFile(already, true);
|
||||||
|
} else {
|
||||||
|
psOpenFile(already, document->openOnSave < 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1654,6 +1654,10 @@ void psExecTelegram() {
|
||||||
_execUpdater(false);
|
_execUpdater(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
void psAutoStart(bool start, bool silent) {
|
void psAutoStart(bool start, bool silent) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -197,6 +197,8 @@ bool psCheckReadyUpdate();
|
||||||
void psExecUpdater();
|
void psExecUpdater();
|
||||||
void psExecTelegram();
|
void psExecTelegram();
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file);
|
||||||
|
|
||||||
void psPostprocessFile(const QString &name);
|
void psPostprocessFile(const QString &name);
|
||||||
void psOpenFile(const QString &name, bool openWith = false);
|
void psOpenFile(const QString &name, bool openWith = false);
|
||||||
void psShowInFolder(const QString &name);
|
void psShowInFolder(const QString &name);
|
||||||
|
|
|
@ -1110,6 +1110,10 @@ bool psCheckReadyUpdate() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file) {
|
||||||
|
return objc_showOpenWithMenu(x, y, file);
|
||||||
|
}
|
||||||
|
|
||||||
void psPostprocessFile(const QString &name) {
|
void psPostprocessFile(const QString &name) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -221,6 +221,8 @@ bool psCheckReadyUpdate();
|
||||||
void psExecUpdater();
|
void psExecUpdater();
|
||||||
void psExecTelegram();
|
void psExecTelegram();
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file);
|
||||||
|
|
||||||
void psPostprocessFile(const QString &name);
|
void psPostprocessFile(const QString &name);
|
||||||
void psOpenFile(const QString &name, bool openWith = false);
|
void psOpenFile(const QString &name, bool openWith = false);
|
||||||
void psShowInFolder(const QString &name);
|
void psShowInFolder(const QString &name);
|
||||||
|
|
|
@ -59,6 +59,8 @@ void objc_debugShowAlert(const QString &str);
|
||||||
void objc_outputDebugString(const QString &str);
|
void objc_outputDebugString(const QString &str);
|
||||||
int64 objc_idleTime();
|
int64 objc_idleTime();
|
||||||
|
|
||||||
|
bool objc_showOpenWithMenu(int x, int y, const QString &file);
|
||||||
|
|
||||||
void objc_showInFinder(const QString &file, const QString &path);
|
void objc_showInFinder(const QString &file, const QString &path);
|
||||||
void objc_openFile(const QString &file, bool openwith);
|
void objc_openFile(const QString &file, bool openwith);
|
||||||
void objc_start();
|
void objc_start();
|
||||||
|
|
|
@ -369,6 +369,209 @@ int64 objc_idleTime() { // taken from https://github.com/trueinteractions/tint/i
|
||||||
return (result == err) ? -1 : int64(result);
|
return (result == err) ? -1 : int64(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@interface OpenWithApp : NSObject {
|
||||||
|
NSString *fullname;
|
||||||
|
NSURL *app;
|
||||||
|
NSImage *icon;
|
||||||
|
}
|
||||||
|
@property (nonatomic, retain) NSString *fullname;
|
||||||
|
@property (nonatomic, retain) NSURL *app;
|
||||||
|
@property (nonatomic, retain) NSImage *icon;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation OpenWithApp
|
||||||
|
@synthesize fullname, app, icon;
|
||||||
|
|
||||||
|
- (void) dealloc {
|
||||||
|
[fullname release];
|
||||||
|
[app release];
|
||||||
|
[icon release];
|
||||||
|
[super dealloc];
|
||||||
|
}
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
@interface OpenFileWithInterface : NSObject {
|
||||||
|
}
|
||||||
|
|
||||||
|
- (id) init:(NSString *)file;
|
||||||
|
- (BOOL) popupAtX:(int)x andY:(int)y;
|
||||||
|
- (void) itemChosen:(id)sender;
|
||||||
|
- (void) dealloc;
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation OpenFileWithInterface {
|
||||||
|
NSString *toOpen;
|
||||||
|
|
||||||
|
NSURL *defUrl;
|
||||||
|
NSString *defBundle, *defName, *defVersion;
|
||||||
|
NSImage *defIcon;
|
||||||
|
|
||||||
|
NSMutableArray *apps;
|
||||||
|
|
||||||
|
NSMenu *menu;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void) fillAppByUrl:(NSURL*)url bundle:(NSString**)bundle name:(NSString**)name version:(NSString**)version icon:(NSImage**)icon {
|
||||||
|
NSBundle *b = [NSBundle bundleWithURL:url];
|
||||||
|
if (b) {
|
||||||
|
NSString *path = [url path];
|
||||||
|
*name = [[NSFileManager defaultManager] displayNameAtPath: path];
|
||||||
|
if (!*name) *name = (NSString*)[b objectForInfoDictionaryKey:@"CFBundleDisplayName"];
|
||||||
|
if (!*name) *name = (NSString*)[b objectForInfoDictionaryKey:@"CFBundleName"];
|
||||||
|
if (*name) {
|
||||||
|
*bundle = [b bundleIdentifier];
|
||||||
|
if (bundle) {
|
||||||
|
*version = (NSString*)[b objectForInfoDictionaryKey:@"CFBundleShortVersionString"];
|
||||||
|
*icon = [[NSWorkspace sharedWorkspace] iconForFile: path];
|
||||||
|
if (*icon && [*icon isValid]) [*icon setSize: CGSizeMake(16., 16.)];
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*bundle = *name = *version = nil;
|
||||||
|
*icon = nil;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (id) init:(NSString*)file {
|
||||||
|
toOpen = file;
|
||||||
|
if (self = [super init]) {
|
||||||
|
NSURL *url = [NSURL fileURLWithPath:file];
|
||||||
|
defUrl = [[NSWorkspace sharedWorkspace] URLForApplicationToOpenURL:url];
|
||||||
|
if (defUrl) {
|
||||||
|
[self fillAppByUrl:defUrl bundle:&defBundle name:&defName version:&defVersion icon:&defIcon];
|
||||||
|
if (!defBundle || !defName) {
|
||||||
|
defUrl = nil;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
NSArray *appsList = (NSArray*)LSCopyApplicationURLsForURL(CFURLRef(url), kLSRolesAll);
|
||||||
|
NSMutableDictionary *data = [NSMutableDictionary dictionaryWithCapacity:16];
|
||||||
|
int fullcount = 0;
|
||||||
|
for (id app in appsList) {
|
||||||
|
if (fullcount > 15) break;
|
||||||
|
|
||||||
|
NSString *bundle = nil, *name = nil, *version = nil;
|
||||||
|
NSImage *icon = nil;
|
||||||
|
[self fillAppByUrl:(NSURL*)app bundle:&bundle name:&name version:&version icon:&icon];
|
||||||
|
if (bundle && name) {
|
||||||
|
if ([bundle isEqualToString:defBundle] && [version isEqualToString:defVersion]) continue;
|
||||||
|
NSString *key = [[NSArray arrayWithObjects:bundle, name, nil] componentsJoinedByString:@"|"];
|
||||||
|
if (!version) version = @"";
|
||||||
|
|
||||||
|
NSMutableDictionary *versions = (NSMutableDictionary*)[data objectForKey:key];
|
||||||
|
if (!versions) {
|
||||||
|
versions = [NSMutableDictionary dictionaryWithCapacity:2];
|
||||||
|
[data setValue:versions forKey:key];
|
||||||
|
}
|
||||||
|
if (![versions objectForKey:version]) {
|
||||||
|
[versions setValue:[NSArray arrayWithObjects:name, icon, app, nil] forKey:version];
|
||||||
|
++fullcount;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (fullcount || defUrl) {
|
||||||
|
apps = [NSMutableArray arrayWithCapacity:fullcount];
|
||||||
|
for (id key in data) {
|
||||||
|
NSMutableDictionary *val = (NSMutableDictionary*)[data objectForKey:key];
|
||||||
|
for (id ver in val) {
|
||||||
|
NSArray *app = (NSArray*)[val objectForKey:ver];
|
||||||
|
OpenWithApp *a = [[OpenWithApp alloc] init];
|
||||||
|
NSString *fullname = (NSString*)[app objectAtIndex:0], *version = (NSString*)ver;
|
||||||
|
BOOL showVersion = ([val count] > 1);
|
||||||
|
if (!showVersion) {
|
||||||
|
NSError *error = NULL;
|
||||||
|
NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:@"^\\d+\\.\\d+\\.\\d+(\\.\\d+)?$" options:NSRegularExpressionCaseInsensitive error:&error];
|
||||||
|
showVersion = ![regex numberOfMatchesInString:version options:NSMatchingWithoutAnchoringBounds range:{0,[version length]}];
|
||||||
|
}
|
||||||
|
if (showVersion) fullname = [[NSArray arrayWithObjects:fullname, @" (", version, @")", nil] componentsJoinedByString:@""];
|
||||||
|
[a setFullname:fullname];
|
||||||
|
[a setIcon:(NSImage*)[app objectAtIndex:1]];
|
||||||
|
[a setApp:(NSURL*)[app objectAtIndex:2]];
|
||||||
|
[apps addObject:a];
|
||||||
|
[a release];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[apps sortUsingDescriptors:[NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:@"fullname" ascending:YES]]];
|
||||||
|
[appsList release];
|
||||||
|
menu = nil;
|
||||||
|
}
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (BOOL) popupAtX:(int)x andY:(int)y {
|
||||||
|
if (![apps count] && !defName) return NO;
|
||||||
|
menu = [[NSMenu alloc] initWithTitle:@"Open With"];
|
||||||
|
|
||||||
|
int index = 0;
|
||||||
|
if (defName) {
|
||||||
|
NSMenuItem *item = [menu insertItemWithTitle:[[NSArray arrayWithObjects:defName, @" (default)", nil] componentsJoinedByString:@""] action:@selector(itemChosen:) keyEquivalent:@"" atIndex:index++];
|
||||||
|
if (defIcon) [item setImage:defIcon];
|
||||||
|
[item setTarget:self];
|
||||||
|
[menu insertItem:[NSMenuItem separatorItem] atIndex:index++];
|
||||||
|
}
|
||||||
|
if ([apps count]) {
|
||||||
|
for (id a in apps) {
|
||||||
|
OpenWithApp *app = (OpenWithApp*)a;
|
||||||
|
NSMenuItem *item = [menu insertItemWithTitle:[a fullname] action:@selector(itemChosen:) keyEquivalent:@"" atIndex:index++];
|
||||||
|
if ([app icon]) [item setImage:[app icon]];
|
||||||
|
[item setTarget:self];
|
||||||
|
}
|
||||||
|
[menu insertItem:[NSMenuItem separatorItem] atIndex:index++];
|
||||||
|
}
|
||||||
|
NSMenuItem *item = [menu insertItemWithTitle:@"Other..." action:@selector(itemChosen:) keyEquivalent:@"" atIndex:index++];
|
||||||
|
[item setTarget:self];
|
||||||
|
|
||||||
|
[menu popUpMenuPositioningItem:nil atLocation:CGPointMake(x, y) inView:nil];
|
||||||
|
|
||||||
|
return YES;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void) itemChosen:(id)sender {
|
||||||
|
NSArray *items = [menu itemArray];
|
||||||
|
NSURL *url = nil;
|
||||||
|
for (int i = 0, l = [items count]; i < l; ++i) {
|
||||||
|
if ([items objectAtIndex:i] == sender) {
|
||||||
|
if (defName) i -= 2;
|
||||||
|
if (i < 0) {
|
||||||
|
url = defUrl;
|
||||||
|
} else if (i < int([apps count])) {
|
||||||
|
url = [(OpenWithApp*)[apps objectAtIndex:i] app];
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (url) {
|
||||||
|
[[NSWorkspace sharedWorkspace] openFile:toOpen withApplication:[url path]];
|
||||||
|
} else {
|
||||||
|
objc_openFile(objcString(toOpen), true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void) dealloc {
|
||||||
|
if (apps) [apps release];
|
||||||
|
[super dealloc];
|
||||||
|
if (menu) [menu release];
|
||||||
|
}
|
||||||
|
|
||||||
|
@end
|
||||||
|
|
||||||
|
bool objc_showOpenWithMenu(int x, int y, const QString &f) {
|
||||||
|
NSString *file = QNSString(f).s();
|
||||||
|
@try {
|
||||||
|
OpenFileWithInterface *menu = [[OpenFileWithInterface alloc] init:file];
|
||||||
|
QRect r = QApplication::desktop()->screenGeometry(QPoint(x, y));
|
||||||
|
y = r.y() + r.height() - y;
|
||||||
|
return !![menu popupAtX:x andY:y];
|
||||||
|
}
|
||||||
|
@catch (NSException *exception) {
|
||||||
|
}
|
||||||
|
@finally {
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
void objc_showInFinder(const QString &file, const QString &path) {
|
void objc_showInFinder(const QString &file, const QString &path) {
|
||||||
[[NSWorkspace sharedWorkspace] selectFile:QNSString(file).s() inFileViewerRootedAtPath:QNSString(path).s()];
|
[[NSWorkspace sharedWorkspace] selectFile:QNSString(file).s() inFileViewerRootedAtPath:QNSString(path).s()];
|
||||||
}
|
}
|
||||||
|
|
|
@ -2295,6 +2295,10 @@ void psExecTelegram() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
void _manageAppLnk(bool create, bool silent, int path_csidl, const wchar_t *args, const wchar_t *description) {
|
void _manageAppLnk(bool create, bool silent, int path_csidl, const wchar_t *args, const wchar_t *description) {
|
||||||
WCHAR startupFolder[MAX_PATH];
|
WCHAR startupFolder[MAX_PATH];
|
||||||
HRESULT hres = SHGetFolderPath(0, path_csidl, 0, SHGFP_TYPE_CURRENT, startupFolder);
|
HRESULT hres = SHGetFolderPath(0, path_csidl, 0, SHGFP_TYPE_CURRENT, startupFolder);
|
||||||
|
|
|
@ -199,6 +199,8 @@ bool psCheckReadyUpdate();
|
||||||
void psExecUpdater();
|
void psExecUpdater();
|
||||||
void psExecTelegram();
|
void psExecTelegram();
|
||||||
|
|
||||||
|
bool psShowOpenWithMenu(int x, int y, const QString &file);
|
||||||
|
|
||||||
void psPostprocessFile(const QString &name);
|
void psPostprocessFile(const QString &name);
|
||||||
void psOpenFile(const QString &name, bool openWith = false);
|
void psOpenFile(const QString &name, bool openWith = false);
|
||||||
void psShowInFolder(const QString &name);
|
void psShowInFolder(const QString &name);
|
||||||
|
|
|
@ -1813,7 +1813,7 @@
|
||||||
CODE_SIGN_IDENTITY = "";
|
CODE_SIGN_IDENTITY = "";
|
||||||
COPY_PHASE_STRIP = NO;
|
COPY_PHASE_STRIP = NO;
|
||||||
CURRENT_PROJECT_VERSION = 0.7.8;
|
CURRENT_PROJECT_VERSION = 0.7.8;
|
||||||
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
|
DEBUG_INFORMATION_FORMAT = dwarf;
|
||||||
DYLIB_COMPATIBILITY_VERSION = 0.7;
|
DYLIB_COMPATIBILITY_VERSION = 0.7;
|
||||||
DYLIB_CURRENT_VERSION = 0.7.8;
|
DYLIB_CURRENT_VERSION = 0.7.8;
|
||||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||||
|
|
Loading…
Reference in New Issue