애플에서  푸쉬서비스 토큰 방식이 추가되어서 알아봤다.


중요개선사항

1. HTTP/2 로 영구 푸시를 기반으로하는 새로운 프로토콜을 도입했다. 

   HTTP/2는 단일 연결을 통해 여러개의 스트림을 지원하는 매우 빠른 바이너리 프로토콜입니다.

2. 프로토콜은 더 이상 활성화되지 않은 장치 토큰을 나타내는 즉각적인 피드백 제공을 지원합니다. 푸쉬를 보내는 API를 호출했을 때, PUSH가 전달되지 않으면 Response로 알려주는게 추가되었다.

3. 페이로드가 최대 4KB로 증가되었다.


하나의 생성된 Key 토큰으로 개발자의 푸쉬를 보낼수가 있습니다.

이로 인해 많은 개발자가 여러 인증서를 관리, 갱신 및 취소 할 수있는 어려움이 줄어 들었습니다.



WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





NSDate *pastDate = ...;

NSDate *nowDate = [NSDate date]; NSCalendar *calendar = [NSCalendar currentCalendar]; NSUInteger calendarUnit = NSCalendarUnitDay | NSCalendarUnitHour | NSCalendarUnitMinute | NSCalendarUnitSecond;


NSDateComponents *dateComp = [calendar components: calendarUnit fromDate: nowDate toDate: pastDate options:0]; NSInteger day = dateComp.day; NSInteger hour = dateComp.hour; NSInteger min = dateComp.minute; NSInteger sec = dateComp.second;



WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





iOS는 왜 해킹에 강할까? 라는 의문점을 가지고 찾아 보는 중에 DEP라는 매커니즘이 적용된 것을 알 수 있었다.


01 DEP란?

프로세서가 데이터 영역은 실행하지 않고 코드 영역만 구동하도록 메모리에서 코드 영역과 데이터 영역을 구분하기 위한 메커니즘.

즉, 공격 코드가 다운로드한 코드를 실행시키는 것을 막을 수 있다는 뜻이다.


다운로드한 코드는 DEP를 통해 데이터로 취급되어서 실행을 할 수 없게 된다.

그래서 iOS를 공격할 때는 ROP를 사용한단다. ( ROP에 대해선 나중에..)


02 iOS의 코드서명.

iOS의 코드서명은 매커니즘은 DEP와 비슷하지만 좀 더 강력하다.

DEP를 매커니즘이 되어있다면, ROP를 활용해 쓰고 실행할 수 있는 메모리 영역을 먼저 생성한 후, 공격에 필요한 페이로드를 작성하고 실행시키면 되지만,

iOS의 코드서명의 경우 신뢰할 수 있는 기관으로 부터 서명되지 않은 코드는 아무것도 실행시킬 수 없다.

따라서 ROP를 실행시킬 수 있을 뿐 페이로드는 실행하기 어렵다.




WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





iOS10.0이 나온지 꽤 시간이 지났지만,

iOS 개발자로써 iOS가 새로 나올때마다 정리를 해둬야 될 것 같아서 뒤늦게 애플문서를 보고 정리를 해야겠다 생각하고 글을 쓴다.

자세한 내용

