// bbldc_daterangedaycount.h                                          -*-C++-*-
#ifndef INCLUDED_BBLDC_DATERANGEDAYCOUNT
#define INCLUDED_BBLDC_DATERANGEDAYCOUNT

#include <bsls_ident.h>
BSLS_IDENT("$Id: $")

//@PURPOSE: Provide a protocol for date-range limited day-count calculations.
//
//@CLASSES:
//  bbldc::DateRangeDayCount: protocol for date-range limited day-counts
//
//@SEE_ALSO: bbldc_basicdaterangedaycountadapter,
//           bbldc_perioddaterangedaycountadapter
//
//@DESCRIPTION: This component provides a protocol, 'bbldc::DateRangeDayCount',
// for implementing an arbitrary day-count convention.  Concrete
// implementations of this protocol may implement, say, the BUS-252 day-count
// convention, or a custom day-count convention appropriate for some niche
// market.
//
// Several of the components in 'bbldc' provide individual day-count convention
// support through interfaces that are functionally identical to the abstract
// interface provided by this component, except that they do not inherit from
// 'bbldc::DateRangeDayCount'.  In conjunction with the adapter components
// (e.g., 'bbldc_basicdaterangedaycountadapter'), 'bbldc::DateRangeDayCount' is
// intended to allow run-time binding of these and other similar day-count
// implementations.
//
// This protocol requires two methods, 'firstDate' and 'lastDate', that define
// a date range for which calculations are valid, to reflect the valid range
// of, say, a calendar required for the computations.
//
///Usage
///-----
// This section illustrates intended use of this component.
//
///Example 1: Definition and Use of a Concrete Day-Count Convention
/// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// This example shows the definition and use of a simple concrete day-count
// convention.  This functionality suffices to demonstrate the requisite steps
// for having a working day-count convention:
//: * Define a concrete day-count type derived from 'bbldc::DateRangeDayCount'.
//:
//: * Implement the four pure virtual methods.
//:
//: * Instantiate and use an object of the concrete type.
//
// First, define the (derived) 'my_DayCountConvention' class and implement its
// constructor inline (for convenience, directly within the derived-class
// definition):
//..
//  // my_daycountconvention.h
//
//  class my_DayCountConvention : public bbldc::DateRangeDayCount {
//      bdlt::Date d_firstDate;
//      bdlt::Date d_lastDate;
//    public:
//      my_DayCountConvention()
//                         : d_firstDate(1, 1, 1), d_lastDate(9999, 12, 31) { }
//      virtual ~my_DayCountConvention();
//      virtual int daysDiff(const bdlt::Date& beginDate,
//                           const bdlt::Date& endDate) const;
//          // Return the (signed) number of days between the specified ...
//      virtual const bdlt::Date& firstDate() const;
//          // Return a reference providing non-modifiable access to the ...
//      virtual const bdlt::Date& lastDate() const;
//          // Return a reference providing non-modifiable access to the ...
//      virtual double yearsDiff(const bdlt::Date& beginDate,
//                               const bdlt::Date& endDate) const;
//          // Return the (signed fractional) number of years between the ...
//  };
//..
// Then, implement the destructor.  Note, however, that we always implement a
// virtual destructor (non-inline) in the .cpp file (to indicate the *unique*
// location of the class's virtual table):
//..
//  // my_daycountconvention.cpp
//
//  // ...
//
//  my_DayCountConvention::~my_DayCountConvention() { }
//..
// Next, we implement the (virtual) 'daysDiff', 'firstDate', 'lastDate', and
// 'yearsDiff' methods, which incorporate the "policy" of what it means for
// this day-count convention to calculate day count, year fraction, and the
// valid range of the convention instance:
//..
//  int my_DayCountConvention::daysDiff(const bdlt::Date& beginDate,
//                                      const bdlt::Date& endDate) const
//  {
//      return endDate - beginDate;
//  }
//
//  const bdlt::Date& my_DayCountConvention::firstDate() const
//  {
//      return d_firstDate;
//  }
//
//  const bdlt::Date& my_DayCountConvention::lastDate() const
//  {
//      return d_lastDate;
//  }
//
//  double my_DayCountConvention::yearsDiff(const bdlt::Date& beginDate,
//                                          const bdlt::Date& endDate) const
//  {
//      return static_cast<double>(endDate - beginDate) / 365.0;
//  }
//..
// Then, create two 'bdlt::Date' variables, 'd1' and 'd2', to use with the
// 'my_DayCountConvention' object and its day-count convention methods:
//..
//  const bdlt::Date d1(2003, 10, 19);
//  const bdlt::Date d2(2003, 12, 31);
//..
// Next, we obtain a 'bbldc::DateRangeDayCount' reference from an instantiated
// 'my_DayCountConvention':
//..
//  my_DayCountConvention           myDcc;
//  const bbldc::DateRangeDayCount& dcc = myDcc;
//..
// Now, we compute the day count between the two dates:
//..
//  const int daysDiff = dcc.daysDiff(d1, d2);
//  assert(73 == daysDiff);
//..
// Finally, we compute the year fraction between the two dates:
//..
//  const double yearsDiff = dcc.yearsDiff(d1, d2);
//  // Need fuzzy comparison since 'yearsDiff' is a 'double'.
//  assert(0.1999 < yearsDiff && 0.2001 > yearsDiff);
//..

#include <bblscm_version.h>

#include <bdlt_date.h>

namespace BloombergLP {
namespace bbldc {

                         // =======================
                         // class DateRangeDayCount
                         // =======================

class DateRangeDayCount {
    // This 'class' provides a protocol for determining values based on dates
    // according to derived implementations of specific day-count conventions.
    // The methods 'firstDate' and 'lastDate' define a date range for which
    // calculations are valid, to reflect the valid range of, say, a calendar
    // required for the computations.

  public:
    // CREATORS
    virtual ~DateRangeDayCount();
        // Destroy this object.

    // ACCESSORS
    virtual int daysDiff(const bdlt::Date& beginDate,
                         const bdlt::Date& endDate) const = 0;
        // Return the (signed) number of days between the specified 'beginDate'
        // and 'endDate'.  If 'beginDate <= endDate', then the result is
        // non-negative.  The behavior is undefined unless
        // 'firstDate() <= beginDate <= lastDate()' and
        // 'firstDate() <= endDate <= lastDate()'.  Note that reversing the
        // order of 'beginDate' and 'endDate' negates the result.

    virtual const bdlt::Date& firstDate() const = 0;
        // Return a reference providing non-modifiable access to the earliest
        // date in the valid range of this day-count convention.

    virtual const bdlt::Date& lastDate() const = 0;
        // Return a reference providing non-modifiable access to the latest
        // date in the valid range of this day-count convention.

    virtual double yearsDiff(const bdlt::Date& beginDate,
                             const bdlt::Date& endDate) const = 0;
        // Return the (signed fractional) number of years between the specified
        // 'beginDate' and 'endDate'.  If 'beginDate <= endDate', then the
        // result is non-negative.  The behavior is undefined unless
        // 'firstDate() <= beginDate <= lastDate()' and
        // 'firstDate() <= endDate <= lastDate()'.  Note that reversing the
        // order of 'beginDate' and 'endDate' negates the result; specifically,
        // '|yearsDiff(b, e) + yearsDiff(e, b)| <= 1.0e-15' for all dates 'b'
        // and 'e'.
};

}  // close package namespace
}  // close enterprise namespace

#endif

// ----------------------------------------------------------------------------
// Copyright 2015 Bloomberg Finance L.P.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// ----------------------------- END-OF-FILE ----------------------------------