zoukankan      html  css  js  c++  java
  • Android6.0 源码修改之 Contacts应用

    一、Contacts应用的主界面和联系人详情界面增加顶部菜单添加退出按钮

    通过Hierarchy View 工具可以发现

    主界面对应的类为 PeopleActivity

    联系人详情界面对应的类为 QuickContactActivity

    左上角的退出按钮其实很简单,系统actionBar已经帮我们实现了这一功能,只是没有显示出来而已。在onCreate()方法中,在setContentView()方法之后,添加如下代码即可显示返回的箭头

    	ActionBar mActionBar = getActionBar();
        if (mActionBar != null) {
            Log.i(TAG, "getSupportActionBar != null....");
            mActionBar.setDisplayHomeAsUpEnabled(true);
            mActionBar.setHomeButtonEnabled(true);
        }
    

    接下来在onOptionsItemSelected()中监听返回按钮的事件即可

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
    	switch (item.getItemId()) {
            case android.R.id.home: {
                     finish();
                }
                return true;
            }
    		....
    }
    

    图1 左上角返回退出功能

    二、第三方app拉起主界面时直接显示模糊查询对应的联系人列表

    简单分析一下,模糊查询需要对应的查询联系人名称,可以通过intent传递参数,这里定义为String类型,当传递参数不为null时,模拟手动点击搜索框对应的逻辑。如下在 PeopleActivity 的 onCreate()方法中增加获取参数的代码

    final String queryString = getIntent().getStringExtra("queryString");
        if (!TextUtils.isEmpty(queryString)) {
             new Handler().postDelayed(new Runnable() {
                @Override
                public void run() {
                    showQueryTextFragment(queryString);
                }
            }, 100);//让搜索逻辑延迟100ms执行
        }
    

    通过测试发现,不加延迟触发搜索框对应的逻辑并不会显示模糊查询结果界面。接下来我们分析点击搜索框对应的逻辑代码,找到搜索框对应的控件id,menu_search, 回到刚刚的菜单监听方法 onOptionsItemSelected()中

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
    	...
    
        switch (item.getItemId()) {
    	 case R.id.menu_search: {
                onSearchRequested();
                return true;
            }
    	}
    	...
    }
    
    @Override
    public boolean onSearchRequested() { // Search key pressed.
        Log.d(TAG, "[onSearchRequested]");
    	//不在搜索模式下,也就是没有点击过搜索框
        if (!mActionBarAdapter.isSelectionMode()) {
    		//获取焦点,弹出键盘
            mActionBarAdapter.setSearchMode(true);
        }
        return true;
    }
    

    从上面不难看出最终调用 mActionBarAdapter 的方法,我们接着跟进去

    源码位置 packages/apps/Contacts/src/com/android/contacts/activities/ActionBarAdapter.java

    public void setSearchMode(boolean flag) {
        if (mSearchMode != flag) {
            mSearchMode = flag;
            update(false /* skipAnimation */);
            if (mSearchView == null) {
                return;
            }
            if (mSearchMode) {
                mSearchView.setEnabled(true);
                setFocusOnSearchView();
            } else {
                // Disable search view, so that it doesn't keep the IME visible.
                mSearchView.setEnabled(false);
            }
            setQueryString(null);
        } else if (flag) {
            // Everything is already set up. Still make sure the keyboard is up
    		//需要注释此处,不然多次调用并退出再次拉起容易出现键盘弹出的情况
            //if (mSearchView != null) setFocusOnSearchView();
        }
    }
    
    public void setFocusOnSearchView() {
    	//mSearchView获取焦点(先获取焦点才能弹出键盘)
        mSearchView.requestFocus();
    	//弹出键盘
        showInputMethod(mSearchView); // Workaround for the "IME not popping up" issue.
    }
    
    private void showInputMethod(View view) {
        final InputMethodManager imm = (InputMethodManager) mActivity.getSystemService(
                Context.INPUT_METHOD_SERVICE);
        if (imm != null) {
            imm.showSoftInput(view, 0);
        }
    }
    

    看到这里我们可以猜想到 mSearchView 肯定设置了文字改变监听,继续查找 addTextChangedListener

    ...
    mSearchView.setInputType(EditorInfo.TYPE_CLASS_TEXT
            | EditorInfo.TYPE_TEXT_VARIATION_EMAIL_ADDRESS);
    mSearchView.addTextChangedListener(new SearchTextWatcher());
    ...
    
    private class SearchTextWatcher implements TextWatcher {
    
        @Override
        public void onTextChanged(CharSequence queryString, int start, int before, int count) {
            if (queryString.equals(mQueryString)) {
                return;
            }
    		//当前输入的模糊查询的名称
            mQueryString = queryString.toString();
            if (!mSearchMode) {
                if (!TextUtils.isEmpty(queryString)) {
                    setSearchMode(true);
                }
            } else if (mListener != null) {
    			//回调通知 PeopleActivity 改变界面
                mListener.onAction(Action.CHANGE_SEARCH_QUERY);
            }
            mClearSearchView.setVisibility(
                    TextUtils.isEmpty(queryString) ? View.GONE : View.VISIBLE);
        }
    
        @Override
        public void afterTextChanged(Editable s) {}
    
        @Override
        public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
    }
    

    回到 PeopleActivity 中找到监听 Action.CHANGE_SEARCH_QUERY 的代码如下

     @Override
    public void onAction(int action) {
        Log.d(TAG,"[onAction]action = " + action);
        /// M: [vcs] @{
        if (mVcsController != null) {
            mVcsController.onActionVcs(action);
        }
        /// @}
        switch (action) {
    	...
    	case ActionBarAdapter.Listener.Action.CHANGE_SEARCH_QUERY:
    			//获取当前输入的模糊查询姓名
                final String queryString = mActionBarAdapter.getQueryString();
    			//显示对应的fragment
                setQueryTextToFragment(queryString);
                updateDebugOptionsVisibility(
                        ENABLE_DEBUG_OPTIONS_HIDDEN_CODE.equals(queryString));
                break;
            default:
                throw new IllegalStateException("Unkonwn ActionBarAdapter action: " + action);
        }
    }
    

    到此,搜索框模糊查询对应的逻辑就分析完了,那么我们就模拟调用对应的逻辑就ok了,再来把整体流程捋一遍,

    点击搜索框->获取焦点->弹出键盘->输入姓名->收到文字内容改变的监听->将输入的内容回调给 PeopleActivity->收到回调显示对应的结果Fragment

    好了,通过调用EditText.setText()方法也能触发文字内容改变的监听,前提是要先获取焦点,那么我们的 showQueryTextFragment() 实现如下

    private void showQueryTextFragment(String queryString){
        Log.d(TAG, "[showQueryTextFragment]");
        if (!mActionBarAdapter.isSelectionMode()) {
            Log.e(TAG, "[queryString==]"+queryString);
            mActionBarAdapter.setSearchMode(true);
            mActionBarAdapter.setQueryString(queryString);
        }
    }
    


    图2 第三方app拉起主界面显示对应的联系人

    三、第三方app拉起联系人详情界面只滑动到一半显示的问题

    -

    图3 拉起只显示一半

    图4 拉起完全显示

    首先从系统的联系人列表界面点击进入详情界面是能完整显示的,所以猜想应该是传递的参数不太一样。所以还是从onCreate()方法看下来

    public class QuickContactActivity extends ContactsActivity {
    
    		/**
         * QuickContacts immediately takes up the full screen. All possible information is shown.
         * This value for {@link android.provider.ContactsContract.QuickContact#EXTRA_MODE}
         * should only be used by the Contacts app.
         */
        public static final int MODE_FULLY_EXPANDED = 4;
    	//看上面的注释就知道了肯定是跟这个变量有关系, 立刻显示全屏,应当只用于 联系人 app 使用
    
    	 @Override
        protected void onCreate(Bundle savedInstanceState) {
            Trace.beginSection("onCreate()");
            super.onCreate(savedInstanceState);
    
            if (RequestPermissionsActivity.startPermissionActivity(this)) {
                return;
            }
    
            getWindow().setStatusBarColor(Color.TRANSPARENT);
    		//处理Intent传递的参数
            processIntent(getIntent());
    
     		.....
    		//Scroller初始化,传递滚动模式
    		mScroller.initialize(mMultiShrinkScrollerListener, mExtraMode == MODE_FULLY_EXPANDED);
            // mScroller needs to perform asynchronous measurements after initalize(), therefore
            // we can't mark this as GONE.
            mScroller.setVisibility(View.INVISIBLE);
    		...
    	}
    
    	private void processIntent(Intent intent) {
            ...
    		//获取传递的EXTRA_MODE,不传默认为large,查看api对应的int值为3,MODE_FULLY_EXPANDED为4, 所以不传递参数或者参数对应值不为4就只显示半屏
            mExtraMode = getIntent().getIntExtra(QuickContact.EXTRA_MODE, QuickContact.MODE_LARGE);
            
    		...
        }
    
    }
    

    通过上面的分析 intent需要传递 QuickContact.EXTRA_MODE 参数, 当你点进去 QuickContact中发现并没有对应4的变量(猜想应该是留了一手不让第三方app直接全屏显示)

    正确的打开姿势

    private void gotoContact(){
        Uri personUri = ContentUris.withAppendedId(Contacts.People.CONTENT_URI, 1);
        Intent intent = new Intent();
        intent.setAction(Intent.ACTION_VIEW);
        intent.setData(personUri);
    	//这句比较关键
        intent.putExtra(ContactsContract.QuickContact.EXTRA_MODE, 4);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intent);
    }
    

    总结

    1、话说当把navigationBar去掉以后,给每个activity添加返回按钮是个很麻烦的工作,可以借鉴一下苹果的思路,直接在屏幕(Window)中添加一个悬浮的按钮处理返回点击事件。具体实现可以看这篇Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮

    2、源码没那么可怕,干起来。

  • 相关阅读:
    codeforces 732E(贪心)
    codeforces 732D(二分)
    codeforces 731C(DFS)
    codeforces 651C(map、去重)
    codeforces 723D(DFS)
    codeforces 721C (拓排 + DP)
    2018 Multi-University Training Contest 3
    2018 Multi-University Training Contest 2
    2018 Multi-University Training Contest 1
    The 2018 ACM-ICPC Chinese Collegiate Programming Contest
  • 原文地址:https://www.cnblogs.com/cczheng-666/p/10770147.html
Copyright © 2011-2022 走看看