[Fixed]-How to automatically get user's timezone from client in Django?

14πŸ‘

βœ…

From the documentation:

Selecting the current time zone

The current time zone is the
equivalent of the current locale for translations. However, there’s no
equivalent of the Accept-Language HTTP header that Django could use to
determine the user’s time zone automatically. Instead, Django provides
time zone selection functions. Use them to build the time zone
selection logic that makes sense for you.

You can try setting timezone cookie via javascript by utilizing getTimezoneOffset function or try to do some geoip magic and figure timezone by location. Probably the most reliable way would be to ask the user directly and save this information in user profile/session.

14πŸ‘

I’ve simplified it even further, and you can plug in in here: https://github.com/Miserlou/django-easy-timezones
or
http://gun.io/blog/django-easy-timezones/

πŸ‘€Rich Jones

9πŸ‘

I was hunting around for the sam thing yesterday. In the end I ended up putting together a Django app to what BluesRockAddict suggests above (i.e. use getTimezoneOffset):

https://github.com/adamcharnock/django-tz-detect

I hope someone finds that useful.

πŸ‘€Adam Charnock

2πŸ‘

There is a nice APP for django to activate timezone https://pypi.python.org/pypi/django-visitor-information-middleware/0.1.0. Which having two middleware

TimezoneMiddleware

The middleware activates a timezone for an authenticated user.

VisitorInformationMiddleware

This middleware adds the following keys to the request.visitor dictionary:

country – country the visitor is based in.

city – city the visitor is based in

location.timezone – timezone used in the location visitor is based in

location.unit_system – unit system used in the location visitor is based in

user.timezone – timezone of the currently authenticated user

user.unit_system – unit system of the currently authenticated user.

cookie_notice – True if a cookie consent notice should be displayed for the current visitor.

Note: Location of the user is determined based on the user's IP address.
πŸ‘€Ranju R

2πŸ‘

There are 2 ways without and with django-tz-detect to automatically detect and apply the user’s current timezone rather than applying only one timezone set to TIME_ZONE in settings.py. *I recommend the way without django-tz-detect and I use Django 4.2.3.

First, create middleware folder with __init__.py(Empty file) and custom.py in core/ and copy base.html from django/contrib/admin/templates/admin/base.html in your virtual environment to templates/admin/ as shown below. *You can see my answer explaining how to create middlewares and you can see my answer explaining how to set templates folder and you can see my answer explaining how to override django admin templates:

django-project
 |-core
 |  |-settings.py
 |  β””-middleware # Here
 |     |-__init__.py
 |     β””-custom.py
 |-my_app1
 |-my_app2
 β””-templates
    |-my_app1
    |  β””-base.html
    |-my_app2
    |  β””-base.html
    β””-admin
       β””-base.html # Here

<Without django-tz-detect>

Now, put the code below to custom.py:

# "core/middleware/custom.py"

import zoneinfo
from django.utils import timezone
from django.shortcuts import render

class TimezoneMiddleware:
    def __init__(self, get_response):
        self.get_response = get_response

    def __call__(self, request):
        tz = request.COOKIES.get("mytz")
        if tz:
            timezone.activate(zoneinfo.ZoneInfo(tz))
        else:
            timezone.activate(zoneinfo.ZoneInfo("UTC"))
        return self.get_response(request)

Then, put <script></script> just before </body> in templates/my_app1/base.html and templates/my_app2/base.html as shown below:

{% "templates/my_app1/base.html" %}
{% "templates/my_app2/base.html" %}

<script>
let tz = Intl.DateTimeFormat().resolvedOptions().timeZone;
if (!tz) {
    tz = "UTC"
}
document.cookie = "mytz=" + tz + ";path=/";
</script>
</body>

Then, put <script></script> just before </body> tag in templates/admin/base.html as shown below:

{% "templates/admin/base.html" %}

