Minecraft Wiki
편집 요약 없음
잔글편집 요약 없음
(사용자 4명의 중간 판 27개는 보이지 않습니다)
3번째 줄: 3번째 줄:
 
local args = f
 
local args = f
 
if f == mw.getCurrentFrame() then
 
if f == mw.getCurrentFrame() then
args = require( 'Module:ProcessArgs' ).merge( true )
+
args = require( '모듈:ProcessArgs' ).merge( true )
 
else
 
else
 
f = mw.getCurrentFrame()
 
f = mw.getCurrentFrame()
 
end
 
end
  +
 
local data = args.data and mw.loadData( '모듈:' .. args.data ) or {}
  +
local settings = data.settings
  +
  +
local Autolink = require( '모듈:Autolink' )
 
 
 
-- Default settings
 
-- Default settings
18번째 줄: 23번째 줄:
 
 
 
local defaultStyle = default
 
local defaultStyle = default
if args.settings then
+
if settings then
local settings = mw.loadData( 'Module:' .. args.settings )
 
 
if not settings.stylesheet then
 
if not settings.stylesheet then
 
-- Make a separate clone of the current default settings
 
-- Make a separate clone of the current default settings
34번째 줄: 38번째 줄:
 
 
 
local sprite = mw.html.create( 'span' ):addClass( 'sprite' )
 
local sprite = mw.html.create( 'span' ):addClass( 'sprite' )
sprite:tag( 'br' )
 
 
 
 
-- mw.html's css method performs very slow escaping, which doubles the time it takes
 
-- mw.html's css method performs very slow escaping, which doubles the time it takes
41번째 줄: 44번째 줄:
 
local styles = {}
 
local styles = {}
 
 
  +
if not setting( 'nourl' ) and setting( 'url' ) then
  +
