Module:File link: Difference between revisions
| imported>Mr. Stradivarius  (add start, end, and thumbtime) |  (Created page with "-- This module provides a library for formatting file wikilinks.  local yesno = require('Module:Yesno') local checkType = require('libraryUtil').checkType  local p = {}  function p._main(args) 	checkType('_main', 1, args, 'table')  	-- This is basically libraryUtil.checkTypeForNamedArg, but we are rolling our 	-- own function to get the right error level. 	local function checkArg(key, val, level) 		if type(val) ~= 'string' then 			error(string.format( 				"type error in...") | ||
| (9 intermediate revisions by 3 users not shown) | |||
| Line 1: | Line 1: | ||
| -- This module provides a library for formatting file wikilinks. | -- This module provides a library for formatting file wikilinks. | ||
| local  | local yesno = require('Module:Yesno') | ||
| local checkType = libraryUtil.checkType | local checkType = require('libraryUtil').checkType | ||
| local  | local p = {} | ||
| function  | function p._main(args) | ||
| 	checkType(' | 	checkType('_main', 1, args, 'table') | ||
| 	local obj, data = {}, {} | |||
| 	-- This is basically libraryUtil.checkTypeForNamedArg, but we are rolling our | |||
| 	-- own function to get the right error level. | |||
| 	local checkSelf = libraryUtil.makeCheckSelfFunction( | |||
| 	local function checkArg(key, val, level) | |||
| 		'fileLink', | |||
| 		if type(val) ~= 'string' then | |||
| 		'fileLink', | |||
| 		obj, | |||
| 		'fileLink object' | |||
| 	) | |||
| 	-- Set the filename if we were passed it as an input to fileLink.new. | |||
| 	if filename then | |||
| 		data.theName = filename | |||
| 	end | |||
| 	function data:name(s) | |||
| 		checkSelf(self, 'name') | |||
| 		checkType('fileLink:name', 1, s, 'string') | |||
| 		data.theName = s | |||
| 		return self | |||
| 	end | |||
| 	function data:format(s, filename) | |||
| 		checkSelf(self, 'format') | |||
| 		checkType('fileLink:format', 1, s, 'string', true) | |||
| 		checkType('fileLink:format', 2, format, 'string', true) | |||
| 		local validFormats = { | |||
| 			thumb = true, | |||
| 			thumbnail = true, | |||
| 			frame = true, | |||
| 			framed = true, | |||
| 			frameless = true | |||
| 		} | |||
| 		if s == nil or validFormats[s] then | |||
| 			data.theFormat = s | |||
| 			data.theFormatFilename = filename | |||
| 		else | |||
| 			error(string.format( | 			error(string.format( | ||
| 				" | 				"type error in '%s' parameter of '_main' (expected string, got %s)", | ||
| 				key, type(val) | |||
| 			),  | 			), level) | ||
| 		end | 		end | ||
| 		return self | |||
| 	end | 	end | ||
| 	local  | 	local ret = {} | ||
| 		-- Used for formatting duplication errors in size-related methods. | |||
| 	-- Adds a positional parameter to the buffer. | |||
| 		error(string.format( | |||
| 	local function addPositional(key) | |||
| 			"duplicate size argument detected in '%s'" | |||
| 		local val = args[key] | |||
| 			.. " ('upright' cannot be used in conjunction with height or width)", | |||
| 		if not val then | |||
| 			methodName | |||
| 			return nil | |||
| 	end | |||
| 	function data:width(px) | |||
| 		checkSelf(self, 'width') | |||
| 		checkType('fileLink:width', 1, px, 'number', true) | |||
| 		if px and data.isUpright then | |||
| 			sizeError('fileLink:width') | |||
| 		end | 		end | ||
| 		checkArg(key, val, 4) | |||
| 		data.theWidth = px | |||
| 		ret[#ret + 1] = val | |||
| 		return self | |||
| 	end | 	end | ||
| 	-- Adds a named parameter to the buffer. We assume that the parameter name | |||
| 	function data:height(px) | |||
| 	-- is the same as the argument key. | |||
| 		checkSelf(self, 'height') | |||
| 	local function addNamed(key) | |||
| 		checkType('fileLink:height', 1, px, 'number', true) | |||
| 		local val = args[key] | |||
| 		if px and data.isUpright then | |||
| 		if not val then | |||
| 			sizeError('fileLink:height') | |||
| 			return nil | |||
| 		end | 		end | ||
| 		checkArg(key, val, 4) | |||
| 		data.theHeight = px | |||
| 		ret[#ret + 1] = key .. '=' .. val | |||
| 		return self | |||
| 	end | 	end | ||
| 	-- Filename | |||
| 	function data:upright(isUpright, factor) | |||
| 	checkArg('file', args.file, 3) | |||
| 		checkSelf(self, 'upright') | |||
| 	ret[#ret + 1] = 'File:' .. args.file | |||
| 		checkType('fileLink:upright', 1, isUpright, 'boolean', true) | |||
| 		checkType('fileLink:upright', 2, factor, 'number', true) | |||
| 	-- Format | |||
| 		if isUpright and (data.theWidth or data.theHeight) then | |||
| 	if args.format then | |||
| 			sizeError('fileLink:upright') | |||
| 		checkArg('format', args.format) | |||
| 		end | |||
| 		if args.formatfile then | |||
| 		data.isUpright = isUpright | |||
| 			checkArg('formatfile', args.formatfile) | |||
| 		data.uprightFactor = factor | |||
| 			ret[#ret + 1] = args.format .. '=' .. args.formatfile | |||
| 		return self | |||
| 	end | |||
| 	function data:resetSize() | |||
| 		checkSelf(self, 'resetSize') | |||
| 		for i, field in ipairs{'theWidth', 'theHeight', 'isUpright', 'uprightFactor'} do | |||
| 			data[field] = nil | |||
| 		end | |||
| 		return self | |||
| 	end | |||
| 	function data:location(s) | |||
| 		checkSelf(self, 'location') | |||
| 		checkType('fileLink:location', 1, s, 'string', true) | |||
| 		local validLocations = { | |||
| 			right = true, | |||
| 			left = true, | |||
| 			center = true, | |||
| 			none = true | |||
| 		} | |||
| 		if s == nil or validLocations[s] then | |||
| 			data.theLocation = s | |||
| 		else | 		else | ||
| 			ret[#ret + 1] = args.format | |||
| 				"bad argument #1 to 'fileLink:location' ('%s' is not a valid location)", | |||
| 				s | |||
| 			), 2) | |||
| 		end | 		end | ||
| 		return self | |||
| 	end | 	end | ||
| 	-- Border | |||
| 	function data:alignment(s) | |||
| 	if yesno(args.border) then | |||
| 		checkSelf(self, 'alignment') | |||
| 		ret[#ret + 1] = 'border' | |||
| 		checkType('fileLink:alignment', 1, s, 'string', true) | |||
| 		local validAlignments = { | |||
| 			baseline = true, | |||
| 			middle = true, | |||
| 			sub = true, | |||
| 			super = true, | |||
| 			['text-top'] = true, | |||
| 			['text-bottom'] = true, | |||
| 			top = true, | |||
| 			bottom = true | |||
| 		} | |||
| 		if s == nil or validAlignments[s] then | |||
| 			data.theAlignment = s | |||
| 		else | |||
| 			error(string.format( | |||
| 				"bad argument #1 to 'fileLink:alignment' ('%s' is not a valid alignment)", | |||
| 				s | |||
| 			), 2) | |||
| 		end | |||
| 		return self | |||
| 	end | |||
| 	function data:border(hasBorder) | |||
| 		checkSelf(self, 'border') | |||
| 		checkType('fileLink:border', 1, hasBorder, 'boolean', true) | |||
| 		data.hasBorder = hasBorder | |||
| 		return self | |||
| 	end | |||
| 	function data:link(s) | |||
| 		checkSelf(self, 'link') | |||
| 		checkType('fileLink:link', 1, s, 'string', true) | |||
| 		data.theLink = s | |||
| 		return self | |||
| 	end | |||
| 	function data:alt(s) | |||
| 		checkSelf(self, 'alt') | |||
| 		checkType('fileLink:alt', 1, s, 'string', true) | |||
| 		data.theAlt = s | |||
| 		return self | |||
| 	end | |||
| 	function data:page(num) | |||
| 		checkSelf(self, 'page') | |||
| 		checkType('fileLink:page', 1, num, 'number', true) | |||
| 		data.thePage = s | |||
| 		return self | |||
| 	end | |||
| 	function data:class(s) | |||
| 		checkSelf(self, 'class') | |||
| 		checkType('fileLink:class', 1, s, 'string', true) | |||
| 		data.theClass = s | |||
| 		return self | |||
| 	end | |||
| 	function data:lang(s) | |||
| 		checkSelf(self, 'lang') | |||
| 		checkType('fileLink:lang', 1, s, 'string', true) | |||
| 		data.theLang = s | |||
| 		return self | |||
| 	end | 	end | ||
| 	addPositional('location') | |||
| 	local function checkTypeStringOrNum(funcName, pos, arg) | |||
| 	addPositional('alignment') | |||
| 		local argType = type(arg) | |||
| 	addPositional('size') | |||
| 		if argType ~= 'nil' and argType ~= 'string' and argType ~= 'number' then | |||
| 	addNamed('upright') | |||
| 			error(string.format( | |||
| 	addNamed('link') | |||
| 				"bad argument #%d to '%s' (string or number expected, got %s)", | |||
| 	addNamed('alt') | |||
| 				pos, | |||
| 	addNamed('page') | |||
| 				funcName, | |||
| 	addNamed('class') | |||
| 				argType | |||
| 	addNamed('lang') | |||
| 			), 3) | |||
| 	addNamed('start') | |||
| 		end | |||
| 	addNamed('end') | |||
| 	addNamed('thumbtime') | |||
| 	addPositional('caption') | |||
| 	return string.format('[[%s]]', table.concat(ret, '|')) | |||
| end | |||
| function p.main(frame) | |||
| 	local origArgs = require('Module:Arguments').getArgs(frame, { | |||
| 		wrappers = 'Template:File link' | |||
| 	}) | |||
| 	if not origArgs.file then | |||
| 		error("'file' parameter missing from [[Template:File link]]", 0) | |||
| 	end | 	end | ||
| 	function data:startTime(time) | |||
| 		checkSelf(self, 'startTime') | |||
| 		checkTypeStringOrNum('fileLink:startTime', 1, time) | |||
| 		data.theStartTime = time | |||
| 		return self | |||
| 	end | |||
| 	function data:endTime(time) | |||
| 		checkSelf(self, 'endTime') | |||
| 		checkTypeStringOrNum('fileLink:endTime', 1, time) | |||
| 		data.theEndTime = time | |||
| 		return self | |||
| 	end | |||
| 	function data:thumbTime(time) | |||
| 		checkSelf(self, 'thumbTime') | |||
| 		checkTypeStringOrNum('fileLink:thumbTime', 1, time) | |||
| 		data.theThumbTime = time | |||
| 		return self | |||
| 	end | |||
| 	function data:caption(s) | |||
| 		checkSelf(self, 'caption') | |||
| 		checkType('fileLink:caption', 1, s, 'string', true) | |||
| 		data.theCaption = s | |||
| 		return self | |||
| 	end | |||
| 	function data:render() | |||
| 		checkSelf(self, 'render') | |||
| 		local ret = {} | |||
| 		-- Filename | |||
| 		if not data.theName then | |||
| 			error('fileLink:render: no filename was found') | |||
| 		end | |||
| 		ret[#ret + 1] = 'File:' .. data.theName | |||
| 		-- Format | |||
| 		if data.theFormat and data.theFormatFilename then | |||
| 			ret[#ret + 1] = data.theFormat .. '=' .. data.theFormatFilename | |||
| 		elseif data.theFormat then | |||
| 			ret[#ret + 1] = data.theFormat | |||
| 		end | |||
| 		-- Border | |||
| 		if data.hasBorder then | |||
| 			ret[#ret + 1] = 'border' | |||
| 		end | |||
| 		-- Location | |||
| 		ret[#ret + 1] = data.theLocation | |||
| 	-- Copy the arguments that were passed to a new table to avoid looking up | |||
| 		-- Alignment | |||
| 	-- every possible parameter in the frame object. | |||
| 		ret[#ret + 1] = data.theAlignment | |||
| 	local args = {} | |||
| 	for k, v in pairs(origArgs) do | |||
| 		-- Size | |||
| 		-- Make _BLANK a special argument to add a blank parameter. For use in | |||
| 		if data.isUpright and data.uprightFactor then | |||
| 		-- conditional templates etc. it is useful for blank arguments to be | |||
| 			ret[#ret + 1] = 'upright=' .. tostring(data.uprightFactor) | |||
| 		-- ignored, but we still need a way to specify them so that we can do | |||
| 		elseif data.isUpright then | |||
| 		-- things like [[File:Example.png|link=]]. | |||
| 			ret[#ret + 1] = 'upright' | |||
| 		if v == '_BLANK' then | |||
| 			v = '' | |||
| 			ret[#ret + 1] = string.format('%dx%dpx', data.theWidth, data.theHeight) | |||
| 		elseif data.theWidth then | |||
| 			ret[#ret + 1] = tostring(data.theWidth) .. 'px' | |||
| 		elseif data.theHeight then | |||
| 			ret[#ret + 1] = string.format('x%dpx', data.theHeight) | |||
| 		end | 		end | ||
| 		args[k] = v | |||
| 		-- Render named parameters. | |||
| 		-- That includes link, alt, page, class, lang, start, end, and thumbtime. | |||
| 		do | |||
| 			local namedParameters = { | |||
| 				{'link', 'theLink'}, | |||
| 				{'alt', 'theAlt'}, | |||
| 				{'page', 'thePage'}, | |||
| 				{'class', 'theClass'}, | |||
| 				{'lang', 'theLang'}, | |||
| 				{'start', 'theStartTime'}, | |||
| 				{'end', 'theEndTime'}, | |||
| 				{'thumbtime', 'theThumbTime'} | |||
| 			} | |||
| 			for i, t in ipairs(namedParameters) do | |||
| 				local parameter = t[1] | |||
| 				local value = data[t[2]] | |||
| 				if value then | |||
| 					ret[#ret + 1] = parameter .. '=' .. tostring(value) | |||
| 				end | |||
| 			end | |||
| 		end | |||
| 		-- Caption | |||
| 		ret[#ret + 1] = data.theCaption | |||
| 		return string.format('[[%s]]', table.concat(ret, '|')) | |||
| 	end | 	end | ||
| 	return p._main(args) | |||
| 	local privateFields = { | |||
| 		theName = true, | |||
| 		theFormat = true, | |||
| 		theFormatFilename = true, | |||
| 		theWidth = true, | |||
| 		theHeight = true, | |||
| 		isUpright = true, | |||
| 		uprightFactor = true, | |||
| 		theLocation = true, | |||
| 		theAlignment = true, | |||
| 		hasBorder = true, | |||
| 		theLink = true, | |||
| 		theAlt = true, | |||
| 		thePage = true, | |||
| 		theClass = true, | |||
| 		theLang = true, | |||
| 		theCaption = true | |||
| 	} | |||
| 	local readOnlyFields = {} | |||
| 	for field in pairs(data) do | |||
| 		readOnlyFields[field] = true | |||
| 	end | |||
| 	readOnlyFields.theName = nil -- This is set if a filename is given to fileLink.new, so remove it. | |||
| 	local function restrictedFieldError(key, restriction) | |||
| 		error(string.format( | |||
| 			"fileLink object field '%s' is %s", | |||
| 			tostring(key), | |||
| 			restriction | |||
| 		), 3) | |||
| 	end | |||
| 	setmetatable(obj, { | |||
| 		__index = function (t, key) | |||
| 			if privateFields[key] then | |||
| 				restrictedFieldError(key, 'private') | |||
| 			else | |||
| 				return data[key] | |||
| 			end | |||
| 		end, | |||
| 		__newindex = function (t, key, value) | |||
| 			if privateFields[key] then | |||
| 				restrictedFieldError(key, 'private') | |||
| 			elseif readOnlyFields[key] then | |||
| 				restrictedFieldError(key, 'read-only') | |||
| 			else | |||
| 				data[key] = value | |||
| 			end | |||
| 		end, | |||
| 		__tostring = function (t) | |||
| 			return t:render() | |||
| 		end, | |||
| 		__pairs = function () | |||
| 			local temp = {} | |||
| 			for k, v in pairs(data) do | |||
| 				if not privateFields[k] then | |||
| 					temp[k] = v | |||
| 				end | |||
| 			end | |||
| 			return pairs(temp) | |||
| 		end | |||
| 	}) | |||
| 	return obj | |||
| end | end | ||
| return  | return p | ||
Latest revision as of 13:41, 1 February 2025
|  | This Lua module is used on approximately 76,000 pages and changes may be widely noticed. Test changes in the module's /sandbox or /testcases subpages, or in your own module sandbox. Consider discussing changes on the talk page before implementing them. | 
|  | This module is subject to page protection. It is a highly visible module in use by a very large number of pages, or is substituted very frequently. Because vandalism or mistakes would affect many pages, and even trivial editing might cause substantial load on the servers, it is protected from editing. | 
|  | This module depends on the following other modules: | 
This module is used to construct wikitext links to files. It is primarily useful for templates and modules that use complicated logic to make file links. Simple file links should be made with wikitext markup directly, as it uses less resources than calling this module. For help with wikitext file markup please refer to the documentation at mediawiki.org.
Usage from wikitext
From wikitext, this module should be called from a template, usually {{file link}}. Please see the template page for documentation. However, it can also be called using the syntax {{#invoke:File link|main|arguments}}.
Usage from Lua
First, you need to import the module.
local mFileLink = require('Module:File link')
Then you can make file links using the _main function.
mFileLink._main(args)
args is a table of arguments that can have the following keys:
- file- the filename. (required)
- format- the file format, e.g. 'thumb', 'thumbnail', 'frame', 'framed', or 'frameless'.
- formatfile- a filename to specify with the 'thumbnail' format option. The filename specified will be used instead of the automatically generated thumbnail.
- border- set this to true or "yes" (or any other value recognized as true by Module:Yesno) to set a border for the image.
- location- the horizontal alignment of the file, e.g. 'right', 'left', 'center', or 'none'.
- alignment- the vertical alignment of the file, e.g. 'baseline', 'middle', 'sub', 'super', 'text-top', 'text-bottom', 'top', or 'bottom'.
- size- the size of the image, e.g. '100px', 'x100px' or '100x100px'.
- upright- the 'upright' parameter, used for setting the size of tall and thin images.
- link- the page that the file should link to. Use the blank string '' to suppress the default link to the file description page.
- alt- the alt text. Use the blank string '' to suppress the default alt text.
- caption- a caption for the file.
- page- sets a page number for multi-paged files such as PDFs.
- class- adds a- classparameter to image links. The MediaWiki software adds this parameter to the- class="..."attribute of the image's- <img />element when the page is rendered into HTML.
- lang- adds a language attribute to specify what language to render the file in.
- start- specifies a start time for audio and video files.
- end- specifies an end time for audio and video files.
- thumbtime- specifies the time to use to generate the thumbnail image for video files.
To see the effect of each of these parameters, see the images help page on mediawiki.org.
Examples
With the file only:
mFileLink.main{file = 'Example.png'}
-- Renders as [[File:Example.png]]
With format, size, link and caption options:
mFileLink.main{
	file = 'Example.png',
	format = 'thumb',
	size = '220px',
	link = 'Wikipedia:Sandbox',
	caption = 'An example.'
}
-- Renders as [[File:Example.png|thumb|220px|link=Wikipedia:Sandbox|An example.]]
With format, size, and border:
mFileLink.main{
	file = 'Example.png',
	format = 'frameless',
	size = '220px',
	border = true
}
-- Renders as [[File:Example.png|frameless|border|220px]]
-- This module provides a library for formatting file wikilinks.
local yesno = require('Module:Yesno')
local checkType = require('libraryUtil').checkType
local p = {}
function p._main(args)
	checkType('_main', 1, args, 'table')
	-- This is basically libraryUtil.checkTypeForNamedArg, but we are rolling our
	-- own function to get the right error level.
	local function checkArg(key, val, level)
		if type(val) ~= 'string' then
			error(string.format(
				"type error in '%s' parameter of '_main' (expected string, got %s)",
				key, type(val)
			), level)
		end
	end
	local ret = {}
	-- Adds a positional parameter to the buffer.
	local function addPositional(key)
		local val = args[key]
		if not val then
			return nil
		end
		checkArg(key, val, 4)
		ret[#ret + 1] = val
	end
	-- Adds a named parameter to the buffer. We assume that the parameter name
	-- is the same as the argument key.
	local function addNamed(key)
		local val = args[key]
		if not val then
			return nil
		end
		checkArg(key, val, 4)
		ret[#ret + 1] = key .. '=' .. val
	end
	-- Filename
	checkArg('file', args.file, 3)
	ret[#ret + 1] = 'File:' .. args.file
	-- Format
	if args.format then
		checkArg('format', args.format)
		if args.formatfile then
			checkArg('formatfile', args.formatfile)
			ret[#ret + 1] = args.format .. '=' .. args.formatfile
		else
			ret[#ret + 1] = args.format
		end
	end
	-- Border
	if yesno(args.border) then
		ret[#ret + 1] = 'border'
	end
	addPositional('location')
	addPositional('alignment')
	addPositional('size')
	addNamed('upright')
	addNamed('link')
	addNamed('alt')
	addNamed('page')
	addNamed('class')
	addNamed('lang')
	addNamed('start')
	addNamed('end')
	addNamed('thumbtime')
	addPositional('caption')
	return string.format('[[%s]]', table.concat(ret, '|'))
end
function p.main(frame)
	local origArgs = require('Module:Arguments').getArgs(frame, {
		wrappers = 'Template:File link'
	})
	if not origArgs.file then
		error("'file' parameter missing from [[Template:File link]]", 0)
	end
	-- Copy the arguments that were passed to a new table to avoid looking up
	-- every possible parameter in the frame object.
	local args = {}
	for k, v in pairs(origArgs) do
		-- Make _BLANK a special argument to add a blank parameter. For use in
		-- conditional templates etc. it is useful for blank arguments to be
		-- ignored, but we still need a way to specify them so that we can do
		-- things like [[File:Example.png|link=]].
		if v == '_BLANK' then
			v = ''
		end
		args[k] = v
	end
	return p._main(args)
end
return p







