/* * Jitsi, the OpenSource Java VoIP and Instant Messaging client. * * Distributable under LGPL license. * See terms of license at gnu.org. */ package net.java.sip.communicator.impl.googlecontacts; import java.util.*; import java.util.regex.*; import net.java.sip.communicator.service.contactsource.*; import net.java.sip.communicator.service.googlecontacts.*; import net.java.sip.communicator.service.protocol.*; /** * Implements ContactQuery for Google Contacts. * * @author Sebastien Vincent */ public class GoogleContactsQuery extends AsyncContactQuery { /** * Maximum results for Google Contacts query. */ public static final int GOOGLECONTACTS_MAX_RESULTS = 20; /** * Maximum number of results for this instance. */ private final int count; /** * The Google query. */ private GoogleQuery gQuery = null; /** * Initializes a new GoogleContactsQuery instance which is to * perform a specific query on behalf of a specific * contactSource. * * @param contactSource the ContactSourceService which is to * perform the new ContactQuery instance * @param query the Pattern for which contactSource is * being queried * @param count maximum number of results */ protected GoogleContactsQuery(GoogleContactsSourceService contactSource, Pattern query, int count) { super(contactSource, query); this.count = count; } /** * Create a SourceContact from a GoogleContactsEntry. * * @param entry GoogleContactsEntry */ private void onGoogleContactsEntry(GoogleContactsEntry entry) { String displayName = entry.getFullName(); if(displayName == null || displayName.length() == 0) { if((entry.getGivenName() == null || entry.getGivenName().length() == 0) && (entry.getFamilyName() == null || entry.getFamilyName().length() == 0)) { return; } displayName = entry.getGivenName() + " " + entry.getFamilyName(); } List contactDetails = getContactDetails(entry); if (!contactDetails.isEmpty()) { GenericSourceContact sourceContact = new GenericSourceContact( getContactSource(), displayName, contactDetails); try { byte img[] = GoogleContactsServiceImpl.downloadPhoto( ((GoogleContactsEntryImpl)entry).getPhotoLink(), getContactSource().getConnection(). getGoogleService()); sourceContact.setImage(img); } catch (OutOfMemoryError oome) { // Ignore it, the image is not vital. } addQueryResult(sourceContact); } } /** * Performs this AsyncContactQuery in a background Thread. * * @see AsyncContactQuery#run() */ @Override protected void run() { GoogleContactsServiceImpl service = GoogleContactsActivator.getGoogleContactsService(); gQuery = new GoogleQuery(query); GoogleContactsConnection cnx = getContactSource().getConnection(); if(cnx == null) { return; } service.searchContact( cnx, gQuery, count, new GoogleEntryCallback() { public void callback(GoogleContactsEntry entry) { onGoogleContactsEntry(entry); } }); } @Override public synchronized void start() { boolean hasStarted = false; try { super.start(); hasStarted = true; } finally { if (!hasStarted) { getContactSource().removeQuery(this); } } } /** * Gets the contactDetails to be set on a SourceContact. * * @param entry GoogleContactsEntry * @return the contactDetails to be set on a SourceContact */ private List getContactDetails(GoogleContactsEntry entry) { List ret = new LinkedList(); List homeMails = entry.getHomeMails(); List workMails = entry.getWorkMails(); List mobilePhones = entry.getMobilePhones(); List homePhones = entry.getHomePhones(); List workPhones = entry.getWorkPhones(); Map ims = entry.getIMAddresses(); ContactDetail detail = null; for(String mail : homeMails) { List> supportedOpSets = new ArrayList>(1); // can be added as contacts supportedOpSets.add(OperationSetPersistentPresence.class); detail = new ContactDetail(mail, ContactDetail.Category.Email, new ContactDetail.SubCategory[]{ ContactDetail.SubCategory.Home}); detail.setSupportedOpSets(supportedOpSets); ret.add(detail); } for(String mail : workMails) { List> supportedOpSets = new ArrayList>(1); // can be added as contacts supportedOpSets.add(OperationSetPersistentPresence.class); detail = new ContactDetail(mail, ContactDetail.Category.Email, new ContactDetail.SubCategory[]{ ContactDetail.SubCategory.Work}); detail.setSupportedOpSets(supportedOpSets); ret.add(detail); } for(String homePhone : homePhones) { List> supportedOpSets = new ArrayList>(2); supportedOpSets.add(OperationSetBasicTelephony.class); // can be added as contacts supportedOpSets.add(OperationSetPersistentPresence.class); homePhone = GoogleContactsActivator.getPhoneNumberI18nService() .normalize(homePhone); detail = new ContactDetail(homePhone, ContactDetail.Category.Phone, new ContactDetail.SubCategory[]{ ContactDetail.SubCategory.Home}); detail.setSupportedOpSets(supportedOpSets); ret.add(detail); } for(String workPhone : workPhones) { List> supportedOpSets = new ArrayList>(2); supportedOpSets.add(OperationSetBasicTelephony.class); // can be added as contacts supportedOpSets.add(OperationSetPersistentPresence.class); workPhone = GoogleContactsActivator.getPhoneNumberI18nService() .normalize(workPhone); detail = new ContactDetail(workPhone, ContactDetail.Category.Phone, new ContactDetail.SubCategory[]{ ContactDetail.SubCategory.Work}); detail.setSupportedOpSets(supportedOpSets); ret.add(detail); } for(String mobilePhone : mobilePhones) { List> supportedOpSets = new ArrayList>(2); supportedOpSets.add(OperationSetBasicTelephony.class); // can be added as contacts supportedOpSets.add(OperationSetPersistentPresence.class); mobilePhone = GoogleContactsActivator.getPhoneNumberI18nService() .normalize(mobilePhone); detail = new ContactDetail(mobilePhone, ContactDetail.Category.Phone, new ContactDetail.SubCategory[]{ ContactDetail.SubCategory.Mobile}); detail.setSupportedOpSets(supportedOpSets); ret.add(detail); } for(Map.Entry im : ims.entrySet()) { if(im.getValue() != GoogleContactsEntry.IMProtocol.OTHER) { ContactDetail.SubCategory imSubCat; switch(im.getValue()) { case AIM: imSubCat = ContactDetail.SubCategory.AIM; break; case ICQ: imSubCat = ContactDetail.SubCategory.ICQ; break; case YAHOO: imSubCat = ContactDetail.SubCategory.Yahoo; break; case JABBER: imSubCat = ContactDetail.SubCategory.Jabber; break; case MSN: imSubCat = ContactDetail.SubCategory.MSN; break; case GOOGLETALK: imSubCat = ContactDetail.SubCategory.GoogleTalk; break; default: imSubCat = null; break; } detail = new ContactDetail( im.getKey(), ContactDetail.Category.InstantMessaging, new ContactDetail.SubCategory[] { imSubCat }); setIMCapabilities(detail, im.getValue()); // can be added as contacts detail.getSupportedOperationSets().add( OperationSetPersistentPresence.class); ret.add(detail); } } return ret; } /** * Sets the IM capabilities of a specific ContactDetail (e.g. * supportedOpSets). * * @param contactDetail the ContactDetail to set the capabilities * of * @param protocol protocol * @return contactDetail */ private ContactDetail setIMCapabilities( ContactDetail contactDetail, GoogleContactsEntry.IMProtocol protocol) { List> supportedOpSets = new LinkedList>(); Map, String> preferredProtocols = new HashMap, String>(); switch (protocol) { case GOOGLETALK: supportedOpSets.add(OperationSetBasicInstantMessaging.class); preferredProtocols.put( OperationSetBasicInstantMessaging.class, ProtocolNames.AIM); break; case ICQ: supportedOpSets.add(OperationSetBasicInstantMessaging.class); preferredProtocols.put( OperationSetBasicInstantMessaging.class, ProtocolNames.ICQ); break; case JABBER: supportedOpSets.add(OperationSetBasicInstantMessaging.class); preferredProtocols.put( OperationSetBasicInstantMessaging.class, ProtocolNames.JABBER); supportedOpSets.add(OperationSetBasicTelephony.class); preferredProtocols.put( OperationSetBasicTelephony.class, ProtocolNames.JABBER); break; case YAHOO: supportedOpSets.add(OperationSetBasicInstantMessaging.class); preferredProtocols.put( OperationSetBasicInstantMessaging.class, ProtocolNames.YAHOO); break; default: break; } contactDetail.setSupportedOpSets(supportedOpSets); if (!preferredProtocols.isEmpty()) contactDetail.setPreferredProtocols(preferredProtocols); return contactDetail; } /** * Notifies this GoogleContactsQuery that it has stopped performing * in the associated background Thread. * * @param completed true if this ContactQuery has * successfully completed, false if an error has been encountered * during its execution * @see AsyncContactQuery#stopped(boolean) */ @Override protected void stopped(boolean completed) { try { super.stopped(completed); } finally { getContactSource().stopped(this); } } /** * Cancels this ContactQuery. * * @see ContactQuery#cancel() */ @Override public void cancel() { if(gQuery != null) { gQuery.cancel(); } super.cancel(); } }