Модуль:Autosorting: различия между версиями
Перейти к навигации
Перейти к поиску
м (1 версия импортирована: Импорт из Википедии) |
w>Stjn (не знаю, как это работает, но лучше на такое не полагаться и уточнять поведение кода) |
||
Строка 49: | Строка 49: | ||
end | end | ||
− | -- Получить | + | -- Получить num (по умолчанию все) первых значений из свойства в Викиданных |
-- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка | -- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка | ||
− | local function getWikidataProperty( frame, property, entityId ) | + | local function getWikidataProperty( frame, property, entityId, num ) |
frame = frame or mw.getCurrentFrame() | frame = frame or mw.getCurrentFrame() | ||
Строка 68: | Строка 68: | ||
result = mw.text.split( result, ', ' ) | result = mw.text.split( result, ', ' ) | ||
− | return { unpack( result, 1, | + | if isEmpty( num ) then |
+ | return result | ||
+ | else | ||
+ | return { unpack( result, 1, num ) } | ||
+ | end | ||
end | end | ||
Строка 81: | Строка 85: | ||
frame = frame or mw.getCurrentFrame() | frame = frame or mw.getCurrentFrame() | ||
− | local propValues = getWikidataProperty( frame, property, entityId ) | + | local propValues = getWikidataProperty( frame, property, entityId, 3 ) |
local catName = getCategoryName( name, property ) | local catName = getCategoryName( name, property ) | ||
− | local instanceOf = getWikidataProperty( frame, 'p31', entityId ) | + | local instanceOf = getWikidataProperty( frame, 'p31', entityId, 3 ) |
instanceOf = #instanceOf > 0 and instanceOf[ 1 ] or '' | instanceOf = #instanceOf > 0 and instanceOf[ 1 ] or '' | ||
local pageTitle = mw.title.getCurrentTitle().fullText | local pageTitle = mw.title.getCurrentTitle().fullText | ||
Строка 103: | Строка 107: | ||
local result = '' | local result = '' | ||
+ | local catName = '' | ||
+ | local occupationExists = false | ||
+ | local validCatsCounter = 0 | ||
for key, val in pairs( propValues ) do | for key, val in pairs( propValues ) do | ||
local value = mwLang:lcfirst( val ) | local value = mwLang:lcfirst( val ) | ||
− | + | catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, value ) | |
+ | occupationExists = true | ||
+ | |||
+ | if isValidCategory( name, catName, property ) then | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | validCatsCounter = validCatsCounter + 1 | ||
+ | end | ||
− | -- | + | -- Ограничение количества выводимых категорий |
− | if | + | if validCatsCounter >= 3 then |
break | break | ||
end | end | ||
+ | end | ||
+ | |||
+ | -- Подходящей категории нет, либо в ВД нет рода занятий, но есть дефолтные значения, | ||
+ | -- тогда поставляем категорию на их основе при её наличии | ||
+ | local args = getArgs( frame ) | ||
+ | local defaultOccupation = args[ 'default-occupation' ] | ||
+ | |||
+ | if result == '' and not isEmpty( defaultOccupation ) then | ||
+ | occupationExists = true | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, defaultOccupation ) | ||
+ | if isValidCategory( name, catName, property ) then | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Есть род занятий (на ВД или через параметр), но подходящей категории нет, | ||
+ | -- тогда подставляем служебную категорию "не распределён" при её наличии | ||
+ | if result == '' and occupationExists then | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: не распределён)', name ) | ||
if isValidCategory( name, catName, property ) then | if isValidCategory( name, catName, property ) then | ||
result = result .. string.format( '[[Category:%s]]', catName ) | result = result .. string.format( '[[Category:%s]]', catName ) | ||
Строка 129: | Строка 161: | ||
local property = 'p31' | local property = 'p31' | ||
local propValues = getWikidataProperty( frame, property, entityId ) | local propValues = getWikidataProperty( frame, property, entityId ) | ||
+ | |||
+ | local args = getArgs( frame ) | ||
+ | local defaultType = args[ 'default-type' ] | ||
+ | local defaultOccupation = args[ 'default-occupation' ] | ||
local defaultCatKey | local defaultCatKey | ||
local result = '' | local result = '' | ||
+ | local catName = '' | ||
+ | local validCatsCounter = 0 | ||
for key, val in pairs( propValues ) do | for key, val in pairs( propValues ) do | ||
− | + | catName = getCategoryName( 'Википедия:%s (тип: %s)', name, mwLang:lcfirst( val ) ) | |
-- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка | -- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка | ||
if val == 'человек' or val == 'human' then | if val == 'человек' or val == 'human' then | ||
Строка 142: | Строка 180: | ||
result = result .. occupations | result = result .. occupations | ||
end | end | ||
+ | validCatsCounter = validCatsCounter + 1 -- "человек" считается всегда валидной | ||
else | else | ||
if isValidCategory( name, catName, property ) then | if isValidCategory( name, catName, property ) then | ||
result = result .. string.format( '[[Category:%s]]', catName ) | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | validCatsCounter = validCatsCounter + 1 | ||
else | else | ||
if isEmpty( defaultCatKey ) then | if isEmpty( defaultCatKey ) then | ||
defaultCatKey = val | defaultCatKey = val | ||
end | end | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Ограничение количества выводимых категорий | ||
+ | if validCatsCounter >= 3 then | ||
+ | break | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Если ничего не нашлось, попытка добавить категорию на основе переданных дефолтных значений | ||
+ | if result == '' and not isEmpty( defaultType ) then | ||
+ | if defaultType == 'человек' and not isEmpty( defaultOccupation ) then -- человек, есть занятие | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, defaultOccupation ) | ||
+ | if isValidCategory( name, catName, property ) then | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | |||
+ | else -- человек, есть занятие, но категория для занятия не прошла проверки | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: человек)', name ) | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | end | ||
+ | |||
+ | elseif defaultType == 'человек' then -- человек, нет занятия | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: человек)', name ) | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
+ | |||
+ | else -- нечеловек | ||
+ | catName = getCategoryName( 'Википедия:%s (тип: %s)', name, defaultType ) | ||
+ | if isValidCategory( name, catName, property ) then | ||
+ | result = result .. string.format( '[[Category:%s]]', catName ) | ||
end | end | ||
end | end | ||
Строка 175: | Строка 244: | ||
local property = 'p131' | local property = 'p131' | ||
− | local propValues = getWikidataProperty( frame, property, entityId ) | + | local propValues = getWikidataProperty( frame, property, entityId, 3 ) |
local result = '' | local result = '' | ||
Строка 197: | Строка 266: | ||
local property = 'p17' | local property = 'p17' | ||
− | local propValues = getWikidataProperty( frame, property, entityId ) | + | local propValues = getWikidataProperty( frame, property, entityId, 3 ) |
local result = '' | local result = '' | ||
Строка 249: | Строка 318: | ||
-- Игнорировать при наличии изображений в указанных свойствах | -- Игнорировать при наличии изображений в указанных свойствах | ||
for _, val in pairs( localFileProps ) do | for _, val in pairs( localFileProps ) do | ||
− | local propValue = getWikidataProperty( frame, val, entityId ) | + | local propValue = getWikidataProperty( frame, val, entityId, 3 ) |
if #propValue > 0 then | if #propValue > 0 then | ||
return '' | return '' |
Версия от 16:36, 23 мая 2024
Для документации этого модуля может быть создана страница Модуль:Autosorting/doc
require( 'strict' )
local p = {}
local mwLang = mw.getContentLanguage()
local getArgs = require( 'Module:Arguments' ).getArgs
local config = mw.loadData( 'Module:Autosorting/config' )
local function isEmpty( val )
return val == nil or val == ''
end
-- Получить назвние категории
local function getCategoryName( str, name, val )
val = val or ''
return string.format( str, mwLang:ucfirst( name ), val )
end
-- Получить стандартный лимит
local function getPageLimit( name, property )
name = mwLang:lcfirst( name )
property = mwLang:uc( property )
local propertyLimits = config.limits[ property ]
if isEmpty( propertyLimits ) then
return -1
end
return propertyLimits[ name ] or propertyLimits.default
end
-- Получить соответствие категории критериям
local function isValidCategory( name, catName, property, doNotCheck )
if not doNotCheck then
local success, title = pcall( mw.title.new, 'Category:' .. catName )
if success and not isEmpty( title ) and title.exists then
return true
end
end
local success, catCount = pcall( mw.site.stats.pagesInCategory, catName, 'pages' )
if success then
local pageLimit = getPageLimit( name, property )
return catCount > pageLimit
end
return false
end
-- Получить num (по умолчанию все) первых значений из свойства в Викиданных
-- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка
local function getWikidataProperty( frame, property, entityId, num )
frame = frame or mw.getCurrentFrame()
local callArgs = {
property,
}
if not isEmpty( entityId ) then
callArgs.from = entityId
end
local success, result = pcall( frame.callParserFunction, frame, '#property', callArgs )
if success and not isEmpty( result ) then
if mw.ustring.len( result ) > 255 then
return {}
end
result = mw.text.split( result, ', ' )
if isEmpty( num ) then
return result
else
return { unpack( result, 1, num ) }
end
end
return {}
end
-- Получить категорию для отсутствия изображений
local function getFileCategory( frame, name, property, entityId )
if isEmpty( name ) then
return ''
end
frame = frame or mw.getCurrentFrame()
local propValues = getWikidataProperty( frame, property, entityId, 3 )
local catName = getCategoryName( name, property )
local instanceOf = getWikidataProperty( frame, 'p31', entityId, 3 )
instanceOf = #instanceOf > 0 and instanceOf[ 1 ] or ''
local pageTitle = mw.title.getCurrentTitle().fullText
local result = string.format( '[[Category:%s|%s%s]]', catName, instanceOf, pageTitle )
return #propValues, result
end
-- Сортировка по профессиям
function p._byOccupation( frame, name, entityId )
if isEmpty( name ) then
return ''
end
frame = frame or mw.getCurrentFrame()
local property = 'p106'
local propValues = getWikidataProperty( frame, property, entityId )
local result = ''
local catName = ''
local occupationExists = false
local validCatsCounter = 0
for key, val in pairs( propValues ) do
local value = mwLang:lcfirst( val )
catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, value )
occupationExists = true
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
validCatsCounter = validCatsCounter + 1
end
-- Ограничение количества выводимых категорий
if validCatsCounter >= 3 then
break
end
end
-- Подходящей категории нет, либо в ВД нет рода занятий, но есть дефолтные значения,
-- тогда поставляем категорию на их основе при её наличии
local args = getArgs( frame )
local defaultOccupation = args[ 'default-occupation' ]
if result == '' and not isEmpty( defaultOccupation ) then
occupationExists = true
catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, defaultOccupation )
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
end
end
-- Есть род занятий (на ВД или через параметр), но подходящей категории нет,
-- тогда подставляем служебную категорию "не распределён" при её наличии
if result == '' and occupationExists then
catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: не распределён)', name )
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
end
end
return result
end
-- Сортировка по типам
function p._byType( frame, name, entityId )
if isEmpty( name ) then
return ''
end
frame = frame or mw.getCurrentFrame()
local property = 'p31'
local propValues = getWikidataProperty( frame, property, entityId )
local args = getArgs( frame )
local defaultType = args[ 'default-type' ]
local defaultOccupation = args[ 'default-occupation' ]
local defaultCatKey
local result = ''
local catName = ''
local validCatsCounter = 0
for key, val in pairs( propValues ) do
catName = getCategoryName( 'Википедия:%s (тип: %s)', name, mwLang:lcfirst( val ) )
-- TODO: переделать на получение Q-элементов, чтобы не зависеть от языка
if val == 'человек' or val == 'human' then
local occupations = p._byOccupation( frame, name, entityId )
if isEmpty( occupations ) then
result = result .. string.format( '[[Category:%s]]', catName )
else
result = result .. occupations
end
validCatsCounter = validCatsCounter + 1 -- "человек" считается всегда валидной
else
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
validCatsCounter = validCatsCounter + 1
else
if isEmpty( defaultCatKey ) then
defaultCatKey = val
end
end
end
-- Ограничение количества выводимых категорий
if validCatsCounter >= 3 then
break
end
end
-- Если ничего не нашлось, попытка добавить категорию на основе переданных дефолтных значений
if result == '' and not isEmpty( defaultType ) then
if defaultType == 'человек' and not isEmpty( defaultOccupation ) then -- человек, есть занятие
catName = getCategoryName( 'Википедия:%s (тип: человек; род занятий: %s)', name, defaultOccupation )
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
else -- человек, есть занятие, но категория для занятия не прошла проверки
catName = getCategoryName( 'Википедия:%s (тип: человек)', name )
result = result .. string.format( '[[Category:%s]]', catName )
end
elseif defaultType == 'человек' then -- человек, нет занятия
catName = getCategoryName( 'Википедия:%s (тип: человек)', name )
result = result .. string.format( '[[Category:%s]]', catName )
else -- нечеловек
catName = getCategoryName( 'Википедия:%s (тип: %s)', name, defaultType )
if isValidCategory( name, catName, property ) then
result = result .. string.format( '[[Category:%s]]', catName )
end
end
end
-- Добавить стандартную категорию только при отсутствии иных
local defaultCatName = getCategoryName( 'Википедия:%s (не распределённые по типам)', name )
if result == '' and not isEmpty( defaultCatKey ) then
local pageTitle = mw.title.getCurrentTitle().fullText
result = result .. string.format( '[[Category:%s|%s%s]]', defaultCatName, defaultCatKey, pageTitle )
end
if result == '' then
return getCategoryName( '[[Category:Википедия:%s (тип: не указан)]]', name )
end
return result
end
-- Сортировка по АТЕ
function p._bySubdivision( frame, name, entityId )
if isEmpty( name ) then
return ''
end
frame = frame or mw.getCurrentFrame()
local property = 'p131'
local propValues = getWikidataProperty( frame, property, entityId, 3 )
local result = ''
for key, val in pairs( propValues ) do
local catName = getCategoryName( 'Википедия:%s (АТЕ: %s)', name, val )
if isValidCategory( name, catName, property, false ) then
result = result .. string.format( '[[Category:%s]]', catName )
end
end
return result
end
-- Сортировка по странам
function p._byCountry( frame, name, entityId )
if isEmpty( name ) then
return ''
end
frame = frame or mw.getCurrentFrame()
local property = 'p17'
local propValues = getWikidataProperty( frame, property, entityId, 3 )
local result = ''
for key, val in pairs( propValues ) do
if isEmpty( val ) then
break
end
local catName = getCategoryName( 'Википедия:%s (страна: %s)', name, val )
result = result .. string.format( '[[Category:%s]]', catName )
if isValidCategory( name, catName, property, false ) then
result = result .. p._bySubdivision( frame, name, entityId )
end
end
return result
end
-- Сортировка по наличию/отсутствию изображений
function p._byImage( frame, file, localFileProps, entityId )
frame = frame or mw.getCurrentFrame()
-- Возможный сброс значения с Викиданных
if file ~= nil and mw.text.trim( file ) == '-' then
file = nil
end
-- Вывести категории при заполненном несуществующем файле (= файле с Викисклада)
if not isEmpty( file ) then
local success, title = pcall( mw.title.new, 'File:' .. file )
-- Игнорировать при заполненном локальном файле
if success and not isEmpty( title ) and title.exists then
return nil
end
local catName = 'Википедия:Статьи с изображениями: заполнить свойство %s в Викиданных'
local result = ''
local p18, p18Category = getFileCategory( frame, catName, 'p18', entityId )
local p373, p373Category = getFileCategory( frame, catName, 'p373', entityId )
if p18 == 0 then
result = result .. p18Category
end
if p373 == 0 then
result = result .. p373Category
end
return result
end
-- Игнорировать при наличии изображений в указанных свойствах
for _, val in pairs( localFileProps ) do
local propValue = getWikidataProperty( frame, val, entityId, 3 )
if #propValue > 0 then
return ''
end
end
-- Вывести категории при отсутствии игнорируемых свойств
local catName = 'Википедия:Статьи без изображений (указано в Викиданных: %s)'
local result = p._byCountry( frame, 'статьи без изображений', entityId )
local p18, p18Category = getFileCategory( frame, catName, 'p18', entityId )
local p242, p242Category = getFileCategory( frame, catName, 'p242', entityId )
local p373, p373Category = getFileCategory( frame, catName, 'p373', entityId )
if p18 > 0 then
result = result .. p18Category
end
if p242 > 0 then
result = result .. p242Category
end
if p373 > 0 then
result = result .. p373Category
end
result = result .. p._byType( frame, 'статьи без изображений', entityId )
return result
end
-- Шаблон сортировки по типам
function p.byType( frame )
local args = getArgs( frame )
local name = args[ 1 ]
local entityId = args[ 'from' ]
if isEmpty( name ) or not isEmpty( args.nocat ) then
return nil
end
if mw.ustring.find( name, 'статьи' ) and mw.title.getCurrentTitle().namespace ~= 0 then
return nil
end
return p._byType( frame, name, entityId )
end
-- Шаблон сортировки по странам
function p.byCountry( frame )
local args = getArgs( frame )
local name = args[ 1 ]
local entityId = args[ 'from' ]
if isEmpty( name ) or not isEmpty( args.nocat ) then
return nil
end
if mw.ustring.find( name, 'статьи' ) and mw.title.getCurrentTitle().namespace ~= 0 then
return nil
end
return p._byCountry( frame, name, entityId )
end
-- Шаблон сортировки по изображениям
function p.byImage( frame )
if mw.title.getCurrentTitle().namespace ~= 0 then
return nil
end
local args = getArgs( frame )
local file = args[ 1 ]
local entityId = args[ 'from' ]
if not isEmpty( args.nocat ) then
return nil
end
-- Игнорирование по умолчанию статей с указанным p18
local uses = args[ 'uses' ]
if isEmpty( uses ) then
uses = 'p18'
end
local localFileProps = mw.text.split( uses, ', ' )
if uses == '-' then
localFileProps = {}
end
return p._byImage( frame, file, localFileProps, entityId )
end
return p