自定义Dialog Theme的Activity_龚十一-CSDN博客

使用Hook方式优雅绕过检查来实现效果,在实现全屏显示Dialog显示时,一定要注意适配Android 8.0

##背景

最近公司开发NFC开门的功能,咋一眼看功能很简单,直接用Dialog方式封装就能解决问题。但是我们公司是模块化的方式搭建开发App,为了让模块间的代码层次清晰与扩展及易与阅读,因此我们最终用Activity的方式实现Dialog的样式及功能。

Dialog 1

dialog 2

以下的开发过程以dialog1图片举例说明

##开发过程

###1.创建Theme样式

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">&lt;style name="activityTransparentTheme" parent="@style/AppTheme.NoActionBar"&gt;</p><p>   &lt;item name="android:windowFrame"&gt;@null&lt;/item&gt; &lt;!-- 无windowFrame --&gt;</p><p>   &lt;item name="android:windowIsFloating"&gt;true&lt;/item&gt;  &lt;!-- 浮在activity之上 --&gt;</p><p>   &lt;item name="android:windowIsTranslucent"&gt;false&lt;/item&gt;  &lt;!-- 半透明 --&gt;</p><p>   &lt;item name="android:windowNoTitle"&gt;true&lt;/item&gt; &lt;!-- 隐藏标题 --&gt;</p><p>   &lt;item name="windowNoTitle"&gt;true&lt;/item&gt; &lt;!-- 隐藏标题 --&gt;</p><p>   &lt;item name="android:background"&gt;@android:color/transparent&lt;/item&gt;</p><p>   &lt;item name="android:windowBackground"&gt;@android:color/transparent&lt;/item&gt; &lt;!-- 设置系统给定的透明值 --&gt;</p><p>   &lt;item name="android:backgroundDimEnabled"&gt;true&lt;/item&gt;  &lt;!-- 背景是否变暗--&gt;</p><p>   &lt;item name="android:windowAnimationStyle"&gt;@style/AnimationFade&lt;/item&gt;</p><p>   &lt;/style&gt;   &lt;!-- activity显示方式 --&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li><li style="color: rgb(153, 153, 153);">8</li><li style="color: rgb(153, 153, 153);">9</li><li style="color: rgb(153, 153, 153);">10</li><li style="color: rgb(153, 153, 153);">11</li></ul>

1、如果继承的Activity是AppCompatActivity,需要用@style/AppTheme XX的样式

2、如果阴影部分出现文字,用windowNoTitle和android:windowNoTitle设置为True即可

扩展:1.windowNoTitle = false 并且 android:windowNoTitle = false 时,会出现两个标题,位于下方的是AppCompatActivity的标题栏。

2.windowNoTitle = true并且 android:windowNoTitle = true时 ,无标题。

###2.创建Activity动画

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;"> &lt;style name="AnimationFade" parent="@android:style/Animation.Activity"&gt;</p><p>     &lt;item name="android:activityOpenEnterAnimation"&gt;@anim/fade_in&lt;/item&gt;</p><p>     &lt;item name="android:activityOpenExitAnimation"&gt;@anim/fade_out&lt;/item&gt;</p><p>     &lt;item name="android:activityCloseEnterAnimation"&gt;@anim/fade_in&lt;/item&gt;</p><p>     &lt;item name="android:activityCloseExitAnimation"&gt;@anim/fade_out&lt;/item&gt;</p><p> &lt;/style&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li></ul>

####2.1 淡出淡入动画

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">&lt;?xml version="1.0" encoding="utf-8"?&gt;</p><p>&lt;alpha xmlns:android="http://schemas.android.com/apk/res/android"</p><p>       android:duration="200"</p><p>       android:fillAfter="true"</p><p>       android:fromAlpha="0"</p><p>       android:toAlpha="1"/&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li></ul>

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">&lt;?xml version="1.0" encoding="utf-8"?&gt;</p><p>&lt;alpha xmlns:android="http://schemas.android.com/apk/res/android"</p><p>       android:duration="200"</p><p>       android:fillAfter="true"</p><p>       android:fromAlpha="1"</p><p>       android:toAlpha="0"/&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li></ul>

###3.创建自定义Dialog实现

