जेली बीन DatePickerDialog - क्या रद्द करने का कोई तरीका है?


148

--- मध्यस्थों पर ध्यान दें: आज (15 जुलाई), मैंने देखा है कि किसी को पहले से ही इस समस्या का सामना करना पड़ रहा है । लेकिन मुझे यकीन नहीं है कि यह एक डुप्लिकेट के रूप में बंद करना उचित है, क्योंकि मुझे लगता है कि मैंने इस मुद्दे का बहुत बेहतर विवरण प्रदान किया है। मुझे यकीन नहीं है कि मुझे दूसरे प्रश्न को संपादित करना चाहिए और इस सामग्री को वहां पेस्ट करना चाहिए, लेकिन मैं किसी और के प्रश्न को बहुत अधिक बदलने में सहज नहीं हूं। ---

मेरे यहाँ कुछ अजीब है।

मुझे नहीं लगता कि समस्या इस बात पर निर्भर करती है कि आप किस एसडीके के खिलाफ निर्माण करते हैं। डिवाइस ओएस संस्करण क्या मायने रखता है।

समस्या # 1: डिफ़ॉल्ट रूप से असंगतता

DatePickerDialogबदल गया था (?) Jelly Bean में और अब केवल एक प्रदान करता है डन बटन। पिछले संस्करणों में एक रद्द बटन शामिल था , और यह उपयोगकर्ता के अनुभव (असंगतता, पिछले Android संस्करणों से मांसपेशियों की स्मृति) को प्रभावित कर सकता है।

प्रतिकृति: एक मूल परियोजना बनाएँ। इसे इसमें डालेंonCreate:

DatePickerDialog picker = new DatePickerDialog(
        this,
        new OnDateSetListener() {
            @Override
            public void onDateSet(DatePicker v, int y, int m, int d) {
                Log.d("Picker", "Set!");
            }
        },
        2012, 6, 15);
picker.show();

अपेक्षित: संवाद में प्रकट होनेवाला रद्द करें बटन।

वर्तमान: एक रद्द करें बटन दिखाई नहीं देता है।

स्क्रीनशॉट: 4.0.3 (ओके) और 4.1.1 (संभवतः गलत है?)।

समस्या # 2: गलत बर्खास्तगी व्यवहार

संवाद कॉल जो भी यह सुनने वास्तव में फोन करना चाहिए, और फिर हमेशा कहता OnDateSetListenerश्रोता। रद्द करना अभी भी सेट विधि को कॉल करता है, और इसे सेट करना दो बार विधि को कॉल करता है।

प्रतिकृति: # 1 कोड का उपयोग करें, लेकिन नीचे कोड जोड़ें (आप इस # 1 हल करेंगे, लेकिन केवल नेत्रहीन / UI देखें):

picker.setButton(DialogInterface.BUTTON_NEGATIVE, "Cancel", 
        new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                Log.d("Picker", "Cancel!");
            }
        });

अपेक्षित होना:

  • BACK कुंजी दबाने या डायलॉग के बाहर क्लिक करने से कुछ नहीं होना चाहिए ।
  • "रद्द करें" दबाकर पिकर रद्द करना चाहिए !
  • "सेट" दबाने पर पिकर सेट को प्रिंट करना चाहिए !

वर्तमान:

  • BACK कुंजी दबाने या डायलॉग पिकर सेट के बाहर क्लिक करने पर !
  • दबाने "रद्द" प्रिंट बीनने रद्द! और फिर पिकर सेट!
  • "सेट" दबाकर पिकर सेट को प्रिंट करता है ! और फिर पिकर सेट!

व्यवहार दिखाने वाली लॉग लाइनें:

07-15 12:00:13.415: D/Picker(21000): Set!

07-15 12:00:24.860: D/Picker(21000): Cancel!
07-15 12:00:24.876: D/Picker(21000): Set!

07-15 12:00:33.696: D/Picker(21000): Set!
07-15 12:00:33.719: D/Picker(21000): Set!

अन्य नोट्स और टिप्पणियां

  • चारों ओर लपेटने से DatePickerFragmentकोई फर्क नहीं पड़ता। मैंने आपके लिए समस्या का सरलीकरण किया है, लेकिन मैंने इसका परीक्षण किया है।

बधाई हो, आपको एंड्रॉइड में बग मिला है। आप इसे यहाँ रिपोर्ट कर सकते हैं
माइकल हैम्पटन

1
बहुत अच्छी तरह से लिखित बग रिपोर्ट। मैं कोड को चलाने का परीक्षण किए बिना इसे पूरी तरह से समझ सकता हूं।
चेओक यान चेंग सेप

6
मैं इस मुद्दे को वोट करने के लिए सभी से आग्रह करता हूं! अंक 34833
ब्रैडले

क्या आप बटन कार्य को ओवरराइड नहीं कर सकते जैसे कि संवाद के बाहर छूने के कारण इसे खारिज कर दिया गया था?
कार्तिक बालकृष्णन

2
बग अभी भी 2 साल बाद खुला है ... अविश्वसनीय।
इरेडमस्टर

जवाबों:


115

नोट: लॉलीपॉप के रूप में फिक्स्ड , यहाँ स्रोतग्राहकों में उपयोग के लिए स्वचालित वर्ग (सभी एंड्रॉइड संस्करणों के साथ संगत) अपडेट किया गया।

टीएल; डीआर: एक वैश्विक समाधान के लिए 1-2-3 मृत आसान कदम:

  1. इस वर्ग को डाउनलोड करें
  2. OnDateSetListenerअपनी गतिविधि में लागू करें (या अपनी आवश्यकताओं के अनुरूप कक्षा बदलें)।
  3. इस कोड के साथ संवाद ट्रिगर करें (इस नमूने में, मैं इसे अंदर उपयोग करता हूं Fragment):

    Bundle b = new Bundle();
    b.putInt(DatePickerDialogFragment.YEAR, 2012);
    b.putInt(DatePickerDialogFragment.MONTH, 6);
    b.putInt(DatePickerDialogFragment.DATE, 17);
    DialogFragment picker = new DatePickerDialogFragment();
    picker.setArguments(b);
    picker.show(getActivity().getSupportFragmentManager(), "frag_date_picker");

और यह सब लेता है! कारण मैं अभी भी अपना जवाब "स्वीकृत" के रूप में रखता हूं, क्योंकि मैं अभी भी अपने समाधान को प्राथमिकता देता हूं क्योंकि ग्राहक कोड में यह बहुत छोटा पदचिह्न है, यह मूलभूत मुद्दे (श्रोता को फ्रेमवर्क क्लास में बुलाया जा रहा है) को संबोधित करता है, विन्यास परिवर्तनों में ठीक काम करता है और यह पिछले लॉजिक में डिफ़ॉल्ट कार्यान्वयन के लिए कोड तर्क को रूट करता है जो इस बग से ग्रस्त नहीं है (वर्ग स्रोत देखें)।

मूल उत्तर (ऐतिहासिक और उपदेशात्मक कारणों के लिए रखा गया):

बग स्रोत

ठीक है, ऐसा लगता है कि यह वास्तव में एक बग है और किसी और ने इसे पहले से ही भर दिया है। 34,833 जारी

मैंने पाया है कि समस्या संभवतः है DatePickerDialog.java। यह कहां पढ़ता है:

private void tryNotifyDateSet() {
    if (mCallBack != null) {
        mDatePicker.clearFocus();
        mCallBack.onDateSet(mDatePicker, mDatePicker.getYear(),
                mDatePicker.getMonth(), mDatePicker.getDayOfMonth());
    }
}

@Override
protected void onStop() {
    tryNotifyDateSet();
    super.onStop();
}

मुझे लगता है कि यह हो सकता था:

@Override
protected void onStop() {
    // instead of the full tryNotifyDateSet() call:
    if (mCallBack != null) mDatePicker.clearFocus();
    super.onStop();
}

अब अगर कोई मुझे बताए कि मैं Android को पैच / बग रिपोर्ट कैसे प्रस्तावित कर सकता हूं, तो मुझे खुशी होगी। इस बीच, मैंने DatePickerDialog.javaवहां जारी अंक में संलग्न संस्करण के रूप में एक संभावित सुधार (सरल) का सुझाव दिया ।

बग से बचने के लिए संकल्पना

nullकंस्ट्रक्टर में श्रोता को सेट करें और BUTTON_POSITIVEबाद में अपना खुद का बटन बनाएं । यह वह है, नीचे विवरण।

