如何允许单击禁用的首选项(或模仿禁用的首选项功能)?

时间:2016-11-08 15:18:31

标签: android android-preferences

背景

某些应用功能需要权限。如果未授予权限,我们需要对其进行“半”禁用(并且如果勾选或不勾选则不进行更改),并且文本告诉它由于没有足够的授予权限而无法使用,因此它只是部分禁用(文本将保持黑色,警告为红色)。它看起来不像普通的禁用偏好。

与正常的禁用首选项一样,当它处于此状态时,依赖于它的所有首选项都将被禁用且不可单击。

但是,与正常的禁用偏好不同,点击此类偏好后,它将转到授予所需权限的屏幕。

问题

有多种方法可以处理点击偏好设置,但我找不到任何方法来点击已禁用的偏好设置。

由于各种其他偏好取决于这样的偏好(因此根据是否启用它们来启用/禁用它们),我需要让它在需要时被禁用,但允许点击它。

我尝试了什么

我尝试了各种方法来处理点击:

  • onPreferenceTree点击PreferenceFragment
  • 首选项本身的setOnPreferenceClickListener
  • 试图覆盖performClick的偏好,但它被认为是一件坏事,因为它是隐藏的
  • 覆盖onClick of preference。
  • 在首选项的onCreateView或onBindView中,在创建的视图上使用setOnClickListener。
  • 包装首选项的onCreateView视图,并在视图或包装它的新视图上使用setOnClickListener。

当禁用首选项时,所有这些都不会被调用,并且您尝试单击首选项。

问题

  1. 是否可以停用偏好设置但仍处理点击次数?

  2. 或者,是否可以扩展首选项依赖项的功能?要使其他偏好依赖于此偏好,而不仅仅是基于它是否已启用?意味着只有在启用了这个并且它不处于半启用状态时才启用它们?

    目前,只有CheckboxPreference在应用程序上用作需要这种新状态的应用程序,因此很容易让其复​​选框看起来处于禁用状态,但让首选项本身保持启用状态(我只是使用 onCreateView方法中的view.findViewById(android.R.id.checkbox)

1 个答案:

答案 0 :(得分:3)

好的,我找到了解决方法:

class CheckboxPreferenceEx : android.preference.CheckBoxPreference {
    var alwaysAllowClickingEnabled: Boolean = false
    private val onPreferenceViewClickListener: View.OnClickListener

    constructor(context: Context) : super(context)
    constructor(context: Context, attrs: AttributeSet) : super(context, attrs)
    constructor(context: Context, attrs: AttributeSet, defStyle: Int) : super(context, attrs, defStyle)

    init {
        onPreferenceViewClickListener = View.OnClickListener {
            if (onPreferenceChangeListener != null && !isEnabled)
                if (onPreferenceChangeListener.onPreferenceChange(this@CheckboxPreferenceEx, !isChecked))
                    isChecked = !isChecked
            if (this@CheckboxPreferenceEx.isEnabled)
                this@CheckboxPreferenceEx.onClick()
        }
    }

    override fun onCreateView(parent: ViewGroup): View {
        val view = super.onCreateView(parent)
        setPreferenceTitleTextViewToHaveMultipleLines(view)
        //mTitleTextView = (TextView) view.findViewById(android.R.id.title);
        //mSummaryTextView = (TextView) view.findViewById(android.R.id.summary);
        return view
    }

    override fun onBindView(view: View) {
        super.onBindView(view)
        view.setOnClickListener(onPreferenceViewClickListener)
        val dependency = dependency
        if (!TextUtils.isEmpty(dependency) && !isEnabled) {
            //if the reason this preference is disabled is because of dependency, act as normal
            val dependencyPreference = preferenceManager.findPreference(dependency)
            if (!dependencyPreference!!.isEnabled || dependencyPreference is TwoStatePreference && !dependencyPreference.isChecked)
                return
        }
        if (alwaysAllowClickingEnabled) {
            //we chose to enable all except for the checkbox, which will still look disabled
            view.isEnabled = true
            setEnabledToViewAndItsDescendants(view, true)
        }
    }

    companion object {

        fun setPreferenceTitleTextViewToHaveMultipleLines(v: View) {
            if (v is TextView && v.getId() == android.R.id.title) {
                v.setSingleLine(false)
                return
            }
            if (v is ViewGroup) {
                for (i in 0 until v.childCount) {
                    val child = v.getChildAt(i)
                    setPreferenceTitleTextViewToHaveMultipleLines(child)
                }
            }
        }

        fun setEnabledToViewAndItsDescendants(v: View?, enabled: Boolean) {
            if (v == null)
                return
            v.isEnabled = enabled
            if (v is ViewGroup) {
                val viewGroups = ArrayList<ViewGroup>()
                viewGroups.add(v)
                while (!viewGroups.isEmpty()) {
                    val viewGroup = viewGroups.removeAt(0)
                    val childCount = viewGroup.childCount
                    for (i in 0 until childCount) {
                        val childView = viewGroup.getChildAt(i)
                        childView.isEnabled = enabled
                        if (childView is ViewGroup)
                            viewGroups.add(childView)
                    }
                }
            }
        }
    }

}

有这样的偏好,当我在其上调用setAlwaysAllowClickingEnabled(true)时,会导致首选项被禁用,但是当你点击它时,它会调用回调。

在我们的示例中,我们希望禁用复选框,但启用标题和摘要TextView,但您可以根据需要执行操作。更新了代码。