web-dev-qa-db-ja.com

SquareのCardCaseアプリは、どのようにしてアドレス帳からユーザーの詳細を自動的に入力しますか?

Squareの新しいカードケースiOSアプリには、「アカウントの作成」機能があります。それをタップすると、アドレス帳からのユーザーのエントリでPREPOPULATEDフォームが表示されます。

これはどのように可能ですか?誰か知ってる?この方法でユーザーの情報を取得することは不可能だと思いました。それはiOS5.0のものではありません。

21
Genericrich

私が思いついた唯一の解決策は、デバイス名を使用してから、アドレス帳で一致するものを検索することでした。これは、誰かが特定の命名規則を使用することを前提としています。たとえば、デバイス名として「Nik'siPhone」を使用します。私はアドレス帳にある唯一のNikでもあるので、私のシナリオでは、の前のテキストを所有者名として使用するとうまくいきます。

これは、Erica SadunによるABAddressBookの非常に便利なラッパーを利用しています ABContactHelper 。配列インデックスを0で使用する代わりに、列挙コードを残しました。一致するものが少数返される可能性が高いため、展開してユーザーに「詳細」を選択するオプションを提供できます。これは、正方形のケースのソリューションと正確には一致しませんが、うまく機能します。私見では。

NSString *firstname;
NSString *lastname;

NSString *ownerName = [[UIDevice currentDevice] name];

NSRange t = [ownerName rangeOfString:@"'s"];
if (t.location != NSNotFound) {
    ownerName = [ownerName substringToIndex:t.location];
} 

NSArray *matches = [ABContactsHelper contactsMatchingName:ownerName];
if(matches.count == 1){ 
    for (ABContact *contact in matches){
        firstname = [contact firstname];
        lastname = [contact lastname];

    }
}
56
Nik Burns

アプリがアドレス帳にアクセスするためのアクセス許可を要求する必要がある変更のため、この方法は機能しなくなりました。 Nik Burnsの回答はうまくいきましたが、アドレス帳にアクセスする必要があります。

Square Wallet(元の質問のSquare Card Caseリファレンスの後継)をダウンロードしましたが、サインアップフォームに事前入力されなくなりました。パスは、デバイス名のアドレス帳のトリックを実行するためにも使用されますが、サインアップフォームに事前入力されなくなりました。

上記の方法を使用すると、サインアップフォームafter連絡先アクセスの要求 を事前入力できますが、目的の "が失われます。魔法」体験。

23
Joshua Dance

私は同じ問題を抱えていましたが、Swiftで、Nik Burnsの提案を使用して、それに関するブログ投稿を書きました: http://paulpeelen.com/2016/01/20/prefill-an-signup-form-in -ios-using-Swift-and-cncontact /

これは基本的にSwiftの最終結果です。

import Contacts

...

    @IBOutlet weak var nameFirst: UILabel!
    @IBOutlet weak var nameLast: UILabel!
    @IBOutlet weak var email: UILabel!
    @IBOutlet weak var phoneNo: UILabel!
    @IBOutlet weak var address: UILabel!

    /**
     Search the addressbook for the contact
     */
    private func getMe() {
        let ownerName = getOwnerName()
        let store = CNContactStore()

        do {
            // Ask permission from the addressbook and search for the user using the owners name
            let contacts = try store.unifiedContactsMatchingPredicate(CNContact.predicateForContactsMatchingName(ownerName), keysToFetch:[CNContactGivenNameKey, CNContactFamilyNameKey, CNContactPhoneNumbersKey, CNContactPostalAddressesKey, CNContactEmailAddressesKey])

            //assuming contain at least one contact
            if let contact = contacts.first {
                // Checking if phone number is available for the given contact.
                if (contact.isKeyAvailable(CNContactPhoneNumbersKey)) {
                    // Populate the fields
                    populateUsingContact(contact)
                } else {
                    //Refetch the keys
                    let keysToFetch = [CNContactGivenNameKey, CNContactFamilyNameKey, CNContactPhoneNumbersKey, CNContactPostalAddressesKey, CNContactEmailAddressesKey]
                    let refetchedContact = try store.unifiedContactWithIdentifier(contact.identifier, keysToFetch: keysToFetch)

                    // Populate the fields
                    populateUsingContact(refetchedContact)
                }
            }
        } catch let e as NSError {
            print(e.localizedDescription)
        }

    }

    /**
     Get the device owners name

     - returns: The owners name
     */
    private func getOwnerName() -> String {
        // Get the device owners name
        var ownerName = UIDevice.currentDevice().name.trimmedString.stringByReplacingOccurrencesOfString("'", withString: "")
        // Get the model of the device
        let model = UIDevice.currentDevice().model

        // Remove the device name from the owners name
        if let t = ownerName.rangeOfString("s \(model)") {
            ownerName = ownerName.substringToIndex(t.startIndex)
        }

        return ownerName.trimmedString
    }

    /**
     Populate the fields using a contact

     - parameter contact: The contact to use
     */
    private func populateUsingContact(contact: CNContact) {
        // Set the name fields
        nameFirst.text = contact.givenName
        nameLast.text = contact.familyName

        // Check if there is an address available, it might be empty
        if (contact.isKeyAvailable(CNContactPostalAddressesKey)) {
            if let
                addrv = contact.postalAddresses.first,
                addr = addrv.value as? CNPostalAddress where addrv.value is CNPostalAddress
            {
                let address = "\(addr.street)\n\(addr.postalCode) \(addr.city)\n\(addr.country)"
                self.address.text = address
            }
        }

        // Check if there is a phonenumber available, it might be empty
        if (contact.isKeyAvailable(CNContactPhoneNumbersKey)) {
            if let
                phonenumberValue = contact.phoneNumbers.first,
                pn = phonenumberValue.value as? CNPhoneNumber where phonenumberValue.value is CNPhoneNumber
            {
                phoneNo.text = pn.stringValue
            }
        }
    }

そして拡張機能:

extension String {

    /// Trim a string
    var trimmedString: String {
        return stringByTrimmingCharactersInSet(NSCharacterSet.whitespaceAndNewlineCharacterSet())
    }

}
1
Paul Peelen

デバイス名を使用することに満足できなかったので、「ME」に対応するアドレス帳の最初のレコードを使用することにしました。

ABAddressBookRef addressBook = ABAddressBookCreate( );
CFArrayRef allPeople = ABAddressBookCopyArrayOfAllPeople( addressBook );
ABRecordRef ref = CFArrayGetValueAtIndex( allPeople, 0 );
ABContact * contact = [ABContact contactWithRecord:ref];
1
damien murphy.

いいえ、できません

http://developer.Apple.com/library/ios/#documentation/ContactData/Conceptual/AddressBookProgrammingGuideforiPhone/Introduction.html#//Apple_ref/doc/uid/TP40007744

4.Xで利用可能だったことは知っています。 5.0の機能だけではありません...以前に利用可能だったかどうかはわかりません。

0
Ross R