समस्या तब होती है DatePickerDialog.java, क्योंकि जैसा कि आप स्रोत में देख सकते हैं, एक वैश्विक चर ( mCallBack) को कॉल करता है जो उस श्रोता को संग्रहीत करता है जो कंस्ट्रक्टर में पारित किया गया था:

    /**
 * @param context The context the dialog is to run in.
 * @param callBack How the parent is notified that the date is set.
 * @param year The initial year of the dialog.
 * @param monthOfYear The initial month of the dialog.
 * @param dayOfMonth The initial day of the dialog.
 */
public DatePickerDialog(Context context,
        OnDateSetListener callBack,
        int year,
        int monthOfYear,
        int dayOfMonth) {
    this(context, 0, callBack, year, monthOfYear, dayOfMonth);
}

    /**
 * @param context The context the dialog is to run in.
 * @param theme the theme to apply to this dialog
 * @param callBack How the parent is notified that the date is set.
 * @param year The initial year of the dialog.
 * @param monthOfYear The initial month of the dialog.
 * @param dayOfMonth The initial day of the dialog.
 */
public DatePickerDialog(Context context,
        int theme,
        OnDateSetListener callBack,
        int year,
        int monthOfYear,
        int dayOfMonth) {
    super(context, theme);

    mCallBack = callBack;
    // ... rest of the constructor.
}

तो, चाल nullश्रोता के रूप में संग्रहीत होने के लिए एक श्रोता प्रदान करना है, और फिर अपने स्वयं के बटन सेट करें (नीचे # 1 से मूल कोड है, अपडेट किया गया):

    DatePickerDialog picker = new DatePickerDialog(
        this,
        null, // instead of a listener
        2012, 6, 15);
    picker.setCancelable(true);
    picker.setCanceledOnTouchOutside(true);
    picker.setButton(DialogInterface.BUTTON_POSITIVE, "OK",
        new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                Log.d("Picker", "Correct behavior!");
            }
        });
    picker.setButton(DialogInterface.BUTTON_NEGATIVE, "Cancel", 
        new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                Log.d("Picker", "Cancel!");
            }
        });
picker.show();

अब यह संभव सुधार के कारण काम करेगा जो मैंने ऊपर पोस्ट किया था।

और क्योंकि DatePickerDialog.javaएक के लिए चेक nullजब भी इसे पढ़ता है mCallback( एपीआई के दिनों से 3 / 1.5 ऐसा लगता है --- निश्चित रूप से हनीकोम्ब जांच नहीं कर सकता), यह अपवाद ट्रिगर नहीं करेगा। लॉलीपॉप के मुद्दे को ध्यान में रखते हुए, मैं इस पर ध्यान देने वाला नहीं हूं: बस डिफ़ॉल्ट कार्यान्वयन (मेरे द्वारा प्रदान की गई कक्षा में शामिल) का उपयोग करें।

सबसे पहले मुझे फोन न करने का डर था clearFocus(), लेकिन मैंने यहां परीक्षण किया है और लॉग लाइनें साफ थीं। इसलिए मैंने जो लाइन प्रस्तावित की थी वह सब के बाद भी आवश्यक नहीं हो सकती है, लेकिन मुझे नहीं पता।

पिछले एपीआई स्तरों के साथ संगतता (संपादित)

जैसा कि मैंने नीचे टिप्पणी में बताया है, यह एक अवधारणा थी, और आप अपने Google ड्राइव खाते से जिस वर्ग का उपयोग कर रहे हैं, उसे डाउनलोड कर सकते हैं । जिस तरह से मैंने उपयोग किया, डिफ़ॉल्ट सिस्टम कार्यान्वयन बग से प्रभावित नहीं होने वाले संस्करणों पर उपयोग किया जाता है।

मैंने कुछ अनुमान (बटन नाम आदि) लिए जो मेरी जरूरतों के लिए उपयुक्त हैं क्योंकि मैं ग्राहक वर्गों में बॉयलरप्लेट कोड को कम से कम करना चाहता था। पूर्ण उपयोग उदाहरण:

class YourActivity extends SherlockFragmentActivity implements OnDateSetListener

// ...

Bundle b = new Bundle();
b.putInt(DatePickerDialogFragment.YEAR, 2012);
b.putInt(DatePickerDialogFragment.MONTH, 6);
b.putInt(DatePickerDialogFragment.DATE, 17);
DialogFragment picker = new DatePickerDialogFragment();
picker.setArguments(b);
picker.show(getActivity().getSupportFragmentManager(), "fragment_date_picker");

11
अनुसंधान पर बहुत बढ़िया काम! दुख की बात है कि यह आवश्यक था, लेकिन फिर भी भयानक।
कॉमन्सवेअर

1
बहुत अच्छा! मैं इस मुद्दे पर दीवार के खिलाफ अपने सिर पर हथौड़ा मार रहा हूं जो मेरे टेक्स्टफिल्ड्स को अपडेट कर रहा था और सही कॉलबैक कॉलिंग / कॉलिंग नहीं कर रहा था। धन्यवाद! मुझे आश्चर्य है कि क्या सुझाया गया वर्कअराउंड "भविष्य का प्रमाण" है या यदि आपको लगता है कि बग ठीक होने पर यह समस्या पैदा करेगा?
अवधि

1
@RomainGuidoux अंत में अद्यतन उत्तर देखें। लिंक में मौजूद क्लास में जैली बीन में केवल उस विधि को कॉल करने के लिए स्मार्ट है। नीचे किसी भी चीज़ पर यह बाईपास हो जाएगा और डिफ़ॉल्ट सिस्टम कार्यान्वयन का उपयोग करेगा, आपकी गतिविधि के लिए ondateset के लिए सिस्टम कॉल को रूट करना। यह सिर्फ इतना है कि जेली बीन में कॉलबैक को रूट करने से पहले अतिरिक्त उपाय (बग से बचना) होता है, और इसमें उस छत्ते की विधि को कॉल करना शामिल है। लेकिन फिर, केवल जेबी में।
davidcsb

1
यहाँ बहुत बढ़िया काम। मेरे पास इस बात के लिए शब्द नहीं हैं कि यह मुद्दा कितना हास्यास्पद है।
बिल फिलिप्स

5
कैसे TimePickerDialog के बारे में? ऐसा लगता है कि TimePickerDialog में गेटटाइमपिकर नहीं है ()
lokoko

16

मैं समाधान पर अपनी खुद की रिफ़ जोड़ने जा रहा हूं, डेविड सीजरिनो ने पोस्ट किया है, यदि आप फ्रेगमेंट का उपयोग नहीं कर रहे हैं, और इसे सभी संस्करणों में ठीक करने का एक आसान तरीका चाहते हैं (2.1 थ्रू 4.1):

public class FixedDatePickerDialog extends DatePickerDialog {
  //I use a Calendar object to initialize it, but you can revert to Y,M,D easily
  public FixedDatePickerDialog(Calendar dateToShow, Context context, OnDateSetListener callBack) {
    super(context, null, dateToShow.get(YEAR), dateToShow.get(MONTH), dateToShow.get(DAY_OF_MONTH));
    initializePicker(callBack);
  }

  public FixedDatePickerDialog(Calendar dateToShow, Context context, int theme,
    OnDateSetListener callBack) {
    super(context, theme, null, dateToShow.get(YEAR), dateToShow.get(MONTH), dateToShow.get(DAY_OF_MONTH));
    initializePicker(callBack);
  }

  private void initializePicker(final OnDateSetListener callback) {
    try {
      //If you're only using Honeycomb+ then you can just call getDatePicker() instead of using reflection
      Field pickerField = DatePickerDialog.class.getDeclaredField("mDatePicker");
      pickerField.setAccessible(true);
      final DatePicker picker = (DatePicker) pickerField.get(this);
      this.setCancelable(true);
      this.setButton(DialogInterface.BUTTON_NEGATIVE, getContext().getText(android.R.string.cancel), (OnClickListener) null);
      this.setButton(DialogInterface.BUTTON_POSITIVE, getContext().getText(android.R.string.ok),
          new DialogInterface.OnClickListener() {
              @Override
              public void onClick(DialogInterface dialog, int which) {
                picker.clearFocus(); //Focus must be cleared so the value change listener is called
                callback.onDateSet(picker, picker.getYear(), picker.getMonth(), picker.getDayOfMonth());
              }
          });
    } catch (Exception e) { /* Reflection probably failed*/ }
  }
}

बस याद रखना: यदि आप बटन नामों को ठीक और रद्द करने के लिए सख्त कर रहे हैं, तो उपयोगकर्ता के स्वयं के बजाय मानक android.R.string.okऔर android.R.string.cancelफ़ील्ड का उपयोग करना बेहतर होगा । और उत्तर के लिए धन्यवाद।
davidcsb 16

