Merge remote-tracking branch 'refs/remotes/tsyesika/394-fuzzy-timestamp'

This commit is contained in:
Christopher Allan Webber 2013-04-13 11:42:34 -05:00
commit 6432755db3
7 changed files with 184 additions and 4 deletions

View File

@ -104,7 +104,7 @@
<td>{{ media_entry.id }}</td>
<td>{{ media_entry.get_uploader.username }}</td>
<td><a href="{{ media_entry.url_for_self(request.urlgen) }}">{{ media_entry.title }}</a></td>
<td>{{ media_entry.created.strftime("%F %R") }}</td>
<td><span title='{{ media_entry.created.strftime("%F %R") }}'>{{ timesince(media_entry.created) }}</span></td>
</tr>
{% endfor %}
</table>

View File

@ -125,7 +125,9 @@
comment=comment.id,
user=media.get_uploader.username,
media=media.slug_or_id) }}#comment">
{{- comment.created.strftime("%I:%M%p %Y-%m-%d") -}}
<span title='{{- comment.created.strftime("%I:%M%p %Y-%m-%d") -}}'>
{{ timesince(comment.created) }}
</span>
</a>:
</div>
<div class="comment_content">
@ -141,9 +143,9 @@
{% endif %}
</div>
<div class="media_sidebar">
{% trans date=media.created.strftime("%Y-%m-%d") -%}
{% trans date=media.created.strftime("%Y-%m-%d"), formatted_time=timesince(media.created) -%}
<h3>Added on</h3>
<p>{{ date }}</p>
<p><span title="{{ date }}">{{ formatted_time }}</span></p>
{%- endtrans %}
{% if media.tags %}
{% include "mediagoblin/utils/tags.html" %}

View File

@ -0,0 +1,57 @@
# GNU MediaGoblin -- federated, autonomous media hosting
# Copyright (C) 2011, 2012 MediaGoblin contributors. See AUTHORS.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
from datetime import datetime, timedelta
from mediagoblin.tools.timesince import is_aware, timesince
def test_timesince(test_app):
test_time = datetime.now()
# it should ignore second and microseconds
assert timesince(test_time, test_time + timedelta(microseconds=1)) == "0 minutes"
assert timesince(test_time, test_time + timedelta(seconds=1)) == "0 minutes"
# test minutes, hours, days, weeks, months and years (singular and plural)
assert timesince(test_time, test_time + timedelta(minutes=1)) == "1 minute"
assert timesince(test_time, test_time + timedelta(minutes=2)) == "2 minutes"
assert timesince(test_time, test_time + timedelta(hours=1)) == "1 hour"
assert timesince(test_time, test_time + timedelta(hours=2)) == "2 hours"
assert timesince(test_time, test_time + timedelta(days=1)) == "1 day"
assert timesince(test_time, test_time + timedelta(days=2)) == "2 days"
assert timesince(test_time, test_time + timedelta(days=7)) == "1 week"
assert timesince(test_time, test_time + timedelta(days=14)) == "2 weeks"
assert timesince(test_time, test_time + timedelta(days=30)) == "1 month"
assert timesince(test_time, test_time + timedelta(days=60)) == "2 months"
assert timesince(test_time, test_time + timedelta(days=365)) == "1 year"
assert timesince(test_time, test_time + timedelta(days=730)) == "2 years"
# okay now we want to test combinations
# e.g. 1 hour, 5 days
assert timesince(test_time, test_time + timedelta(days=5, hours=1)) == "5 days, 1 hour"
assert timesince(test_time, test_time + timedelta(days=15)) == "2 weeks, 1 day"
assert timesince(test_time, test_time + timedelta(days=97)) == "3 months, 1 week"
assert timesince(test_time, test_time + timedelta(days=2250)) == "6 years, 2 months"

View File

@ -29,9 +29,11 @@ from mediagoblin import _version
from mediagoblin.tools import common
from mediagoblin.tools.translate import get_gettext_translation
from mediagoblin.tools.pluginapi import get_hook_templates
from mediagoblin.tools.timesince import timesince
from mediagoblin.meddleware.csrf import render_csrf_form_token
SETUP_JINJA_ENVS = {}
@ -73,6 +75,9 @@ def get_jinja_env(template_loader, locale):
template_env.filters['urlencode'] = url_quote_plus
# add human readable fuzzy date time
template_env.globals['timesince'] = timesince
# allow for hooking up plugin templates
template_env.globals['get_hook_templates'] = get_hook_templates