(https://developer.apple.com/library/prerelease/content/releasenotes/General/WhatsNewIniOS/Articles/iOS10.html#//apple_ref/doc/uid/TP40017084-SW1)


Providing Haptic Feedback

아이폰7, 아이폰7Plus 에서 기존 맥북에 있던 트랙패드처럼 홈버튼이 햅틱방식으로 변경되었다.

UIKit의 UIFeedbackGenerator로 사용할 수 있다.

아이폰7이 출시 된 후 만져본 느낌으론 어색했지만 적응이 된다면 상당히 괜찮을것 같다.


SiriKit

SiriKit이 오픈되었다. 일반 앱에서도 Siri를 이용한 앱을 만들수 있는데, 현재는 아래 종류의 앱만이 이용할 수 있다.

  • 음성/영상 전화
  • 메시지
  • 돈 주고받기
  • 사진찾기
  • 얘약관련
  • 운동 매니저
  • 카플레이
  • 레스토랑예약(애플제공한에서)
사용자가 서비스를 포함하는 요청을 하면 시리 킷은 사용자의 요청을 설명하고 요청에 해당되는 데이터를 제공할 수 있다.
자세한 사항은 SiriKit Programming Guide을 참고 해야 될 것 같다.

Proactive Suggestions

프로액티비는 검색을 통해 사용자에게 필요한 앱을 제안하는 기능이다. iOS9 에서는 App search를 적용했다면, Spotlight나 Safari 검색결과, Handoff, Siri 제안 등을 노출시킬 수 있었는데, iOS10에서는 사용자가 앱에서 어떤 동작을 하는지에 대한 정보를 제공해서 지도, 카플레이, 앱 전환, Siri대화, 잠금화면 등 앱을 추가적으로 노출시킬 수 있다.

Integrating with the Messages App

iOS10에서 크게 바뀐 것중 하나가 메시지 앱이다. 
코드를 건드릴 필요 없이 이미지만으로 스티커 앱을 만들 수있고, 커스텀 UI를 통해 텍스트,이미지,미디어 파일을 대화 안에서 생성하고 보내고 업데이트 할 수 있다. 단, Messages프레임워크를 사용해야 한다.

User Notifications

이번에 추가된 User Notifications 프레임워크는 기존에 로컬알림과 원격알림을 모두 관리할 수 있게 되었다. 플랫폼 간의 사용 용법이 통일화 되었고 playload를 디코드하여 이미지를 표시하거나 end-to-end  암호화를 지원할 수 있게 되었다.

Speech Recognition

연속적으로 음성 인식을 할 수있는 API가 추가되어, 녹음되거나 실시간으로 재생되는 음성을 인식하고 텍스트로 변환할 수 있게 되었다.
아래의 간단한 코드로 구현이 가능하다.

let recognizer = SFSpeechRecognizer()  
let request = SFSpeechURLRecognitionRequest(url: audioFileURL)  
recognizer?.recognitionTask(with: request, resultHandler: { (result, error) in  
     print (result?.bestTranscription.formattedString)
})
다만, 음성인식의 정확도를 높이기 위해 음성정보가 애플 서버에 임시적으로 저장되기 때문에 이를 위해 Info.plist에 사용자가 알 수 있도록 미리 표시해야 한다.

Wide Color

아이패드 프로에서 기존 sRGB보다 더 넓어진 확장된 sRGB(extended sRGB) 색영역을 표현할 수 있다. 이를 지원하기 위해 Core Graphics, Core Image, Metal, AVFoundation 프레임워크 들이 내부적으로 변경되었다.
  • iOS10에서, UIColor 클래스 확장 된 sRGB 색 공간을 사용하며, 그 초기화는 더 이상 0.0 내지 1.0의 원료 성분 값을 고정하지 않는다. (당신은 색상을 만들거나 그 구성 요소 값의 색상을 요구하는지 여부) 앱 구성 요소 값을 클램프은 UIKit에 의존하는 경우, 당신은 당신이 iOS10 에 연결하면 앱의 동작을 변경해야합니다.
  • 아이 패드 프로 (9.7 인치)에 UIView의 그리기 사용자 정의 수행 할 때, 기본 도면 환경을 확장 sRGB 색 공간으로 구성되어 있습니다.
  • 앱이 사용자 정의 이미지 오브젝트를 렌더링하는 경우, 대상 비트 맵이 확장 된 범위 또는 표준 범위 형식을 사용하여 생성되는지 여부를 제어하는 ​​새로운 UIGraphicsImageRenderer 클래스를 사용합니다.
  • 만약 그러한 코어 그래픽 또는 금속과 같은 낮은 레벨의 API를 사용하여 와이드 영역 기기에 자신의 화상 처리를 수행하는 경우에는 확장 된 범위의 컬러 공간 및 16 비트 부동 소수점 성분 값을 지원하는 픽셀 포맷을 사용한다. 색상 값의 클램핑하는 것이 필요한 경우, 당신은 이렇게 명시 적으로 수행해야합니다.
  • 코어 그래픽, 코어 이미지 및 금속 성능 쉐이더 쉽게 색 공간 사이의 색상과 이미지를 변환하기위한 새로운 옵션을 제공합니다.

Adapting to the True Tone Display

아이패드 프로 9.7인치에서 추가된 트루톤 디스플레이는 다양한 빛 환경에서 최적화된 색상으로 화면을 보여주게 된다. 이는 Info.plist 파일의 UIWhitePointAdaptivityStyle 키 값을 적절하게 설정할 수 있다.

App Search Enhancements

iOS10에서 CoreSpotlight 프레임워크를 통해 다음의 검색이 개선 되었다.
  • 앱 내의 검색
  • 이어서 검색 계속하기
  • 차동 개인 정보 보호와 딥 링크 인기를 크라우드 소싱
  • 유효성검증 결과의 시각화

Widget Enhancements

잠금화면에서도 위젯을 볼 수 있으며, 어떤 배경에서도 잘 보이게 개선되었다.

Apple Pay Enhancements

애플페이가 많이 확장되었다.

Security and Privacy Enhancements

iOS10은 몇 가지 변경 및 코드의 보안을 강화하고 사용자 데이터의 개인 정보를 유지하는 데 개선되었다.. 이러한 항목에 대한 자세한 내용은 https://developer.apple.com/security/를 참고할 수 있다

CallKit

CallKit 이 추가되어 VoIP앱이 일반 전화가 오는 것과 동일한 화면에 표시 할 수 있다. VoIP앱을 이요한 기록이 네이티브 전화 앱의 즐겨찾기와 최근 사용목록에도 표시되고 extension 을 통해 전화 관리나 전화를 건 사람에 대한 정보 표시가 가능해 우리나라 악질적인 스팸을 아이폰에서도 막을 수 있게 되었다.

News Publisher Enhancements

뉴스 게시자 쉽게 애플 뉴스 포맷을 사용 애플 뉴스 아름답게 디자인 뉴스, 잡지, 웹 콘텐츠를 제공 할 수있다. 누구나 독립 출판사와 블로거 주요 잡지 또는 뉴스 조직에서 가입 할 수 있습니다. 
자세한 사항은 visit https://newsresources.apple.com. 

Video Subscriber Account

iOS10에서 추가된 Video Subscriber Account를 통해 케이블 사업자나 위성 사업자에 구독중인 계정을 인증하고 이를 통해 앱에서 권한을 조절할 수 있게 되었다. Apple TV에서 각 채널별로 회원가입하고 로그인하고 복잡하게 진행된던 점이 SSO로 해결을 할 수 있을것으로 예상된다.

App Extensions
iOS 10에서 추가된 앱 익스텐션은 다음과 같다.
  • Call Directory

  • Intents

  • Intents UI

  • Messages

  • Notification Content

  • Notification Service

  • Sticker Pack



WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





1. NSOperation

- 고수준의 편리한 API제공한다.(KVO 키 감시, operation cancel제어등..)

- 일반 스레드보다 약간 처리속도에 손실이 있다고 한다. 그럼에도 불구하고 편리한 메소드들은 이러한 단점을 커버하기 충분하다.

- GCD의 객체형이라고 볼 수 있을까?

사용후 알아서 메모리 해제.


2. GCD

- 블록으로 구현되어 있어 간단하게 사용가능하고 또한 코드 가독성도 좋다.

- 저수준 스레드 구현가능되기에 속도면에서 좋다고 한다. 다만, 상태감시, 오퍼레이션 캔슬 등등 다 직접 만들어야 한다는 귀찮음을 동반한다.

GCD는 NSOperation과 유사합니다. GCD는 C언어 기반이라는 차이점을 갖습니다.

사용후 직접 메모리 해제.



NSOperation과 GCD의 차이가 뭘지 궁금해서 검색해서 찾아본 결과인데,

잘못된 정보가 있다면 댓글 부탁드립니다.




WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  2개가 달렸습니다.
  1. 비밀댓글입니다
secret





- ViewController.m : 아이폰에서 값을 보낼 화면에서 다음과 같이 하시면 됩니다.


#import <WatchConnectivity/WatchConnectivity.h>

@interface ViewController () <WCSessionDelegate>


@end


- (void)viewDidLoad {

    [super viewDidLoad];

    //WCSession을 activate합니다.

    if(WCSession.isSupported){

        WCSession* session = WCSession.defaultSession;

        session.delegate = self;

        [session activateSession];

    }

}

-(void)packageAndSendMessage{

    //보낼 Dictionary를 설정합니다.

    NSDictionary* request = "보낼 Dictionary";

    

    if(WCSession.isSupported){

        WCSession* session = WCSession.defaultSession;

        session.delegate = self;

        [session activateSession];

        if(session.reachable)

        {

            //sendMessage로 Dictionary를 전송합니다.

            [session sendMessage:request replyHandler: ^(NSDictionary<NSString *,id> * __nonnull replyMessage)

             {

                 dispatch_async(dispatch_get_main_queue(), ^{

                     NSDictionary* message = replyMessage;

                     NSString* response = message[@"response"];

                     if(response) {

                         NSLog(@"response = %@", response);

                     }

                     else {

                         NSLog(@"nil");

                     }

                 });

             }

                errorHandler:^(NSError * __nonnull error) {

                    dispatch_async(dispatch_get_main_queue(), ^{

                        NSLog(@"error = %@",error.localizedDescription);

                    });

                }

             ];

        }

        else

        {

            NSLog(@"Session Not reachable");

        }

        

    }

    else

    {

        NSLog(@"Session Not Supported");

    }

}




- InterfaceController.m

#import <WatchConnectivity/WatchConnectivity.h>

@interface InterfaceController() <WCSessionDelegate>
- (void)session:(WCSession *)session didReceiveMessage:(NSDictionary<NSString *, id> *)message replyHandler:(void(^)(NSDictionary<NSString *, id> *replyMessage))replyHandler
{
    if(message){
        NSString* command = [message objectForKey:@"보낸 Dictionary로 값을 가져오면 됩니다."];
        NSLog(@"command = %@", command);
    }
}

/**
 Standard WatchKit delegate
 */
-(void)sessionWatchStateDidChange:(nonnull WCSession *)session
{
    if(WCSession.isSupported){
        WCSession* session = WCSession.defaultSession;
        session.delegate = self;
        [session activateSession];
        
    }
}



WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





KeyChain 이란?

KeyChain은 iOS의 다양한 응용 프로그램에서 비밀번호나 계정등을 저장하는 암호화 되어있는 저장소입니다.

안전한 앱을 만들기 위해서 보안이 필요한 요소들을 분리하고 암호화 하는 노력이 필요합니다.

여기에서는 안전하게 데이터를 저장하기 위한 장소인 Keychain에 대해서 설명하고 어떻게 사용하는지 알아보겠습니다.


KeychainItemWrapper 클래스 사용하기

Keychain의 API들은 Security.framework에 정의되어 있는데 C함수로 정의되어 있습니다. 이들을 사용하기 위해 애플에서 제공해주는 KeychainItemWrapper를 사용하는 경우가 많습니다. (https://developer.apple.com/library/ios/samplecode/GenericKeychain/Introduction/Intro.html#//apple_ref/doc/uid/DTS40007797-Intro-DontLinkElementID_2)

하지만 arc가 적용되어 있지 않기 때문에 arc가 적용된 버젼을 첨부파일에 첨부하겠습니다.


 KeychainItemWrapper.h



KeyChain을 사용하는 방법

알아볼 예제에서는 UDID를 불러와서 저장하는 방법을 해보겠습니다.


KeychainItemWrapper *wrapper = [[KeychainItemWrapper alloc] initWithIdentifier:@"UUID" accessGroup:nil];


NSString *uuid = [wrapper objectForKey:(__bridge id)(kSecAttrAccount)];


if( uuid == nil || uuid.length == 0)

{

    // if there is not UUID in keychain, make UUID and save it.

    uuid = [[[UIDevice currentDevice] identifierForVendor] UUIDString];

    // save UUID in keychain

    [wrapper setObject:uuid forKey:(__bridge id)(kSecAttrAccount)];

}


NSLog(@"UUID = %@", uuid);


정리

앱에서 저장해야할 중요한 정보를 저장하는게 좋을것 같습니다. 간단한 정보는 UserDefaults에 저장하면 간단하겠지만, 아이디라던가 지워지면 안되는 정보들을 저장할 용도로 쓰는 것이 좋을 것 같습니다.


WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret





간단하게 비디오만 재생시키는 샘플입니다.


1. .h 파일

@property (strong, nonatomic) NSURL *videoURL;

@property (strong, nonatomic) AVPlayerViewController *avVideoController;


2. Play Video

-(void) playMovie{

    

    self.avVideoController = [[AVPlayerViewController alloc] init];

    AVPlayer *player = [AVPlayer playerWithURL:[self localMovieURL]];

    self.avVideoController.player = player;

    

    [self.avVideoController.view setFrame:CGRectMake (0, 0, self.view.frame.size.width, self.view.frame.size.height)];

    [self.view addSubview:self.avVideoController.view];

    

    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(videoPlayBackDidFinish:)

    name:AVPlayerItemDidPlayToEndTimeNotification

    object:[self.avVideoController.player currentItem]];

    

    [player play];

    

}


-(NSURL *)localMovieURL

{

    NSURL *theMovieURL = nil;

    NSBundle *bundle = [NSBundle mainBundle];

    if (bundle)

    {

        NSString *moviePath = [bundle pathForResource:@"IMG_0075" ofType:@"MOV"];

        if (moviePath)

        {

            theMovieURL = [NSURL fileURLWithPath:moviePath];

        }

    }

    return theMovieURL;

}


3. DidFinishNotification

- (void)videoPlayBackDidFinish:(NSNotification *)notification {

    

    [[NSNotificationCenter defaultCenter]removeObserver:self name:AVPlayerItemDidPlayToEndTimeNotification object:nil];

    

    [self.avVideoController.view removeFromSuperview];

    self.avVideoController = nil;

    

    UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"Video Playback" message:@"Just finished the video playback. The video is now removed." preferredStyle:UIAlertControllerStyleAlert];

    UIAlertAction *okayAction = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:nil];

    [alertController addAction:okayAction];

    [self presentViewController:alertController animated:YES completion:nil];

    

}


WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  1개가 달렸습니다.
  1. 잘보고갑니다 2016.04.08 10:02
    좋은정보 잘 배워갑니다 감사합니다
secret





Contacts.framework 는 iOS9 부터 기존 ABAddressBook.framework 를 대체합니다.


샘플프로젝트는 깃허브에 올려놓았습니다.

https://github.com/minjoongkim/iOS9-Contacts.framework-AddressBook-Sample



1. 주소록 불러오기

CNAuthorizationStatus status = [CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts];
if( status == CNAuthorizationStatusDenied || status == CNAuthorizationStatusRestricted)
{
    NSLog(@"access denied");
}
else
{
    //Create repository objects contacts
    CNContactStore *contactStore = [[CNContactStore alloc] init];

    //Select the contact you want to import the key attribute  ( https://developer.apple.com/library/watchos/documentation/Contacts/Reference/CNContact_Class/index.html#//apple_ref/doc/constant_group/Metadata_Keys )

    NSArray *keys = [[NSArray alloc]initWithObjects:CNContactIdentifierKey, CNContactEmailAddressesKey, CNContactBirthdayKey, CNContactImageDataKey, CNContactPhoneNumbersKey, CNContactViewController.descriptorForRequiredKeys, nil];

    // Create a request object
    CNContactFetchRequest *request = [[CNContactFetchRequest alloc] initWithKeysToFetch:keys];
    request.predicate = nil;

    [contactStore enumerateContactsWithFetchRequest:request
                                              error:nil
                                         usingBlock:^(CNContact* __nonnull contact, BOOL* __nonnull stop)
     {
         // Contact one each function block is executed whenever you get
         NSString *phoneNumber = @"";
         if( contact.phoneNumbers)
             phoneNumber = [[[contact.phoneNumbers firstObject] value] stringValue];

         NSLog(@"phoneNumber = %@", phoneNumber);
         NSLog(@"givenName = %@", contact.givenName);
         NSLog(@"familyName = %@", contact.familyName);
         NSLog(@"email = %@", contact.emailAddresses);


         [contactList addObject:contact];
     }];

    [contactTableView reloadData];
}