आह, अच्छा, मैंने ध्यान नहीं दिया कि उन्होंने ओके और रद्द पाठ प्रदान किया है। धन्यवाद!
बजे

सुपर (संदर्भ, विषय, अशक्त, dateToShow.get (YEAR), dateToShow.get (MONTH), dateToShow.get (DAY_OF_MITTH) पर nullpointer अपवाद प्राप्त करना;
किशोर

इर्रर ... क्या आप एक अशक्त गुजर रहे हैं dateToShow? अन्य अशक्तता वास्तव में "फिक्स" है, इसलिए वहां होना चाहिए। आप किस संस्करण पर हैं?
१५

क्या आप मुझे बता सकते हैं importकि आपके पास क्या है Field? मेरे पास 6 विकल्प हैं और उनमें से कोई भी काम नहीं किया है।
आदिल मलिक

8

जब तक बग को ठीक नहीं किया जाएगा, मेरा सुझाव है कि DatePickerDialog या TimePickerDialog का उपयोग न करें। TimePicker / DatePicker विजेट के साथ कस्टम मेड AlertDialog का उपयोग करें;

बदलें TimePickerDialog के साथ;

    final TimePicker timePicker = new TimePicker(this);
    timePicker.setIs24HourView(true);
    timePicker.setCurrentHour(20);
    timePicker.setCurrentMinute(15);

    new AlertDialog.Builder(this)
            .setTitle("Test")
            .setPositiveButton(android.R.string.ok, new OnClickListener() {

                @Override
                public void onClick(DialogInterface dialog, int which) {
                    Log.d("Picker", timePicker.getCurrentHour() + ":"
                            + timePicker.getCurrentMinute());
                }
            })
            .setNegativeButton(android.R.string.cancel,
                    new OnClickListener() {

                        @Override
                        public void onClick(DialogInterface dialog,
                                int which) {
                            Log.d("Picker", "Cancelled!");
                        }
                    }).setView(timePicker).show();

के साथ DatePickerDialog बदलें;

    final DatePicker datePicker = new DatePicker(this);
    datePicker.init(2012, 10, 5, null);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
        datePicker.setCalendarViewShown(false);
    }

    new AlertDialog.Builder(this)
            .setTitle("Test")
            .setPositiveButton(android.R.string.ok, new OnClickListener() {

                @Override
                public void onClick(DialogInterface dialog, int which) {
                    Log.d("Picker", datePicker.getYear() + " "
                            + (datePicker.getMonth() + 1) + " "
                            + datePicker.getDayOfMonth());
                }
            })
            .setNegativeButton(android.R.string.cancel,
                    new OnClickListener() {

                        @Override
                        public void onClick(DialogInterface dialog,
                                int which) {
                            Log.d("Picker", "Cancelled!");
                        }
                    }).setView(datePicker).show();

4

डेविड सेसरिनो द्वारा समाधान पर आधारित टाइमपिकर के लिए एक , "टीएल; डीआर: एक वैश्विक समाधान के लिए 1-2-3 मृत आसान उपाय"

TimePickerDialog, DatePickerDialog.getDatePicker जैसी कार्यक्षमता प्रदान नहीं करता है। तो, OnTimeSetListener श्रोता प्रदान किया जाना है। बस DatePicker वर्कअराउंड समाधान के साथ समानता रखने के लिए, मैंने पुराने mListener अवधारणा को बनाए रखा है। जरूरत पड़ने पर आप इसे बदल सकते हैं।

कॉलिंग और श्रोता मूल समाधान के समान है। बस शामिल करें

import android.app.TimePickerDialog;
import android.app.TimePickerDialog.OnTimeSetListener;

अभिभावक वर्ग का विस्तार करें,

... implements OnDateSetListener, OnTimeSetListener

लागू

 @Override
 public void onTimeSet(TimePicker view, int hourOfDay, int minute) {
 ...
 }

उदाहरण कॉलिंग

    Calendar cal = Calendar.getInstance();
    int hour = cal.get(Calendar.HOUR_OF_DAY);
    int minute = cal.get(Calendar.MINUTE);


    Bundle b = new Bundle();
    b.putInt(TimePickerDialogFragment.HOUR, hour);
    b.putInt(TimePickerDialogFragment.MINUTE, minute);

    DialogFragment picker = new TimePickerDialogFragment();
    picker.setArguments(b);
    picker.show(getSupportFragmentManager(), "frag_time_picker");

(रद्द करने के लिए अद्यतन)

public class TimePickerDialogFragment extends DialogFragment {

    public static final String HOUR = "Hour";
    public static final String MINUTE = "Minute";

    private boolean isCancelled = false; //Added to handle cancel
    private TimePickerDialog.OnTimeSetListener mListener;

    //Added to handle parent listener
    private TimePickerDialog.OnTimeSetListener mTimeSetListener = new TimePickerDialog.OnTimeSetListener() {
        public void onTimeSet(TimePicker view, int hourOfDay, int minute) {
            if (!isCancelled)
            {
                mListener.onTimeSet(view,hourOfDay,minute);
            }
        }
    };
    //
    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        this.mListener = (TimePickerDialog.OnTimeSetListener) activity;
    }

    @Override
    public void onDetach() {
        this.mListener = null;
        super.onDetach();
    }

    @TargetApi(11)
    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        Bundle b = getArguments();
        int h = b.getInt(HOUR);
        int m = b.getInt(MINUTE);

        final TimePickerDialog picker = new TimePickerDialog(getActivity(), getConstructorListener(), h, m,DateFormat.is24HourFormat(getActivity()));

        //final TimePicker timePicker = new TimePicker(getBaseContext());
        if (hasJellyBeanAndAbove()) {
            picker.setButton(DialogInterface.BUTTON_POSITIVE,
                    getActivity().getString(android.R.string.ok),
                    new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialog, int which) {
                            isCancelled = false; //Cancel flag, used in mTimeSetListener
                        }
                    });
            picker.setButton(DialogInterface.BUTTON_NEGATIVE,
                    getActivity().getString(android.R.string.cancel),
                    new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialog, int which) {
                            isCancelled = true; //Cancel flag, used in mTimeSetListener
                        }
                    });
        }
        return picker;
    }
    private boolean hasJellyBeanAndAbove() {
        return Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN;
    }

    private TimePickerDialog.OnTimeSetListener getConstructorListener() {
        return hasJellyBeanAndAbove() ? mTimeSetListener : mListener; //instead of null, mTimeSetListener is returned.
    }
}

यह मेरे लिए काम नहीं कर रहा है जब यह s3 एपीआई 18 पर कोशिश कर रहा है
अब्देलहेडी

3

यदि कोई त्वरित समाधान चाहता है, तो यहां मैंने जो कोड इस्तेमाल किया है:

public void showCustomDatePicker () {

final DatePicker mDatePicker = (DatePicker) getLayoutInflater().
        inflate(R.layout.date_picker_view, null);
//Set an initial date for the picker
final Calendar c = Calendar.getInstance();
int year = c.get(Calendar.YEAR);
int month = c.get(Calendar.MONTH);
int day = c.get(Calendar.DAY_OF_MONTH);
//Set the date now
mDatePicker.updateDate(year, month, day);

//create the dialog
AlertDialog.Builder mBuilder = new Builder(this);
//set the title
mBuilder.setTitle(getString(R.string.date_picker_title))
    //set our date picker
    .setView(mDatePicker)
    //set the buttons 
.setPositiveButton(android.R.string.ok, new OnClickListener() {

    @Override
    public void onClick(DialogInterface dialog, int which) {
        //whatever method you choose to handle the date changes
            //the important thing to know is how to retrieve the data from the picker
        handleOnDateSet(mDatePicker.getYear(), 
                mDatePicker.getMonth(), 
                mDatePicker.getDayOfMonth());
    }
})
.setNegativeButton(android.R.string.cancel, new OnClickListener() {

    @Override
    public void onClick(DialogInterface dialog, int which) {
        dialog.dismiss();
    }
})
//create the dialog and show it.
.create().show();

}

जहाँ Layout.date_picker_view डेटपेकर के साथ एक साधारण लेआउट संसाधन है क्योंकि यह केवल तत्व है:

<!xml version="1.0" encoding="utf-8">
<DatePicker xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/date_picker"
android:layout_width="fill_parent"   
android:spinnersShown="true" 
android:calendarViewShown="false"
android:layout_height="fill_parent"/>

