如何实现安卓的分秒选择器?

4
我需要实现一个对话框,用户可以在其中输入分钟和秒数。例如:6分钟30秒。我还需要对分钟数设置上限(即,分钟范围为0-10,秒数为0-60)。
但正如在这里讨论的那样,Android TimePicker仅显示小时和分钟。我看到有一些帖子关于添加秒数,但没有关于删除小时并添加秒数的内容。可能也没有办法限制分钟数。
我看到有一些帖子推荐使用NumberPicker创建自定义时间选择器,但也看到了警告,称NumberPicker仅用于内部目的,不应该被使用。这似乎是个进退两难的局面。
我可以尝试修改TimePicker或NumberPicker的源代码,但我不太喜欢花费大量的时间去理解它们的工作原理,并维护一个自定义版本。
我想最好的方法是实现一个自定义的分钟-秒钟选择器。使用上下按钮移动数字看起来不太难,但我不知道如何允许用户滑动滚动每个“轮子”中的数字。
有什么建议吗?
2个回答

6

使用这个SO答案给TimePickerDialog添加分钟间隔后,我创建了一个类似的子类,将小时选项卡扩展到0-60,并将小时:分钟选项卡视为分钟:秒:

import android.app.TimePickerDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.widget.NumberPicker;
import android.widget.TimePicker;

import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;

public class DurationPicker extends TimePickerDialog {

    private TimePicker timePicker;
    private final OnTimeSetListener callback;

    private int initialMinutes;

    public DurationPicker(Context context, OnTimeSetListener callBack, int minutes, int seconds) {
        super(context, callBack, minutes, seconds, true); // the last argument removes the am/pm spinner
        this.callback = callBack;
        this.initialMinutes = minutes; // we'll have to set this again after modifying the "hours" spinner
        this.setTitle("Set duration");
    }

    @Override
    public void onClick(DialogInterface dialog, int which) {
        if (callback != null && timePicker != null) {
            timePicker.clearFocus();
            callback.onTimeSet(timePicker, timePicker.getCurrentHour(), timePicker.getCurrentMinute());
        }
    }

    @Override
    public void onAttachedToWindow() {
        super.onAttachedToWindow();
        try {
            Class<?> classForid = Class.forName("com.android.internal.R$id");
            Field timePickerField = classForid.getField("timePicker");
            this.timePicker = (TimePicker) findViewById(timePickerField.getInt(null));
            Field field = classForid.getField("hour");

            // modify the hours spinner to cover the maximum number of minutes we want to support
            int maxMinutes = 60;
            NumberPicker mHourSpinner = (NumberPicker)timePicker.findViewById(field.getInt(null));
            mHourSpinner.setMinValue(0);
            mHourSpinner.setMaxValue(maxMinutes);
            List<String> displayedValues = new ArrayList<String>();
            for (int i = 0; i <= maxMinutes; i++) {
                displayedValues.add(String.format("%d", i));
            }
            mHourSpinner.setDisplayedValues(displayedValues.toArray(new String[0]));
            mHourSpinner.setValue(initialMinutes); // we can set this again now that we've modified the hours spinner
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

}

在回调函数中,只需将第二个参数的值视为分钟,第三个参数的值视为秒钟。
可能在Android的未来版本中,当“分钟”微调器设置为24以上的数字时,会返回不正确的结果,但我已经阅读了当前TimePicker代码,并没有发现任何问题。此外,可能可以修改或替换上午/下午微调器为秒微调器,然后选择器就可以支持小时:分钟:秒以及分钟:秒。

对话框中的时间选择器显示为时钟,而不是微调器。如何更改模式? - CodeMonkey
@YonatanNir 请看这个答案 https://dev59.com/5lsW5IYBdhLWcg3wyJl3#48163947 - Zezi Reeds

0

不幸的是,@arlomedia的代码对我没有起作用,因为它没有找到hour字段。 我根据this的答案(Kotlin版本)修改了这个类:


import android.app.TimePickerDialog
import android.content.Context
import android.content.DialogInterface
import android.content.res.Resources
import android.view.View
import android.widget.NumberPicker
import android.widget.TimePicker
import java.util.*


class DurationPickerDialog(context: Context?, private val callback: OnTimeSetListener?, private val initialMinutes: Int, seconds: Int) :
    TimePickerDialog(context, callback, initialMinutes, seconds, true) {

    init {
        this.setTitle("Set duration")
    }

    private var timePicker: TimePicker? = null

    override fun onClick(dialog: DialogInterface, which: Int) {
        if (callback != null && timePicker != null) {
            timePicker!!.clearFocus()
            callback.onTimeSet(timePicker, timePicker!!.currentHour, timePicker!!.currentMinute)
        }
    }

    override fun onAttachedToWindow() {
        super.onAttachedToWindow()
        try {

          
            timePicker = findViewById<View>(
                Resources.getSystem().getIdentifier("timePicker", "id", "android")
            ) as TimePicker


            val hoursPicker = timePicker!!.findViewById(
                Resources.getSystem().getIdentifier("hour", "id", "android")
            ) as NumberPicker

            // Modify the hours spinner to cover the maximum number of minutes we want to support

            val maxMinutes = 60
            hoursPicker.minValue = 0
            hoursPicker.maxValue = maxMinutes

            val displayedValues: MutableList<String> = ArrayList()

            for (i in 0..maxMinutes) displayedValues.add(String.format("%d", i))

            hoursPicker.displayedValues = displayedValues.toTypedArray()
            hoursPicker.value = initialMinutes

        } catch (e: Exception) {
            e.printStackTrace()
        }
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接