模块:Db:修订间差异
MJ Hamster(留言 | 贡献) 小无编辑摘要 |
MJ Hamster(留言 | 贡献) 小无编辑摘要 |
||
| 第18行: | 第18行: | ||
-- [[------------------------------------------------------------------]] | -- [[------------------------------------------------------------------]] | ||
local displayNames = { | local displayNames = { | ||
['title'] = '歌曲标题', ['composer'] = '艺术家', ['bpm'] = 'BPM', ['singer'] = '歌手', ['genre'] = '流派', | |||
['title'] = '歌曲标题', | ['id'] = '歌曲ID', ['key'] = 'KEY音', ['bga'] = 'BGA作者', ['duration'] = '时长', | ||
['id'] = '歌曲ID', | |||
['4b_nm'] = '4B NM', ['4b_nm_notes'] = '4B NM按键数', ['4b_hd'] = '4B HD', ['4b_hd_notes'] = '4B HD按键数', | ['4b_nm'] = '4B NM', ['4b_nm_notes'] = '4B NM按键数', ['4b_hd'] = '4B HD', ['4b_hd_notes'] = '4B HD按键数', | ||
['4b_mx'] = '4B MX', ['4b_mx_notes'] = '4B MX按键数', ['4b_sc'] = '4B SC', ['4b_sc_notes'] = '4B SC按键数', | ['4b_mx'] = '4B MX', ['4b_mx_notes'] = '4B MX按键数', ['4b_sc'] = '4B SC', ['4b_sc_notes'] = '4B SC按键数', | ||
| 第56行: | 第44行: | ||
end | end | ||
-- 格式化单个值的私有函数 ( | -- 格式化单个值的私有函数 (基础格式化,确保 ' - ' 被视为字符串) | ||
local function formatSingleValue(value, fieldType) | local function formatSingleValue(value, fieldType) | ||
if value == nil then | if value == nil then | ||
| 第65行: | 第53行: | ||
return p.valueMappingMethod[fieldType](value) | return p.valueMappingMethod[fieldType](value) | ||
else | else | ||
-- 核心:将所有值转换为字符串,包括 '-' | |||
return tostring(value) | return tostring(value) | ||
end | end | ||
| 第77行: | 第66行: | ||
end | end | ||
-- 获取难度模式 | |||
-- 获取难度模式 | |||
local function getDifficultyDetails(field) | local function getDifficultyDetails(field) | ||
local checkField = field and string.lower(mw.text.trim(field)) or '' | local checkField = field and string.lower(mw.text.trim(field)) or '' | ||
| 第90行: | 第78行: | ||
if difficultyFields[checkField] == true then | if difficultyFields[checkField] == true then | ||
local mode = checkField:sub(-2) | local mode = checkField:sub(-2) | ||
return checkField:match("^(%d+)b"), mode | return checkField:match("^(%d+)b"), mode | ||
end | end | ||
return nil, nil | return nil, nil | ||
end | end | ||
-- **核心:难度样式应用 ( | -- **核心:难度样式应用 (仅用于表格,并处理可能存在的 'sc' 前缀)** | ||
local function formatDifficultyStyles(valueString, mode) | local function formatDifficultyStyles(valueString, mode) | ||
-- valueString 预期是经过 formatSingleValue 处理的字符串 (例如 '15', ' - ', 或 '无') | -- valueString 预期是经过 formatSingleValue 处理的字符串 (例如 '15', ' - ', 或 '无') | ||
-- 1. 特殊值检查:如果是 '-' 或 '无',直接返回 | |||
if valueString == '-' or valueString == i18n['none'] then | |||
return valueString | |||
end | |||
local displayString = valueString -- 用于最终输出的字符串 (默认为原始字符串) | |||
local numValue = tonumber(valueString) | local numValue = tonumber(valueString) | ||
-- | if mode == 'sc' and numValue == nil then | ||
-- 核心修复:如果是 SC 字段且不是纯数字,则尝试剥离 'sc'/'SC' 前缀进行转换 | |||
local stripped = valueString:match("^[Ss][Cc](%d+)$") | |||
if stripped then | |||
numValue = tonumber(stripped) -- 使用剥离后的数字进行样式判断 | |||
displayString = stripped -- 使用剥离后的数字字符串进行显示 | |||
end | |||
end | |||
-- 如果仍不是数字,则返回原始字符串 | |||
if numValue == nil then | if numValue == nil then | ||
return valueString | return valueString | ||
| 第139行: | 第140行: | ||
if style ~= '' then | if style ~= '' then | ||
-- 返回带样式的纯数字字符串 ( | -- 返回带样式的纯数字字符串 (例如 <span...>15</span>) | ||
return '<span style="' .. style .. '">' .. | return '<span style="' .. style .. '">' .. displayString .. '</span>' | ||
end | end | ||
return | return displayString -- 返回原始数字字符串(例如 8) | ||
end | end | ||
| 第150行: | 第151行: | ||
local isSC = string.match(fieldType, '_sc$') | local isSC = string.match(fieldType, '_sc$') | ||
-- 仅当是 SC | -- 仅当是 SC 字段,且值存在且不为 '-' 或 '无' 时,添加 "sc" 前缀 | ||
if isSC and valueString ~= i18n['none'] and valueString ~= '-' then | if isSC and valueString ~= i18n['none'] and valueString ~= '-' then | ||
return 'sc' .. valueString | if not valueString:lower():find('sc') then | ||
return 'sc' .. valueString | |||
end | |||
end | end | ||
| 第214行: | 第217行: | ||
local formattedValue = formatSingleValue(value, argType) | local formattedValue = formatSingleValue(value, argType) | ||
-- 步骤 2: 应用 SC 前缀 | -- 步骤 2: 应用 SC 前缀 | ||
formattedValue = applySCPrefix(formattedValue, argType) | formattedValue = applySCPrefix(formattedValue, argType) | ||
| 第221行: | 第224行: | ||
return table.concat(resultTable, ' / ') | return table.concat(resultTable, ' / ') | ||
end | end | ||
| 第307行: | 第276行: | ||
local output = {} | local output = {} | ||
-- 表格样式 | -- 表格样式 | ||
table.insert(output, '{| class="wikitable sortable" style="width: 100%; margin: 0 auto; text-align: center;"') | table.insert(output, '{| class="wikitable sortable" style="width: 100%; margin: 0 auto; text-align: center;"') | ||
-- 构造表格头部 | -- 构造表格头部 | ||
table.insert(output, '|-') | table.insert(output, '|-') | ||
for _, field in ipairs(displayFields) do | for _, field in ipairs(displayFields) do | ||
| 第320行: | 第289行: | ||
for _, songId in ipairs(matchingKeys) do | for _, songId in ipairs(matchingKeys) do | ||
local songEntry = songData[songId] | local songEntry = songData[songId] | ||
table.insert(output, '|-') | table.insert(output, '|-') | ||
for _, field in ipairs(displayFields) do | for _, field in ipairs(displayFields) do | ||
field = mw.text.trim(field) | field = mw.text.trim(field) | ||
| 第333行: | 第301行: | ||
-- 步骤 2: **仅在表格中**应用特殊样式 | -- 步骤 2: **仅在表格中**应用特殊样式 | ||
if field == 'bpm' then | if field == 'bpm' then | ||
formattedValue = formatBPMValue(formattedValue) | formattedValue = formatBPMValue(formattedValue) | ||
end | end | ||
local _, mode = getDifficultyDetails(field) | local _, mode = getDifficultyDetails(field) | ||
if mode then | if mode then | ||
-- | -- 核心:formatDifficultyStyles 现在能正确处理 '-' 值 | ||
formattedValue = formatDifficultyStyles(formattedValue, mode) | formattedValue = formatDifficultyStyles(formattedValue, mode) | ||
end | end | ||
-- | -- MediaWiki 表格单元格开始符 ' | ' | ||
table.insert(output, '|' .. formattedValue) | table.insert(output, '|' .. formattedValue) | ||
end | end | ||
| 第352行: | 第318行: | ||
table.insert(output, '|}') | table.insert(output, '|}') | ||
return table.concat(output, '\n') | return table.concat(output, '\n') | ||
end | end | ||
return p | return p | ||
2025年11月29日 (六) 13:22的版本
本模块为自动表核心功能程序,根据输入的信息自动筛选数据。
数据库储存在Module:db_song,通过Template:db可以调用数据库中的信息,也可以通过Template:songlist生成自动表格。
local p = {}
-- [[------------------------------------------------------------------]]
-- [[ 1. 国际化和常量 ]]
-- [[------------------------------------------------------------------]]
local i18n = {
['true'] = '是',
['false'] = '否',
['none'] = '无',
['category_missing_value'] = '缺失%s',
['error_no_type'] = '<span class="error">Db:调用时必须提供type参数。</span>',
['error_find_keys_args'] = '<span class="error">Db:findKeysByValue必须提供field和value参数。</span>',
['error_generate_table_args'] = '<span class="error">Db:generateListTable调用缺少field, value或columns参数。</span>',
}
-- [[------------------------------------------------------------------]]
-- [[ 2. 表格标题映射 ]]
-- [[------------------------------------------------------------------]]
local displayNames = {
['title'] = '歌曲标题', ['composer'] = '艺术家', ['bpm'] = 'BPM', ['singer'] = '歌手', ['genre'] = '流派',
['id'] = '歌曲ID', ['key'] = 'KEY音', ['bga'] = 'BGA作者', ['duration'] = '时长',
['4b_nm'] = '4B NM', ['4b_nm_notes'] = '4B NM按键数', ['4b_hd'] = '4B HD', ['4b_hd_notes'] = '4B HD按键数',
['4b_mx'] = '4B MX', ['4b_mx_notes'] = '4B MX按键数', ['4b_sc'] = '4B SC', ['4b_sc_notes'] = '4B SC按键数',
['5b_nm'] = '5B NM', ['5b_nm_notes'] = '5B NM按键数', ['5b_hd'] = '5B HD', ['5b_hd_notes'] = '5B HD按键数',
['5b_mx'] = '5B MX', ['5b_mx_notes'] = '5B MX按键数', ['5b_sc'] = '5B SC', ['5b_sc_notes'] = '5B SC按键数',
['6b_nm'] = '6B NM', ['6b_nm_notes'] = '6B NM按键数', ['6b_hd'] = '6B HD', ['6b_hd_notes'] = '6B HD按键数',
['6b_mx'] = '6B MX', ['6b_mx_notes'] = '6B MX按键数', ['6b_sc'] = '6B SC', ['6b_sc_notes'] = '6B SC按键数',
['8b_nm'] = '8B NM', ['8b_nm_notes'] = '8B NM按键数', ['8b_hd'] = '8B HD', ['8b_hd_notes'] = '8B HD按键数',
['8b_mx'] = '8B MX', ['8b_mx_notes'] = '8B MX按键数', ['8b_sc'] = '8B SC', ['8b_sc_notes'] = '8B SC按键数',
}
-- [[------------------------------------------------------------------]]
-- [[ 3. 本地辅助函数定义 ]]
-- [[------------------------------------------------------------------]]
local function GetValuesTable()
return mw.loadData('Module:db_song')
end
local function ValueFromValuesByKey(values, key)
if values and key then
return values[key]
end
return nil
end
-- 格式化单个值的私有函数 (基础格式化,确保 ' - ' 被视为字符串)
local function formatSingleValue(value, fieldType)
if value == nil then
return i18n['none']
elseif type(value) == 'boolean' then
return value and i18n['true'] or i18n['false']
elseif type(p.valueMappingMethod[fieldType]) == 'function' then
return p.valueMappingMethod[fieldType](value)
else
-- 核心:将所有值转换为字符串,包括 '-'
return tostring(value)
end
end
-- 仅在表格中应用的BPM样式
local function formatBPMValue(valueString)
if string.find(valueString, '-') then
return '<span style="color:red;">' .. valueString .. '</span>'
end
return valueString
end
-- 获取难度模式
local function getDifficultyDetails(field)
local checkField = field and string.lower(mw.text.trim(field)) or ''
local difficultyFields = {
['4b_nm'] = true, ['4b_hd'] = true, ['4b_mx'] = true, ['4b_sc'] = true,
['5b_nm'] = true, ['5b_hd'] = true, ['5b_mx'] = true, ['5b_sc'] = true,
['6b_nm'] = true, ['6b_hd'] = true, ['6b_mx'] = true, ['6b_sc'] = true,
['8b_nm'] = true, ['8b_hd'] = true, ['8b_mx'] = true, ['8b_sc'] = true,
}
if difficultyFields[checkField] == true then
local mode = checkField:sub(-2)
return checkField:match("^(%d+)b"), mode
end
return nil, nil
end
-- **核心:难度样式应用 (仅用于表格,并处理可能存在的 'sc' 前缀)**
local function formatDifficultyStyles(valueString, mode)
-- valueString 预期是经过 formatSingleValue 处理的字符串 (例如 '15', ' - ', 或 '无')
-- 1. 特殊值检查:如果是 '-' 或 '无',直接返回
if valueString == '-' or valueString == i18n['none'] then
return valueString
end
local displayString = valueString -- 用于最终输出的字符串 (默认为原始字符串)
local numValue = tonumber(valueString)
if mode == 'sc' and numValue == nil then
-- 核心修复:如果是 SC 字段且不是纯数字,则尝试剥离 'sc'/'SC' 前缀进行转换
local stripped = valueString:match("^[Ss][Cc](%d+)$")
if stripped then
numValue = tonumber(stripped) -- 使用剥离后的数字进行样式判断
displayString = stripped -- 使用剥离后的数字字符串进行显示
end
end
-- 如果仍不是数字,则返回原始字符串
if numValue == nil then
return valueString
end
local style = ''
if mode == 'sc' then
-- SC 样式 (包括颜色和光晕)
local blueGlow = ' text-shadow: 0 0 5px #3D66FF, 0 0 8px #3D66FF;'
style = 'font-weight: bold;'
if numValue >= 1 and numValue <= 5 then
style = style .. ' color: #E00075;'
elseif numValue >= 6 and numValue <= 10 then
style = style .. ' color: #C604E3;'
elseif numValue >= 11 and numValue <= 12 then
style = style .. ' color: #3D66FF;'
elseif numValue == 13 then
style = style .. ' color: orange;' .. blueGlow
elseif numValue == 14 then
style = style .. ' color: red;' .. blueGlow
elseif numValue == 15 then
style = style .. ' color: purple;' .. blueGlow
end
elseif (mode == 'nm' or mode == 'hd' or mode == 'mx') then
-- NM/HD/MX 样式 (仅 14/15)
if numValue == 14 then
style = 'font-weight: bold; color: orange;'
elseif numValue == 15 then
style = 'font-weight: bold; color: red;'
end
end
if style ~= '' then
-- 返回带样式的纯数字字符串 (例如 <span...>15</span>)
return '<span style="' .. style .. '">' .. displayString .. '</span>'
end
return displayString -- 返回原始数字字符串(例如 8)
end
-- **SC 前缀应用 (仅用于 p.value)**
local function applySCPrefix(valueString, fieldType)
local isSC = string.match(fieldType, '_sc$')
-- 仅当是 SC 字段,且值存在且不为 '-' 或 '无' 时,添加 "sc" 前缀
if isSC and valueString ~= i18n['none'] and valueString ~= '-' then
if not valueString:lower():find('sc') then
return 'sc' .. valueString
end
end
return valueString
end
-- 4. 值格式化方法 (valueMappingMethod)
p.valueMappingMethod = {
['duration'] = (function (value)
if type(value) == 'number' then
local minutes = math.floor(value / 60)
local seconds = value % 60
return string.format('%d:%02d', minutes, seconds)
end
return value
end),
}
-- [[------------------------------------------------------------------]]
-- [[ 5. 外部调用函数 (p.*) ]]
-- [[------------------------------------------------------------------]]
function p.value(f)
local args = f
local frame = mw.getCurrentFrame()
if f == frame then
args = require('Module:ProcessArgs').merge(true)
end
local argTargetName = mw.text.trim(args[1] or '')
local argTypeString = args.type
local argNocat = args.nocat
if not argTypeString then
return i18n['error_no_type']
end
local values = GetValuesTable()
local songEntry = ValueFromValuesByKey(values, argTargetName)
if not songEntry then
if not argNocat then
local title = mw.title.getCurrentTitle()
if title.namespace == 0 and not title.isSubpage then
return '?' .. '[[Category:' .. string.format(i18n['category_missing_value'], '歌曲ID ' .. argTargetName) .. ']]'
end
end
return i18n['none']
end
local fieldNames = mw.text.split(argTypeString, ',')
local resultTable = {}
for _, argType in ipairs(fieldNames) do
argType = mw.text.trim(argType)
local value = ValueFromValuesByKey(songEntry, argType)
-- 步骤 1: 基础格式化
local formattedValue = formatSingleValue(value, argType)
-- 步骤 2: 应用 SC 前缀
formattedValue = applySCPrefix(formattedValue, argType)
table.insert(resultTable, formattedValue)
end
return table.concat(resultTable, ' / ')
end
function p.generateListTable(f)
local args = require('Module:ProcessArgs').merge(true)
local conditionField = mw.text.trim(args.field or '')
local conditionValue = args.value
local displayFieldsString = mw.text.trim(args.columns or '')
-- 预处理 conditionValue 的类型 (保持不变)
if type(conditionValue) == 'string' then
local trimmedValue = mw.text.trim(conditionValue)
local lowerValue = string.lower(trimmedValue)
if lowerValue == 'true' or trimmedValue == '是' then
conditionValue = true
elseif lowerValue == 'false' or trimmedValue == '否' then
conditionValue = false
else
local num = tonumber(trimmedValue)
if num ~= nil then
conditionValue = num
else
conditionValue = trimmedValue
end
end
end
if conditionField == '' or conditionValue == nil or displayFieldsString == '' then
return i18n['error_generate_table_args']
end
local songData = GetValuesTable()
local displayFields = mw.text.split(displayFieldsString, ',')
local matchingKeys = {}
for songId, songEntry in pairs(songData) do
local currentValue = ValueFromValuesByKey(songEntry, conditionField)
if currentValue == conditionValue then
table.insert(matchingKeys, songId)
end
end
table.sort(matchingKeys)
if #matchingKeys == 0 then
return '未找到符合条件的歌曲。'
end
local output = {}
-- 表格样式
table.insert(output, '{| class="wikitable sortable" style="width: 100%; margin: 0 auto; text-align: center;"')
-- 构造表格头部
table.insert(output, '|-')
for _, field in ipairs(displayFields) do
field = mw.text.trim(field)
table.insert(output, '! ' .. (displayNames[field] or field))
end
-- 构造表格行
for _, songId in ipairs(matchingKeys) do
local songEntry = songData[songId]
table.insert(output, '|-')
for _, field in ipairs(displayFields) do
field = mw.text.trim(field)
local rawValue = ValueFromValuesByKey(songEntry, field)
-- 步骤 1: 基础格式化
local formattedValue = formatSingleValue(rawValue, field)
-- 步骤 2: **仅在表格中**应用特殊样式
if field == 'bpm' then
formattedValue = formatBPMValue(formattedValue)
end
local _, mode = getDifficultyDetails(field)
if mode then
-- 核心:formatDifficultyStyles 现在能正确处理 '-' 值
formattedValue = formatDifficultyStyles(formattedValue, mode)
end
-- MediaWiki 表格单元格开始符 ' | '
table.insert(output, '|' .. formattedValue)
end
end
table.insert(output, '|}')
return table.concat(output, '\n')
end
return p