यदि आप रुचि रखते हैं तो यहां पूर्ण ट्यूटोरियल है।


इसने मेरे लिए अद्भुत काम किया। समझने में आसान, लागू करने में आसान! 4.4
इरेडमिस्टर

3

मेरा सरल उपाय। जब आप इसे फिर से प्राप्त करना चाहते हैं तो बस "रीसेटफ़ायर" चलाएं (फिर से डायलॉग खोलने पर)।

private class FixedDatePickerDialogListener implements DatePickerDialog.OnDateSetListener{
    private boolean fired;

    public void resetFired(){
        fired = false;
    }

    public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth) {
        if (fired) {
            Log.i("DatePicker", "Double fire occurred.");
            return;//ignore and return.
        } 
        //put your code here to handle onDateSet
        fired = true;//first time fired 
    }
}

@AbdelHady आपका कोड संपादन गलत था लेकिन मैंने इसे अधिक स्पष्ट बनाने के लिए इसे संपादित किया है। कोई ".JellyBeanOrAbove ()" विधि जोड़ने की आवश्यकता नहीं है, कोई फायदा नहीं है, बस अनावश्यक समानता जोड़ता है। कॉल को रीसेट करना सस्ता है।
डैनियल रयान

यहां कोड गलत है, क्योंकि onDateSetएक बार कॉल किया जाएगा जब संवाद किसी भी तरह से बंद किया जा रहा हो, और अगर इसका मतलब समय निर्धारित करना था तो इसे फिर से निकाल दिया जाएगा, इसलिए हमें दूसरी कॉल को पकड़ने की जरूरत है, पहले वाले को नहीं जैसा कि आपने किया था,
अब्देलहैडी

isJellyBeanOrAbove()जेलीबीन की तुलना में कम संस्करणों के बारे में , बग यह नहीं है कि यह पूरा प्रश्न सभी के बारे में है, और यह विचार करते हुए कि हम दूसरी कॉल को पकड़ना चाहते हैं, कोड तब तक नहीं चलेगा जब तक हम यह जाँच नहीं करते, मेरा विश्वास करो, मैंने अपना प्रयास किया है एमुलेटर और वास्तविक उपकरणों पर कोड (विभिन्न संस्करणों के साथ) कई बार और यह एक आकर्षण की तरह काम कर रहा है
अब्देलहेडी

मुझे यकीन नहीं है कि अगर एक मूल्य के नीचे है। मैंने इसे 2 साल पहले पोस्ट किया था, यह तब से हमारे व्यावसायिक अनुप्रयोग में है जब से बस ठीक काम कर रहा है। हमारे क्यूए टीमों या हमारे हजारों उपयोगकर्ताओं द्वारा कोई बग रिपोर्ट नहीं की गई। यह एक सरल उत्तर है जिसका लोग विस्तार कर सकते हैं। जब आप फिर से फायर करना चाहें, तो "resetFired" कॉल करें।
डैनियल रयान

हमारे app में "JellyBeanOrAbove "की जरूरत नहीं है। यदि आप सही क्षेत्रों में "रीसेटफ़ायर" कहते हैं तो ऐप सभी संस्करणों में ठीक काम करेगा।
डैनियल रयान

3

इसी तरह के मुद्दे पर अंकुर चौधरी के शानदार जवाब के अनुसार TimePickerDialog, अगर हम onDateSetदिए गए दृश्य के अंदर isShown()जांचते हैं या नहीं, तो यह न्यूनतम अंक के साथ पूरे मामले को हल करेगा, जिसमें पिकर को निकालने या कोड के चारों ओर जाने वाले कुछ घृणित झंडे की जांच करने की कोई आवश्यकता नहीं है। या यहाँ तक कि OS संस्करण के लिए जाँच, बस निम्नलिखित करें:

public void onDateSet(DatePicker view, int year, int month, int day) {
    if (view.isShown()) {
        // read the date here :)
    }
}

और निश्चित onTimeSetरूप से अंकुर के उत्तर के अनुसार ही किया जा सकता है


1
अन्य सभी से सर्वश्रेष्ठ उत्तर!
hiew1

2

जिस तरह से मैंने इस स्थिति को प्रबंधित किया वह एक ध्वज का उपयोग कर रहा था और ऑनस्केल और ऑनडिस्मस विधियों को ओवरराइड कर रहा था।

onCancel केवल तभी कॉल किया जाता है जब उपयोगकर्ता संवाद या बैक बटन के बाहर छूता है। onDismiss हमेशा कहा जाता है

ऑनसेल विधि में एक ध्वज सेट करने से उपयोगकर्ता के इरादे को ऑनडिस्मस विधि में फ़िल्टर करने में मदद मिल सकती है: कार्रवाई को रद्द करें या कार्रवाई करें। कुछ कोड के नीचे जो विचार दिखाता है।

public class DatePickerDialogFragment extends DialogFragment implements DatePickerDialog.OnDateSetListener {

    private boolean cancelDialog = false;
    private int year;
    private int month;
    private int day;

    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        DatePickerDialog dpd = new DatePickerDialog(getActivity(), this, year, month, day);
        return dpd;
    }

    public void setDatePickerDate(int year, int month, int day) {
        this.year = year;
        this.month = month;
        this.day = day;
    }

    @Override
    public void onCancel(DialogInterface dialog) {
        super.onCancel(dialog);
        cancelDialog = true;
    }

    @Override
    public void onDismiss(DialogInterface dialog) {
        super.onDismiss(dialog);
        if (!cancelDialog) {
          #put the code you want to execute if the user clicks the done button
        }
    }

    @Override
    public void onDateSet(DatePicker view, int year, int monthOfYear, int dayOfMonth) {
        setDatePickerDate(year, monthOfYear, dayOfMonth);
    }
}

1

बहुत ही सरल वर्कअराउंड है, यदि आपका एप्लिकेशन एक्शन बार का उपयोग नहीं करता है। वैसे, ध्यान दें कि कुछ ऐप काम करने के लिए इस कार्यक्षमता पर भरोसा करते हैं, क्योंकि डेट पिकर को रद्द करने का एक विशेष अर्थ होता है (जैसे यह तारीख फ़ील्ड को एक रिक्त स्ट्रिंग में साफ़ करता है, जो कुछ ऐप्स के लिए एक मान्य और सार्थक प्रकार का इनपुट है ) और ओके पर दो बार सेट होने से रोकने के लिए बूलियन झंडे का उपयोग करना इस मामले में आपकी मदद नहीं करेगा।

पुन। वास्तविक फिक्स, आपको नए बटन या अपना स्वयं का संवाद बनाने की आवश्यकता नहीं है। बिंदु दोनों के साथ संगत होना है, एंड्रॉइड के पुराने संस्करण, छोटी गाड़ी वाले (4. ) और भविष्य के किसी भी, हालांकि बाद के बारे में निश्चित होना असंभव है। ध्यान दें कि Android 2 में , onStop () android.app.Dialog के लिए कुछ भी नहीं करता है, और 4 में। * * यह mActionBar.setShowHideAnimationEnabled (झूठा) करता है जो केवल महत्वपूर्ण है यदि आपके ऐप में एक्शन बार है। डेटापिक्क डाइलोग में ऑनटॉप (), जो डायलॉग से विरासत में मिला है, केवल mDatePicker.clearFocus () (Android के सूत्रों के नवीनतम फिक्स 4.3 के रूप में) का योगदान देता है, जो आवश्यक नहीं लगता है।

इसलिए onStop () को एक ऐसी विधि के साथ बदलना जो कुछ भी नहीं करता है कई उदाहरणों में आपके ऐप को ठीक करना चाहिए और यह सुनिश्चित करना चाहिए कि यह भविष्य के लिए ऐसा ही रहेगा। इस प्रकार बस अपने खुद के साथ DatePickerDialog क्लास का विस्तार करें और ऑनटॉप () को एक डमी विधि से ओवरराइड करें। आपको अपनी आवश्यकताओं के अनुसार एक या दो निर्माणकर्ता भी उपलब्ध कराने होंगे। यह भी ध्यान दें कि किसी को सीधे इस गतिविधि के साथ कुछ करने का प्रयास करके इस फिक्स को अधिक करने की कोशिश करने की परीक्षा नहीं होनी चाहिए, क्योंकि यह आपकी संगतता को केवल Android के नवीनतम संस्करणों में सीमित कर देगा। यह भी ध्यान दें कि DatePicker के onStop () के लिए सुपर को कॉल करने में सक्षम होना अच्छा होगा क्योंकि बग केवल DatePickerDialog में onStop () में है, लेकिन DatePickerDialog के सुपर क्लास में नहीं। हालाँकि, इसके लिए आपको अपने कस्टम वर्ग से super.super.onStop () कॉल करना होगा, जो जावा आपको करने नहीं देगा, क्योंकि यह एनकैप्सुलेशन दर्शन के खिलाफ जाता है :) नीचे मेरा छोटा वर्ग है जो मैंने DatePickerDialog को विरूपित किया था। मुझे उम्मीद है कि यह टिप्पणी किसी के लिए उपयोगी होगी। वोजटेक जारोज़

public class myDatePickerDialog extends DatePickerDialog {

public myDatePickerDialog(Context context, OnDateSetListener callBack, int year, int monthOfYear, int dayOfMonth) {
    super(context, callBack, year, monthOfYear, dayOfMonth);
}

@Override
protected void onStop() {
    // Replacing tryNotifyDateSet() with nothing - this is a workaround for Android bug https://android-review.googlesource.com/#/c/61270/A

    // Would also like to clear focus, but we cannot get at the private members, so we do nothing.  It seems to do no harm...
    // mDatePicker.clearFocus();

    // Now we would like to call super on onStop(), but actually what we would mean is super.super, because
    // it is super.onStop() that we are trying NOT to run, because it is buggy.  However, doing such a thing
    // in Java is not allowed, as it goes against the philosophy of encapsulation (the Creators never thought
    // that we might have to patch parent classes from the bottom up :)
    // However, we do not lose much by doing nothing at all, because in Android 2.* onStop() in androd.app.Dialog //actually
    // does nothing and in 4.* it does:
    //      if (mActionBar != null) mActionBar.setShowHideAnimationEnabled(false); 
    // which is not essential for us here because we use no action bar... QED
    // So we do nothing and we intend to keep this workaround forever because of users with older devices, who might
    // run Android 4.1 - 4.3 for some time to come, even if the bug is fixed in later versions of Android.
}   

}


यहां तक ​​कि अगर एक ActionBar का उपयोग किया जाता है तो यह एक स्वीकार्य समाधान हो सकता है यदि आप ActionBar को कभी नहीं छिपाते / दिखाते हैं। चीजों को अतिरिक्त सुरक्षित बनाने के लिए आप कुछ भी नहीं करने के लिए onStart को ओवरराइड कर सकते हैं (तब एनीमेशन के लिए कॉल सुरक्षित रूप से अनहुक हो जाएगा)। और यहां तक ​​कि अगर आप इसे छिपाते हैं / दिखाते हैं, तो यह सिर्फ एनीमेशन है जो अक्षम है।
डैनियल

0


नीचे के कॉन्सेप्ट को आजमाएं।

DatePickerDialog picker = new DatePickerDialog(
        this,
        new OnDateSetListener() {
            @Override
            public void onDateSet(DatePicker v, int y, int m, int d) {
                Log.d("Picker", "Set!");
            }
        },
        2012, 6, 15);
picker.show();


onDateSet () विधि दो बार कॉल करती है (यदि आप दो बार एमुलेटर में जाँच कर रहे हैं। यह दो बार कॉल करता है। यदि आप वास्तविक डिवाइस का उपयोग कर रहे हैं तो यह सही समय पर कॉल करेगा। यदि आप एमुलेटर का उपयोग कर रहे हैं तो काउंटर का उपयोग करें। यदि आप वास्तविक डिवाइस में काम कर रहे हैं तो काउंटर वेरिएबल को नजरअंदाज करें। असली डिवाइस के लिए मेरे लिए काम करना।)
जब उपयोगकर्ता DatePickerDialog में बटन पर क्लिक करता है।
इसके लिए आपको एक काउंटर वैल्यू बनाए रखनी चाहिए और कुछ भी नहीं करना चाहिए जब मैथोड पहली बार कॉल करता है और जब ऑपरेशन दूसरी बार कॉल करता है तो ऑपरेशन करते हैं।
नीचे दिए गए कोडिंग स्निपेट देखें

   static int counter=0;       //Counter will be declared globally.

    DatePickerDialog picker = new DatePickerDialog(
            this,
            new OnDateSetListener() {
                @Override
                public void onDateSet(DatePicker v, int y, int m, int d) {

                   counter++;
                   if(counter==1) return;
                   counter=0;
                   //Do the operations here

                }
            },
            2012, 6, 15);
    picker.show();



डेटकिकर परिश्रम को रद्द करने के लिए, मेरे लिए काम कर रहा है। इसके लिए एमुलेटर नहीं

DialogInterface.OnClickListener dialogOnClickListener=new DialogInterface.OnClickListener()
        {

            @Override
            public void onClick(DialogInterface dialog, int which) {
                // TODO Auto-generated method stub

                if(which==Dialog.BUTTON_NEGATIVE)
                {
                    Log.i(tagName, "dialog negative button clicked");
                    dialog.dismiss();
                }

            }

        };

        mDatePickerDialog.setButton(Dialog.BUTTON_NEGATIVE, "Cancel", dialogOnClickListener);


एक असली डिवाइस के लिए मेरे लिए काम कर रहा है। लेकिन एमुलेटर के लिए सही ढंग से काम नहीं कर रहा है। मुझे लगता है कि यह एक Android एमुलेटर बग है।


0

एक सरल समाधान दूसरे रन को छोड़ने के लिए बूलियन का उपयोग करना होगा

boolean isShow = false; // define global variable


// when showing time picker
TimePickerDialog timeDlg = new TimePickerDialog( this, new OnTimeSetListener()
            {

                @Override
                public void onTimeSet( TimePicker view, int hourOfDay, int minute )
                {
                    if ( isShow )
                    {
                        isShow = false;
                        // your code
                    }

                }
            }, 8, 30, false );

timeDlg.setButton( TimePickerDialog.BUTTON_NEGATIVE, "Cancel", new DialogInterface.OnClickListener()
            {
                @Override
                public void onClick( DialogInterface dialog, int which )
                {
                    isShow = false;
                }
            } );
timeDlg.setButton( TimePickerDialog.BUTTON_POSITIVE, "Set", new DialogInterface.OnClickListener()
            {
                @Override
                public void onClick( DialogInterface dialog, int which )
                {
                    isShow = true;
                }
            } );

timeDlg.show();

एक बार फिर, जैसे मैंने आपके सामने दूसरे आदमी से कहा, इससे बग का समाधान नहीं होता । मैं अशिष्ट ध्वनि नहीं करना चाहता हूं, लेकिन आप लोग यहां पोस्ट करने से पहले अपने स्वयं के समाधान का परीक्षण कर रहे होंगे ... बस मेरी बात को साबित करने के लिए, अपने कोड का उपयोग करें और उपयोगकर्ता को संवाद रद्द करने के लिए वापस देखें और देखें कि मेरा क्या मतलब है। कारण बग के है onStopविधि जब यह नहीं है ... यह दो बार फायरिंग चाहिए बुला बग का एक परिणाम है।
दाविदस्कब

यदि मैं पर्याप्त स्पष्ट नहीं था, तो मुझे रहने दो: डायलॉग कॉल्स को रद्द करने के लिए वापस दबाएं onDateSet। इस प्रकार, टूट गया।
davidcsb

त्रुटि दिखाने के लिए धन्यवाद। मैं जवाब को संपादित करता हूं ताकि यह बैक बटन के साथ काम करे। आपका उत्तर काम कर रहा है लेकिन DatePicker dp = picker.getDatePicker (); TimePickers के साथ काम नहीं कर रहा है क्योंकि getTimePicker () विधि जोड़ा नहीं गया है। तो यह एक मान्य उत्तर होगा
chamikaw

हम दूसरे रन को कैसे छोड़ना चाहते हैं? !!, मैंने इसे कई बार आज़माया है, जब किसी भी तरह से डायलॉग को बंद करना onDateSetएक बार बुलाया जाएगा, लेकिन जब "किया गया" या "सेट" चुनना हो तो इसे दो बार कहा जाएगा। इसके लिए हमें केवल पहले एक को छोड़ने की आवश्यकता है, इसलिए यदि इसे दो बार कहा जाता है और केवल तभी हमारे पास सही तिथि है
अब्देलहेडी

0

आप नकारात्मक उपयोगकर्ता कार्यों का पता लगाने के लिए onCancel () का उपयोग कर सकते हैं और setOnDismissListener () का उपयोग कर सकते हैं। और DatePickerDialog.BUTTON_POSITIVE के साथ आप जानते हैं कि उपयोगकर्ता एक नई तारीख निर्धारित करना चाहता है।

 DatePickerDialog mDPD = new DatePickerDialog(
                      getActivity(), mOnDateSetListener, mYear, mMonth, mDay);
 mDPD.setOnCancelListener(new OnCancelListener() {
    @Override
    public void onCancel(DialogInterface dialog) {
        // do something onCancek
        setDate = false;
    }
 });

 mDPD.setOnDismissListener(new OnDismissListener() {
    @Override
    public void onDismiss(DialogInterface arg0) {
        // do something onDismiss
        setDate = false;
    }
});

mDPD.setButton(DatePickerDialog.BUTTON_POSITIVE, "Finish", new DatePickerDialog.OnClickListener() {

    @Override
    public void onClick(DialogInterface dialog, int which) {
        // user set new date
        setDate = true;
    }
});

फिर सेटडेट की जांच करें:

public void onDateSet(DatePicker view, int year, int month, int day) {
    if(setDate){
        //do something with new date
    }
}

0

यहाँ रद्द बटन पर DatePickerDialog के लिए मेरा वर्कअराउंड क्लास है और साथ ही इसे बैक बटन द्वारा छोड़ दिया गया है। DatePickerDialog की शैली में प्रतिलिपि बनाएं और उपयोग करें (क्योंकि श्रोता स्टेटफुल है, हमें उपयोग करते समय नए उदाहरण बनाने होंगे, अन्यथा इसे काम करने के लिए अधिक कोड की आवश्यकता है)

उपयोग:

new FixedDatePickerDialog(this,
            new FixedOnDateSetListener() {

                @Override
                public void onDateSet(DatePicker view, int year,
                        int monthOfYear, int dayOfMonth) {
                    if (isOkSelected()) {
                        // when DONE button is clicked
                    }
                }

            }, year, month, day).show();

वर्ग:

public class FixedDatePickerDialog extends DatePickerDialog {
private final FixedOnDateSetListener fixedCallback;
public FixedDatePickerDialog(Context context,
        FixedOnDateSetListener callBack, int year, int monthOfYear,
        int dayOfMonth) {
    super(context, callBack, year, monthOfYear, dayOfMonth);
    fixedCallback = callBack;
    this.setButton(DialogInterface.BUTTON_NEGATIVE,
            context.getString(R.string.cancel), this);
    this.setButton(DialogInterface.BUTTON_POSITIVE,
            context.getString(R.string.done), this);
}

@Override
public void onClick(DialogInterface dialog, int which) {
    if (which == BUTTON_POSITIVE) {
        fixedCallback.setOkSelected(true);
    } else {
        fixedCallback.setOkSelected(false);
    }
    super.onClick(dialog, which);
}

public abstract static class FixedOnDateSetListener implements
        OnDateSetListener {
    private boolean okSelected = false;

    @Override
    abstract public void onDateSet(DatePicker view, int year,
            int monthOfYear, int dayOfMonth);

    public void setOkSelected(boolean okSelected) {
        this.okSelected = okSelected;
    }

    public boolean isOkSelected() {
        return okSelected;
    }
}

}


0

मैं डेट पिकर, टाइम पिकर और नंबर पिकर का उपयोग कर रहा हूं। जब भी पिकर खारिज किया जाता है, उससे पहले जब भी उपयोगकर्ता एक नंबर का चयन करता है, तो नंबर लेने वाले ऑनवैल्यूड करते हैं।

public int interimValue;
public int finalValue;

public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
    this.interimValue = newVal;
}

public void onDismiss(DialogInterface dialog) {
    super.onDismiss(dialog);
    this.finalValue = this.interimValue;
}

मैंने अपने बटनों के लिए कस्टम onClickListeners सेट करने के लिए इसे बढ़ाया, एक तर्क के साथ कि कौन सा बटन क्लिक किया गया था। अब मैं जांच सकता हूं कि मैंने अपना अंतिम मूल्य निर्धारित करने से पहले कौन सा बटन टैप किया था:

public int interimValue;
public int finalValue;
public boolean saveButtonClicked;

public void setup() {
    picker.setButton(DialogInterface.BUTTON_POSITIVE, getString(R.string.BUTTON_SAVE), new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int which) {
            picker.onClick(dialog, which); // added for Android 5.0
            onButtonClicked(true);
        }
    });
    picker.setButton(DialogInterface.BUTTON_NEGATIVE, getString(R.string.BUTTON_CANCEL), new DialogInterface.OnClickListener() {
        public void onClick(DialogInterface dialog, int which) {
            picker.onClick(dialog, which); // added for Android 5.0
            onButtonClicked(false);
        }
    });
}

public void onValueChange(NumberPicker picker, int oldVal, int newVal) {
    this.interimValue = newVal;
}

public void onButtonClicked(boolean save) {
    this.saveButtonClicked = save;
}

public void onDismiss(DialogInterface dialog) {
    super.onDismiss(dialog);
    if (this.saveButtonClicked) {
        // save
        this.finalValue = this.interimValue;
    } else {
        // cancel
    }
}

और फिर मैंने तिथि और समय लेने वालों के साथ तारीख और समय के प्रकार के साथ-साथ नंबर लेने वालों के लिए अंतर प्रकार के साथ काम करने के लिए इसे बढ़ाया।

मैंने इसे पोस्ट किया क्योंकि मुझे लगा कि ऊपर दिए गए कुछ समाधानों की तुलना में यह सरल था, लेकिन अब जब मैंने सभी कोड शामिल कर लिए हैं, तो मुझे लगता है कि यह अधिक सरल नहीं है! लेकिन यह उस संरचना में अच्छी तरह से फिट है जो मेरे पास पहले से थी।

लॉलीपॉप के लिए अपडेट: जाहिरा तौर पर यह बग सभी एंड्रॉइड 4.1-4.4 डिवाइसों पर नहीं होता है, क्योंकि मुझे उन उपयोगकर्ताओं से कुछ रिपोर्टें मिली थीं जिनकी तिथि और समय लेने वाले ऑनडेटेट और ऑनटाइमसेट कॉलबैक नहीं कह रहे थे। और बग को आधिकारिक तौर पर एंड्रॉइड 5.0 में तय किया गया था। मेरे दृष्टिकोण ने केवल उन उपकरणों पर काम किया है जहां बग मौजूद है, क्योंकि मेरे कस्टम बटन ने डायल के ऑनक्लिक हैंडलर को कॉल नहीं किया था, जो कि एकमात्र जगह है जो ऑनडेटेट और ऑनटाइमसेट को बुलाया जाता है जब बग मौजूद नहीं होता है। मैंने डायलॉग के ऑनक्लिक पर कॉल करने के लिए ऊपर अपना कोड अपडेट किया है, इसलिए अब यह काम करता है कि बग मौजूद है या नहीं।


0

मुझे ऊपर डेविड सेसरिनो का जवाब पसंद आया, लेकिन कुछ ऐसा चाहता था जो टूटे हुए संवाद के बदले में आए और किसी भी ऐसे संवाद पर काम करेगा जो शायद रद्द हो / जिसमें गलत व्यवहार रद्द हो। यहाँ DatePickerDialog / TimePickerDialog के लिए व्युत्पन्न वर्ग हैं जो प्रतिस्थापन में गिरावट के रूप में काम करना चाहिए। ये कस्टम विचार नहीं हैं। यह सिस्टम संवाद का उपयोग करता है, लेकिन उम्मीद के मुताबिक काम करने के लिए रद्द / बैक बटन व्यवहार को बदल देता है।

यह एपीआई स्तर 3 और उच्चतर पर काम करना चाहिए। तो, मूल रूप से एंड्रॉइड का कोई भी संस्करण (मैंने इसे जेलीबीन और लॉलीपॉप पर विशेष रूप से परीक्षण किया)।

DatePickerDialog:

package snappy_company_name_here;

import android.content.Context;
import android.content.DialogInterface;
import android.widget.DatePicker;

/**
 * This is a modified version of DatePickerDialog that correctly handles cancellation behavior since it's broken on jellybean and
 * kitkat date pickers.
 *
 * Here is the bug: http://code.google.com/p/android/issues/detail?id=34833
 * Here is an SO post with a bunch of details: http://stackoverflow.com/questions/11444238/jelly-bean-datepickerdialog-is-there-a-way-to-cancel
 *
 * @author stuckj, created on 5/5/15.
 */
public class DatePickerDialog extends android.app.DatePickerDialog implements DialogInterface.OnClickListener
{
    final CallbackHelper callbackHelper;

