Merge pull request #3025 from tractorcow/pulls/3.2-date-relevance

API DateTime.Ago better infers significance of date units.
This commit is contained in:
Sean Harvey 2014-09-25 18:28:30 +12:00
commit 04e26d4a36
3 changed files with 135 additions and 67 deletions

View File

@ -221,9 +221,10 @@ class Date extends DBField {
* Returns the number of seconds/minutes/hours/days or months since the timestamp. * Returns the number of seconds/minutes/hours/days or months since the timestamp.
* *
* @param boolean $includeSeconds Show seconds, or just round to "less than a minute". * @param boolean $includeSeconds Show seconds, or just round to "less than a minute".
* @param int $significance Minimum significant value of X for "X units ago" to display
* @return String * @return String
*/ */
public function Ago($includeSeconds = true) { public function Ago($includeSeconds = true, $significance = 2) {
if($this->value) { if($this->value) {
$time = SS_Datetime::now()->Format('U'); $time = SS_Datetime::now()->Format('U');
if(strtotime($this->value) == $time || $time > strtotime($this->value)) { if(strtotime($this->value) == $time || $time > strtotime($this->value)) {
@ -231,14 +232,14 @@ class Date extends DBField {
'Date.TIMEDIFFAGO', 'Date.TIMEDIFFAGO',
"{difference} ago", "{difference} ago",
'Natural language time difference, e.g. 2 hours ago', 'Natural language time difference, e.g. 2 hours ago',
array('difference' => $this->TimeDiff($includeSeconds)) array('difference' => $this->TimeDiff($includeSeconds, $significance))
); );
} else { } else {
return _t( return _t(
'Date.TIMEDIFFIN', 'Date.TIMEDIFFIN',
"in {difference}", "in {difference}",
'Natural language time difference, e.g. in 2 hours', 'Natural language time difference, e.g. in 2 hours',
array('difference' => $this->TimeDiff($includeSeconds)) array('difference' => $this->TimeDiff($includeSeconds, $significance))
); );
} }
} }
@ -246,79 +247,68 @@ class Date extends DBField {
/** /**
* @param boolean $includeSeconds Show seconds, or just round to "less than a minute". * @param boolean $includeSeconds Show seconds, or just round to "less than a minute".
* @return String * @param int $significance Minimum significant value of X for "X units ago" to display
* @return string
*/ */
public function TimeDiff($includeSeconds = true) { public function TimeDiff($includeSeconds = true, $significance = 2) {
if(!$this->value) return false;
$time = SS_Datetime::now()->Format('U');
$ago = abs($time - strtotime($this->value));
if($ago < 60 && !$includeSeconds) {
return _t('Date.LessThanMinuteAgo', 'less than a minute');
} elseif($ago < $significance * 60 && $includeSeconds) {
return $this->TimeDiffIn('seconds');
} elseif($ago < $significance * 3600) {
return $this->TimeDiffIn('minutes');
} elseif($ago < $significance * 86400) {
return $this->TimeDiffIn('hours');
} elseif($ago < $significance * 86400 * 30) {
return $this->TimeDiffIn('days');
} elseif($ago < $significance * 86400 * 365) {
return $this->TimeDiffIn('months');
} else {
return $this->TimeDiffIn('years');
}
}
/**
* Gets the time difference, but always returns it in a certain format
*
* @param string $format The format, could be one of these:
* 'seconds', 'minutes', 'hours', 'days', 'months', 'years'.
* @return string The resulting formatted period
*/
public function TimeDiffIn($format) {
if(!$this->value) return false; if(!$this->value) return false;
$time = SS_Datetime::now()->Format('U'); $time = SS_Datetime::now()->Format('U');
$ago = abs($time - strtotime($this->value)); $ago = abs($time - strtotime($this->value));
if($ago < 60 && $includeSeconds) {
$span = $ago;
$result = ($span != 1) ? "{$span} "._t("Date.SECS", "secs") : "{$span} "._t("Date.SEC", "sec");
} elseif($ago < 60) {
$result = _t('Date.LessThanMinuteAgo', 'less than a minute');
} elseif($ago < 3600) {
$span = round($ago/60);
$result = ($span != 1) ? "{$span} "._t("Date.MINS", "mins") : "{$span} "._t("Date.MIN", "min");
} elseif($ago < 86400) {
$span = round($ago/3600);
$result = ($span != 1) ? "{$span} "._t("Date.HOURS", "hours") : "{$span} "._t("Date.HOUR", "hour");
} elseif($ago < 86400*30) {
$span = round($ago/86400);
$result = ($span != 1) ? "{$span} "._t("Date.DAYS", "days") : "{$span} "._t("Date.DAY", "day");
} elseif($ago < 86400*365) {
$span = round($ago/86400/30);
$result = ($span != 1) ? "{$span} "._t("Date.MONTHS", "months") : "{$span} "._t("Date.MONTH", "month");
} elseif($ago > 86400*365) {
$span = round($ago/86400/365);
$result = ($span != 1) ? "{$span} "._t("Date.YEARS", "years") : "{$span} "._t("Date.YEAR", "year");
}
// Replace duplicate spaces, backwards compat with existing translations
$result = preg_replace('/\s+/', ' ', $result);
return $result;
}
/**
* Gets the time difference, but always returns it in a certain format
* @param string $format The format, could be one of these:
* 'seconds', 'minutes', 'hours', 'days', 'months', 'years'.
*
* @return string
*/
public function TimeDiffIn($format) {
if($this->value) {
$ago = abs(time() - strtotime($this->value));
switch($format) { switch($format) {
case "seconds": case "seconds":
$span = $ago; $span = $ago;
return ($span != 1) ? "{$span} seconds" : "{$span} second"; return ($span != 1) ? "{$span} "._t("Date.SECS", "secs") : "{$span} "._t("Date.SEC", "sec");
break;
case "minutes": case "minutes":
$span = round($ago/60); $span = round($ago/60);
return ($span != 1) ? "{$span} minutes" : "{$span} minute"; return ($span != 1) ? "{$span} "._t("Date.MINS", "mins") : "{$span} "._t("Date.MIN", "min");
break;
case "hours": case "hours":
$span = round($ago/3600); $span = round($ago/3600);
return ($span != 1) ? "{$span} hours" : "{$span} hour"; return ($span != 1) ? "{$span} "._t("Date.HOURS", "hours") : "{$span} "._t("Date.HOUR", "hour");
break;
case "days": case "days":
$span = round($ago/86400); $span = round($ago/86400);
return ($span != 1) ? "{$span} days" : "{$span} day"; return ($span != 1) ? "{$span} "._t("Date.DAYS", "days") : "{$span} "._t("Date.DAY", "day");
break;
case "months": case "months":
$span = round($ago/86400/30); $span = round($ago/86400/30);
return ($span != 1) ? "{$span} months" : "{$span} month"; return ($span != 1) ? "{$span} "._t("Date.MONTHS", "months") : "{$span} "._t("Date.MONTH", "month");
break;
case "years": case "years":
$span = round($ago/86400/365); $span = round($ago/86400/365);
return ($span != 1) ? "{$span} years" : "{$span} year"; return ($span != 1) ? "{$span} "._t("Date.YEARS", "years") : "{$span} "._t("Date.YEAR", "year");
break;
}
} }
} }

View File

@ -163,6 +163,30 @@ class DateTest extends SapphireTest {
public function testAgoInPast() { public function testAgoInPast() {
SS_Datetime::set_mock_now('2000-12-31 12:00:00'); SS_Datetime::set_mock_now('2000-12-31 12:00:00');
$this->assertEquals(
'1 month ago',
DBField::create_field('Date', '2000-11-26')->Ago(true, 1),
'Past match on days, less than two months, lowest significance'
);
$this->assertEquals(
'50 days ago', // Rounded from 49.5 days up
DBField::create_field('Date', '2000-11-12')->Ago(),
'Past match on days, less than two months'
);
$this->assertEquals(
'2 months ago',
DBField::create_field('Date', '2000-10-27')->Ago(),
'Past match on days, over two months'
);
$this->assertEquals(
'66 days ago', // rounded from 65.5 days up
DBField::create_field('Date', '2000-10-27')->Ago(true, 3),
'Past match on days, over two months, significance of 3'
);
$this->assertEquals( $this->assertEquals(
'10 years ago', '10 years ago',
DBField::create_field('Date', '1990-12-31')->Ago(), DBField::create_field('Date', '1990-12-31')->Ago(),
@ -177,6 +201,12 @@ class DateTest extends SapphireTest {
$this->assertEquals( $this->assertEquals(
'1 year ago', '1 year ago',
DBField::create_field('Date', '1999-12-30')->Ago(true, 1),
'Approximate past match in singular, lowest significance'
);
$this->assertEquals(
'12 months ago',
DBField::create_field('Date', '1999-12-30')->Ago(), DBField::create_field('Date', '1999-12-30')->Ago(),
'Approximate past match in singular' 'Approximate past match in singular'
); );
@ -195,6 +225,12 @@ class DateTest extends SapphireTest {
$this->assertEquals( $this->assertEquals(
'in 1 day', 'in 1 day',
DBField::create_field('Date', '2001-01-01')->Ago(true, 1),
'Approximate past match on minutes'
);
$this->assertEquals(
'in 24 hours',
DBField::create_field('Date', '2001-01-01')->Ago(), DBField::create_field('Date', '2001-01-01')->Ago(),
'Approximate past match on minutes' 'Approximate past match on minutes'
); );

View File

@ -106,6 +106,12 @@ class SS_DatetimeTest extends SapphireTest {
$this->assertEquals( $this->assertEquals(
'1 year ago', '1 year ago',
DBField::create_field('SS_Datetime', '1999-12-30 12:00:12')->Ago(true, 1),
'Approximate past match in singular, significance=1'
);
$this->assertEquals(
'12 months ago',
DBField::create_field('SS_Datetime', '1999-12-30 12:00:12')->Ago(), DBField::create_field('SS_Datetime', '1999-12-30 12:00:12')->Ago(),
'Approximate past match in singular' 'Approximate past match in singular'
); );
@ -128,6 +134,36 @@ class SS_DatetimeTest extends SapphireTest {
'Approximate past match on seconds with $includeSeconds=false' 'Approximate past match on seconds with $includeSeconds=false'
); );
$this->assertEquals(
'1 min ago',
DBField::create_field('SS_Datetime', '2000-12-31 11:58:50')->Ago(false),
'Test between 1 and 2 minutes with includeSeconds=false'
);
$this->assertEquals(
'70 secs ago',
DBField::create_field('SS_Datetime', '2000-12-31 11:58:50')->Ago(true),
'Test between 1 and 2 minutes with includeSeconds=true'
);
$this->assertEquals(
'4 mins ago',
DBField::create_field('SS_Datetime', '2000-12-31 11:55:50')->Ago(),
'Past match on minutes'
);
$this->assertEquals(
'1 hour ago',
DBField::create_field('SS_Datetime', '2000-12-31 10:50:58')->Ago(true, 1),
'Past match on hours, significance=1'
);
$this->assertEquals(
'3 hours ago',
DBField::create_field('SS_Datetime', '2000-12-31 08:50:58')->Ago(),
'Past match on hours'
);
SS_Datetime::clear_mock_now(); SS_Datetime::clear_mock_now();
} }
@ -142,6 +178,12 @@ class SS_DatetimeTest extends SapphireTest {
$this->assertEquals( $this->assertEquals(
'in 1 hour', 'in 1 hour',
DBField::create_field('SS_Datetime', '2000-12-31 1:01:05')->Ago(true, 1),
'Approximate past match on minutes, significance=1'
);
$this->assertEquals(
'in 61 mins',
DBField::create_field('SS_Datetime', '2000-12-31 1:01:05')->Ago(), DBField::create_field('SS_Datetime', '2000-12-31 1:01:05')->Ago(),
'Approximate past match on minutes' 'Approximate past match on minutes'
); );