阻止iOS通话中状态栏在屏幕外推送Phonegap UIWebView

时间:2014-02-07 05:10:33

标签: ios iphone objective-c cordova statusbar

我的问题是:每当iPhone用户正在通话,或者正在使用他或她的手机作为热点时,iOS 7状态栏会被放大,从而将我的Phonegap应用程序的UIWebView推离屏幕底部。放大的状态栏被称为“通话状态栏”。见下图:

Wonky in-call status bar.

Stack Overflow答案我试图解决这个问题:

Iphone- How to resize view when call status bar is toggled?

How In-Call status bar impacts UIViewController's view size ? (and how to handle it properly)

此外,Phonegap似乎没有任何类型的事件通知我状态栏的更改。听取Phonegap“暂停”事件是没用的,因为1)it's known to have quirks in iOS和2)它并没有真正涵盖热点案例。

我的Objective-C技能非常小,我只需要在进行必要的4个多小时的谷歌搜索,堆栈溢出,嚎叫等等之后提出这类问题......

堆叠神的溢出,让我觉得你的书呆子愤怒。

2 个答案:

答案 0 :(得分:9)

根据Jef的建议提出以下解决方案。你想要做的是以下几点:

  • 观察原生didChangeStatusBarFrame代表
  • 通过原生statusBarFrame
  • 获取有关状态栏的尺寸信息
  • 通过触发传递信息的事件将信息公开给您的网页视图

我已经设置了Github repo,其中包含您在此答案中找到的所有代码。

在AppDelegate中设置通知

// Appdelegate.m
- (void)application:(UIApplication *)application didChangeStatusBarFrame:(CGRect)oldStatusBarFrame
{
  NSMutableDictionary *statusBarChangeInfo = [[NSMutableDictionary alloc] init];
  [statusBarChangeInfo setObject:@"statusbarchange" 
                     forKey:@"frame"];

  [[NSNotificationCenter defaultCenter] postNotificationName:@"statusbarchange" 
                                        object:self 
                                        userInfo:statusBarChangeInfo];
 }

使statusBarChange选择器可用

// MainViewController.h
@protocol StatusBarChange <NSObject>
    -(void)onStatusbarChange:(NSNotification*)notification;
@end

设置监听器。这样,originsize字典会在statusBarFrame更改时随时更改并在Web视图中触发传递此数据的事件。

// MainViewController.m
- (void)onStatusbarChange:(NSNotification*)notification
{
    // Native code for
    NSMutableDictionary *eventInfo = [self getStatusBarInfo];
    [self notifiy:notification.name withInfo:eventInfo];
}

- (void)notifiy:(NSString*)event withInfo:(NSMutableDictionary*)info
{
    NSString *json = [self toJSON:info];
    NSString *cmd = [NSString stringWithFormat:@"cordova.fireWindowEvent('\%@\', %@)", event, json];
    [self.webView stringByEvaluatingJavaScriptFromString:cmd];
}

- (NSMutableDictionary *)getStatusBarInfo
{
    CGRect statusBarFrame = [[UIApplication sharedApplication] statusBarFrame];

    NSMutableDictionary *statusBarInfo = [[NSMutableDictionary alloc] init];
    NSMutableDictionary *size = [[NSMutableDictionary alloc] init];
    NSMutableDictionary *origin = [[NSMutableDictionary alloc] init];

    size[@"height"] = [NSNumber numberWithInteger:((int) statusBarFrame.size.height)];
    size[@"width"] = [NSNumber numberWithInteger:((int) statusBarFrame.size.width)];

    origin[@"x"] = [NSNumber numberWithInteger:((int) statusBarFrame.origin.x)];
    origin[@"y"] = [NSNumber numberWithInteger:((int) statusBarFrame.origin.y)];

    statusBarInfo[@"size"] = size;
    statusBarInfo[@"origin"] = origin;

    return statusBarInfo;
}

- (NSString *) toJSON:(NSDictionary *)dictionary {
    NSError *error;
    NSData *jsonData = [NSJSONSerialization dataWithJSONObject:dictionary options:NSJSONWritingPrettyPrinted error:&error];
    return [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
}

- (void)dealloc
{
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

所有这些都可以让您收听window.statusbarchange事件,例如像这样:

// www/js/index.js
window.addEventListener('statusbarchange', function(e){
  // Use e.size.height to adapt to the changing status bar      

}, false)

答案 1 :(得分:1)

我会说当你从背景回来时总会发生这种情况,不是吗?

在其他情况下,如果没有您的应用通过来电等方式将应用程序至少短暂地推到后台,那么该栏可能会放大吗?

如果确定你可以查询代表中的状态栏高度 - (void)会(orDid)恢复并相应调整吗?

如果它确实发生而没有让前景变得更加困难,我们需要找出要观察的通知,我知道有一个audioSession中断通知,如果有来电,不确定热点事情,但肯定也有通知...

在这里编辑确定,选择其中一个通知观察..

UIApplicationWillChangeStatusBarFrameNotification

UIApplicationDidChangeStatusBarFrameNotification

或者在委托中实现其中一个回调

-application:willChangeStatusBarFrame:

-application:didChangeStatusBarFrame: