How to localize a date in Liquid?
Asked Answered
W

4

11

In my Rails 5 app I am using Liquid to let my users generate content.

Based on my users input, I init my template with something like this:

string   = "Order {{ order.id }} was created {{ order.date | date: '%A %d/%m-%Y' }}"
template = Liquid::Template.parse(string)
result   = template.render({'order' => {'id' => '123', 'date' => order.date}})

This would print something a la:

'Order 123 was created Sunday 14/01-2018'

But how do I build Liquid date localization into my Rails app?

It does not seem to be supported in the documentation. However Shopify themselves seems to have build localization into their Liquid implementation.

I suppose I would need to pass my template a locale (en, fr, etc.) and a locale file. My Rails locale file looks like this:

en:
  datetime: &datetime
    month_names:
      [~, January, February, March, April, May, June, July, August, September, October, November, December]
    day_names:
      [Sunday, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday]
    abbr_day_names:
      [~, Sun, Mon, Thue, Wed, Thu, Fri, Sat]
    formats:
       default: "%d/%m/%Y"
       long: "%A %d/%m-%Y"

And I call it with:

l(order.date, :format => :long, :locale => 'en')

I would love to have access to a similar date localization inside my Liquid template.

Worser answered 23/1, 2018 at 15:24 Comment(3)
You might be able to deliberately call I18n.localize ? stackednotion.com/blog/2015/01/03/… seems relevant - it shows how to deliberately pass in a locale and use a customised date format (but note I've spent like maybe 5 whole seconds thinking about this...)Gains
gist.github.com/biow0lf/1486342 ?Abortifacient
Thanks @Abortifacient I've used this approach to register a filter that does l(order.date, :format => :long, :locale => 'en'). I'm gonna write it in as an answer if no one else comes up with a better solution. But it's not a perfect one, because it limits me to a set of predefined formats.Worser
W
2

I ended up using the link proveded by @Ben, and wrote up this solution:

# config/initializers/liquid_filters.rb
module I18nFilters
  def l(date, format)
    I18n.l(date, :format => format.to_sym)
  end
end

Liquid::Template.register_filter(I18nFilters)

Now, when invoking that new method:

{{ order.date | l: 'long' }}

it will print out:

'Sunday 14/01-2018'

with Sunday being whatever is in day_names withing the locale file.

Worser answered 9/3, 2018 at 6:48 Comment(0)
M
0

When you pass your date object with order.date

result   = template.render({'order' => {'id' => '123', 'date' => order.date}})

Can you pass a localized date or localized datetime? Then when it gets rendered it should be correct.

Otherwise, you could add the date to a span with class name you can target (for fallback) and "fix" it on the client side with a combination of pure Javascript and a library like Moment.js

# https://mcmap.net/q/265872/-how-to-get-the-exact-local-time-of-client
const timezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
Manganite answered 4/3, 2018 at 21:25 Comment(2)
Hi @Manganite modifying with js is a hack that I want to avoid. But the first suggestion seems more like a question to me. I'm not sure I understand it.Worser
sorry words are hard. :) When you pass order.date to the liquid template, I would assume you know the locale of the person you're displaying the data for, if not you can do this within rails, guides.rubyonrails.org/i18n.html. This will allow you to pass the correctly localized date to the template.Manganite
T
0

It looks like you're missing only a reference to t method.

Instead of

string   = "Order {{ order.id }} was created {{ order.date | date: '%A %d/%m-%Y' }}"

Try with

string   = "Order {{ order.id }} was created {{ order.date | t: date: '%A %d/%m-%Y' }}"
Tropaeolin answered 9/3, 2018 at 4:16 Comment(0)
D
0

You can do it by using the JavaScript Intl API.

Add this filter to .eleventy.js:

  eleventyConfig.addFilter("displayDate", function(date, locale) {
    return new Intl.DateTimeFormat(locale, { dateStyle: "long" }).format(date);
  });

Now you can show localized dates by using this in, for example, your Liquid template:

{{ date | displayDate: locale }}

Here, date contains the date and locale contains the locale.

Dorolisa answered 13/11, 2021 at 15:24 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.