Mutually restricting begin and end date-times using p:calendar (no validation)

后端 未结 1 415
梦毁少年i
梦毁少年i 2021-02-01 01:51

We have a requirement to present two p:calendar components to the user, representing a start and end date each. Both datetimes have dates, hours and minutes. PrimeFaces has perf

1条回答
  •  心在旅途
    2021-02-01 02:39

    Preface:

    I don't work with JSF, but there are a couple of things that might steer you back to where you want to be:

    a) when working with just the date portion of a dateTime in a standard calendar, consider using:

    someCalendar.set(Calendar.MILLISECOND, 0)
    

    b) consider using joda-time, as it seems to be frequently recommended (here, here , and many other places) over the standard library for correctness, performance, and ease of use in many situations.

    c) Make sure your bean scope is surviving each ajax call (not redirecting, only sending standard post-backs, etc) and each event handler is getting the faces context (eg. FacesContext facesContext = FacesContext.getCurrentInstance();)

    d) mindate and the like probably don't work like you expect , and I don't expect that automatic behavior can be quite so easily interjected.

    When those options aren't available, and you have to do it all yourself with what you have:

    Philisophical / UX: The first thing I would do is remove the expectation of arrangement or perspective from the pair of dates. Don't treat the pair as a vector that exposes or expects a direction on the timeline.

    • In other words, is a start or from date always less than or earlier than an end or to date? No, as can be seen for a query of historical data, or for applying corrections to events that have either yet to happen or have already happened?

      This connotation can easily confuse a user as to whether they are going 'back to' or 'forward from' (and can easily confuse yourself). Instead I would treat a pair of dates with a time-period between them as just and simply that a pair of dates or a range or a period that declares an interval, and infer their relative position on the timeline depending on the any consequently chosen values. In this way you can honor the respective and inherent requirements that the dates never be equal, and the left is always to the left, the right always to the right.

    We can't infer what 'start' or 'from' means, but we can infer some meaning and relative relationship: a right, a left, and a between on a chronological timeline. Note: Always resolve dates to UTC before doing any calculation or comparison.

    long oneDateValue = oneDate.toUtc().toMilliseconds();
    long anotherDateValue = anotherDate.toUtc().toMilliseconds();
    
    long right = max (oneDateValue, anotherDateValue);
    long left = min (oneDateValue, anotherDateValue);
    

    Evaluating Precision: The second thing I would look at when working with a range of dates in any language is similar to how you might deal with floating point numbers. For comparisons, do not compare for equality, but instead compare the delta to an "acceptable error level". In other words, the application is really only concerned with a certain degree of precision, so make sure that only that precision is captured and considered:

    const int dateTimeResolutionInMs = 86400000; // milliseconds per day
    
    public bool areEssentiallySame(long left, long right) {
    
       // the difference between right and left is less than our precision 
       // requires, thus dates are effectively the same
       return (right - left < dateTimeResolutionInMs);
    }
    

    Coercing Precision: Thirdly, how do we resolve the difference in values even if within the range of the resolution? (Out application was given more precision than it can handle or expect or needs).

    long diff = value % dateTimeResolutionInMs;

    1. Truncate: return value - diff;

    2. Nearest (w/bias): return value + (diff < dateTimeResolutionInMs/ 2) ? -1 * diff : dateTimeResolutionInMs - diff;

    3. Others: there are lots of other strategies for either shrinking or expanding a value to a preferred resolution or precision

    Addendum: As far as getting post-backs/Ajax calls to return a view with the values you expect for the events fired by a calendar element, you may want to separate that concern off to a new question if the note in the preface didn't get you anywhere, and you know for certain your bean is properly registered and recognized. You may have some browser/browser-version specific issues that contribute to the undesired behavior, and like anything else, there are issues, both known and unknown.

    0 讨论(0)
提交回复
热议问题