Razor
Razor

Reputation: 29678

Freemarker/Velocity - date manipulation

I have a fairly simple question about these 2 templating engines.
I'm trying to make a future/past date, a fixed time from now, e.g. 18 months ago, or tomorrow.

I know that it is possible to do this with a java date object inside a velocity/freemarker template (something like $date.add(2,-18)), but I would like to do this with DateTool or freemarker core.

This is something that I see as purely presentational (just think at the default dates you see in flight booking forms), so I can't see any reason why a templating engine shouldn't be able to do this.

Is it possible though? If so, how?

Upvotes: 4

Views: 11765

Answers (7)

gce
gce

Reputation: 1663

Just in case that helps someone, posting my result for Freemarker, based on previous answers:

<#function minusSeconds date, numSec >
    <#return (date?long - numSec * 1000)?number_to_datetime?iso_utc
</#function>

Upvotes: 0

joshwhatk
joshwhatk

Reputation: 156

I found that per @Stefan Haberl, ?long does work on dates to get the same value as java.util.Date.getTime() as a Long. However, I needed a little more explanation to be able to compare dates or modify dates.

Here's the gist:

  1. ?long = java.util.Date.getTime() returns epoch time in milliseconds
  2. At this point, you can add/subtract any number of milliseconds from that number for your manipulation

I like working in seconds instead of milliseconds (less unnecessary zeros, I don't care about milliseconds, etc.), which looks like this:

[#function convertToUnix date]
  [#return (date?date?long / 1000)]
[/#function]

[#-- Output Unix Timestamp --]
${convertToUnix(.now)}

At this point, 86400 = 1 day (because we are in "seconds" now), so you can simply add/subtract that to manipulate the date.

[#assign
  day = 86400
  week = 7 * day
  avgMonth = 365.25 / 12 * day

  testingEndOfDay = convertToUnix(.now) < (convertToUnix(sameDay) + day)
  testingYesterday = convertToUnix(.now) < (convertToUnix(yesterday) + day)
]
${testingEndOfDay?c} # true, .now is less than the end of the day
${testingYesterday?c} # false, .now is greater than the end of yesterday

Note: I am ignoring the time of day, we received dates that started at 12:00AM and wanted to check against .now for the end of the day.

Now, if I want to get a date back from the Unix format (in seconds), I can convert it back using the ?number_to_date builtin

[#assign
  nowAsUnix = convertToUnix(.now)
  prettyDate = (nowAsUnix * 1000)?number_to_date
]

Note: I'm open to edits/improvements as I'm not sure why much of this was required ¯\_(ツ)_/¯

Upvotes: 2

Stefan Haberl
Stefan Haberl

Reputation: 10559

I know this is post is really old, but for people from the future still looking for an answer: Date manipulation can be done converting dates to/from milliseconds:

${(mydate?long - 18 * 86400000)?number_to_date?string("yyyy-MM-dd")}

for example would subtract 18 days from mydate. (Note that 86400000 is the amount of milliseconds per day)

Upvotes: 10

koljaTM
koljaTM

Reputation: 10260

You can write your own methods to use in FreeMarker: http://freemarker.sourceforge.net/docs/pgui_datamodel_method.html

build a DataAddMethod that executes this logic.

Upvotes: 0

A. Ionescu
A. Ionescu

Reputation: 2146

You can do it in Velocity too, but not with the DateTool (that belongs the Velocity Extras anyway).

It is a good practice to have a "Format" object with various utilities that are practical for presentational purposes only (you can see the various frameworks that use Velocity how they have "Format" objects).

So your code would look like $format.dateFromNow(), and there would be still a presentational approach.

Upvotes: 1

serg
serg

Reputation: 111345

In Velocity Tools there is no such method.

Upvotes: 1

Istao
Istao

Reputation: 7585

For freemarker, perhaps : Built-ins for dates.

Upvotes: 1

Related Questions