Module:Portal bar/sandbox: Difference between revisions

Content deleted Content added
sync from live
better implementation of nbsp
 
(79 intermediate revisions by 6 users not shown)
Line 1:
-- This module implements {{portal bar}}.
 
require('Module:No globalsstrict')
 
-- determine whether we're being called from a sandbox
local p = {}
local isSandbox = mw.getCurrentFrame():getTitle():find('sandbox', 1, true)
local function checkPortalExists(portal)
local sandbox = isSandbox and '/sandbox' or ''
return not (mw.title.makeTitle(100, portal).id == 0)
 
end
local getImageNameportalModule = require( 'Module:Portal' ..sandbox).image
local getImageName = portalModule._image
local checkPortals = portalModule._checkPortals
local processPortalArgs = portalModule._processPortalArgs
local yesno = require( 'Module:Yesno' )
local getArgs = require('Module:Arguments').getArgs
local p = {}
 
local function sandboxVersion(s)
local trackingEnabled = true
return isSandbox and s.."-sand" or s
 
-- Check whether to do tracking in this namespace
-- Returns true unless the page is one of the banned namespaces
local function checkTrackingNamespace()
local thisPage = mw.title.getCurrentTitle()
if (thisPage.namespace == 1) -- Talk
or (thisPage.namespace == 2) -- User
or (thisPage.namespace == 3) -- User talk
or (thisPage.namespace == 5) -- Wikipedia talk
or (thisPage.namespace == 7) -- File talk
or (thisPage.namespace == 11) -- Template talk
or (thisPage.namespace == 15) -- Category talk
or (thisPage.namespace == 101) -- Portal talk
or (thisPage.namespace == 118) -- Draft
or (thisPage.namespace == 119) -- Draft talk
or (thisPage.namespace == 829) -- Module talk
then
return false
end
return true
end
 
-- Check whether to do tracking on this pagename
-- Returns false if the page title matches one of the banned strings
-- Otherwise returns true
local function checkTrackingPagename()
local thisPage = mw.title.getCurrentTitle()
local thisPageLC = mw.ustring.lower(thisPage.text)
if (string.match(thisPageLC, "/archive") ~= nil) then
return false
end
if (string.match(thisPageLC, "/doc") ~= nil) then
return false
end
if (string.match(thisPageLC, "/test") ~= nil) then
return false
end
return true
end
 
 
-- Builds the portal bar used by {{portal bar}}.
function p._main( portals, args )
-- check for sensible args
if #portals < 1 then return '' end -- Don't display a blank navbox if no portals were specified.
args = type(args) == "table" and args or {}
-- Normalize arguments
for key, default in pairs({border=true,redlinks=false,tracking=true}) do
if args[key] == nil then args[key] = default end
args[key] = yesno(args[key], default)
end
 
local nav = mw.html.create( 'div' )
:addClass( sandboxVersion('noprint metadataportal-bar' ))
:addClass( 'noprint metadata noviewer' )
:attr( 'role', 'navigation' )
:attr( 'aria-label' , 'Portals' )
:addClass(sandboxVersion(args.border and 'portal-bar-bordered' or 'portal-bar-unbordered'))
:css( 'font-weight', 'bold' )
if yesno( args.border ) == false then
nav
:css( 'padding', '0.3em 1.7em 0.1em' )
:css( 'font-size', '88%' )
:css( 'text-align', 'center' )
else
nav
:addClass( 'navbox' )
:css( 'padding', '0.4em 2em' )
end
local trackingCat = ''
if (args.tracking == 'no') or (args.tracking == 'n') or (args.tracking == 'false') then
-- Allow any number of portals
trackingEnabled = false
args.minPortals = 0
end
args.maxPortals = -1
if (checkTrackingNamespace() == false) then
-- Check to see whether there are redlinks, filter out unless args.redlink is true
trackingEnabled = false
portals, trackingCat = checkPortals(portals, args)
end
nav:wikitext(trackingCat)
if (checkTrackingPagename() == false) then
if #portals == 0 then
trackingEnabled = false
return trackingCat
end
 