####3.1 layout布局

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">  &lt;LinearLayout</p><p>        android:id="@+id/content_layout"</p><p>        android:layout_width="match_parent"</p><p>        android:layout_height="wrap_content"</p><p>        android:layout_gravity="center"</p><p>        android:layout_marginEnd="24dp"</p><p>        android:layout_marginStart="24dp"</p><p>        android:background="@drawable/common_rounded_dialog_bg"</p><p>        android:orientation="vertical"</p><p>        android:paddingBottom="32dp"</p><p>        android:paddingEnd="16dp"</p><p>        android:paddingStart="16dp"</p><p>        android:paddingTop="32dp"&gt;</p><p>        &lt;TextView</p><p>            android:id="@+id/tv_message"</p><p>            android:layout_width="match_parent"</p><p>            android:layout_height="wrap_content"</p><p>            android:textColor="@color/colorBlack"</p><p>            android:textSize="@dimen/size_p2"/&gt;</p><p>        &lt;TextView</p><p>            android:id="@+id/tv_confirm"</p><p>            android:layout_width="match_parent"</p><p>            android:layout_height="48dp"</p><p>            android:layout_marginTop="32dp"</p><p>            android:background="@drawable/btn_black_shape_25"</p><p>            android:gravity="center"</p><p>            android:textColor="@color/colorWhite"</p><p>            android:textStyle="bold"/&gt;</p><p>    &lt;/LinearLayout&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li><li style="color: rgb(153, 153, 153);">8</li><li style="color: rgb(153, 153, 153);">9</li><li style="color: rgb(153, 153, 153);">10</li><li style="color: rgb(153, 153, 153);">11</li><li style="color: rgb(153, 153, 153);">12</li><li style="color: rgb(153, 153, 153);">13</li><li style="color: rgb(153, 153, 153);">14</li><li style="color: rgb(153, 153, 153);">15</li><li style="color: rgb(153, 153, 153);">16</li><li style="color: rgb(153, 153, 153);">17</li><li style="color: rgb(153, 153, 153);">18</li><li style="color: rgb(153, 153, 153);">19</li><li style="color: rgb(153, 153, 153);">20</li><li style="color: rgb(153, 153, 153);">21</li><li style="color: rgb(153, 153, 153);">22</li><li style="color: rgb(153, 153, 153);">23</li><li style="color: rgb(153, 153, 153);">24</li><li style="color: rgb(153, 153, 153);">25</li><li style="color: rgb(153, 153, 153);">26</li><li style="color: rgb(153, 153, 153);">27</li><li style="color: rgb(153, 153, 153);">28</li><li style="color: rgb(153, 153, 153);">29</li><li style="color: rgb(153, 153, 153);">30</li><li style="color: rgb(153, 153, 153);">31</li><li style="color: rgb(153, 153, 153);">32</li></ul>

#####3.1.1 btn_black_shape_25.xml

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">&lt;?xml version="1.0" encoding="utf-8"?&gt;</p><p>&lt;shape xmlns:android="http://schemas.android.com/apk/res/android"</p><p>    android:shape="rectangle"&gt;</p><p>    &lt;corners android:radius="25dp" /&gt;</p><p>    &lt;solid android:color="@color/colorBlack"/&gt;</p><p>&lt;/shape&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li></ul>

#####3.1.2 common_rounded_dialog_bg.xml

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">&lt;?xml version="1.0" encoding="utf-8"?&gt;</p><p>&lt;shape xmlns:android="http://schemas.android.com/apk/res/android"</p><p>    android:shape="rectangle" &gt;</p><p>    &lt;corners</p><p>        android:radius="15dp" /&gt;</p><p>    &lt;gradient</p><p>        android:endColor="@android:color/white"</p><p>        android:startColor="@android:color/white" /&gt;</p><p>&lt;/shape&gt;</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li><li style="color: rgb(153, 153, 153);">8</li><li style="color: rgb(153, 153, 153);">9</li><li style="color: rgb(153, 153, 153);">10</li></ul>

####3.2 activity实现

在实现全屏显示Dialog显示时,一定要注意适配Android 8.0,网络上上大多说“删除AndroidManifest.xml中相应Activity的 android:screenOrientation=”“属性;或者删除相应Activity的android:theme=””属性中对应style中true属性”,我认为使用Hook方式优雅绕过检查来实现效果会更好。

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">class UnlockDoorActivity : BaseDataBindingActivity() {</p><p>    var hasFinishView = true</p><p>    override fun onCreate(savedInstanceState: Bundle?) {</p><p>        &lt;!-- Hook方式 解决Android8.0 的bug  --&gt;</p><p>        ActivityHook.hookOrientation(this)</p><p>        super.onCreate(savedInstanceState)</p><p>        val windowManager: WindowManager? = this.windowManager</p><p>        val display: Display? = windowManager?.defaultDisplay</p><p>        val lp = this.window?.attributes</p><p>        lp?.width = display?.width</p><p>        lp?.gravity = Gravity.BOTTOM</p><p>        this.window?.attributes = lp</p><p>    }</p><p>   ....</p><p>    override val layoutId: Int = R.layout.activity_unlock_door</p><p>   ....</p><p>    override fun finish() {</p><p>        super.finish()</p><p>        //activity消失时的动画</p><p>        overridePendingTransition(R.anim.fade_in, R.anim.fade_out)</p><p>    }</p><p>}</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li><li style="color: rgb(153, 153, 153);">8</li><li style="color: rgb(153, 153, 153);">9</li><li style="color: rgb(153, 153, 153);">10</li><li style="color: rgb(153, 153, 153);">11</li><li style="color: rgb(153, 153, 153);">12</li><li style="color: rgb(153, 153, 153);">13</li><li style="color: rgb(153, 153, 153);">14</li><li style="color: rgb(153, 153, 153);">15</li><li style="color: rgb(153, 153, 153);">16</li><li style="color: rgb(153, 153, 153);">17</li><li style="color: rgb(153, 153, 153);">18</li><li style="color: rgb(153, 153, 153);">19</li><li style="color: rgb(153, 153, 153);">20</li><li style="color: rgb(153, 153, 153);">21</li><li style="color: rgb(153, 153, 153);">22</li><li style="color: rgb(153, 153, 153);">23</li><li style="color: rgb(153, 153, 153);">24</li><li style="color: rgb(153, 153, 153);">25</li><li style="color: rgb(153, 153, 153);">26</li><li style="color: rgb(153, 153, 153);">27</li></ul>

3.2.2 ActivityHook.java