View File

@ -0,0 +1,95 @@
# Copyright (c) Django Software Foundation and individual contributors.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without modification,
# are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# 3. Neither the name of Django nor the names of its contributors may be used
# to endorse or promote products derived from this software without
# specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
# ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
from __future__ import unicode_literals
import datetime
import pytz
from mediagoblin.tools.translate import pass_to_ugettext, lazy_pass_to_ungettext as _
"""UTC time zone as a tzinfo instance."""
utc = pytz.utc if pytz else UTC()
def is_aware(value):
"""
Determines if a given datetime.datetime is aware.
The logic is described in Python's docs:
http://docs.python.org/library/datetime.html#datetime.tzinfo
"""
return value.tzinfo is not None and value.tzinfo.utcoffset(value) is not None
def timesince(d, now=None, reversed=False):
"""
Takes two datetime objects and returns the time between d and now
as a nicely formatted string, e.g. "10 minutes". If d occurs after now,
then "0 minutes" is returned.
Units used are years, months, weeks, days, hours, and minutes.
Seconds and microseconds are ignored. Up to two adjacent units will be
displayed. For example, "2 weeks, 3 days" and "1 year, 3 months" are
possible outputs, but "2 weeks, 3 hours" and "1 year, 5 days" are not.
Adapted from http://blog.natbat.co.uk/archive/2003/Jun/14/time_since
"""
chunks = (
(60 * 60 * 24 * 365, lambda n: _('year', 'years', n)),
(60 * 60 * 24 * 30, lambda n: _('month', 'months', n)),
(60 * 60 * 24 * 7, lambda n : _('week', 'weeks', n)),
(60 * 60 * 24, lambda n : _('day', 'days', n)),
(60 * 60, lambda n: _('hour', 'hours', n)),
(60, lambda n: _('minute', 'minutes', n))
)
# Convert datetime.date to datetime.datetime for comparison.
if not isinstance(d, datetime.datetime):
d = datetime.datetime(d.year, d.month, d.day)
if now and not isinstance(now, datetime.datetime):
now = datetime.datetime(now.year, now.month, now.day)
if not now:
now = datetime.datetime.now(utc if is_aware(d) else None)
delta = (d - now) if reversed else (now - d)
# ignore microseconds
since = delta.days * 24 * 60 * 60 + delta.seconds
if since <= 0:
# d is in the future compared to now, stop processing.
return '0 ' + pass_to_ugettext('minutes')
for i, (seconds, name) in enumerate(chunks):
count = since // seconds
if count != 0:
break
s = pass_to_ugettext('%(number)d %(type)s') % {'number': count, 'type': name(count)}
if i + 1 < len(chunks):
# Now get the second item
seconds2, name2 = chunks[i + 1]
count2 = (since - (seconds * count)) // seconds2
if count2 != 0:
s += pass_to_ugettext(', %(number)d %(type)s') % {'number': count2, 'type': name2(count2)}
return s

View File

@ -123,6 +123,16 @@ def pass_to_ugettext(*args, **kwargs):
*args, **kwargs)
def pass_to_ungettext(*args, **kwargs):
"""
Pass a translation on to the appropriate ungettext method.
The reason we can't have a global ugettext method is because
mg_globals gets swapped out by the application per-request.
"""
return mg_globals.thread_scope.translations.ungettext(
*args, **kwargs)
def lazy_pass_to_ugettext(*args, **kwargs):
"""
Lazily pass to ugettext.
@ -158,6 +168,16 @@ def lazy_pass_to_ngettext(*args, **kwargs):
"""
return LazyProxy(pass_to_ngettext, *args, **kwargs)
def lazy_pass_to_ungettext(*args, **kwargs):
"""
Lazily pass to ungettext.
This is useful if you have to define a translation on a module
level but you need it to not translate until the time that it's
used as a string.
"""
return LazyProxy(pass_to_ungettext, *args, **kwargs)
def fake_ugettext_passthrough(string):
"""

View File

@ -61,6 +61,7 @@ setup(
'sqlalchemy-migrate',
'mock',
'itsdangerous',
'pytz',
## This is optional!
# 'translitcodec',
## For now we're expecting that users will install this from