Salman Ali
Salman Ali

Reputation: 255

Export Picture in VCard Swift

i am trying to develop an app that takes input of some fields from the user about a contact and then export that contact as VCard. I can successfully export the contact but the when i try to import the contact using Contacts at iCloud.com, the picture is not being imported. All the other information is being imported but not the picture.

There are two ways i tried to export the VCard.

First Way Here

public  func getVCard() -> Data?{

    var contact = CNMutableContact()

    let homeEmail = CNLabeledValue(label:CNLabelHome, value:self.email as! NSString)
    contact.emailAddresses = [homeEmail]

    let phone = CNLabeledValue(label: CNLabelPhoneNumberMobile, value:CNPhoneNumber(stringValue: self.phone!))
    contact.phoneNumbers = [phone]

    contact.givenName = self.name!
    if let d = self.data{
        contact.imageData = d

    }


    let address = CNMutablePostalAddress()
    address.country = self.address!
    contact.postalAddresses = [CNLabeledValue(label: CNLabelHome, value: address)]

    contact.organizationName = self.company!

    contact.jobTitle = title!

    let sc1 = CNSocialProfile(urlString: "\(socialMedia1!)", username: self.name!, userIdentifier: self.name!, service: CNContactSocialProfilesKey)
    let sc2 = CNSocialProfile(urlString: "\(socialMedia2!)", username: self.name!, userIdentifier: self.name!, service: CNContactSocialProfilesKey)
    //contact.socialProfiles = [CNSocialProfile(]
    let sc3 = CNSocialProfile(urlString: "\(socialMedia3!)", username: self.name!, userIdentifier: self.name!, service: CNContactSocialProfilesKey)
    contact.socialProfiles = [CNLabeledValue(label: "Profile1", value: sc1), CNLabeledValue(label: "Profile2", value: sc2), CNLabeledValue(label: "Profile3", value: sc3)]

    contact.note = self.notes!
    let urlAddresses = CNLabeledValue(label: CNLabelURLAddressHomePage, value: self.web as! NSString)
    contact.urlAddresses = [urlAddresses]

    var data: Data?

    do{
        data = try CNContactVCardSerialization.data(with: [contact])
    }catch let error{
        print(error.localizedDescription)
    }

    return data
}

When i debug this function, i can see the picture data in the contact object but when that VCard is exported and i try to import it on iCloud.com it does not import picture of that contact.

Now the second method i used to export the contact is as follow and i would prefer the answer using this method because this is the best fit for me.

func textBasedVCard()-> Data?{

    var string = "BEGIN:VCARD\nVERSION:3.0\n"
    string += "N:\(self.name!);\nFN:\(self.name!)\nORG:\(self.company!)\nTITLE:\(self.title!)\nTEL;TYPE=WORK,VOICE:\(self.phone!)\nADR;TYPE=WORK:;;\(self.address!)\nNOTE:\(self.notes!)\nitem1.URL:\(self.web!)\nitem2.URL:\(self.blog!)\nitem3.URL:\(self.socialMedia1!)\nitem4.URL:\(self.socialMedia2!)\nitem5.URL:\(self.socialMedia3!)\nEMAIL;TYPE=PREF,INTERNET:\(self.email!)\nEND:VCARD"
    print(string)
    let utf8str = string.data(using: String.Encoding.utf8)
    //utf8str?.base64EncodedStringWithOptions(NSData.Base64EncodingOptions(rawValue: 0))
    if let base64Encoded = utf8str?.base64EncodedString(options: .init(rawValue: 0))
    {
        return Data(base64Encoded: base64Encoded)!

    }
    return nil
}

In this method i do not know how to attach picture data. Please guide me through this problem. I will be really grateful. Thanks.

Edit:

This question in no way a duplicate of any other question. You can see the link yourself someone posted just to make himself feel proud.

Upvotes: 3

Views: 3026

Answers (1)

nathan
nathan

Reputation: 9395

From the response posted above (ABPersonCreatePeopleInSourceWithVCardRepresentation):

Address Book supports vCard version 3.0.

VCARD > PHOTO - An image or photograph of the individual associated with the vCard. It may point to an external URL or may be embedded in the vCard as a Base64 encoded block of text.

2.1: PHOTO;JPEG:http://example.com/photo.jpg
2.1: PHOTO;JPEG;ENCODING=BASE64:[base64-data]
3.0: PHOTO;TYPE=JPEG;VALUE=URI:http://example.com/photo.jpg
=> 3.0: PHOTO;TYPE=JPEG;ENCODING=b:[base64-data]             // Works in iOS/macOS
4.0: PHOTO;MEDIATYPE=image/jpeg:http://example.com/photo.jpg
4.0: PHOTO:data:image/jpeg;base64,[base64-data]

How to create a VCF 3.0 card for iOS/macOS

Testing shows the 2nd way works: encode the JPEG photo using base64.

PHOTO;ENCODING=b;TYPE=JPEG:\(base64EncodedImage)

Working PoC:

struct VCFBuilder {
    private let name: String? = "Name1"
    private let company: String? = "Company"
    private let title: String? = "Title1"
    private let phone: String? = "555-1234-1234"
    private let address: String? = "aaa"
    private let notes: String? = "PROnotes"
    private let web: String? = "web"
    private let blog: String? = "bolgg"
    private let socialMedia1: String? = "social1"
    private let socialMedia2: String? = "social2"
    private let socialMedia3: String? = "social3"
    private let email: String? = "[email protected]"
    private let imageUrl: String? = "http://0.gravatar.com/avatar/3f009d72559f51e7e454b16e5d0687a1"

    func vcf() -> String? {

        do {
            let url = URL(string: self.imageUrl!)
            let data = try Data(contentsOf: url!).base64EncodedString()
            return generateVCF(withEncodedImage: data)
        } catch {
            print(error)
            return nil
        }
    }

    // TODO: Don't !
    private func generateVCF(withEncodedImage imageBase64: String) -> String {
        return """
        BEGIN:VCARD
        VERSION:3.0
        N:\(self.name!);
        FN:\(self.name!)
        ORG:\(self.company!)
        TITLE:\(self.title!)
        TEL;TYPE=WORK,VOICE:\(self.phone!)
        ADR;TYPE=WORK:;;\(self.address!)
        NOTE:\(self.notes!)
        PHOTO;ENCODING=b;TYPE=JPEG:\(imageBase64)
        item1.URL:\(self.web!)
        item2.URL:\(self.blog!)
        item3.URL:\(self.socialMedia1!)
        item4.URL:\(self.socialMedia2!)
        item5.URL:\(self.socialMedia3!)
        EMAIL;TYPE=PREF,INTERNET:\(self.email!)
        END:VCARD
        """
    }
}

let vcf = VCFBuilder().vcf()
print(vcf)

How to import VCF 3.0 card in iOS/macOS:

  • Manually using Contacts.app
  • iOS <= 9: ABPersonCreatePeopleInSourceWithVCardRepresentation(_:_:) (docs)
  • iOS >= 9: CNContactVCardSerialization.contacts(with:) (docs)
guard let data = VCFBuilder().vcf()?.data(using: .utf8),
      let contact = try? CNContactVCardSerialization.contacts(with: data).first
        else { return }

print(contact.imageData!.base64EncodedString())

Imported using iOS' Contacts.app:

enter image description here

Upvotes: 3

Related Questions