Skip to content

Commit

Permalink
feat: full syntax for strftime, close #177
Browse files Browse the repository at this point in the history
- strftime syntax: <flags><width><modifier><conversion>
- new conversions: %n, %t
- fixed conversions: %N
- flags: _ ^ - 0 # :
- modifiers: E, O (ignored)
  • Loading branch information
harttle committed Dec 15, 2019
1 parent c69ad53 commit ba5ff3f
Show file tree
Hide file tree
Showing 7 changed files with 370 additions and 255 deletions.
4 changes: 2 additions & 2 deletions src/template/filter/filter.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Expression } from '../../render/expression'
import { Context } from '../../context/context'
import { isArray } from '../../util/underscore'
import { isArray, identify } from '../../util/underscore'
import { FilterImplOptions } from './filter-impl-options'

type KeyValuePair = [string?, string?]
Expand All @@ -18,7 +18,7 @@ export class Filter {
if (!impl && strictFilters) throw new TypeError(`undefined filter: ${name}`)

this.name = name
this.impl = impl || (x => x)
this.impl = impl || identify
this.args = args
}
public * render (value: any, context: Context) {
Expand Down
292 changes: 139 additions & 153 deletions src/util/strftime.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { padStart } from './underscore'
import { changeCase, padStart, padEnd } from './underscore'

const rFormat = /%([-_0^#:]+)?(\d+)?([EO])?(.)/
const monthNames = [
'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August',
'September', 'October', 'November', 'December'
Expand All @@ -15,170 +16,155 @@ const suffixes = {
3: 'rd',
'default': 'th'
}
interface FormatOptions {
flags: object;
width?: string;
modifier?: string;
}

function abbr (str: string) {
return str.slice(0, 3)
}

// prototype extensions
const _date = {
daysInMonth: function (d: Date) {
const feb = _date.isLeapYear(d) ? 29 : 28
return [31, feb, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
},

getDayOfYear: function (d: Date) {
let num = 0
for (let i = 0; i < d.getMonth(); ++i) {
num += _date.daysInMonth(d)[i]
}
return num + d.getDate()
},

getWeekOfYear: function (d: Date, startDay: number) {
// Skip to startDay of this week
const now = this.getDayOfYear(d) + (startDay - d.getDay())
// Find the first startDay of the year
const jan1 = new Date(d.getFullYear(), 0, 1)
const then = (7 - jan1.getDay() + startDay)
return padStart(String(Math.floor((now - then) / 7) + 1), 2, '0')
},

isLeapYear: function (d: Date) {
const year = d.getFullYear()
return !!((year & 3) === 0 && (year % 100 || (year % 400 === 0 && year)))
},

getSuffix: function (d: Date) {
const str = d.getDate().toString()
const index = parseInt(str.slice(-1))
return suffixes[index] || suffixes['default']
},

century: function (d: Date) {
return parseInt(d.getFullYear().toString().substring(0, 2), 10)
function daysInMonth (d: Date) {
const feb = isLeapYear(d) ? 29 : 28
return [31, feb, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
}
function getDayOfYear (d: Date) {
let num = 0
for (let i = 0; i < d.getMonth(); ++i) {
num += daysInMonth(d)[i]
}
return num + d.getDate()
}
function getWeekOfYear (d: Date, startDay: number) {
// Skip to startDay of this week
const now = getDayOfYear(d) + (startDay - d.getDay())
// Find the first startDay of the year
const jan1 = new Date(d.getFullYear(), 0, 1)
const then = (7 - jan1.getDay() + startDay)
return String(Math.floor((now - then) / 7) + 1)
}
function isLeapYear (d: Date) {
const year = d.getFullYear()
return !!((year & 3) === 0 && (year % 100 || (year % 400 === 0 && year)))
}
function getSuffix (d: Date) {
const str = d.getDate().toString()
const index = parseInt(str.slice(-1))
return suffixes[index] || suffixes['default']
}
function century (d: Date) {
return parseInt(d.getFullYear().toString().substring(0, 2), 10)
}

// default to 0
const padWidths = {
d: 2,
e: 2,
H: 2,
I: 2,
j: 3,
k: 2,
l: 2,
L: 3,
m: 2,
M: 2,
S: 2,
U: 2,
W: 2
}

// default to '0'
const padChars = {
a: ' ',
A: ' ',
b: ' ',
B: ' ',
c: ' ',
e: ' ',
k: ' ',
l: ' ',
p: ' ',
P: ' '
}
const formatCodes = {
a: function (d: Date) {
return dayNamesShort[d.getDay()]
},
A: function (d: Date) {
return dayNames[d.getDay()]
},
b: function (d: Date) {
return monthNamesShort[d.getMonth()]
},
B: function (d: Date) {
return monthNames[d.getMonth()]
},
c: function (d: Date) {
return d.toLocaleString()
},
C: function (d: Date) {
return _date.century(d)
},
d: function (d: Date) {
return padStart(d.getDate(), 2, '0')
},
e: function (d: Date) {
return padStart(d.getDate(), 2)
},
H: function (d: Date) {
return padStart(d.getHours(), 2, '0')
},
I: function (d: Date) {
return padStart(String(d.getHours() % 12 || 12), 2, '0')
},
j: function (d: Date) {
return padStart(_date.getDayOfYear(d), 3, '0')
},
k: function (d: Date) {
return padStart(d.getHours(), 2)
},
l: function (d: Date) {
return padStart(String(d.getHours() % 12 || 12), 2)
},
L: function (d: Date) {
return padStart(d.getMilliseconds(), 3, '0')
},
m: function (d: Date) {
return padStart(d.getMonth() + 1, 2, '0')
},
M: function (d: Date) {
return padStart(d.getMinutes(), 2, '0')
},
p: function (d: Date) {
return (d.getHours() < 12 ? 'AM' : 'PM')
},
P: function (d: Date) {
return (d.getHours() < 12 ? 'am' : 'pm')
},
q: function (d: Date) {
return _date.getSuffix(d)
},
s: function (d: Date) {
return Math.round(d.valueOf() / 1000)
},
S: function (d: Date) {
return padStart(d.getSeconds(), 2, '0')
},
u: function (d: Date) {
return d.getDay() || 7
},
U: function (d: Date) {
return _date.getWeekOfYear(d, 0)
},
w: function (d: Date) {
return d.getDay()
},
W: function (d: Date) {
return _date.getWeekOfYear(d, 1)
},
x: function (d: Date) {
return d.toLocaleDateString()
},
X: function (d: Date) {
return d.toLocaleTimeString()
},
y: function (d: Date) {
return d.getFullYear().toString().substring(2, 4)
},
Y: function (d: Date) {
return d.getFullYear()
},
z: function (d: Date) {
const tz = d.getTimezoneOffset() / 60 * 100
return (tz > 0 ? '-' : '+') + padStart(String(Math.abs(tz)), 4, '0')
},
'%': function () {
return '%'
}
a: (d: Date) => dayNamesShort[d.getDay()],
A: (d: Date) => dayNames[d.getDay()],
b: (d: Date) => monthNamesShort[d.getMonth()],
B: (d: Date) => monthNames[d.getMonth()],
c: (d: Date) => d.toLocaleString(),
C: (d: Date) => century(d),
d: (d: Date) => d.getDate(),
e: (d: Date) => d.getDate(),
H: (d: Date) => d.getHours(),
I: (d: Date) => String(d.getHours() % 12 || 12),
j: (d: Date) => getDayOfYear(d),
k: (d: Date) => d.getHours(),
l: (d: Date) => String(d.getHours() % 12 || 12),
L: (d: Date) => d.getMilliseconds(),
m: (d: Date) => d.getMonth() + 1,
M: (d: Date) => d.getMinutes(),
N: (d: Date, opts: FormatOptions) => {
const width = Number(opts.width) || 9
const str = String(d.getMilliseconds()).substr(0, width)
return padEnd(str, width, '0')
},
p: (d: Date) => (d.getHours() < 12 ? 'AM' : 'PM'),
P: (d: Date) => (d.getHours() < 12 ? 'am' : 'pm'),
q: (d: Date) => getSuffix(d),
s: (d: Date) => Math.round(d.valueOf() / 1000),
S: (d: Date) => d.getSeconds(),
u: (d: Date) => d.getDay() || 7,
U: (d: Date) => getWeekOfYear(d, 0),
w: (d: Date) => d.getDay(),
W: (d: Date) => getWeekOfYear(d, 1),
x: (d: Date) => d.toLocaleDateString(),
X: (d: Date) => d.toLocaleTimeString(),
y: (d: Date) => d.getFullYear().toString().substring(2, 4),
Y: (d: Date) => d.getFullYear(),
z: (d: Date, opts: FormatOptions) => {
const offset = d.getTimezoneOffset()
const nOffset = Math.abs(offset)
const h = Math.floor(nOffset / 60)
const m = nOffset % 60
return (offset > 0 ? '-' : '+') +
padStart(h, 2, '0') +
(opts.flags[':'] ? ':' : '') +
padStart(m, 2, '0')
},
't': () => '\t',
'n': () => '\n',
'%': () => '%'
};
(formatCodes as any).h = formatCodes.b;
(formatCodes as any).N = formatCodes.L
(formatCodes as any).h = formatCodes.b

export default function (d: Date, format: string) {
export default function (d: Date, formatStr: string) {
let output = ''
let remaining = format

while (true) {
const r = /%./g
const results = r.exec(remaining)

// No more format codes. Add the remaining text and return
if (!results) {
return output + remaining
}

// Add the preceding text
output += remaining.slice(0, r.lastIndex - 2)
remaining = remaining.slice(r.lastIndex)

// Add the format code
const ch = results[0].charAt(1)
const func = formatCodes[ch]
output += func ? func(d) : '%' + ch
let remaining = formatStr
let match
while ((match = rFormat.exec(remaining))) {
output += remaining.slice(0, match.index)
remaining = remaining.slice(match.index + match[0].length)
output += format(d, match)
}
return output + remaining
}

function format (d: Date, match: RegExpExecArray) {
const [input, flagStr = '', width, modifier, conversion] = match
const convert = formatCodes[conversion]
if (!convert) return input
const flags = {}
for (const flag of flagStr) flags[flag] = true
let ret = String(convert(d, { flags, width, modifier }))
let padChar = padChars[conversion] || '0'
let padWidth = width || padWidths[conversion] || 0
if (flags['^']) ret = ret.toUpperCase()
else if (flags['#']) ret = changeCase(ret)
if (flags['_']) padChar = ' '
else if (flags['0']) padChar = '0'
if (flags['-']) padWidth = 0
return padStart(ret, padWidth, padChar)
}
19 changes: 18 additions & 1 deletion src/util/underscore.ts
Original file line number Diff line number Diff line change
Expand Up @@ -101,8 +101,25 @@ export function range (start: number, stop: number, step = 1) {
}

export function padStart (str: any, length: number, ch = ' ') {
return pad(str, length, ch, (str, ch) => ch + str)
}

export function padEnd (str: any, length: number, ch = ' ') {
return pad(str, length, ch, (str, ch) => str + ch)
}

export function pad (str: any, length: number, ch: string, add: (str: string, ch: string) => string) {
str = String(str)
let n = length - str.length
while (n-- > 0) str = ch + str
while (n-- > 0) str = add(str, ch)
return str
}

export function identify<T> (val: T): T {
return val
}

export function changeCase (str: string): string {
const hasLowerCase = [...str].some(ch => ch >= 'a' && ch <= 'z')
return hasLowerCase ? str.toUpperCase() : str.toLowerCase()
}
2 changes: 1 addition & 1 deletion test/integration/liquid/fs-option.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ describe('LiquidOptions#fs', function () {
engine = new Liquid({
root: '/root/',
fs
})
} as any)
})
it('should be used to read templates', function () {
return engine.renderFile('files/foo')
Expand Down
4 changes: 2 additions & 2 deletions test/integration/liquid/liquid.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@ describe('Liquid', function () {
})
it('should call plugin with Liquid', async function () {
const engine = new Liquid()
engine.plugin(function (Liquid) {
this.registerFilter('t', x => isFalsy(x))
engine.plugin(function () {
this.registerFilter('t', isFalsy)
})
const html = await engine.parseAndRender('{{false|t}}')
expect(html).to.equal('true')
Expand Down
Loading

0 comments on commit ba5ff3f

Please sign in to comment.