<script>
let tz = Intl.DateTimeFormat().resolvedOptions().timeZone;
if (!tz) {
    tz = "UTC"
}
document.cookie = "mytz=" + tz + ";path=/";
</script>
</body>

Finally, set TimezoneMiddleware to MIDDLEWARE, then every time you load a django website, the user’s current timezone is automatically detected and applied:

# "core/settings.py"

MIDDLEWARE = [
    ...
    'core.middleware.custom.TimezoneMiddleware'
]

<With django-tz-detect>

Now, install django-tz-detect as shown below:

pip install django-tz-detect

Then, set tz_detect and TimezoneMiddleware to INSTALLED_APPS and MIDDLEWARE respectively in settings.py as shown below. *Make sure that django.template.context_processors.request is set to TEMPLATESβ€˜s OPTIONSβ€˜s context_processors:

# "core/settings.py"

INSTALLED_APPS = [
    ...
    'tz_detect' # Here
]

MIDDLEWARE = [
    ...
    'tz_detect.middleware.TimezoneMiddleware', # Here
]

...

TEMPLATES = [
    {
        ...
        'DIRS': [
            BASE_DIR / 'templates'
        ],
        ...
        'OPTIONS': {
            'context_processors': [
                ...
                'django.template.context_processors.request', # Here
                ...
            ],
        },
    },
]

Then, add the path below to urlpatterns in core/urls.py as shown below:

# "core/urls.py"

urlpatterns = [
    path('admin/', admin.site.urls),
    path('my_app1/', include('my_app1.urls')),
    path('my_app2/', include('my_app2.urls')),
    path('tz_detect/', include('tz_detect.urls')) # Here
]

Then, load tz_detect.py, then use tz_detect tag just before </body> tag in templates/my_app1/base.html and templates/my_app2/base.html as shown below:

{% "templates/my_app1/base.html" %}
{% "templates/my_app2/base.html" %}

{% load tz_detect %}
...
{% tz_detect %}
</body>

Finally, load tz_detect.py, then use tz_detect tag just before </body> tag in templates/admin/base.html as shown below:

{% "templates/admin/base.html" %}
                               {% ↓ Here ↓ %}
{% load i18n static i18n_switcher tz_detect %}<!DOCTYPE html>
...
<!-- END SVGs -->
{% tz_detect %}
</body>

1πŸ‘

I currently created a middleware class (following Django’s documentation) in which I rely on MaxMind geoip database (http://dev.maxmind.com/geoip/legacy/geolite) and GeoDjango (https://docs.djangoproject.com/en/1.5/ref/contrib/gis/) to retrive user’s country code and then set the timezone dynamically using pytz:

class TimezoneMiddleware(object):


    def __getUserTimeZone(self, request):
        info = IPResolver(request).getGeoInfo()
        return pytz.country_timezones[info['country_code']][0]


    def process_request(self, request):
        try:
            tz = self.__getUserTimeZone(request)
            timezone.activate(tz)
            logger.debug('Time zone "%s" activated' % str(tz))
        except Exception as e:
            logger.error('Unable to set timezone: %s' % str(e))

pytz.country_timezones returns a collection of time zones available for the given country, so I basically choose the first one returned.

IPResolver is a personal utility class I wrote on top of django.contrib.gis.utils.GeoIP

πŸ‘€daveoncode

0πŸ‘

I just did this in my program. In my js:

<script> 
  var today = new Date()
  var offset = today.getTimezoneOffset()
  const serializedData = 'csrfmiddlewaretoken={{ csrf_token }}&tz_offset= ' 
  + offset.toString() 
</script>

Then on the django side:

def your_function(request):
    tz_offset = request.POST.get('tz_offset')
    now = datetime.datetime.now() - timedelta(minutes=int(tz_offset))
    return now

I pass the serializedData into the POST request and then access that in my function

<script>
   function update_transaction(serializedData){
    $.post('{% url "your_url" %}', serializedData, function(data){
        //do stuff
    });
}
</script>

 
πŸ‘€B.Shreve

Leave a comment