溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Android中Presentation如何實(shí)現(xiàn)雙屏異顯

發(fā)布時(shí)間:2021-09-27 11:00:38 來源:億速云 閱讀:222 作者:小新 欄目:編程語言

這篇文章給大家分享的是有關(guān)Android中Presentation如何實(shí)現(xiàn)雙屏異顯的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過來看看吧。

一、概述

現(xiàn)在越來越多的Android設(shè)備有多個(gè)屏幕,雙屏異顯應(yīng)用場(chǎng)景最多的應(yīng)該就是類似于收銀平臺(tái)那種設(shè)備,在主屏上店員能夠?qū)c(diǎn)商品進(jìn)行選擇錄入,副屏則是展示給我們的賬單詳情,但是它只通過了一個(gè)軟件系統(tǒng)就實(shí)現(xiàn)了雙屏異顯這個(gè)功能,而Presentation正是這其中的關(guān)鍵。

二、Presentation分析

1.簡(jiǎn)述:首先從它的繼承關(guān)系上來看Presentation是繼承自Dialog的,就是說它其實(shí)就是一種特殊的Dialog用于在第二個(gè)屏幕上顯示內(nèi)容的,它在創(chuàng)建時(shí)會(huì)和它的目標(biāo)展示屏幕相關(guān)聯(lián),包括它的context和一些配置參數(shù)等。

2.Context:然而這里提到的context和它的容器所處的context會(huì)有不同,它會(huì)用自身的context去加載presentation的布局和相關(guān)資源以此來確保在目標(biāo)屏幕上能夠展示正確的大小和獲取合適的屏幕密度。

3.自動(dòng)移除:雖說presentation和它相關(guān)聯(lián)的Activity的context不同,但是他們也不是完全分離的關(guān)系,當(dāng)和presentation相關(guān)聯(lián)的屏幕被移除后,presentation也會(huì)自動(dòng)的被移除,所以當(dāng)Activity處于pause和resume的狀態(tài)時(shí)Presentation也需要特別注意當(dāng)前顯示的內(nèi)容的狀態(tài)。

4.屏幕選擇:因?yàn)橛袝r(shí)候我們的Android設(shè)備有多個(gè)屏幕,所以選擇合適的屏幕去展示就顯得非常重要了,所以在我們顯示Presentation的時(shí)候需要去讓我們的系統(tǒng)去選擇合適的屏幕來進(jìn)行展示,以下是兩種方式去選擇一個(gè)合適的屏幕。

這是我們選擇展示presentation最簡(jiǎn)單的一種方式,media router是一種系統(tǒng)層級(jí)的服務(wù),它能夠追蹤到系統(tǒng)當(dāng)中所有可用的音頻和視屏route,當(dāng)有路徑被選中或取消選中,還有當(dāng)適合用presentation進(jìn)行顯示的時(shí)候的route改變的時(shí)候它會(huì)發(fā)送一個(gè)通知,然后應(yīng)用本身會(huì)監(jiān)控這個(gè)通知自動(dòng)的去選擇presentation的展示或者隱藏,這里的推薦使用的presentation display其實(shí)只是media router推薦的,如果我們的應(yīng)用需要在第二個(gè)屏幕上進(jìn)行顯示就使用,如果不用的話就用本地來展示內(nèi)容。

利用media router去選擇presentation的顯示屏幕利用display manager去選擇persentation的顯示屏幕

DisplayManager能夠監(jiān)控到我們系統(tǒng)當(dāng)中的所有連接上的顯示設(shè)備,然而不是所有的設(shè)備都適用于作為副屏來進(jìn)行內(nèi)容展示的,比如當(dāng)一個(gè)Activity想顯示一個(gè)presentation在主屏幕上,其實(shí)效果就會(huì)相當(dāng)于在主Activity當(dāng)中顯示了一個(gè)特殊的Dialog,所以當(dāng)我們選擇這種方式去選用Presentation去顯示的時(shí)候就必須給它綁定一個(gè)可用的display。

三、源碼分析

首先來看它的構(gòu)造函數(shù)

public Presentation(Context outerContext, Display display, int theme) { super(createPresentationContext(outerContext, display, theme), theme, false); mDisplay = display; mDisplayManager = (DisplayManager)getContext().getSystemService(DISPLAY_SERVICE); final Window w = getWindow(); final WindowManager.LayoutParams attr = w.getAttributes(); attr.token = mToken; w.setAttributes(attr); w.setGravity(Gravity.FILL); w.setType(TYPE_PRESENTATION); setCanceledOnTouchOutside(false); }

在它的形參中第一個(gè)Context參數(shù)非常重要,這是應(yīng)用正在展示presentation的一個(gè)context,它是Presentation自己創(chuàng)建的它自己的一個(gè)context,基于這個(gè)context才能正確的在它所關(guān)聯(lián)的屏幕上展示合適的信息。然后代碼里面設(shè)置了這個(gè)window的相關(guān)屬性。

接著我們看看它自身的Context的創(chuàng)建

private static Context createPresentationContext( Context outerContext, Display display, int theme) { //首先判斷傳入的context和display是否為空,為空則拋出異常 if (outerContext == null) { throw new IllegalArgumentException("outerContext must not be null"); } if (display == null) { throw new IllegalArgumentException("display must not be null"); } Context displayContext = outerContext.createDisplayContext(display); //這里是對(duì)它的主題的判斷,為0即為默認(rèn)主題 if (theme == 0) { TypedValue outValue = new TypedValue(); displayContext.getTheme().resolveAttribute(  com.android.internal.R.attr.presentationTheme, outValue, true); theme = outValue.resourceId; } // Derive the display's window manager from the outer window manager. // We do this because the outer window manager have some extra information // such as the parent window, which is important if the presentation uses // an application window type. final WindowManagerImpl outerWindowManager = (WindowManagerImpl)outerContext.getSystemService(WINDOW_SERVICE); final WindowManagerImpl displayWindowManager = outerWindowManager.createPresentationWindowManager(displayContext); //因?yàn)镃ontextThemeWrapper的父類是我們的Context //所以這里最終返回的就是Presentation給我們創(chuàng)建好的Context return new ContextThemeWrapper(displayContext, theme) { //在這個(gè)方法中又返回的是Object對(duì)象 @Override public Object getSystemService(String name) { if (WINDOW_SERVICE.equals(name)) {  return displayWindowManager;  //如果和這個(gè)傳入的name相同的話返回的就是上面windowManager創(chuàng)建出來的WindowManager的一個(gè)具體實(shí)現(xiàn) } //否則就返回的是Context這個(gè)抽象類中的一種服務(wù)類型 return super.getSystemService(name); } }; }

接著我們繼續(xù)看一下Presentation對(duì)屏幕增加、移除和改變的監(jiān)聽

private final DisplayListener mDisplayListener = new DisplayListener() { @Override public void onDisplayAdded(int displayId) { } @Override public void onDisplayRemoved(int displayId) { if (displayId == mDisplay.getDisplayId()) { handleDisplayRemoved(); } } @Override public void onDisplayChanged(int displayId) { if (displayId == mDisplay.getDisplayId()) { handleDisplayChanged(); } } };

這里我們看到它對(duì)add并沒有進(jìn)行處理,所以我們進(jìn)一步去看一下onDisplayRemoved中的關(guān)鍵handlerDisplayRemoved和onDisplayChanged中的核心handlerDisplayChanged的實(shí)現(xiàn)

handlerDisplayChanged:

private void handleDisplayChanged() { onDisplayChanged(); // We currently do not support configuration changes for presentations // (although we could add that feature with a bit more work). // If the display metrics have changed in any way then the current configuration // is invalid and the application must recreate the presentation to get // a new context. if (!isConfigurationStillValid()) { Log.i(TAG, "Presentation is being dismissed because the "  + "display metrics have changed since it was created."); cancel(); } }

在這個(gè)方法中,我們遺憾的發(fā)現(xiàn)Google程序員并沒有對(duì)當(dāng)前屏幕配置發(fā)生改變后做特殊的處理,所以當(dāng)我們的屏幕尺寸等信息改變時(shí),我們的presentation必須重新Create去重新獲取context,再重新進(jìn)行顯示。

@Override protected void onStart() { super.onStart(); mDisplayManager.registerDisplayListener(mDisplayListener, mHandler); // Since we were not watching for display changes until just now, there is a // chance that the display metrics have changed. If so, we will need to // dismiss the presentation immediately. This case is expected // to be rare but surprising, so we'll write a log message about it. if (!isConfigurationStillValid()) { Log.i(TAG, "Presentation is being dismissed because the "  + "display metrics have changed since it was created."); mHandler.sendEmptyMessage(MSG_CANCEL); } }

同時(shí)在onStart中我們也能發(fā)現(xiàn),因?yàn)樗鼤簳r(shí)還沒有對(duì)display change進(jìn)行監(jiān)聽,而這里又是有可能會(huì)改變的,所以在這種情況下它是打印了一個(gè)log去通知一下。

handlerDisplayRemoved這個(gè)方法的話是系統(tǒng)自動(dòng)去發(fā)送一個(gè)消息,然后調(diào)用后把presentation自動(dòng)取消掉。

感謝各位的閱讀!關(guān)于“Android中Presentation如何實(shí)現(xiàn)雙屏異顯”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI