sclass EnglishDateParser > DateStructures { replace P with ParsedWithTokens. bool assumeFuture = true; // e.g. for "tuesday" [not used] LS tok; int maxTokens = 3; L> longestParses(S s) { ret pwt_topDogs(pwt_filterByType SomeDate(allParses(s))); } SomeDate parse(S s) { ret getVar(first(longestParses(s))); } ItIt

allParses(S s) { // tokenize, initialize tok = javaTok(s); new L

out; L> codeTokens = asList(lai_codeTokens(tok)); // find numbers L> numbers = concatMap parseNumber(codeTokens); // find ordinals (1st, 2nd, ...) new L> ordinals; for (P number : numbers) for (P ord : parseToTheRight(fixedToken("st", "nd", "rd", "th"), number)) ordinals.add(pwt_combine(number, ord)); //print(+ordinals); // "in days" for (P number : numbers) for (P in : print("in", parseToTheLeft(fixedToken("in"), number))) for (P days : parseToTheRight(fixedToken("day", "days"), number)) out.add(pwt_combine(new TodayPlus(number!), in, days)); // " days from now" for (P number : numbers) for (P daysFromNow : parseToTheRight(fixedToken("day from now", "days from now"), number)) out.add(pwt_combine(new TodayPlus(number!), number, daysFromNow)); L> years = concatMap parseYear(codeTokens); out.addAll(years); L> weekdays = concatMap parseWeekday(codeTokens); out.addAll(weekdays); // month names L> monthNames = concatMap parseMonthName(codeTokens); out.addAll(monthNames); // month name + year, e.g. "February 2020" out.addAll(pwt_combine(monthNames, years, (month, year) -> new Month(month->month, new Year(year!)))); /*for (P month : monthNames) for (P year : pwt_toTheRightOf(years, month)) out.add(pwt_combine(new Month(month->month, new Year(year!)), month, year));*/ // month name + ordinal, e.g. "March 4th" //for (P month : monthNames) // "next " for (P weekday : weekdays) for (P next : parseToTheLeft(fixedToken("next"), weekday)) out.add(pwt_combine(new Weekday(weekday->weekday, new CurrentWeekPlus(1)), next, weekday)); for (P year : years) for (P slash : parseToTheRight(fixedToken("/"), year)) for (P month : parseMonth(slash.remaining())) for (P slash2 : parseToTheRight(fixedToken("/"), month)) for (P day : parseDayOfMonth(slash2.remaining())) out.add(pwt_combine(new Day(day!, new Month(month!, new Year(year!))), year, day)); ret itIt(out); } ItIt> parseNumber(ListAndIndex toParse) { ret mapI_nonNulls(pwt_followingTokens(1, maxTokens, toParse), p -> isInteger(p!) ? p.withValue(parseInt(p!)) : null); } ItIt> parseYear(ListAndIndex toParse) { ret filterI(parseNumber(toParse), p -> between(p!, 1900, 2100)); } ItIt> parseMonth(ListAndIndex toParse) { ret filterI(parseNumber(toParse), p -> between(p!, 1, 12)); } ItIt> parseDayOfMonth(ListAndIndex toParse) { ret filterI(parseNumber(toParse), p -> between(p!, 1, 31)); } IF1 fixedToken(S... tokens) { ret fixedToken(litciset(tokens)); } IF1 fixedToken(Set set) { ret t -> contains(set, t) ? t : null; } ItIt> parseFixedToken(S token, ListAndIndex toParse) { ret filterI(pwt_followingTokens(1, maxTokens, toParse), p -> eqic(p!, token)); } ItIt> parseFixedToken_before(S token, ListAndIndex toParse) { ret filterI(pwt_precedingTokens(1, maxTokens, toParse), p -> eqic(p!, token)); } ItIt> parseOneOfFixedTokens(Set tokens, ListAndIndex toParse) { ret filterI(pwt_followingTokens(1, maxTokens, toParse), p -> contains(tokens, p!)); } ItIt> parseWeekday(ListAndIndex toParse) { ret pwt_mapI(s -> new Weekday(parseEnglishWeekday(s)), parseOneOfFixedTokens(englishWeekdays_ciSet(), toParse)); } ItIt> parseMonthName(ListAndIndex toParse) { ret pwt_mapI(s -> new Month(parseEnglishMonthName(s)), parseOneOfFixedTokens(englishMonthNames_ciSet(), toParse)); } L> parseToTheLeft(IF1 f, ParsedWithTokens p) { ret pwt_transform(f, pwt_precedingTokens(1, maxTokens, p.start())); } L> parseToTheRight(IF1 f, ParsedWithTokens p) { ret pwt_transform(f, pwt_followingTokens(1, maxTokens, p.remaining())); } }