Contacts模块中的QuickContacts

2024-01-12 03:32

本文主要是介绍Contacts模块中的QuickContacts,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在ContactsLIstActiviry中的newView()和bindView()函数中都使用了ContactListItemCache类型的Cache:

    final static class ContactListItemCache {

        public View header;

        public TextView headerText;

        public View divider;

        public TextView nameView;

        public View callView;

        public ImageView callButton;

        public CharArrayBuffer nameBuffer = new CharArrayBuffer(128);

        public TextView labelView;

        public CharArrayBuffer labelBuffer = new CharArrayBuffer(128);

        public TextView dataView;

        public CharArrayBuffer dataBuffer = new CharArrayBuffer(128);

        public ImageView presenceView;

        public QuickContactBadge photoView;

        public ImageView nonQuickContactPhotoView;

}

 

其中QuickContactBadge类型的photoView,就是指的图片,点击图片之后会弹出QuickContacts框。

 

 

 

 

 

看一下QuickContactBadge的代码:

public void onClick(View v) {

…………..……

} else if (mContactEmail != null) {

mQueryHandler.startQuery(TOKEN_EMAIL_LOOKUP_AND_TRIGGER, mContactEmail,

Uri.withAppendedPath(Email.CONTENT_LOOKUP_URI, Uri.encode(mContactEmail)),

EMAIL_LOOKUP_PROJECTION, null, null, null);

…………..……

//在相应点击时触发一个startQuery()方法,这个方法执行完毕之后自动调用mQueryHandler的//onQueryComplete()方法

}

 

protected void onQueryComplete(int token, Object cookie, Cursor cursor) {

…………………………………….

 

if (trigger && lookupUri != null) {

trigger(lookupUri);//在调用trigger()方法。

…………………………………..

}

 

 

}

    private void trigger(Uri lookupUri) {

        QuickContact.showQuickContact(getContext(), this, lookupUri, mMode, mExcludeMimes);

    }

 

对于showQuickContact方法来说,它是ContactsContract类中的一个函数。下面是代码:

 

 

public static void showQuickContact(Context context, Rect target, Uri lookupUri, int mode,

                String[] excludeMimes) {

            // Launch pivot dialog through intent for now

            final Intent intent = new Intent(ACTION_QUICK_CONTACT);//这个就是.UI.QuickContactsActivity的IntentFilter的Action

            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP

                    | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);

 

            intent.setData(lookupUri);

            intent.putExtra(EXTRA_TARGET_RECT, target);

            intent.putExtra(EXTRA_MODE, mode);

            intent.putExtra(EXTRA_EXCLUDE_MIMES, excludeMimes);

            context.startActivity(intent);//此时就会启动.UI.QuickContactsActivity

}

 

QuickContactsActivity类中的代码比较少。它继承了QuickContactWindow.OnDismissListener,使用onDismiss()函数来把QuickContact框dismiss掉。

QuickContactsActivity类在onNewIntent()函数中先将Intent解析出来,再调用show()方法将QuickContactWindow显示出来。

    public void onNewIntent(Intent intent) {

        super.onNewIntent(intent);

                   ……………

        mQuickContact.show(lookupUri, target, mode, excludeMimes);//show()方法是QuickContactWindow类中的函数

    }

 

.UI.QuickContactsActivity界面的绘制就是在QuickContactWindow类中完成的。

 

 

 

首先,在Show()方法中调用mHandler.startQuery()方法

 

   

public synchronized void show(Uri lookupUri, Rect anchor, int mode, String[] excludeMimes) {

                   ………………………………….

        if (mMode == QuickContact.MODE_LARGE) {

            // Select photos, but only super-primary

            mHandler.startQuery(TOKEN_DATA, lookupUri, dataUri, DataQuery.PROJECTION, Data.MIMETYPE

                + "!=? OR (" + Data.MIMETYPE + "=? AND " + Data._ID + "=" + Contacts.PHOTO_ID

                + ")", new String[] { Photo.CONTENT_ITEM_TYPE, Photo.CONTENT_ITEM_TYPE }, null);

        } else {

            // Exclude all photos from cursor

            mHandler.startQuery(TOKEN_DATA, lookupUri, dataUri, DataQuery.PROJECTION, Data.MIMETYPE

                    + "!=?", new String[] { Photo.CONTENT_ITEM_TYPE }, null);

        }

    }

 

然后在onQueryComplete()方法中调用handleData()

   

    public synchronized void onQueryComplete(int token, Object cookie, Cursor cursor) {

        if (cookie != mLookupUri) return;

 

        if (cursor == null) {

            // Problem while running query, so bail without showing

            Log.w(TAG, "Missing cursor for token=" + token);

            this.dismiss();

            return;

        }

                   //绘制QuickContact框的过程基本都在此函数中完成。

        handleData(cursor);

        mHasData = true;

 

        if (!cursor.isClosed()) {

            cursor.close();

        }

 

        considerShowing();

}

再来看handleData()的代码:

 

   

    private void handleData(Cursor cursor) {

        if (cursor == null) return;

…………………………………………………………

        // 使用一个while循环,将从onQueryComplete()方法中得到的cursor遍历。

        while (cursor.moveToNext()) {

            // 将cursor中的数据解出来。

            final long dataId = cursor.getLong(DataQuery._ID);

            final String accountType = cursor.getString(DataQuery.ACCOUNT_TYPE);

            final String resPackage = cursor.getString(DataQuery.RES_PACKAGE);

            final String mimeType = cursor.getString(DataQuery.MIMETYPE);

           

…………………………………………………………

// 中间通过判断mimeType的类型,生成DataAction类的实例,然后使用函数considerAdd()将

// DataAction类型的action以mimeType为键存储在mActions中。

…………………………………………………………

}//end while

        // 将mActions的键取出。

        final Set<String> containedTypes = mActions.keySet();

       

        String[] temp = containedTypes.toArray(new String[containedTypes.size()]);

        int length = temp.length;

       

        // 在此循环中调用inflateAction()函数,将QuickContact框中的图标生成。        

for (String mimeType : ORDERED_MIMETYPES) {

            if (containedTypes.contains(mimeType)) {

                final int index = mTrack.getChildCount() - 1;

                mTrack.addView(inflateAction(mimeType), index); // 将快速联系人的小图标add到mTrack中,在showInternal()函数中会调用startAnimation()将其动画效果显示出来

                containedTypes.remove(mimeType);

            }

        }

…………………………………………………………

 

    }

 

 

   

    private void considerAdd(Action action, String mimeType) {

        if (mResolveCache.hasResolve(action)) {

            mActions.collect(mimeType, action);

        }

}

 

 

 

   

    private View inflateAction(String mimeType) {

……………………………….

        ActionList children = mActions.get(mimeType);

        Action firstInfo = children.get(0);

        if (children.size() == 1) {

            view.setTag(firstInfo);

        } else {

            for (Action action : children) {

                if (action.isPrimary()) {

                    view.setTag(action);

                    isActionSet = true;

                    break;

                }

            }

            if (!isActionSet) {

                view.setTag(children);

            }

        }

 

        // Set icon and listen for clicks

        final CharSequence descrip = mResolveCache.getDescription(firstInfo);

        final Drawable icon = mResolveCache.getIcon(firstInfo);

        view.setChecked(false);

        view.setContentDescription(descrip);

        view.setImageDrawable(icon);

        view.setOnClickListener(this);

        return view;

    }

 

此时,handleData()函数已经完成,接着调用considerShowing(),由considerShowing()调用showInternal()函数来完成绘制工作。

 

 

   

    private void considerShowing() {

        if (mHasData && !mShowing && !mDismissed) {

            if (mMode == QuickContact.MODE_MEDIUM && !mHasValidSocial) {

                // Missing valid social, swap medium for small header

                mHeader.setVisibility(View.GONE);

                mHeader = getHeaderView(QuickContact.MODE_SMALL);

            }

 

            // All queries have returned, pull curtain

            showInternal();

        }

    }

 

 

 

   

    private void showInternal() {

        mDecor = mWindow.getDecorView();

        mDecor.getViewTreeObserver().addOnGlobalLayoutListener(this);

        WindowManager.LayoutParams l = mWindow.getAttributes();

 

        l.width = mScreenWidth + mShadowHoriz + mShadowHoriz;

        l.height = WindowManager.LayoutParams.WRAP_CONTENT;

 

        // Force layout measuring pass so we have baseline numbers

        mDecor.measure(l.width, l.height);

        final int blockHeight = mDecor.getMeasuredHeight();

 

        l.gravity = Gravity.TOP | Gravity.LEFT;

        l.x = -mShadowHoriz;

 

        if (mAnchor.top > blockHeight) {

            // Show downwards callout when enough room, aligning bottom block

            // edge with top of anchor area, and adjusting to inset arrow.

            showArrow(R.id.arrow_down, mAnchor.centerX());

            l.y = mAnchor.top - blockHeight + mShadowVert;

            l.windowAnimations = R.style.QuickContactAboveAnimation;

 

        } else {

            // Otherwise show upwards callout, aligning block top with bottom of

            // anchor area, and adjusting to inset arrow.

            showArrow(R.id.arrow_up, mAnchor.centerX());

            l.y = mAnchor.bottom - mShadowVert;

            l.windowAnimations = R.style.QuickContactBelowAnimation;

 

        }

 

        l.flags = WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN

                | WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS;

 

        mRequestedY = l.y;

        mWindowManager.addView(mDecor, l);

        mShowing = true;

        mQuerying = false;

        mDismissed = false;

 

        mTrack.startAnimation(mTrackAnim);// 调用动画效果

 

        if (TRACE_LAUNCH) {

            android.os.Debug.stopMethodTracing();

            Log.d(TAG, "Window recycled " + mWindowRecycled + " times, chiclets "

                    + mActionRecycled + " times");

        }

    }

 

到此,完了。

若有错,欢迎指正。

这篇关于Contacts模块中的QuickContacts的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/596664

相关文章

Python使用date模块进行日期处理的终极指南

《Python使用date模块进行日期处理的终极指南》在处理与时间相关的数据时,Python的date模块是开发者最趁手的工具之一,本文将用通俗的语言,结合真实案例,带您掌握date模块的六大核心功能... 目录引言一、date模块的核心功能1.1 日期表示1.2 日期计算1.3 日期比较二、六大常用方法详

python中time模块的常用方法及应用详解

《python中time模块的常用方法及应用详解》在Python开发中,时间处理是绕不开的刚需场景,从性能计时到定时任务,从日志记录到数据同步,时间模块始终是开发者最得力的工具之一,本文将通过真实案例... 目录一、时间基石:time.time()典型场景:程序性能分析进阶技巧:结合上下文管理器实现自动计时

Node.js net模块的使用示例

《Node.jsnet模块的使用示例》本文主要介绍了Node.jsnet模块的使用示例,net模块支持TCP通信,处理TCP连接和数据传输,具有一定的参考价值,感兴趣的可以了解一下... 目录简介引入 net 模块核心概念TCP (传输控制协议)Socket服务器TCP 服务器创建基本服务器服务器配置选项服

Python利用自带模块实现屏幕像素高效操作

《Python利用自带模块实现屏幕像素高效操作》这篇文章主要为大家详细介绍了Python如何利用自带模块实现屏幕像素高效操作,文中的示例代码讲解详,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1、获取屏幕放缩比例2、获取屏幕指定坐标处像素颜色3、一个简单的使用案例4、总结1、获取屏幕放缩比例from

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.

多模块的springboot项目发布指定模块的脚本方式

《多模块的springboot项目发布指定模块的脚本方式》该文章主要介绍了如何在多模块的SpringBoot项目中发布指定模块的脚本,作者原先的脚本会清理并编译所有模块,导致发布时间过长,通过简化脚本... 目录多模块的springboot项目发布指定模块的脚本1、不计成本地全部发布2、指定模块发布总结多模

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

Python模块导入的几种方法实现

《Python模块导入的几种方法实现》本文主要介绍了Python模块导入的几种方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录一、什么是模块?二、模块导入的基本方法1. 使用import整个模块2.使用from ... i