<code class="has-numbering" onclick="mdcp.copyCode(event)" style="position: unset;">/**</p><p> * http://www.manongjc.com/article/37655.html</p><p> */</p><p>public class ActivityHook {</p><p>  /**</p><p>   * java.lang.IllegalStateException: Only fullscreen opaque activities can request orientation</p><p>   * &lt;p&gt;</p><p>   * 修复android 8.0存在的问题</p><p>   * &lt;p&gt;</p><p>   * 在Activity中onCreate()中super之前调用</p><p>   *</p><p>   * @param activity Activity</p><p>   */</p><p>  public static void hookOrientation(Activity activity) {</p><p>    //目标版本8.0及其以上</p><p>    if (activity.getApplicationInfo().targetSdkVersion &gt;= Build.VERSION_CODES.O) {</p><p>      if (isTranslucentOrFloating(activity)) {</p><p>        fixOrientation(activity);</p><p>      }</p><p>    }</p><p>  }</p><p>  /**</p><p>   * 设置屏幕不固定,绕过检查</p><p>   *</p><p>   * @param activity Activity</p><p>   */</p><p>  private static void fixOrientation(Activity activity) {</p><p>    try {</p><p>      Class&lt;Activity&gt; activityClass = Activity.class;</p><p>      Field mActivityInfoField = activityClass.getDeclaredField("mActivityInfo");</p><p>      mActivityInfoField.setAccessible(true);</p><p>      ActivityInfo activityInfo = (ActivityInfo) mActivityInfoField.get(activity);</p><p>      //设置屏幕不固定</p><p>      activityInfo.screenOrientation = ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;</p><p>    } catch (Exception e) {</p><p>      e.printStackTrace();</p><p>    }</p><p>  }</p><p>  /**</p><p>   * 检查屏幕 横竖屏或者锁定就是固定</p><p>   *</p><p>   * @param activity Activity</p><p>   * @return true</p><p>   */</p><p>  private static boolean isTranslucentOrFloating(Activity activity) {</p><p>    boolean isTranslucentOrFloating = false;</p><p>    try {</p><p>      Class&lt;?&gt; styleableClass = Class.forName("com.android.internal.R$styleable");</p><p>      Field WindowField = styleableClass.getDeclaredField("Window");</p><p>      WindowField.setAccessible(true);</p><p>      int[] styleableRes = (int[]) WindowField.get(null);</p><p>      //先获取到TypedArray</p><p>      final TypedArray typedArray = activity.obtainStyledAttributes(styleableRes);</p><p>      Class&lt;?&gt; ActivityInfoClass = ActivityInfo.class;</p><p>      //调用检查是否屏幕旋转</p><p>      Method isTranslucentOrFloatingMethod = ActivityInfoClass.getDeclaredMethod("isTranslucentOrFloating", TypedArray.class);</p><p>      isTranslucentOrFloatingMethod.setAccessible(true);</p><p>      isTranslucentOrFloating = (boolean) isTranslucentOrFloatingMethod.invoke(null, typedArray);</p><p>    } catch (Exception e) {</p><p>      e.printStackTrace();</p><p>    }</p><p>    return isTranslucentOrFloating;</p><p>  }</p><p>}</p><p><div class="hljs-button {2}" data-title="复制"></div></code><ul class="pre-numbering" style=""><li style="color: rgb(153, 153, 153);">1</li><li style="color: rgb(153, 153, 153);">2</li><li style="color: rgb(153, 153, 153);">3</li><li style="color: rgb(153, 153, 153);">4</li><li style="color: rgb(153, 153, 153);">5</li><li style="color: rgb(153, 153, 153);">6</li><li style="color: rgb(153, 153, 153);">7</li><li style="color: rgb(153, 153, 153);">8</li><li style="color: rgb(153, 153, 153);">9</li><li style="color: rgb(153, 153, 153);">10</li><li style="color: rgb(153, 153, 153);">11</li><li style="color: rgb(153, 153, 153);">12</li><li style="color: rgb(153, 153, 153);">13</li><li style="color: rgb(153, 153, 153);">14</li><li style="color: rgb(153, 153, 153);">15</li><li style="color: rgb(153, 153, 153);">16</li><li style="color: rgb(153, 153, 153);">17</li><li style="color: rgb(153, 153, 153);">18</li><li style="color: rgb(153, 153, 153);">19</li><li style="color: rgb(153, 153, 153);">20</li><li style="color: rgb(153, 153, 153);">21</li><li style="color: rgb(153, 153, 153);">22</li><li style="color: rgb(153, 153, 153);">23</li><li style="color: rgb(153, 153, 153);">24</li><li style="color: rgb(153, 153, 153);">25</li><li style="color: rgb(153, 153, 153);">26</li><li style="color: rgb(153, 153, 153);">27</li><li style="color: rgb(153, 153, 153);">28</li><li style="color: rgb(153, 153, 153);">29</li><li style="color: rgb(153, 153, 153);">30</li><li style="color: rgb(153, 153, 153);">31</li><li style="color: rgb(153, 153, 153);">32</li><li style="color: rgb(153, 153, 153);">33</li><li style="color: rgb(153, 153, 153);">34</li><li style="color: rgb(153, 153, 153);">35</li><li style="color: rgb(153, 153, 153);">36</li><li style="color: rgb(153, 153, 153);">37</li><li style="color: rgb(153, 153, 153);">38</li><li style="color: rgb(153, 153, 153);">39</li><li style="color: rgb(153, 153, 153);">40</li><li style="color: rgb(153, 153, 153);">41</li><li style="color: rgb(153, 153, 153);">42</li><li style="color: rgb(153, 153, 153);">43</li><li style="color: rgb(153, 153, 153);">44</li><li style="color: rgb(153, 153, 153);">45</li><li style="color: rgb(153, 153, 153);">46</li><li style="color: rgb(153, 153, 153);">47</li><li style="color: rgb(153, 153, 153);">48</li><li style="color: rgb(153, 153, 153);">49</li><li style="color: rgb(153, 153, 153);">50</li><li style="color: rgb(153, 153, 153);">51</li><li style="color: rgb(153, 153, 153);">52</li><li style="color: rgb(153, 153, 153);">53</li><li style="color: rgb(153, 153, 153);">54</li><li style="color: rgb(153, 153, 153);">55</li><li style="color: rgb(153, 153, 153);">56</li><li style="color: rgb(153, 153, 153);">57</li><li style="color: rgb(153, 153, 153);">58</li><li style="color: rgb(153, 153, 153);">59</li><li style="color: rgb(153, 153, 153);">60</li><li style="color: rgb(153, 153, 153);">61</li><li style="color: rgb(153, 153, 153);">62</li><li style="color: rgb(153, 153, 153);">63</li><li style="color: rgb(153, 153, 153);">64</li><li style="color: rgb(153, 153, 153);">65</li><li style="color: rgb(153, 153, 153);">66</li><li style="color: rgb(153, 153, 153);">67</li></ul>

##参考

android 8.0 Only fullscreen opaque activities can request orientation(Hook方式优雅绕过检查 )

来源URL:https://blog.csdn.net/qq_30280379/article/details/93621480