2. 피커뷰로 불러오기

// Create a new contact view
CNContactViewController *contactController = [CNContactViewController viewControllerForContact:contact];
contactController.delegate = self;
contactController.allowsEditing = YES;
contactController.allowsActions = YES;

// Display the view
[self.navigationController pushViewController:contactController animated:YES];

3. 주소록 추가하기

-(void)saveContact:(NSString*)familyName givenName:(NSString*)givenName phoneNumber:(NSString*)phoneNumber {
    CNMutableContact *mutableContact = [[CNMutableContact alloc] init];

    mutableContact.givenName = givenName;
    mutableContact.familyName = familyName;
    CNPhoneNumber * phone =[CNPhoneNumber phoneNumberWithStringValue:phoneNumber];

    mutableContact.phoneNumbers = [[NSArray alloc] initWithObjects:[CNLabeledValue labeledValueWithLabel:CNLabelPhoneNumberiPhone value:phone], nil];
    CNContactStore *store = [[CNContactStore alloc] init];
    CNSaveRequest *saveRequest = [[CNSaveRequest alloc] init];
    [saveRequest addContact:mutableContact toContainerWithIdentifier:store.defaultContainerIdentifier];

    NSError *error;
    if([store executeSaveRequest:saveRequest error:&error]) {
        NSLog(@"save");
        [self reloadContactList];
    }else {
        NSLog(@"save error");
    }
}

4. 주소록 업데이트 하기

-(void)updateContact:(CNContact*)contact memo:(NSString*)memo{
    CNMutableContact *mutableContact = contact.mutableCopy;

    mutableContact.note = memo;

    CNContactStore *store = [[CNContactStore alloc] init];
    CNSaveRequest *saveRequest = [[CNSaveRequest alloc] init];
    [saveRequest updateContact:mutableContact];

    NSError *error;
    if([store executeSaveRequest:saveRequest error:&error]) {
        NSLog(@"save");
    }else {
        NSLog(@"save error : %@", [error description]);
    }
}

5. 주소록 삭제하기

-(void)deleteContact:(CNContact*)contact {
    CNMutableContact *mutableContact = contact.mutableCopy;

    CNContactStore *store = [[CNContactStore alloc] init];
    CNSaveRequest *deleteRequest = [[CNSaveRequest alloc] init];
    [deleteRequest deleteContact:mutableContact];

    NSError *error;
    if([store executeSaveRequest:deleteRequest error:&error]) {
        NSLog(@"delete complete");
        [self reloadContactList];
    }else {
        NSLog(@"delete error : %@", [error description]);
    }

}

6. 주소록 상세보기

-(void)loadContactView:(CNContact*)contact {
    // Create a new contact view
    CNContactViewController *contactController = [CNContactViewController viewControllerForContact:contact];
    contactController.delegate = self;
    contactController.allowsEditing = YES;
    contactController.allowsActions = YES;

    // Display the view
    [self.navigationController pushViewController:contactController animated:YES];
}



WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  2개가 달렸습니다.
  1. 김련호 2016.02.26 09:36
    안녕하세요,

    샘플 코드 잘 보았습니다^^ 많은 도움이 되었네요~

    제 코드에서 뭔가 문제가 있는 줄 알았는데, Github에 올려두신 샘플에서도 동일한 문제가 있는 부분이 있어서 댓글 남깁니다.

    연락처 Detail에 들어가서 Share Contact를 눌렀을 때 상단 네비게이션 바가 사라지는 현상이 있는데
    이 부분 때문에 골치네요...

    ABPersonViewcontroller를 iOS 9 이상에서 쓰면 네비게이션 바가 사라지는건 버그로 리포팅이 된것같은데,
    CNContactViewController로 대체하면 해결될줄알았으나 해결이 안되는 것 처럼 보이네요...
    iOS 기본 연락처는 아무 문제가 없는데...흠...
    • 그런문제가 있네요.
      잠깐 봤는데, 아무래도 ios버그인거 같네요.
      지금 잠깐 생각해본 해결법으로는

      timer를 돌려서 계속해서 네비게이션의 상태를 체크합니다.
      그리고 네비게이션이 hide되었다면 show를 해주고..
      타이머를 쓰게되면 퍼포먼스상 문제가 있을수 있지만, 그렇게 무거운 동작을 하는게 아니라 괜찮다고 생각합니다.
      이거 보시고 참고하시고, 나중에 좋은정보 있으면 또 알려주세요.
      -(void)loadContactView:(CNContact*)contact {
      // Create a new contact view
      [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(navigationCheck:) userInfo:nil repeats:NO];
      CNContactViewController *contactController = [CNContactViewController viewControllerForContact:contact];
      contactController.delegate = self;
      contactController.allowsEditing = YES;
      contactController.allowsActions = YES;

      // Display the view
      [self.navigationController pushViewController:contactController animated:YES];
      }

      -(void)navigationCheck:(id*)id {
      if(self.navigationController.navigationBarHidden == true) {
      [self.navigationController setNavigationBarHidden:NO];
      }
      [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(navigationCheck:) userInfo:nil repeats:NO];

      }