    // NOTE: Must be static since we're using it in a super constructor call. Which is annoying, but necessary
    private static class CallbackHelper implements OnDateSetListener
    {
        private final OnDateSetListener callBack;
        private boolean dialogButtonPressHandled = false; // To prevent setting the date when the dialog is dismissed...

        // NOTE: Must be static since we're using it in a super constructor call. Which is annoying, but necessary
        public CallbackHelper(final OnDateSetListener callBack)
        {
            this.callBack = callBack;
        }

        @Override
        public void onDateSet(final DatePicker view, final int year, final int monthOfYear, final int dayOfMonth)
        {
            if (!dialogButtonPressHandled && (callBack != null))
            {
                callBack.onDateSet(view, year, monthOfYear, dayOfMonth);
            }
        }
    }

    /**
     * Sets the positive and negative buttons to use the dialog callbacks we define.
     */
    private void setButtons(final Context context)
    {
        setButton(DialogInterface.BUTTON_NEGATIVE, context.getString(android.R.string.cancel), this);
        setButton(DialogInterface.BUTTON_POSITIVE, context.getString(android.R.string.ok), this);
    }

    @Override
    public void onClick(final DialogInterface dialog, final int which)
    {
        // ONLY call the super method in the positive case...
        if (which == DialogInterface.BUTTON_POSITIVE)
        {
            super.onClick(dialog, which);
        }

        callbackHelper.dialogButtonPressHandled = true;
    }

    @Override
    public void onBackPressed()
    {
        getButton(DialogInterface.BUTTON_NEGATIVE).performClick();
    }

    // Need this so we can both pass callbackHelper to the super class and save it off as a variable.
    private DatePickerDialog(final Context context,
                             final OnDateSetListener callBack,
                             final int year,
                             final int monthOfYear,
                             final int dayOfMonth,
                             final CallbackHelper callbackHelper)
    {
        super(context, callbackHelper, year, monthOfYear, dayOfMonth);
        this.callbackHelper = callbackHelper;
        setButtons(context);
    }

    /**
     * @param context The context the dialog is to run in.
     * @param callBack How the parent is notified that the date is set.
     * @param year The initial year of the dialog.
     * @param monthOfYear The initial month of the dialog.
     * @param dayOfMonth The initial day of the dialog.
     */
    public DatePickerDialog(final Context context,
                            final OnDateSetListener callBack,
                            final int year,
                            final int monthOfYear,
                            final int dayOfMonth)
    {
        this(context, callBack, year, monthOfYear, dayOfMonth, new CallbackHelper(callBack));
    }

    // Need this so we can both pass callbackHelper to the super class and save it off as a variable.
    private DatePickerDialog(final Context context, final int theme, final OnDateSetListener listener, final int year,
                             final int monthOfYear, final int dayOfMonth, final CallbackHelper callbackHelper)
    {
        super(context, theme, callbackHelper, year, monthOfYear, dayOfMonth);
        this.callbackHelper = callbackHelper;
        setButtons(context);
    }

    /**
     * @param context The context the dialog is to run in.
     * @param theme the theme to apply to this dialog
     * @param listener How the parent is notified that the date is set.
     * @param year The initial year of the dialog.
     * @param monthOfYear The initial month of the dialog.
     * @param dayOfMonth The initial day of the dialog.
     */
    public DatePickerDialog(final Context context, final int theme, final OnDateSetListener listener, final int year,
                            final int monthOfYear, final int dayOfMonth)
    {
        this(context, theme, listener, year, monthOfYear, dayOfMonth, new CallbackHelper(listener));
    }
}

TimePickerDialog:

package snappy_company_name_here;

import android.content.Context;
import android.content.DialogInterface;
import android.widget.TimePicker;

/**
 * This is a modified version of TimePickerDialog that correctly handles cancellation behavior since it's broken on jellybean and
 * kitkat date pickers.
 *
 * Here is the bug: http://code.google.com/p/android/issues/detail?id=34833
 * Here is an SO post with a bunch of details: http://stackoverflow.com/questions/11444238/jelly-bean-datepickerdialog-is-there-a-way-to-cancel
 *
 * @author stuckj, created on 5/5/15.
 */
public class TimePickerDialog extends android.app.TimePickerDialog implements DialogInterface.OnClickListener
{
    final CallbackHelper callbackHelper;

    // NOTE: Must be static since we're using it in a super constructor call. Which is annoying, but necessary
    private static class CallbackHelper implements OnTimeSetListener
    {
        private final OnTimeSetListener callBack;
        private boolean dialogButtonPressHandled = false; // To prevent setting the date when the dialog is dismissed...

        // NOTE: Must be static since we're using it in a super constructor call. Which is annoying, but necessary
        public CallbackHelper(final OnTimeSetListener callBack)
        {
            this.callBack = callBack;
        }

        @Override
        public void onTimeSet(final TimePicker view, final int hourOfDay, final int minute)
        {
            if (!dialogButtonPressHandled && (callBack != null))
            {
                callBack.onTimeSet(view, hourOfDay, minute);
            }
        }
    }

    /**
     * Sets the positive and negative buttons to use the dialog callbacks we define.
     */
    private void setButtons(final Context context)
    {
        setButton(DialogInterface.BUTTON_NEGATIVE, context.getString(android.R.string.cancel), this);
        setButton(DialogInterface.BUTTON_POSITIVE, context.getString(android.R.string.ok), this);
    }

    @Override
    public void onClick(final DialogInterface dialog, final int which)
    {
        // ONLY call the super method in the positive case...
        if (which == DialogInterface.BUTTON_POSITIVE)
        {
            super.onClick(dialog, which);
        }

        callbackHelper.dialogButtonPressHandled = true;
    }

    @Override
    public void onBackPressed()
    {
        getButton(DialogInterface.BUTTON_NEGATIVE).performClick();
    }

    // Need this so we can both pass callbackHelper to the super class and save it off as a variable.
    private  TimePickerDialog(final Context context,
                              final OnTimeSetListener callBack,
                              final int hourOfDay, final int minute, final boolean is24HourView, final CallbackHelper callbackHelper)
    {
        super(context, callbackHelper, hourOfDay, minute, is24HourView);
        this.callbackHelper = callbackHelper;
        setButtons(context);
    }

    /**
     * @param context Parent.
     * @param callBack How parent is notified.
     * @param hourOfDay The initial hour.
     * @param minute The initial minute.
     * @param is24HourView Whether this is a 24 hour view, or AM/PM.
     */
    public TimePickerDialog(final Context context,
                            final OnTimeSetListener callBack,
                            final int hourOfDay, final int minute, final boolean is24HourView)
    {
        this(context, callBack, hourOfDay, minute, is24HourView, new CallbackHelper(callBack));
    }

    // Need this so we can both pass callbackHelper to the super class and save it off as a variable.
    private TimePickerDialog(final Context context, final int theme, final OnTimeSetListener callBack, final int hourOfDay,
                            final int minute, final boolean is24HourView, final CallbackHelper callbackHelper)
    {
        super(context, theme, callbackHelper, hourOfDay, minute, is24HourView);
        this.callbackHelper = callbackHelper;
        setButtons(context);
    }

    /**
     * @param context Parent.
     * @param theme the theme to apply to this dialog
     * @param callBack How parent is notified.
     * @param hourOfDay The initial hour.
     * @param minute The initial minute.
     * @param is24HourView Whether this is a 24 hour view, or AM/PM.
     */
    public TimePickerDialog(final Context context, final int theme, final OnTimeSetListener callBack, final int hourOfDay,
                            final int minute, final boolean is24HourView)
    {
        this(context, theme, callBack, hourOfDay, minute, is24HourView, new CallbackHelper(callBack));
    }
}

यह ऊपर दिए गए समुद्र के उत्तर के समान है।
stuckj

0

लैम्बडा एक्सप्रेशंस का उपयोग करके क्लियरबटन के साथ मेरा काम करने वाला संस्करण:

public class DatePickerFragment extends DialogFragment {
    private OnDateSelectListener dateSelectListener;
    private OnDateClearListener dateClearListener;

    public void setDateSelectListener(OnDateSelectListener dateSelectListener) {
        this.dateSelectListener = dateSelectListener;
    }

    public void setDateClearListener(OnDateClearListener dateClearListener) {
        this.dateClearListener = dateClearListener;
    }

    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        // Use the current date as the default date in the picker
        final Calendar c = Calendar.getInstance();
        int year = c.get(Calendar.YEAR);
        int month = c.get(Calendar.MONTH);
        int day = c.get(Calendar.DAY_OF_MONTH);