styles[#styles + 1] = 'background-image:' .. ( setting( 'url' ).url or setting( 'url' ) )
 
end
 
if setting( 'stylesheet' ) then
 
if setting( 'stylesheet' ) then
 
sprite:addClass(
 
sprite:addClass(
46번째 줄: 52번째 줄:
 
mw.ustring.lower( setting( 'name' ):gsub( ' ', '-' ) ) .. '-sprite'
 
mw.ustring.lower( setting( 'name' ):gsub( ' ', '-' ) ) .. '-sprite'
 
)
 
)
  +
elseif not setting( 'url' ) then
else
 
table.insert( styles, 'background-image:{{FileUrl|' .. (
+
styles[#styles + 1] = 'background-image:' .. p.getUrl(
 
setting( 'image' ) or setting( 'name' ) .. 'Sprite.png'
 
setting( 'image' ) or setting( 'name' ) .. 'Sprite.png'
) .. '}}' )
+
).url
 
end
 
end
 
local class = setting( 'class' )
 
local class = setting( 'class' )
57번째 줄: 63번째 줄:
 
 
 
local size = setting( 'size' )
 
local size = setting( 'size' )
local pos = math.abs( setting( 'pos' ) ) - 1
 
 
local sheetWidth = setting( 'sheetsize' )
 
local sheetWidth = setting( 'sheetsize' )
 
local tiles = sheetWidth / size
 
local tiles = sheetWidth / size
local left = pos % tiles * size
+
local pos = setting( 'pos' ) - 1
local top = math.floor( pos / tiles ) * size
 
 
local scale = setting( 'scale' )
 
local scale = setting( 'scale' )
 
local autoScale = setting( 'autoscale' )
 
local autoScale = setting( 'autoscale' )
 
 
if left > 0 or top > 0 then
+
if pos then
table.insert( styles, 'background-position:-' .. left * scale .. 'px -' .. top * scale .. 'px' )
+
local left = pos % tiles * size * scale
 
local top = math.floor( pos / tiles ) * size * scale
  +
styles[#styles + 1] = 'background-position:-' .. left .. 'px -' .. top .. 'px'
 
end
 
end
  +
 
if not autoScale and scale ~= defaultStyle.scale then
 
if not autoScale and scale ~= defaultStyle.scale then
table.insert( styles, 'background-size:' .. sheetWidth * scale .. 'px auto' )
+
styles[#styles + 1] = 'background-size:' .. sheetWidth * scale .. 'px auto'
 
end
 
end
 
if size ~= defaultStyle.size or ( not autoScale and scale ~= defaultStyle.scale ) then
 
if size ~= defaultStyle.size or ( not autoScale and scale ~= defaultStyle.scale ) then
table.insert( styles, 'height:' .. size * scale .. 'px' )
+
styles[#styles + 1] = 'height:' .. size * scale .. 'px'
table.insert( styles, 'width:' .. size * scale .. 'px' )
+
styles[#styles + 1] = 'width:' .. size * scale .. 'px'
 
end
 
end
  +
 
local align = setting( 'align' )
 
local align = setting( 'align' )
 
if align ~= defaultStyle.align then
 
if align ~= defaultStyle.align then
table.insert( styles, 'vertical-align:' .. align )
+
styles[#styles + 1] = 'vertical-align:' .. align
 
end
 
end
table.insert( styles, setting( 'css' ) )
+
styles[#styles + 1] = setting( 'css' )
  +
 
sprite:cssText( table.concat( styles, ';' ) )
 
sprite:cssText( table.concat( styles, ';' ) )
 
 
86번째 줄: 95번째 줄:
 
local spriteText
 
local spriteText
 
if text then
 
if text then
 
if not args['wrap'] then
root = mw.html.create( 'span' ):addClass( 'nowrap' )
 
spriteText = mw.html.create( 'span' ):addClass( 'sprite-text' ):wikitext( text )
+
root = mw.html.create( 'span' ):addClass( 'nowrap' )
  +
end
  +
spriteText = mw.html.create( 'span' ):addClass( 'sprite-text' ):wikitext( Autolink.invlink(text, 'nolink', args.data) )
 
end
 
end
 
 
112번째 줄: 123번째 줄:
 
-- Internal link
 
-- Internal link
 
local linkPrefix = setting( 'linkprefix' ) or ''
 
local linkPrefix = setting( 'linkprefix' ) or ''
return '[[' .. linkPrefix .. link .. '|' .. tostring( root ) .. ']]'
+
return '[[' .. linkPrefix .. Autolink.invlink(link, 'linkonly', args.data) .. '|' .. tostring( root ) .. ']]'
 
end
 
end
 
 
121번째 줄: 132번째 줄:
 
local args = f
 
local args = f
 
if f == mw.getCurrentFrame() then
 
if f == mw.getCurrentFrame() then
args = require( 'Module:ProcessArgs' ).merge( true )
+
args = require( '모듈:ProcessArgs' ).merge( true )
 
else
 
else
 
f = mw.getCurrentFrame()
 
f = mw.getCurrentFrame()
 
end
 
end
 
 
  +
local Reverselink = require( '모듈:Reverselink' )
  +
  +
local data = args.data and mw.loadData( '모듈:' .. args.data ) or {}
 
local categories = {}
 
local categories = {}
 
local idData = args.iddata
 
local idData = args.iddata
 
if not idData then
 
if not idData then
local default = {}
+
local name = args.name or data.settings.name
 
local id = mw.text.trim( tostring( Reverselink.xlink( args[1] ) or '' ) )
if args.settings then
 
 
idData = data.ids[id] or data.ids[mw.ustring.lower( id ):gsub( '[%s%+]', '-' )]
default = mw.loadData( 'Module:' .. args.settings )
 
end
 
 
local name = args.name or default.name
 
local ids = mw.loadData( 'Module:' .. ( args.ids or default.ids or name .. '/IDs' ) )
 
ids = ids.ids or ids
 
local id = mw.text.trim( tostring( args[1] or '' ) )
 
idData = ids[id] or ids[mw.ustring.lower( id ):gsub( '[%s%+]', '-' )]
 
 
end
 
end
 
 
145번째 줄: 152번째 줄:
 
local disallowCats = args.nocat or title.isTalkPage or title.nsText:find( '^User' )
 
local disallowCats = args.nocat or title.isTalkPage or title.nsText:find( '^User' )
 
if idData then
 
if idData then
if type( idData ) == 'table' then
+
if idData.deprecated then
  +
args.class = ( args.class or '' ) .. ' sprite-deprecated'
if idData.deprecated and not disallowCats then
+
if not disallowCats then
table.insert( categories, f:expandTemplate{ title = 'Translation category', args = { 'Pages using deprecated sprite names', project = 0 } } )
+
categories[#categories + 1] = f:expandTemplate{ title = 'Translation category', args = { '잘못된 스프라이트 이름을 사용하는 문서', project = 0 } }
 
end
 
end
 
args.pos = idData.pos
 
else
 
args.pos = idData
 
 
end
 
end
 
 
args.pos = idData.pos
 
elseif not disallowCats then
 
elseif not disallowCats then
table.insert( categories, f:expandTemplate{ title = 'Translation category', args = { 'Pages with missing sprites', project = 0 } } )
+
categories[#categories + 1] = f:expandTemplate{ title = 'Translation category', args = { '스프라이트가 없는 문서', project = 0 } }
 
end
 
end
 
 
return p.base( args ), table.concat( categories, '' )
+
return p.base( args ), table.concat( categories )
 
end
 
end
   
164번째 줄: 170번째 줄:
 
local args = f
 
local args = f
 
if f == mw.getCurrentFrame() then
 
if f == mw.getCurrentFrame() then
args = require( 'Module:ProcessArgs' ).merge( true )
+
args = require( '모듈:ProcessArgs' ).merge( true )
 
end
 
end
 
 
171번째 줄: 177번째 줄:
 
link = args[1]:match( '^(.-)%+' ) or args[1]
 
link = args[1]:match( '^(.-)%+' ) or args[1]
 
end
 
end
local text = args.text or args[2] or link
+
local text
 
if not args.notext then
  +
text = args.text or args[2] or link
  +
end
 
 
 
args[1] = args.id or args[1]
 
args[1] = args.id or args[1]
178번째 줄: 187번째 줄:
 
 
 
return p.sprite( args )
 
return p.sprite( args )
  +
end
  +
  +
function p.getUrl( image, query, classname )
  +
local f = mw.getCurrentFrame()
  +
local t = {
  +
url = f:expandTemplate{
  +
title = 'FileUrl',
  +
args = { image, query = query }
 
},
  +
}
  +
if classname and classname ~= '' then
  +
t.style = f:expandTemplate{
  +
title = 'FileUrlStyle',
  +
args = { classname, image, query = query }
 
}
  +
end
  +
return t
  +
end
  +
  +
function p.getParsedUrlStyle( f )
  +
local args = f:getParent().args
  +
local module = args[1]
  +
return require( 'Module:' .. module ).settings.url.style
 
end
 
end
   
187번째 줄: 219번째 줄:
 
f = mw.getCurrentFrame()
 
f = mw.getCurrentFrame()
 
end
 
end
local settingsPage = mw.text.trim( args[1] )
+
local dataPage = mw.text.trim( args[1] )
local settings = mw.loadData( 'Module:' .. settingsPage )
+
local data = mw.loadData( '모듈:' .. dataPage )
local idsPage = 'Module:' .. ( settings.ids or settings.name .. '/IDs' )
 
 
 
 
local getProtection = function( title, action, extra )
 
local getProtection = function( title, action, extra )
 
local protections = { 'edit' }
 
local protections = { 'edit' }
 
if extra then
 
if extra then
table.insert( protections, extra )
+
protections[#protections + 1] = extra
 
end
 
end
 
 
204번째 줄: 235번째 줄:
 
end
 
end
 
 
table.insert( protections, protection )
+
protections[#protections + 1] = protection
 
end
 
end
 
 
local direct = title.protectionLevels[action]
+
local direct = title.protectionLevels[action] or {}
 
for _, protection in ipairs( direct ) do
 
for _, protection in ipairs( direct ) do
 
addProtection( protection )
 
addProtection( protection )
213번째 줄: 244번째 줄:
 
local cascading = title.cascadingProtection.restrictions[action] or {}
 
local cascading = title.cascadingProtection.restrictions[action] or {}
 
if #cascading > 0 then
 
if #cascading > 0 then
table.insert( protections, 'protect' )
+
protections[#protections + 1] = 'protect'
 
end
 
end
 
for _, protection in ipairs( cascading ) do
 
for _, protection in ipairs( cascading ) do
222번째 줄: 253번째 줄:
 
end
 
end
 
 
local body
+
local spriteStyle = ''
  +
if data.settings.url and data.settings.url.style then
if args.refresh then
 
  +
spriteStyle = data.settings.url.style
body = mw.html.create( '' )
 
else
 
local idsTitle = mw.title.new( idsPage )
 
local spritesheet = settings.image or settings.name .. 'Sprite.png'
 
local spriteTitle = mw.title.new( 'File:' .. spritesheet )
 
local idsProtection = getProtection( idsTitle, 'edit' )
 
local spriteProtection = getProtection( spriteTitle, 'upload', 'upload,reupload' )
 
body = mw.html.create( 'div' ):attr( {
 
id = 'spritedoc',
 
['data-idspage'] = idsTitle.id,
 
['data-idsprotection'] = idsProtection,
 
['data-idstimestamp'] = f:callParserFunction( 'REVISIONTIMESTAMP', idsPage ),
 
['data-spritesheet'] = spritesheet,
 
['data-spriteprotection'] = spriteProtection,
 
['data-pos'] = settings.pos or 1,
 
['data-refreshtext'] = mw.text.nowiki( '{{#invoke:sprite|doc|' .. settingsPage .. '|refresh=1}}' )
 
} )
 
 
end
 
end
 
 
local data = mw.loadData( idsPage )
+
local dataTitle = mw.title.new( '모듈:' .. dataPage )
  +
-- Temporary until this is updated
  +
local classname = ''
  +
if data.settings.stylesheet then
  +
classname = data.settings.classname or
  +
mw.ustring.lower( data.settings.name:gsub( ' ', '-' ) ) .. '-sprite'
  +
end
 
local spritesheet = data.settings.image or data.settings.name .. 'Sprite.png'
 
local spriteTitle = mw.title.new( 'ファイル:' .. spritesheet )
 
local dataProtection = getProtection( dataTitle, 'edit' )
 
local spriteProtection = getProtection( spriteTitle, 'upload', 'upload,reupload' )
 
local body = mw.html.create( 'div' ):attr( {
 
id = 'spritedoc',
  +
['data-dataprotection'] = dataProtection,
 
['data-datatimestamp'] = f:callParserFunction( 'REVISIONTIMESTAMP', '모듈:' .. dataPage ),
  +
['data-datapage'] = '모듈:' .. dataPage,
 
['data-spritesheet'] = spritesheet,
 
['data-spriteprotection'] = spriteProtection,
  +
['data-urlfunc'] = "require( [[모듈:Sprite]] ).getUrl( '" .. spritesheet .. "', '$1', '" .. classname .. "' )",
 
['data-refreshtext'] = mw.text.nowiki( '{{#invoke:sprite|doc|' .. dataPage .. '|refresh=1}}' ),
  +
['data-settings'] = mw.text.jsonEncode( data.settings ),
  +
} )
 
 
 
local sections = {}
 
local sections = {}
for _, sectionData in ipairs( data.sections or { 'Uncategorized' } ) do
+
for _, sectionData in ipairs( data.sections or { name = 'Uncategorized' } ) do
 
local sectionTag = body:tag( 'div' ):addClass( 'spritedoc-section' ):attr( 'data-section-id', sectionData.id )
 
local sectionTag = body:tag( 'div' ):addClass( 'spritedoc-section' ):attr( 'data-section-id', sectionData.id )
 
sectionTag:tag( 'h3' ):wikitext( sectionData.name )
-- https://phabricator.wikimedia.org/T73594
 
sectionTag:wikitext( '<h3>', sectionData[1], '</h3>' )
 
 
sections[sectionData.id] = { boxes = sectionTag:tag( 'ul' ):addClass( 'spritedoc-boxes' ) }
 
sections[sectionData.id] = { boxes = sectionTag:tag( 'ul' ):addClass( 'spritedoc-boxes' ) }
 
end
 
end
 
 
 
local keyedData = {}
 
local keyedData = {}
  +
local i = 1
 
for name, idData in pairs( data.ids ) do
 
for name, idData in pairs( data.ids ) do
table.insert( keyedData, {
+
keyedData[i] = {
 
sortKey = mw.ustring.lower( name ),
 
sortKey = mw.ustring.lower( name ),
 
name = name,
 
name = name,
 
data = idData
 
data = idData
} )
+
}
  +
i = i + 1
 
end
 
end
 
table.sort( keyedData, function( a, b )
 
table.sort( keyedData, function( a, b )
273번째 줄: 310번째 줄:
 
local box = section.boxes:tag( 'li' ):addClass( 'spritedoc-box' ):attr( 'data-pos', pos )
 
local box = section.boxes:tag( 'li' ):addClass( 'spritedoc-box' ):attr( 'data-pos', pos )
 
box:tag( 'div' ):addClass( 'spritedoc-image' )
 
box:tag( 'div' ):addClass( 'spritedoc-image' )
:wikitext( p.base{ pos = pos, settings = settingsPage } )
+
:wikitext( p.base{ pos = pos, data = dataPage, nourl = spriteStyle ~= '' } )
 
 
 
names = box:tag( 'ul' ):addClass( 'spritedoc-names' )
 
names = box:tag( 'ul' ):addClass( 'spritedoc-names' )
288번째 줄: 325번째 줄:
 
 
 
if args.refresh then
 
if args.refresh then
return '', tostring( body )
+
return '', '', tostring( body )
 
end
 
end
return f:callParserFunction( '#widget:Stylesheet', { page = 'SpriteDoc' } ), tostring( body )
+
local styles = f:callParserFunction( '#widget:SpriteDoc.css' )
  +
return styles, spriteStyle, tostring( body )
 
end
 
end
 
return p
 
return p

2020년 6월 30일 (화) 14:51 판

[만들기 | 역사 | 캐시 제거]설명문서
이 모듈의 설명문서가 없습니다. 이 모듈의 사용 방법을 안다면, 설명문서를 만들어 주십시오.
local p = {}
function p.base( f )
	local args = f
	if f == mw.getCurrentFrame() then 
		args = require( '모듈:ProcessArgs' ).merge( true )
	else
		f = mw.getCurrentFrame()
	end
	
	local data = args.data and mw.loadData( '모듈:' .. args.data ) or {}
	local settings = data.settings
	
	local Autolink = require( '모듈:Autolink' )
	
	-- Default settings
	local default = {
		scale = 1,
		sheetsize = 256,
		size = 16,
		pos = 1,
		align = 'text-top'
	}
	
	local defaultStyle = default
	if settings then
		if not settings.stylesheet then
			-- Make a separate clone of the current default settings
			defaultStyle = mw.clone( default )
		end
		for k, v in pairs( settings ) do
			default[k] = v
		end
	end
	
	local setting = function( arg )
		return args[arg] or default[arg]
	end
	
	local sprite = mw.html.create( 'span' ):addClass( 'sprite' )
	
	-- mw.html's css method performs very slow escaping, which doubles the time it takes
	-- to run, so we'll construct the styles manually, and put them in the cssText
	-- method, which only does html escaping (which isn't slow)
	local styles = {}
	
	if not setting( 'nourl' ) and setting( 'url' ) then
		styles[#styles + 1] = 'background-image:' .. ( setting( 'url' ).url or setting( 'url' ) )
	end
	if setting( 'stylesheet' ) then
		sprite:addClass(
			setting( 'classname' ) or
			mw.ustring.lower( setting( 'name' ):gsub( ' ', '-' ) ) .. '-sprite'
		)
	elseif not setting( 'url' ) then
		styles[#styles + 1] = 'background-image:' .. p.getUrl(
			setting( 'image' ) or setting( 'name' ) .. 'Sprite.png'
		).url
	end
	local class = setting( 'class' )
	if class then
		sprite:addClass( class )
	end
	
	local size = setting( 'size' )
	local sheetWidth = setting( 'sheetsize' )
	local tiles = sheetWidth / size
	local pos = setting( 'pos' ) - 1
	local scale = setting( 'scale' )
	local autoScale = setting( 'autoscale' )
	
	if pos then
		local left = pos % tiles * size * scale
		local top = math.floor( pos / tiles ) * size * scale
		styles[#styles + 1] = 'background-position:-' .. left .. 'px -' .. top .. 'px'
	end
	
	if not autoScale and scale ~= defaultStyle.scale then
		styles[#styles + 1] = 'background-size:' .. sheetWidth * scale .. 'px auto'
	end
	if size ~= defaultStyle.size or ( not autoScale and scale ~= defaultStyle.scale ) then
		styles[#styles + 1] = 'height:' .. size * scale .. 'px'
		styles[#styles + 1] = 'width:' .. size * scale .. 'px'
	end
	
	local align = setting( 'align' )
	if align ~= defaultStyle.align then
		styles[#styles + 1] = 'vertical-align:' .. align
	end
	styles[#styles + 1] = setting( 'css' )
	
	sprite:cssText( table.concat( styles, ';' ) )
	
	local text = setting( 'text' )
	local root
	local spriteText
	if text then
		if not args['wrap'] then
			root = mw.html.create( 'span' ):addClass( 'nowrap' )
		end
		spriteText = mw.html.create( 'span' ):addClass( 'sprite-text' ):wikitext( Autolink.invlink(text, 'nolink', args.data) )
	end
	
	local title = setting( 'title' )
	if title then
		( root or sprite ):attr( 'title', title )
	end
	
	if not root then
		root = mw.html.create( '' )
	end
	root:node( sprite )
	if spriteText then
		root:node( spriteText )
	end
	
	local link = setting( 'link' ) or ''
	if link ~= '' and mw.ustring.lower( link ) ~= 'none' then
		-- External link
		if link:find( '//' ) then
			return '[' .. link .. ' ' .. tostring( root ) .. ']'
		end
		
		-- Internal link
		local linkPrefix = setting( 'linkprefix' ) or ''
		return '[[' .. linkPrefix .. Autolink.invlink(link, 'linkonly', args.data) .. '|' .. tostring( root ) .. ']]'
	end
	
	return tostring( root )
end

function p.sprite( f )
	local args = f
	if f == mw.getCurrentFrame() then
		args = require( '모듈:ProcessArgs' ).merge( true )
	else
		f = mw.getCurrentFrame()
	end
	
	local Reverselink = require( '모듈:Reverselink' )
	
	local data = args.data and mw.loadData( '모듈:' .. args.data ) or {}
	local categories = {}
	local idData = args.iddata
	if not idData then
		local name = args.name or data.settings.name
		local id = mw.text.trim( tostring( Reverselink.xlink( args[1] ) or '' ) )
		idData = data.ids[id] or data.ids[mw.ustring.lower( id ):gsub( '[%s%+]', '-' )]
	end
	
	local title = mw.title.getCurrentTitle()
	-- Remove categories on language pages, talk pages, and in User/UserWiki/UserProfile namespaces
	local disallowCats = args.nocat or title.isTalkPage or title.nsText:find( '^User' )
	if idData then
		if idData.deprecated then
			args.class = ( args.class or '' ) .. ' sprite-deprecated'
			if not disallowCats then
				categories[#categories + 1] = f:expandTemplate{ title = 'Translation category', args = { '잘못된 스프라이트 이름을 사용하는 문서', project = 0 } }
			end
		end
		
		args.pos = idData.pos
	elseif not disallowCats then
		categories[#categories + 1] = f:expandTemplate{ title = 'Translation category', args = { '스프라이트가 없는 문서', project = 0 } }
	end
	
	return p.base( args ), table.concat( categories )
end

function p.link( f )
	local args = f
	if f == mw.getCurrentFrame() then
		args = require( '모듈:ProcessArgs' ).merge( true )
	end
	
	local link = args[1]
	if args[1] and not args.id then
		link = args[1]:match( '^(.-)%+' ) or args[1]
	end
	local text
	if not args.notext then
		text = args.text or args[2] or link
	end
	
	args[1] = args.id or args[1]
	args.link = args.link or link
	args.text = text
	
	return p.sprite( args )
end

function p.getUrl( image, query, classname )
	local f = mw.getCurrentFrame()
	local t = {
		url = f:expandTemplate{
			title = 'FileUrl',
			args = { image, query = query }
		},
	}
	if classname and classname ~= '' then
		t.style = f:expandTemplate{
			title = 'FileUrlStyle',
			args = { classname, image, query = query }
		}
	end
	return t
end

function p.getParsedUrlStyle( f )
	local args = f:getParent().args
	local module = args[1]
	return require( 'Module:' .. module ).settings.url.style
end

function p.doc( f )
	local args = f
	if f == mw.getCurrentFrame() then
		args = f.args
	else
		f = mw.getCurrentFrame()
	end
	local dataPage = mw.text.trim( args[1] )
	local data = mw.loadData( '모듈:' .. dataPage )
	
	local getProtection = function( title, action, extra )
		local protections = { 'edit' }
		if extra then
			protections[#protections + 1] = extra
		end
		
		local addProtection = function( protection )
			if protection == 'autoconfirmed' then
				protection = 'editsemiprotected'
			elseif protection == 'sysop' then
				protection = 'editprotected'
			end
			
			protections[#protections + 1] = protection
		end
		
		local direct = title.protectionLevels[action] or {}
		for _, protection in ipairs( direct ) do
			addProtection( protection )
		end
		local cascading = title.cascadingProtection.restrictions[action] or {}
		if #cascading > 0 then
			protections[#protections + 1] = 'protect'
		end
		for _, protection in ipairs( cascading ) do
			addProtection( protection )
		end
		
		return table.concat( protections, ',' )
	end
	
	local spriteStyle = ''
	if data.settings.url and data.settings.url.style then
		spriteStyle = data.settings.url.style
	end
	
	local dataTitle = mw.title.new( '모듈:' .. dataPage )
	-- Temporary until this is updated
	local classname = ''
	if data.settings.stylesheet then
		classname = data.settings.classname or
			mw.ustring.lower( data.settings.name:gsub( ' ', '-' ) ) .. '-sprite'
	end
	local spritesheet = data.settings.image or data.settings.name .. 'Sprite.png'
	local spriteTitle = mw.title.new( 'ファイル:' .. spritesheet )
	local dataProtection = getProtection( dataTitle, 'edit' )
	local spriteProtection = getProtection( spriteTitle, 'upload', 'upload,reupload' )
	local body = mw.html.create( 'div' ):attr( {
		id = 'spritedoc',
		['data-dataprotection'] = dataProtection,
		['data-datatimestamp'] = f:callParserFunction( 'REVISIONTIMESTAMP', '모듈:' .. dataPage ),
		['data-datapage'] = '모듈:' .. dataPage,
		['data-spritesheet'] = spritesheet,
		['data-spriteprotection'] = spriteProtection,
		['data-urlfunc'] = "require( [[모듈:Sprite]] ).getUrl( '" .. spritesheet .. "', '$1', '" .. classname .. "' )",
		['data-refreshtext'] = mw.text.nowiki( '{{#invoke:sprite|doc|' .. dataPage .. '|refresh=1}}' ),
		['data-settings'] = mw.text.jsonEncode( data.settings ),
	} )
	
	local sections = {}
	for _, sectionData in ipairs( data.sections or { name = 'Uncategorized' } ) do
		local sectionTag = body:tag( 'div' ):addClass( 'spritedoc-section' ):attr( 'data-section-id', sectionData.id )
		sectionTag:tag( 'h3' ):wikitext( sectionData.name )
		sections[sectionData.id] = { boxes = sectionTag:tag( 'ul' ):addClass( 'spritedoc-boxes' ) }
	end
	
	local keyedData = {}
	local i = 1
	for name, idData in pairs( data.ids ) do
		keyedData[i] = {
			sortKey = mw.ustring.lower( name ),
			name = name,
			data = idData
		}
		i = i + 1
	end
	table.sort( keyedData, function( a, b )
		return a.sortKey < b.sortKey
	end )
	
	for _, data in ipairs( keyedData ) do
		local idData = data.data
		local pos = idData.pos
		local section = sections[idData.section]
		local names = section[pos]
		if not names then
			local box = section.boxes:tag( 'li' ):addClass( 'spritedoc-box' ):attr( 'data-pos', pos )
			box:tag( 'div' ):addClass( 'spritedoc-image' )
				:wikitext( p.base{ pos = pos, data = dataPage, nourl = spriteStyle ~= '' } )
			
			names = box:tag( 'ul' ):addClass( 'spritedoc-names' )
			section[pos] = names
		end
		local nameElem = mw.html.create( 'li' ):addClass( 'spritedoc-name' )
		local codeElem = nameElem:tag( 'code' ):wikitext( data.name )
		
		if idData.deprecated then
			codeElem:addClass( 'spritedoc-deprecated' )
		end
		names:wikitext( tostring( nameElem ) )
	end
	
	if args.refresh then
		return '', '', tostring( body )
	end
	local styles = f:callParserFunction( '#widget:SpriteDoc.css' )
	return styles, spriteStyle, tostring( body )
end
return p