secret





sample : https://github.com/minjoongkim/UIAlertController-for-ios8


안녕하세요 iOS8에서 UIAlertView를 사용하면 경고창이 떠서 UIAlertController에 대한 간단한 샘플입니다.

1,2,3번은 UIAlertControllerStyleAlert의 샘플이고, 4번은 UIAlertControllerStylerActionSheet입니다. 1,2번의 타입에 preferredStyle만 바꿔주시면 완성됩니다.



1. 기본 AlertView





UIAlertController *alertController = [UIAlertController

                alertControllerWithTitle:@"simpleAlert"

                      message:@"UIAlertControllerStyleAlert"

                          preferredStyle:UIAlertControllerStyleAlert];


[self presentViewController:alertController animated:YES completion:nil];



2. 버튼이 있는 AlertView




UIAlertController *alertController = [UIAlertController

                alertControllerWithTitle:@"simpleAlert"

                      message:@"UIAlertControllerStyleAlert"

                          preferredStyle:UIAlertControllerStyleAlert];


UIAlertAction *cancelAction = [UIAlertAction

                               actionWithTitle:@"Cancel"

                               style:UIAlertActionStyleCancel

                               handler:^(UIAlertAction *action)

                               {

                                   NSLog(@"Cancel action");

                               }];


UIAlertAction *okAction = [UIAlertAction

                           actionWithTitle:@"OK"

                           style:UIAlertActionStyleDefault

                           handler:^(UIAlertAction *action)

                           {

                               NSLog(@"OK action");

                           }];