        // Create a new instance of DatePickerDialog and return it
        DatePickerDialog dialog = new DatePickerDialog(getActivity(), null, year, month, day);
        dialog.setCancelable(true);
        dialog.setCanceledOnTouchOutside(true);
        dialog.setTitle("Select Date");
        dialog.setButton(BUTTON_POSITIVE, ("Done"), (dialog1, which) -> {
            DatePicker dp = dialog.getDatePicker();
            dialog.dismiss();
            dateSelectListener.onDateSelect(dp.getYear(), dp.getMonth(), dp.getDayOfMonth());
        });
        dialog.setButton(BUTTON_NEUTRAL, ("Clear"), (dialog1, which) -> {
            dialog.dismiss();
            dateClearListener.onDateClear();
        });
        dialog.setButton(BUTTON_NEGATIVE, ("Cancel"), (dialog1, which) -> {
            if (which == DialogInterface.BUTTON_NEGATIVE) {
                dialog.cancel();
            }
        });
        dialog.getDatePicker().setCalendarViewShown(false);
        return dialog;
    }


    public interface OnDateClearListener {
        void onDateClear();
    }

    public interface OnDateSelectListener {
        void onDateSelect(int year, int monthOfYear, int dayOfMonth);
    }
}

0

TimePickerDialog के लिए समाधान निम्नानुसार हो सकता है:

TimePickerDialog createTimePickerDialog(Context context, int themeResId, TimePickerDialog.OnTimeSetListener orignalListener,
                                                         int hourOfDay, int minute, boolean is24HourView) {
        class KitKatTimeSetListener implements TimePickerDialog.OnTimeSetListener {
            private int hour;
            private int minute;

            private KitKatTimeSetListener() {
            }

            @Override
            public void onTimeSet(TimePicker view, int hourOfDay, int minute) {
                this.hour = hourOfDay;
                this.minute = minute;
            }

            private int getHour() { return hour; }
            private int getMinute() {return minute; }
        };

        KitKatTimeSetListener kitkatTimeSetListener = new KitKatTimeSetListener();
        TimePickerDialog timePickerDialog = new TimePickerDialog(context, themeResId, kitkatTimeSetListener, hourOfDay, minute, is24HourView);

        timePickerDialog.setButton(DialogInterface.BUTTON_POSITIVE, context.getString(android.R.string.ok), (dialog, which) -> {
            timePickerDialog.onClick(timePickerDialog, DialogInterface.BUTTON_POSITIVE);
            orignalListener.onTimeSet(new TimePicker(context), kitkatTimeSetListener.getHour(), kitkatTimeSetListener.getMinute());
            dialog.cancel();
        });
        timePickerDialog.setButton(DialogInterface.BUTTON_NEGATIVE, context.getString(android.R.string.cancel), (dialog, which) -> {
            dialog.cancel();
        });

        return timePickerDialog;
    }

मैं KitKatSetTimeListener को सभी घटनाओं को सौंपता हूं, और केवल BUTTON_POSITIVE पर क्लिक करने की स्थिति में मूल OnTimeSetListener में आग लगा देता है।


0

यहाँ पोस्ट किए गए कुछ सूक्तियों का परीक्षण करने के बाद, मुझे व्यक्तिगत रूप से लगता है कि यह समाधान सबसे सरल है। मैं DatePickerDialog कंस्ट्रक्टर में "श्रोता" के रूप में "अशक्त" पास करता हूं, और फिर जब मैं "ओके" बटन पर क्लिक करता हूं तो मैं अपने ऑनडेसट्रीसैटलीस्टनर को कॉल करता हूं:

datePickerDialog = new DatePickerDialog(getContext(), null, dateSearch.get(Calendar.YEAR), dateSearch.get(Calendar.MONTH), dateSearch.get(Calendar.DAY_OF_MONTH));
    datePickerDialog.setCancelable(false);
    datePickerDialog.setButton(DialogInterface.BUTTON_POSITIVE, getString(R.string.dialog_ok), new DialogInterface.OnClickListener() {
        @Override
        public void onClick(DialogInterface dialog, int which) {
            Log.d("Debug", "Correct");
            onDateSearchSetListener.onDateSet(datePickerDialog.getDatePicker(), datePickerDialog.getDatePicker().getYear(), datePickerDialog.getDatePicker().getMonth(), datePickerDialog.getDatePicker().getDayOfMonth());
        }
    });
    datePickerDialog.setButton(DialogInterface.BUTTON_NEGATIVE, getString(R.string.dialog_cancel), new DialogInterface.OnClickListener() {
        @Override
        public void onClick(DialogInterface dialog, int which) {
            Log.d("Debug", "Cancel");
            dialog.dismiss();
        }
    });

-1

मैं जानता हूं कि यह पद यहां लगभग एक साल से है लेकिन मुझे लगा कि मुझे अपने निष्कर्षों को पोस्ट करना चाहिए। आप अभी भी श्रोता (इसे नीरस करने के लिए स्थापित करने के बजाय) रख सकते हैं और अभी भी उम्मीद के मुताबिक यह काम कर सकते हैं। मुख्य रूप से "ओके" या (और) "रद्द" बटन सेट करना है। मैंने इसका परीक्षण किया और यह मेरे लिए कृतज्ञता से काम करता है। सुनने वाले को दो बार निकाल दिया जाता है।

इस उदाहरण को देखें,

private void setTime(){
final Calendar c = Calendar.getInstance();
int hour = c.get(Calendar.HOUR_OF_DAY);
int minute = c.get(Calendar.MINUTE);

final TimePickerDialog timepicker = new TimePickerDialog(this.getActivity(),
        timePickerListener,
        hour, 
        minute, 
        DateFormat.is24HourFormat(getActivity()));

timepicker.setButton(DialogInterface.BUTTON_POSITIVE, "Print", new    
    android.content.DialogInterface.OnClickListener(){
        @Override
        public void onClick(DialogInterface dialog,int which) {
            print = true;
            timepicker.dismiss();           
        }
});

timepicker.setButton(DialogInterface.BUTTON_NEGATIVE, "Cancel", new 
    android.content.DialogInterface.OnClickListener(){
        @Override
        public void onClick(DialogInterface dialog,int which){
            print = false;
            timepicker.dismiss();       
        }
});

timepicker.setCancelable(false);
timepicker.show();
}

सादा और सरल, यह आपके बताए तरीके से काम नहीं करता है। timePickerListenerआप अभी भी अपने संवाद में क्या करते हैं, इसकी परवाह किए बिना कहा जाता है। मुझे यह जानने के लिए परीक्षण करने की भी आवश्यकता नहीं है कि, आपको केवल स्रोतों को देखने की आवश्यकता है : यदि आप इसे सेट नहीं करते हैं null, tryNotifyTimeSet()तो श्रोता onTimeSet()को इसके onClick()और दोनों में कॉल करेंगे onStop()
davidcsb

दूसरे शब्दों में, मूल वर्ग को अपने श्रोता (यानी, इसे कंस्ट्रक्टर में पास करना) के संदर्भ में रखने की अनुमति दें। आप बटनों के साथ कैसे व्यवहार करते हैं यह अप्रासंगिक है।
davidcsb

हाय डेविड, आपने यह परीक्षण नहीं किया और आपने इसे काम नहीं माना। यह कोड का सटीक टुकड़ा है जो मैं वर्तमान में अपने ऐप में उपयोग कर रहा हूं और यह एक विजेता की तरह काम करता है।
मगरमच्छ

1) मैंने कभी नहीं कहा कि यह काम नहीं किया। मैंने कहा कि यह आपके द्वारा कहे गए तरीके से काम नहीं करता है। कृपया फिर से पढ़ें। 2) यदि आप LSP और SRP उल्लंघन किया है, करने के लिए मानव घंटे बर्बाद कर बेकार में बदल सभी अपने पूरे ग्राहक तर्क यह है कि किसी भी परिवर्तन के साथ शुरू की जरूरत नहीं थी। 3) आपका उत्तर प्रश्न को संबोधित करता है, हाँ, अन्यथा मैं इसे "एक उत्तर नहीं" के रूप में हटाने के लिए ध्वजांकित करूंगा, लेकिन 4) आपका उत्तर अभी भी है (इसके बारे में खेद है) बहुत अक्षम है और मौलिक डिजाइन मुद्दे को नहीं सुनता है (श्रोता कहा जाता है) ), इसलिए केवल डाउनवोट। 6) आपने वापस अक्षम कर दिया और बूलियन को मजबूर करने के लिए इसे एक खिड़की बना दिया। तो, वहाँ 6 गंभीर मुद्दे!
दाविदस्कब
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.