Module:multiple images

From Wiktionary, the free dictionary
Jump to navigation Jump to search

Called by {{multiple images}}. Do not use directly.


-- implements [[template:multiple image]]
local p = {}

local autoscaledimages
local nonautoscaledimages

local function isnotempty(s)
	return s and s:match("^%s*(.-)%s*$") ~= ""
end

local function removepx(s)
	return tostring(s or ""):match("^(.*)[Pp][Xx]%s*$") or s
end

local function getdimensions(s, w, h)
	if tonumber(w) and tonumber(h) then
		nonautoscaledimages = true
		return tonumber(w), tonumber(h)
	end
	local file = s and mw.title.new("File:" .. mw.uri.decode(mw.ustring.gsub(s, "%|.*$", ""), "WIKI"))
	file = file and file.file or { width = 0, height = 0 }
	w = tonumber(file.width) or 0
	h = tonumber(file.height) or 0
	autoscaledimages = true
	return w, h
end

local function renderImageCell(image, width, height, link, alt, thumbtime, caption, textalign, istyle)
	local root = mw.html.create("")

	local altstr = "|alt=" .. (alt or "")
	local linkstr = link and ("|link=" .. link) or ""
	local widthstr = "|" .. tostring(width) .. "px"
	local thumbtimestr = ""

	if widthstr == "|-nanpx" then
		widthstr = ""
	end
	if isnotempty(thumbtime) then
		thumbtimestr = "|thumbtime=" .. thumbtime
	end

	local imagediv = root:tag("div")
	imagediv:addClass("thumbimage")
	imagediv:cssText(istyle)
	if height then
		imagediv:css("height", tostring(height) .. "px")
		imagediv:css("overflow", "hidden")
	end
	imagediv:wikitext("[[file:" .. image .. widthstr .. linkstr .. altstr .. thumbtimestr .. "]]")
	if isnotempty(caption) then
		local captiondiv = root:tag("div")
		captiondiv:addClass("thumbcaption")
		if isnotempty(textalign) then
			captiondiv:addClass("text-align-" .. textalign)
		end
		captiondiv:wikitext(caption)
	end
	return tostring(root)
end

local function getWidth(w1, w2)
	local w
	if isnotempty(w1) then
		w = tonumber(w1)
	elseif isnotempty(w2) then
		w = tonumber(w2)
	end
	return w or 200
end

local function getPerRow(pstr, ic)
	-- split string into array using any non-digit as a dilimiter
	local pr = mw.text.split(pstr or "", "[^%d][^%d]*")
	-- if split failed, assume a single row
	if #pr < 1 then
		pr = { tostring(ic) }
	end
	-- convert the array of strings to an array of numbers,
	-- adding any implied/missing numbers at the end of the array
	local r = 1
	local thisrow = tonumber(pr[1] or ic) or ic
	local prownum = {}
	while ic > 0 do
		prownum[r] = thisrow
		ic = ic - thisrow
		r = r + 1
		-- use the previous if the next is missing and
		-- make sure we don't overstep the number of images
		thisrow = math.min(tonumber(pr[r] or thisrow) or ic, ic)
	end
	return prownum
end

