Модуль:Multiple image: различия между версиями
Перейти к навигации
Перейти к поиску
Drakon721 (обсуждение | вклад) м 1 версия импортирована |
Drakon721 (обсуждение | вклад) Нет описания правки |
||
Строка 1: | Строка 1: | ||
-- implements [[template:multiple image]] | -- implements [[template:multiple image]] | ||
local p = {} | local p = {} | ||
local function isnotempty(s) | local function isnotempty(s) | ||
Строка 9: | Строка 6: | ||
end | end | ||
local function renderImageCell(image, width, height, link, alt, thumbtime, caption, textalign, istyle) | |||
local function renderImageCell(image, width, height, link, alt, thumbtime, caption, textalign, istyle) | |||
local root = mw.html.create('') | local root = mw.html.create('') | ||
local altstr = '|alt=' .. (alt or '') | local altstr = '|alt=' .. (alt or '') | ||
local linkstr = link and ('|link=' .. link) or '' | local linkstr = link and ('|link=' .. link) or '' | ||
Строка 34: | Строка 14: | ||
local thumbtimestr = '' | local thumbtimestr = '' | ||
if isnotempty( thumbtime ) then | if isnotempty( thumbtime ) then | ||
thumbtimestr = '|thumbtime=' .. thumbtime | thumbtimestr = '|thumbtime=' .. thumbtime | ||
end | end | ||
local imagediv = root:tag('div') | local imagediv = root:tag('div') | ||
imagediv:addClass('thumbimage') | imagediv:addClass('thumbimage') | ||
Строка 52: | Строка 29: | ||
local captiondiv = root:tag('div') | local captiondiv = root:tag('div') | ||
captiondiv:addClass('thumbcaption') | captiondiv:addClass('thumbcaption') | ||
captiondiv:css('clear', 'left') | |||
if isnotempty(textalign) then | if isnotempty(textalign) then | ||
captiondiv: | captiondiv:css('text-align', textalign) | ||
end | end | ||
captiondiv:wikitext(caption) | captiondiv:wikitext(caption) | ||
Строка 86: | Строка 64: | ||
ic = ic - thisrow | ic = ic - thisrow | ||
r = r + 1 | r = r + 1 | ||
-- use the previous if the next is missing and | -- use the previous if the next is missing and | ||
-- make sure we don't overstep the number of images | -- make sure we don't overstep the number of images | ||
thisrow = math.min(tonumber(pr[r] or thisrow) or ic, ic) | thisrow = math.min(tonumber(pr[r] or thisrow) or ic, ic) | ||
Строка 96: | Строка 74: | ||
local pargs = frame:getParent().args | local pargs = frame:getParent().args | ||
local args = frame.args | local args = frame.args | ||
local width = | local width = pargs['width'] or '' | ||
local dir = pargs['direction'] or '' | local dir = pargs['direction'] or '' | ||
local align = pargs['align'] or args['align'] or '' | |||
local align = pargs['align'] or args['align'] | |||
local capalign = pargs['caption_align'] or args['caption_align'] or '' | local capalign = pargs['caption_align'] or args['caption_align'] or '' | ||
local totalwidth = | local totalwidth = pargs['total_width'] or args['total_width'] or '' | ||
local imgstyle = pargs['image_style'] or args['image_style'] | local imgstyle = pargs['image_style'] or args['image_style'] | ||
local header = pargs['header'] or pargs['title'] or '' | local header = pargs['header'] or pargs['title'] or '' | ||
Строка 128: | Строка 105: | ||
-- sort the imagenumbers | -- sort the imagenumbers | ||
table.sort(imagenumbers) | table.sort(imagenumbers) | ||
-- create an array with the number of images per row | -- create an array with the number of images per row | ||
perrow = getPerRow(dir == 'vertical' and '1' or pargs['perrow'], imagecount) | perrow = getPerRow(dir == 'vertical' and '1' or pargs['perrow'], imagecount) | ||
-- compute the number of rows | -- compute the number of rows | ||
local rowcount = #perrow | local rowcount = #perrow | ||
-- store the image widths and compute row widths and maximum row width | -- store the image widths and compute row widths and maximum row width | ||
local widths = {} | local widths = {} | ||
local widthmax = 0 | local widthmax = 0 | ||
Строка 147: | Строка 123: | ||
if( k <= imagecount ) then | if( k <= imagecount ) then | ||
local i = imagenumbers[k] | local i = imagenumbers[k] | ||
widths[k] = getWidth(width, pargs['width' .. i]) | |||
widthsum[r] = widthsum[r] + widths[k] | widthsum[r] = widthsum[r] + widths[k] | ||
end | end | ||
Строка 160: | Строка 132: | ||
-- make sure the gap is non-negative | -- make sure the gap is non-negative | ||
if imagegap < 0 then imagegap = 0 end | if imagegap < 0 then imagegap = 0 end | ||
-- if total_width has been specified, rescale the image widths | -- if total_width has been specified, rescale the image widths | ||
local heights = {} | |||
if( isnotempty(totalwidth) ) then | if( isnotempty(totalwidth) ) then | ||
totalwidth = tonumber(totalwidth) | totalwidth = tonumber(totalwidth) | ||
Строка 175: | Строка 148: | ||
if( k<= imagecount ) then | if( k<= imagecount ) then | ||
local i = imagenumbers[k] | local i = imagenumbers[k] | ||
local h = | local h = tonumber( pargs['height' .. i] or '' ) or 0 | ||
if (h > 0) then | if (h > 0) then | ||
ar[j] = widths[k]/h | ar[j] = widths[k]/h | ||
Строка 202: | Строка 175: | ||
widthmax = math.max(widthmax, widthsum[r]) | widthmax = math.max(widthmax, widthsum[r]) | ||
end | end | ||
end | end | ||
-- start building the array of images, if there are images | -- start building the array of images, if there are images | ||
Строка 221: | Строка 194: | ||
root:addClass('thumb') | root:addClass('thumb') | ||
root:addClass('tmulti') | root:addClass('tmulti') | ||
root:addClass(thumbclass[align] or 'tright') | root:addClass(thumbclass[align] or 'tright') | ||
if( align == 'center' or align == 'centre' ) then | if( align == 'center' or align == 'centre' ) then | ||
root:addClass('center') | root:addClass('center') | ||
end | |||
if( pargs['margin_top'] or args['margin_top']) then | |||
root:css('margin-top', pargs['margin_top'] or args['margin_top']) | |||
end | |||
if( pargs['margin_bottom'] or args['margin_bottom']) then | |||
root:css('margin-bottom', pargs['margin_bottom'] or args['margin_bottom']) | |||
end | end | ||
if( bg ~= '' ) then | if( bg ~= '' ) then | ||
Строка 237: | Строка 215: | ||
if( bg ~= '' ) then | if( bg ~= '' ) then | ||
div:css('background-color', bg) | div:css('background-color', bg) | ||
end | end | ||
-- add the header | -- add the header | ||
if( isnotempty(header) ) then | if( isnotempty(header) ) then | ||
div:tag('div') | div:tag('div') | ||
: | :css('clear', 'both') | ||
: | :css('font-weight', 'bold') | ||
:css('text-align', pargs['header_align'] or 'center') | |||
:css('background-color', pargs['header_background'] or 'transparent') | |||
:wikitext(header) | |||
end | end | ||
-- loop through the images | -- loop through the images | ||
local k = 0 | local k = 0 | ||
for r=1,rowcount do | for r=1,rowcount do | ||
for j=1,perrow[r] do | for j=1,perrow[r] do | ||
k = k + 1 | k = k + 1 | ||
if( k <= imagecount ) then | if( k <= imagecount ) then | ||
local imagediv = | local imagediv = div:tag('div') | ||
imagediv:addClass('tsingle') | imagediv:addClass('tsingle') | ||
if dir ~= 'vertical' then | |||
imagediv:css('float', 'left') | |||
end | |||
if bg ~= '' then | if bg ~= '' then | ||
imagediv:css('background-color', bg); | imagediv:css('background-color', bg); | ||
end | end | ||
imagediv:css('margin', '1px') | |||
if ((imagegap > 1) and (j < perrow[r])) then | if ((imagegap > 1) and (j < perrow[r])) then | ||
imagediv:css('margin-right', tostring(imagegap) .. 'px') | imagediv:css('margin-right', tostring(imagegap) .. 'px') | ||
Строка 271: | Строка 248: | ||
imagediv:css('width', tostring(2 + w) .. 'px') | imagediv:css('width', tostring(2 + w) .. 'px') | ||
:css('max-width', tostring(2 + w) .. 'px') | :css('max-width', tostring(2 + w) .. 'px') | ||
imagediv:wikitext(renderImageCell(img, w, heights[k], | imagediv:wikitext(renderImageCell(img, w, heights[k], | ||
pargs['link' .. i], pargs['alt' .. i], | pargs['link' .. i], pargs['alt' .. i], | ||
pargs['thumbtime' .. i], pargs['caption' .. i], capalign, imgstyle)) | pargs['thumbtime' .. i], pargs['caption' .. i], capalign, imgstyle)) | ||
end | end | ||
end | |||
-- only float content gives a parent height:0, so add a clearing div | |||
if dir ~= 'vertical' then | |||
div:tag('div') | |||
:css('clear', 'left') | |||
end | end | ||
end | end | ||
-- add the footer | -- add the footer | ||
if( isnotempty(footer) ) then | if( isnotempty(footer) ) then | ||
div:tag('div') | div:tag('div') | ||
:addClass(' | :addClass('thumbcaption') | ||
:css(' | :css('clear', 'left') | ||
:css('text-align', pargs['footer_align'] or args['footer_align'] or 'left') | |||
:css('background-color', pargs['footer_background'] or 'transparent') | |||
:wikitext(footer) | |||
end | end | ||
return tostring(root) | return tostring(root) | ||
Строка 296: | Строка 274: | ||
function p.render( frame ) | function p.render( frame ) | ||
return renderMultipleImages( frame ) | |||
end | end | ||
return p | return p |
Текущая версия от 20:57, 6 июля 2022
Для документации этого модуля может быть создана страница Модуль:Multiple image/doc
-- implements [[template:multiple image]]
local p = {}
local function isnotempty(s)
return s and s:match( '^%s*(.-)%s*$' ) ~= ''
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 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')
captiondiv:css('clear', 'left')
if isnotempty(textalign) then
captiondiv:css('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 = pargs['width'] or ''
local dir = pargs['direction'] or ''
local align = pargs['align'] or args['align'] or ''
local capalign = pargs['caption_align'] or args['caption_align'] or ''
local totalwidth = 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 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]
widths[k] = getWidth(width, pargs['width' .. i])
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
local heights = {}
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 = tonumber( pargs['height' .. i] or '' ) 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(thumbclass[align] or 'tright')
if( align == 'center' or align == 'centre' ) then
root:addClass('center')
end
if( pargs['margin_top'] or args['margin_top']) then
root:css('margin-top', pargs['margin_top'] or args['margin_top'])
end
if( pargs['margin_bottom'] or args['margin_bottom']) then
root:css('margin-bottom', pargs['margin_bottom'] or args['margin_bottom'])
end
if( bg ~= '' ) then
root:css('background-color', bg)
end
local div = root:tag('div')
div:addClass('thumbinner')
div:css('width', tostring(bodywidth) .. 'px')
:css('max-width', tostring(bodywidth) .. 'px')
if( bg ~= '' ) then
div:css('background-color', bg)
end
-- add the header
if( isnotempty(header) ) then
div:tag('div')
:css('clear', 'both')
:css('font-weight', 'bold')
:css('text-align', pargs['header_align'] or 'center')
:css('background-color', pargs['header_background'] or 'transparent')
:wikitext(header)
end
-- loop through the images
local k = 0
for r=1,rowcount do
for j=1,perrow[r] do
k = k + 1
if( k <= imagecount ) then
local imagediv = div:tag('div')
imagediv:addClass('tsingle')
if dir ~= 'vertical' then
imagediv:css('float', 'left')
end
if bg ~= '' then
imagediv:css('background-color', bg);
end
imagediv:css('margin', '1px')
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
-- only float content gives a parent height:0, so add a clearing div
if dir ~= 'vertical' then
div:tag('div')
:css('clear', 'left')
end
end
-- add the footer
if( isnotempty(footer) ) then
div:tag('div')
:addClass('thumbcaption')
:css('clear', 'left')
:css('text-align', pargs['footer_align'] or args['footer_align'] or 'left')
:css('background-color', pargs['footer_background'] or 'transparent')
:wikitext(footer)
end
return tostring(root)
end
return ''
end
function p.render( frame )
return renderMultipleImages( frame )
end
return p