Module:Calendar date/recurring

MyWikiBiz, Author Your Legacy — Tuesday January 14, 2025
Jump to navigationJump to search

Documentation for this module may be created at Module:Calendar date/recurring/doc

--[[ 

Calculates the Gregorian date of a recurring holiday that varies year-to-year, but follows the rule "Nth [day of week] of [month]"

  "month" = month number (1 to 12)
  "weeknumber" = number of week (1 to 4, or -1 to mean "last")
  "dayofweek" = number that represents the day of the week, where 1 = Sunday and 7 = Saturday
  "year" = Gregorian calendar year
               
 ]]

require('Module:No globals')

local p = {}

function p.calculate(frame)
	local args = frame.args

	local ONE_DAY = 86400 -- number of seconds in one day
	local year = tonumber(args.year)
	local month = tonumber(args.month)
	local weeknumber = tonumber(args.weeknumber)
	local dayofweek = tonumber(args.dayofweek)

	local date = os.time{year=year, month=month, day=1}
	local dateparts = os.date("*t", date)
	
	if weeknumber > 0 then
		-- find the first [dayofweek] of this month
		while dateparts["wday"] ~= dayofweek do
			date = date + ONE_DAY
			dateparts = os.date("*t", date)
		end

		-- add the correct number of weeks
		if weeknumber > 1 then
			date = date + ((weeknumber - 1) * (7 * ONE_DAY))
		end
	else
		-- find the first day of the next month
		while dateparts["month"] == month do
			date = date + ONE_DAY
			dateparts = os.date("*t", date)
		end

		-- go back one day to get the last day of the month we want
		date = date - ONE_DAY
		dateparts = os.date("*t", date)

		-- go backwards until we find the right day of week
		while dateparts["wday"] ~= dayofweek do
			date = date - ONE_DAY
			dateparts = os.date("*t", date)
		end
	end
	local result = os.date("%Y-%m-%d", date)
	return result

end

return p