local function renderMultipleImages(frame)
	local pargs = frame:getParent().args
	local args = frame.args
	local width = removepx(pargs["width"] or "")
	local dir = pargs["direction"] or ""
	local border = pargs["border"] or args["border"] or ""
	local align = pargs["align"] or args["align"] or (border == "infobox" and "center" or "")
	local capalign = pargs["caption_align"] or args["caption_align"] or ""
	local totalwidth = removepx(pargs["total_width"] or args["total_width"] or "")
	local imgstyle = pargs["image_style"] or args["image_style"]
	local header = pargs["header"] or pargs["title"] or ""
	local footer = pargs["footer"] or ""
	local imagegap = tonumber(pargs["image_gap"] or "1") or 1
	local perrow = nil
	local thumbclass = {
		["left"] = "tleft",
		["none"] = "tnone",
		["center"] = "tnone",
		["centre"] = "tnone",
		["right"] = "tright",
	}

	-- find all the nonempty images
	local imagenumbers = {}
	local imagecount = 0
	for k, v in pairs(pargs) do
		local i = tonumber(tostring(k):match("^%s*image([%d]+)%s*$") or "0")
		if i > 0 and isnotempty(v) then
			table.insert(imagenumbers, i)
			imagecount = imagecount + 1
		end
	end

	-- sort the imagenumbers
	table.sort(imagenumbers)

	-- create an array with the number of images per row
	perrow = getPerRow(dir == "vertical" and "1" or pargs["perrow"], imagecount)

	-- compute the number of rows
	local rowcount = #perrow

	-- store the image widths and compute row widths and maximum row width
	local heights = {}
	local widths = {}
	local widthmax = 0
	local widthsum = {}
	local k = 0
	for r = 1, rowcount do
		widthsum[r] = 0
		for c = 1, perrow[r] do
			k = k + 1
			if k <= imagecount then
				local i = imagenumbers[k]
				if isnotempty(totalwidth) then
					widths[k], heights[k] =
						getdimensions(pargs["image" .. i], pargs["width" .. i], pargs["height" .. i])
				else
					widths[k] = getWidth(width, pargs["width" .. i])
				end
				widthsum[r] = widthsum[r] + widths[k]
			end
		end
		widthmax = math.max(widthmax, widthsum[r])
	end

	-- make sure the gap is non-negative
	if imagegap < 0 then
		imagegap = 0
	end

	-- if total_width has been specified, rescale the image widths
	if isnotempty(totalwidth) then
		totalwidth = tonumber(totalwidth)
		widthmax = 0
		local k = 0
		for r = 1, rowcount do
			local koffset = k
			local tw = totalwidth - (3 + imagegap) * (perrow[r] - 1) - 12
			local ar = {}
			local arsum = 0
			for j = 1, perrow[r] do
				k = k + 1
				if k <= imagecount then
					local i = imagenumbers[k]
					local h = heights[k] or 0
					if h > 0 then
						ar[j] = widths[k] / h
						heights[k] = h
					else
						ar[j] = widths[k] / 100
					end
					arsum = arsum + ar[j]
				end
			end
			local ht = tw / arsum
			local ws = 0
			k = koffset
			for j = 1, perrow[r] do
				k = k + 1
				if k <= imagecount then
					local i = imagenumbers[k]
					widths[k] = math.floor(ar[j] * ht + 0.5)
					ws = ws + widths[k]
					if heights[k] then
						heights[k] = math.floor(ht)
					end
				end
			end
			widthsum[r] = ws
			widthmax = math.max(widthmax, widthsum[r])
		end
	end

	-- start building the array of images, if there are images
	if imagecount > 0 then
		-- compute width of outer div
		local bodywidth = 0
		for r = 1, rowcount do
			if widthmax == widthsum[r] then
				bodywidth = widthmax + (3 + imagegap) * (perrow[r] - 1) + 12
			end
		end
		-- The body has a min-width of 100, which needs to be taken into account on specific widths
		bodywidth = math.max(100, bodywidth - 8)

		local bg = pargs["background color"] or ""
		-- create the array of images
		local root = mw.html.create("div")
		root:addClass("thumb")
		root:addClass("tmulti")
		-- root:addClass('tmulti-sandbox')
		root:addClass(thumbclass[align] or "tright")

		if align == "center" or align == "centre" then
			root:addClass("center")
		end
		if bg ~= "" then
			root:css("background-color", bg)
		end

		local div = root:tag("div")
		div:addClass("thumbinner multiimageinner")
		div:css("width", tostring(bodywidth) .. "px"):css("max-width", tostring(bodywidth) .. "px")
		if bg ~= "" then
			div:css("background-color", bg)
		end
		if border == "infobox" or border == "none" then
			div:css("border", "none")
		end
		-- add the header
		if isnotempty(header) then
			div:tag("div")
				:addClass("trow")
				:tag("div")
				:addClass("theader")
				:css("text-align", pargs["header_align"])
				:css("background-color", (pargs["header_background"] ~= "") and pargs["header_background"] or nil)
				:wikitext(header)
		end
		-- loop through the images
		local k = 0
		for r = 1, rowcount do
			local rowdiv = div:tag("div"):addClass("trow")
			for j = 1, perrow[r] do
				k = k + 1
				if k <= imagecount then
					local imagediv = rowdiv:tag("div")
					imagediv:addClass("tsingle")
					if bg ~= "" then
						imagediv:css("background-color", bg)
					end
					if (imagegap > 1) and (j < perrow[r]) then
						imagediv:css("margin-right", tostring(imagegap) .. "px")
					end
					local i = imagenumbers[k]
					local img = pargs["image" .. i]
					local w = widths[k]
					imagediv:css("width", tostring(2 + w) .. "px"):css("max-width", tostring(2 + w) .. "px")
					imagediv:wikitext(
						renderImageCell(
							img,
							w,
							heights[k],
							pargs["link" .. i],
							pargs["alt" .. i],
							pargs["thumbtime" .. i],
							pargs["caption" .. i],
							capalign,
							imgstyle
						)
					)
				end
			end
		end
		-- add the footer
		if isnotempty(footer) then
			local falign = string.lower(pargs["footer_align"] or args["footer_align"] or "left")
			falign = (falign == "centre") and "center" or falign
			div:tag("div")
				:addClass("trow")
				:css("display", (falign ~= "left") and "flow-root" or "flex")
				:tag("div")
				:addClass("thumbcaption")
				:css("text-align", (falign ~= "left") and falign or nil)
				:css("background-color", (pargs["footer_background"] ~= "") and pargs["footer_background"] or nil)
				:wikitext(footer)
		end
		return tostring(root)
	end
	return ""
end

function p.render(frame)
	autoscaledimages = false
	nonautoscaledimages = false

	return frame:extensionTag({
		name = "templatestyles",
		args = { src = "multiple images/styles.css", wrapper = ".tmulti" },
	}) .. renderMultipleImages(frame) .. (autoscaledimages and "[[Category:Pages using multiple image with auto scaled images]]" or "") .. (nonautoscaledimages and "[[Category:Pages using multiple image with manual scaled images]]" or "")
end

return p