local related = yesno(args.related)
-- scan for nonexistent portals, if they exist remove them from the portals table. If redlinks=yes, then don't remove
if related then
local portallen = #portals
nav:addClass(sandboxVersion('portal-bar-related'))
-- traverse the list backwards to ensure that no portals are missed (table.remove also moves down the portals in the list, so that the next portal isn't checked if going fowards.
else
-- going backwards allows us to circumvent this issue
local header = nav:tag('span')
for i=portallen,1,-1 do
header:addClass(sandboxVersion('portal-bar-header'))
-- the use of pcall here catches any errors that may occour when attempting to locate pages when the page name is invalid
header:wikitext('[[Wikipedia:Contents/Portals|Portal]]')
-- if pcall returns true, then rerun the function to find if the page exists
if not pcall(checkPortalExists, #portals[i]) or> not checkPortalExists(portals[i])1 then
header:wikitext('s')
-- Getting here means a redlinked portal has been found
if (args.redlinks == 'yes') or (args.redlinks == 'y') or (args.redlinks == 'true') or (args.redlinks == 'include') then
-- if redlinks as been set to yes (or similar), add the cleanup category and then break the loop before the portal is removed from the list
if trackingEnabled then
nav:wikitext('[[Category:Portal templates with redlinked portals]]')
end
break
end
-- remove the portal (this does not happen if redlinks=yes)
table.remove(portals,i)
end
header:wikitext(':')
end
local container = nav:tag('ul')
container:addClass(sandboxVersion('portal-bar-content'))
-- if the length of the table is different, then rows were removed from the table, so portals were removed. If this is the case add the cleanup category
if not (portallen == #portals)related then
container:addClass(sandboxVersion('portal-bar-content-related'))
if #portals == 0 then
if trackingEnabled then
return '[[Category:Portal templates with all redlinked portals]]'
else
return ""
end
end
if trackingEnabled then
nav:wikitext('[[Category:Portal templates with redlinked portals]]')
end
end
local size = related and '27x25px' or '21x19px'
local list = mw.html.create( 'ul' )
:css( 'margin', '0.1em 0 0' )
for _, portal in ipairs( portals ) do
container
list
:tag( 'li' )
:addClass(sandboxVersion('portal-bar-item'))
:css( 'display', 'inline' )
:wikitext( string.format('<span class="nowrap">[[File:%s|%s]] </span>[[Portal:%s|%s]]',
:tag( 'span' ) -- Inline-block on inner span for IE6-7 compatibility.
getImageName(portal,true), size, portal, portal))
:css( 'display', 'inline-block' )
:css( 'white-space', 'nowrap' )
:tag( 'span' )
:css( 'margin', '0 0.5em' )
:wikitext( string.format( '[[File:%s|24x21px]]', getImageName{ portal } ) )
:done()
:wikitext( string.format( '[[Portal:%s|%s portal]]', portal, portal ) )
end
local styleFile = 'Module:Portal bar'..sandbox..'/styles.css'
nav
return mw.getCurrentFrame():extensionTag{
:node( list )
name = 'templatestyles', args = { src = styleFile }
return} .. tostring( nav )
end
 
-- Processes external arguments and sends them to the other functions.
function p.main( frame )
local origArgs = getArgs(frame)
-- If called via #invoke, use the args passed into the invoking
local portals, args = processPortalArgs(origArgs)
-- template, or the args passed to #invoke if any exist. Otherwise
-- assume args are being passed directly in from the debug console
-- or from another Lua module.
local origArgs
if type( frame.getParent ) == 'function' then
origArgs = frame:getParent().args
for k, v in pairs( frame.args ) do
origArgs = frame.args
break
end
else
origArgs = frame
end
-- Process the args to make an array of portal names that can be used with ipairs. We need to use ipairs because we want to list
-- all the portals in the order they were passed to the template, but we also want to be able to deal with positional arguments
-- passed explicitly, for example {{portal|2=Politics}}. The behaviour of ipairs is undefined if nil values are present, so we
-- need to make sure they are all removed.
local portals, args = {}, {}
for k, v in pairs( origArgs ) do
if type( k ) == 'number' and type( v ) == 'string' then -- Make sure we have no non-string portal names.
if mw.ustring.find( v, '%S' ) then -- Remove blank values.
table.insert( portals, k )
end
elseif type( k ) ~= 'number' then -- Separate named arguments from portals.
if type( v ) == 'string' then
v = mw.text.trim( v )
end
args[ k ] = v
end
end
table.sort( portals )
for i, v in ipairs( portals ) do
portals[ i ] = mw.text.trim( origArgs[ v ] ) -- Swap keys with values, trimming whitespace.
end
return p._main( portals, args )
end