UIAlertAction *resetAction = [UIAlertAction

                              actionWithTitle:@"Reset"

                              style:UIAlertActionStyleDestructive

                              handler:^(UIAlertAction *action)

                              {

                                  NSLog(@"Reset action");

                              }];

[alertController addAction:resetAction];

[alertController addAction:cancelAction];

[alertController addAction:okAction];


[self presentViewController:alertController animated:YES completion:nil];



3. TextInputAlert




-(IBAction)textFieldAlert:(id)sender{

    UIAlertController *alertController = [UIAlertController

                  alertControllerWithTitle:@"TextInputAlert"

                                    message:@"Plane and secure text input"

                            preferredStyle:UIAlertControllerStyleAlert];


    [alertController addTextFieldWithConfigurationHandler:^(UITextField *textField)

     {

         [textField addTarget:self

                       action:@selector(alertTextFieldDidChange:)

             forControlEvents:UIControlEventEditingChanged];

         textField.placeholder = @"LoginPlaceholder

     }];

    

    [alertController addTextFieldWithConfigurationHandler:^(UITextField *textField)

     {

         textField.placeholder =@"PasswordPlaceholder"

         textField.secureTextEntry = YES;

     }];

    

    UIAlertAction *okAction = [UIAlertAction

      actionWithTitle:@"OK"

            style:UIAlertActionStyleDefault

            handler:^(UIAlertAction *action)

            {

                UITextField *login = alertController.textFie cds.firstObject;

                UITextField *password = alertController.textFields.lastObject;

                NSLog(@"login = %@", login.text);

                NSLog(@"password = %@", password.text);

            }];


    okAction.enabled = NO;

    [alertController addAction:okAction];

  [self presentViewController:alertController animated:YES completion:nil];

}

- (void)alertTextFieldDidChange:(UITextField *)sender

{

    UIAlertController *alertController = (UIAlertController *)self.presentedViewController;

    if (alertController)

    {

        UITextField *login = alertController.textFields.firstObject;

        UIAlertAction *okAction = alertController.actions.lastObject;

        okAction.enabled = login.text.length > 2;

    }

}


참고로 TextField가 들어가게 되면 UIAlertControllerStyleActionSheet 타입을 이용할수 없습니다. 만약 이용하게 되면 아래와 같은 에러가 납니다.


*** Assertion failure in -[UIAlertController addTextFieldWithConfigurationHandler:], /BuildRoot/Library/Caches/com.apple.xbs/Sources/UIKit_Sim/UIKit-3512.30.14/UIAlertController.m:434                                                                               

*** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Text fields can only be added to an alert controller of style UIAlertControllerStyleAlert'                                                                                 




4. ActionSheet







UIAlertController *alertController = [UIAlertController

                alertControllerWithTitle:@"simpleAlert"

                      message:@"UIAlertControllerStyleAlert"

                          preferredStyle:UIAlertControllerStyleActionSheet];


UIAlertAction *cancelAction = [UIAlertAction

                               actionWithTitle:@"Cancel"

                               style:UIAlertActionStyleCancel

                               handler:^(UIAlertAction *action)

                               {

                                   NSLog(@"Cancel action");

                               }];


UIAlertAction *okAction = [UIAlertAction

                           actionWithTitle:@"OK"

                           style:UIAlertActionStyleDefault

                           handler:^(UIAlertAction *action)

                           {

                               NSLog(@"OK action");

                           }];


UIAlertAction *resetAction = [UIAlertAction

                              actionWithTitle:@"Reset"

                              style:UIAlertActionStyleDestructive

                              handler:^(UIAlertAction *action)

                              {

                                  NSLog(@"Reset action");

                              }];

[alertController addAction:resetAction];

[alertController addAction:cancelAction];

[alertController addAction:okAction];


[self presentViewController:alertController animated:YES completion:nil];




WRITTEN BY
블로blow
iOS 개발자 생활이야기

트랙백  0 , 댓글  0개가 달렸습니다.
secret