模块:沙盒/逆袭的天邪鬼/talkpage
本模块用于统计讨论页上各位用户的发言次数。目前不支持DYK等投票页或VIP等比较特殊的讨论页面。另外,请不要用Flow来欺负我这个电脑高级中手。
{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | analyse | Wikipedia:互助客栈/条目探讨 }}
效果(点此刷新)
用户 | 发言次数 | 最后发言时间 |
---|---|---|
Unt Phesoca | 5 | 2024年10月29日 (二) 00:47 (UTC) |
TuhansiaVuoria | 1 | 2024年12月22日 (日) 01:13 (UTC) |
Sanmosa | 19 | 2024年12月21日 (六) 04:00 (UTC) |
Factrecordor | 2 | 2024年12月17日 (二) 12:10 (UTC) |
FradonStar | 5 | 2024年12月17日 (二) 18:58 (UTC) |
Happyseeu | 2 | 2024年12月19日 (四) 18:35 (UTC) |
Owennson | 1 | 2024年12月16日 (一) 16:58 (UTC) |
Yage Wu | 2 | 2024年12月18日 (三) 11:10 (UTC) |
物灵 | 1 | 2024年10月26日 (六) 10:53 (UTC) |
Outlookxp | 2 | 2024年12月19日 (四) 11:03 (UTC) |
Ryanlo713 | 13 | 2024年11月10日 (日) 19:54 (UTC) |
Ericliu1912 | 10 | 2024年12月19日 (四) 21:22 (UTC) |
Dewadipper | 4 | 2024年12月7日 (六) 05:04 (UTC) |
甜甜圈真好吃 | 2 | 2024年12月17日 (二) 13:45 (UTC) |
红渡厨 | 4 | 2024年12月19日 (四) 08:31 (UTC) |
书画昼尽 | 1 | 2024年10月26日 (六) 01:59 (UTC) |
Jason2016426 | 1 | 2024年12月19日 (四) 09:48 (UTC) |
Yumeto | 2 | 2024年12月20日 (五) 00:20 (UTC) |
自由雨日 | 33 | 2024年12月19日 (四) 06:33 (UTC) |
银色雪莉 | 2 | 2024年12月17日 (二) 18:49 (UTC) |
日期20220626 | 1 | 2024年12月13日 (五) 05:27 (UTC) |
Iokseng | 1 | 2024年12月6日 (五) 03:55 (UTC) |
Dkzzl | 1 | 2024年12月7日 (六) 14:21 (UTC) |
杰里毛斯 | 1 | 2024年12月18日 (三) 11:42 (UTC) |
魔琴 | 30 | 2024年12月17日 (二) 21:43 (UTC) |
Fglffer | 1 | 2024年12月12日 (四) 18:20 (UTC) |
SheltonMartin | 1 | 2024年11月22日 (五) 02:47 (UTC) |
克劳棣 | 1 | 2024年12月19日 (四) 09:12 (UTC) |
HanTsî | 2 | 2024年12月12日 (四) 16:04 (UTC) |
Milkypine | 3 | 2024年11月8日 (五) 14:37 (UTC) |
航站区 | 1 | 2024年12月9日 (一) 06:55 (UTC) |
Nishino Asuka | 2 | 2024年12月20日 (五) 15:44 (UTC) |
George6VI | 3 | 2024年12月6日 (五) 07:50 (UTC) |
HK5201314 | 23 | 2024年12月17日 (二) 13:07 (UTC) |
Abcet10 | 2 | 2024年12月4日 (三) 14:13 (UTC) |
Nostalgiacn | 19 | 2024年12月21日 (六) 16:49 (UTC) |
Sihjee | 15 | 2024年12月5日 (四) 21:07 (UTC) |
糯米花 | 3 | 2024年12月11日 (三) 10:21 (UTC) |
The Puki desu | 1 | 2024年10月20日 (日) 10:24 (UTC) |
AINH | 1 | 2024年12月16日 (一) 17:15 (UTC) |
Ohtashinichiro | 1 | 2024年12月17日 (二) 12:39 (UTC) |
花开夜 | 2 | 2024年12月21日 (六) 20:48 (UTC) |
Kcx36 | 1 | 2024年12月19日 (四) 12:31 (UTC) |
YFdyh000 | 10 | 2024年12月17日 (二) 15:03 (UTC) |
夏土贤 | 5 | 2024年12月12日 (四) 14:03 (UTC) |
Easterlies | 1 | 2024年12月16日 (一) 17:05 (UTC) |
Djhuty | 3 | 2024年12月19日 (四) 08:11 (UTC) |
Example | 4 | 2024年12月24日 (二) 04:46 (UTC) |
Patrickov | 10 | 2024年11月8日 (五) 03:58 (UTC) |
SunAfterRain | 34 | 2024年12月22日 (日) 05:00 (UTC) |
Liu116 | 4 | 2024年12月21日 (六) 10:05 (UTC) |
Cmsth11126a02 | 6 | 2024年12月17日 (二) 18:54 (UTC) |
其中User:Example是未能判断出名字的用户,例如签名格式很不标准等。
如果只关心某一具体话题,可以
{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | analyse | Wikipedia:互助客栈/条目探讨 | topic=某一話題}}
注意不支持简繁转换。
其他参数
编辑- topic:只统计某个话题。如果标题相同(或者多个标题被匹配),那么这些话题都会被统计进去。
- ispattern:表示topic是一个lua的匹配字符串。请不要加括号。
- userlink、ipuserlink:指定用户名的样式,其中“$USER”会被换为真正的用户名。例如{{User|$USER}}。
- type:可取user、topic和all。默认为user,即只统计各用户的发言情况。
topic的表头如下:
主题 | 发言次数 | 参与者人数 | 发起人 | 最后发言者 | 最后发言时间 |
---|
all的表头如下:
主题 | 用户 | 发言次数 | 最后发言时间 |
---|
其他函数
编辑lastsign
编辑获取最后一个发言的人。例如:
{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | lastsign | Wikipedia:互助客栈/条目探讨 }}
效果(刷新):Example 2024年12月24日 (二) 04:46 (UTC)
同样支持上面提到的各个参数。
lastname和lastdate
编辑只显示最后发言人的名字或日期。
count和countuser
编辑显示签名和签名者的数量。count函数支持user参数,即只显示特定用户的签名数量。
示例一:
[[User:Peacearth]]的簽名檔上已經有{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | count | User:Peacearth/訪客簽名區 }}個能被識別出來的簽名,這是由{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | countuser | User:Peacearth/訪客簽名區 }}位用戶簽出來的。
效果:
User:Peacearth的签名档上已经有230个能被识别出来的签名,这是由225位用户签出来的。
示例二:
小曹在Peacearth的討論頁上發了{{ #invoke:沙盒/逆襲的天邪鬼/talkpage | count | User talk:Peacearth | user=小曹2015 }}個留言。
效果:
小曹在Peacearth的讨论页上发了0个留言。
已知bug
编辑
local f = {}
f.DATEPATTERN = '(%d-)年(%d-)月(%d-)日 %([一二三四五六日]-%) (%d-):(%d-) %(UTC%)'
f.USERPREFIX = {
'user:', '用戶:', '用户:', '使用者:', 'u:', 'U:',
'user talk:', 'user_talk:', '用戶討論:', '用户讨论:', '使用者討論:', 'ut:','UT:','Ut:',
'special:用户贡献/', 'special:contributions/', 'special:用戶貢獻/', 'special:使用者貢獻/',
'特殊:用户贡献/', '特殊:用戶貢獻/', '特殊:使用者貢獻/', '特殊:contributions/'
}
f.MINDATE = '1970年1月1日 (四) 00:00 (UTC)'--{{#time:U|1970010100000000}} = 1
f.MAXDATE = mw.getContentLanguage():formatDate( 'Y年n月j日 (D) H:i (T)' ) --now
--- 反序尋找字符串
local function rfind(s, pattern, init)
local x, y
local i = init or #s
local len = #s
x, y = string.find(string.reverse(s), string.reverse(pattern), len-i+1, true)
if x then
return len-y+1, len-x+1
end
end
--- 生成用戶名鏈接。默認是指向用戶頁(IP用戶為貢獻頁)的鏈接。
-- @param name 用戶名
-- @param style 自定義樣式,其中$USER會被換成用戶名。
-- @param ipstyle IP用戶的自定義樣式。若未指定則與style相同。
-- @return 維基格式的用戶名鏈接
local function getuserlink(name, style, ipstyle)
if string.find(name, '^%d-%.%d-%.%d-%.%d-$') then
return string.gsub(ipstyle or style or '[[Special:用户贡献/$USER|$USER]]', '$USER', name)
else
return string.gsub(style or '[[User:$USER|$USER]]', '$USER', name)
end
end
--- TODO 去除掉影響分析的文字,例如「===」、「<pre>」等
-- @param orintext 原始討論串。
-- @return 經過預處理之後的討論串。去掉一些搗亂的內容。
local function preprocess(orintext, titlelevel)
local text = orintext
text = string.gsub(text, titlelevel .. '=', '---')
text = string.gsub(text, '<pre>.-</pre>', '')
text = string.gsub(text, '<nowiki>.-</nowiki>', '')
text = string.gsub(text, '{{移動至|.-}}', '')
text = string.gsub(text, '{{talkback|.-}}', '')
text = string.gsub(text, '{{deltalk|.-}}', '')
text = string.gsub(text, '{{移动自|.-}}', '')
return text
end
--- 從長長的討論串中找到特定主題。如果未指定主題則將所有主題分割開並存入table中。
-- @param orintext 原始討論串。
-- @param topic 待提取的主題。未指定的話則改為分割所有主題。
-- @param ispattern 表示topic是否為lua的匹配字符串。
-- @return 返回一個鍵為討論標題值為討論內容的table。如果沒有則返回nil。
local function extracttopic(args)
local orintext = args.text
local topic = args.topic
local ispattern = args.ispattern
local titlelevel = args.titlelevel=='3' and '===' or '=='
-- 預處理
local text = titlelevel .. '\n\n' .. preprocess(orintext, titlelevel) .. '\n' .. titlelevel
local result = {}
local found = nil
local topic2 = topic and mw.text.trim(topic)
for title, content in string.gmatch(text, '[ ]*(.-)[ ]*'..titlelevel..'[ ]*\n(.-)\n'..titlelevel) do
local ok = false
if topic2 and (not ispattern) and title == topic2 then
ok = true
elseif topic2 and ispattern and (string.match(title, topic2)) then
ok = true
elseif not topic2 then
ok = true
end
if ok then
found = true
local temptitle = title
local i = 1
while result[temptitle] do
i = i + 1
temptitle = title .. ' (' .. i .. ')'
end
result[temptitle] = content
end
end
return found and result
end
--- 將討論串上的日期轉為一個數字。
-- @param date 討論串格式的日期。
-- @return 將數字直接拼起來。不是時間戳。
local function date2num(date)
local y,m,d,h,min = string.gmatch(date, f.DATEPATTERN)()
if y and m and d and h and min then
return y*100000000+m*1000000+d*10000+h*100+min
else
return 0
end
end
--- 比較兩個日期的大小。
-- @param date1
-- @param date2
-- @return 若date1<date2則返回true。
local function comparedate(date1, date2)
return date2num(date1) < date2num(date2)
end
--- 統計討論串中各個簽名的次數和最後發言時間。
-- @param orintext 原始討論串
-- @return @todo TODO 需要重寫
function f._count(orintext)
-- 預處理
local text = '\n' .. orintext
local lowertext = string.lower(text)
-- 尋找日期
local result = {}
local x, y
y = 1
repeat
x, y = text.find(text, f.DATEPATTERN, y+1)
if not x then
break
end
local date = string.sub(text, x, y)
-- 截取當前行以便於尋找用戶名
local t1, t2 = rfind(text, '\n', x)
local curtalk = string.sub(lowertext, t1, x-1)
-- 尋找用戶名
local name = nil
for i, v in ipairs(f.USERPREFIX) do
local t3, t4 = rfind(curtalk, '[[' .. v)
if t4 then
local t5 = (string.find(curtalk, '[/#|%]]', t4+1))
if t5 then
-- 確定用戶名
name = string.sub(text, t1+t4, t1+t5-2)
-- 首字母大寫
local firstletter = string.byte(name, 1, 1)
if firstletter >= 97 and firstletter <=122 then
name = string.upper(string.sub(name, 1, 1)) .. string.sub(name, 2, -1)
end
break
end
end
end
name = name or 'Example'
if not result[name] then
result[name] = {count = 1, firstdate = date, lastdate = date}
else
result[name].count = result[name].count + 1
if comparedate(result[name].lastdate, date) then
result[name].lastdate = date
end
if comparedate(date, result[name].firstdate) then
result[name].firstdate = date
end
end
until false
return result
end
local function maketable(counttable, args)
local tabletype = args.tabletype and mw.text.trim(args.tabletype) or 'user'
local top = '{| class="wikitable sortable"\n|-'
local header
if tabletype == 'all' then
top = '{| class="wikitable"\n|-'
end
local result = { top }
if tabletype == 'topic' then
table.insert(result, '! 主題 !! 發言次數 !! 參與者人數 !! 發起人 !! 最後發言者 !! data-sort-type="number" | 最後發言時間')
for k, v in pairs(counttable.topics) do
if v.count > 0 then
table.insert(result, string.format('|-\n| %s || %s || %s || %s || %s || data-sort-value="%s" | %s',
k, v.count, v.usercount, getuserlink(v.firstname, args.userlink, args.ipuserlink),
getuserlink(v.lastname, args.userlink, args.ipuserlink), date2num(v.lastdate), v.lastdate))
end
end
elseif tabletype == 'topicsimple' then
table.insert(result, '! 主題 !! 發言次數 !! 最後發言者 !! data-sort-type="number" | 最後發言時間')
for k, v in pairs(counttable.topics) do
if v.count > 0 then
table.insert(result, string.format('|-\n| %s || %s || %s || data-sort-value="%s" | %s',
k, v.count, getuserlink(v.lastname, args.userlink, args.ipuserlink), date2num(v.lastdate), v.lastdate))
end
end
elseif tabletype == 'all' then
table.insert(result, '! 主題 !! 用戶 !! 發言次數 !! data-sort-type="number" | 最後發言時間')
for k, v in pairs(counttable.topics) do
if v.count > 0 then
local first = k .. ' || '
if v.usercount > 1 then
first = 'rowspan=' .. v.usercount .. ' | ' .. first
end
for k2, v2 in pairs(v.users) do
table.insert(result, string.format('|-\n| %s%s || %s || data-sort-value="%s" | %s',
first, getuserlink(k2, args.userlink, args.ipuserlink), v2.count, date2num(v2.lastdate), v2.lastdate))
first = ''
end
end
end
else
table.insert(result, '! 用戶 !! 發言次數 !! data-sort-type="number" | 最後發言時間')
for k, v in pairs(counttable.users) do
table.insert(result, string.format('|-\n| %s || %s || data-sort-value="%s" | %s', getuserlink(k, args.userlink, args.ipuserlink), v.count, date2num(v.lastdate), v.lastdate))
end
end
table.insert(result, '|}')
return table.concat(result, '\n')
end
--- 對簽名數據進行匯總
-- @param stat 通過count函數得到的統計數據
-- @return 一個table,發言數、發言人數、最後發言者、最後發言時間
local function getstat(data)
local count = 0
local usercount = 0
local lastdate = f.MINDATE
local lastname = nil
local firstdate = f.MAXDATE
local firstname = nil
for username, userinfo in pairs(data) do
count = count + userinfo.count
usercount = usercount + 1
if comparedate(lastdate, userinfo.lastdate) then
lastdate = userinfo.lastdate
lastname = username
end
if comparedate(userinfo.firstdate, firstdate) then
firstdate = userinfo.firstdate
firstname = username
end
end
return {
count = count,
usercount = usercount,
firstdate = firstdate,
firstname = firstname,
lastdate = lastdate,
lastname = lastname
}
end
--- 統計討論串中各個簽名的次數和最後發言時間。
-- @param args
-- @return
local function getcount(args)
local talk = string.gsub(args.text, '\r', '\n')
local userdata = {}
local result = { users = userdata, topics = {} }
local topics = extracttopic(args) or {}
for title, content in pairs(topics) do
local topicdata = f._count(content)
for username, userinfo in pairs(topicdata) do
local dat = userdata[username] or { count = 0, lastdate = f.MINDATE }
dat.count = dat.count + userinfo.count
dat.lastdate = comparedate(dat.lastdate, userinfo.lastdate) and userinfo.lastdate or dat.lastdate
userdata[username] = dat
end
result.topics[title] = getstat(topicdata)
result.topics[title].users = topicdata
end
return result
end
--- 獲取最後簽名。
-- @param args
-- @return 一個table,包含用戶名和日期。
local function getlastsign(args)
local r = getcount(args)
local lastname = nil
local lastdate = f.MINDATE
for k, v in pairs(r.users) do
if comparedate(lastdate, v.lastdate) then
lastname = k
lastdate = v.lastdate
end
end
return {name=lastname, date=lastdate}
end
--- 將維基百科模板上的參數轉為lua的table。
-- @param frame 維基百科模板上的參數。
-- @return 一個table,包括各個參數。
local function getargs(frame)
return {
text = mw.title.new(frame.args[1]):getContent(),
topic = frame.args.topic,
ispattern = frame.args.ispattern,
userlink = frame.args.userlink,
ipuserlink = frame.args.ipuserlink,
tabletype = frame.args.type,
titlelevel = frame.args.titlelevel
}
end
--- 統計討論頁發言情況。詳情見文檔頁。
-- @param frame 維基百科頁面名。
-- @return 一張表格。
function f.analyse(frame)
local args = getargs(frame)
return maketable(getcount(args), args)
end
--- 獲取最後一個發言的用戶。
-- @param frame 維基百科頁面名。
-- @return 一個用戶名,不使用鏈接。
function f.lastname(frame)
return getlastsign(getargs(frame)).name
end
--- 獲取最後一個發言的日期。
-- @param frame 維基百科頁面名。
-- @return 一個日期。
function f.lastdate(frame)
return getlastsign(getargs(frame)).date
end
--- 獲取最後一個簽名。
-- @param frame 維基百科頁面名。
-- @return 一個簽名鏈接附上一個日期。
function f.lastsign(frame)
local args = getargs(frame)
local lastsign = getlastsign(args)
if lastsign.name then
return string.format('%s %s', getuserlink(lastsign.name, args.userlink, args.ipuserlink), lastsign.date)
end
end
--- 統計簽名數量。
-- @param frame 維基百科頁面名。
-- @return 一個數字。
function f.count(frame)
local data = getcount(getargs(frame))
local username = frame.args.user and mw.text.trim(frame.args.user)
if not username then
local num = 0
for k, v in pairs(data.topics) do
num = num + v.count
end
return num
elseif data.users[username] then
return data.users[username].count
else
return 0
end
end
--- 統計簽名者數量。
-- @param frame 維基百科頁面名。
-- @return 一個數字。
function f.countuser(frame)
local data = getcount(getargs(frame)).users
local count = 0
for _ in pairs(data) do
count = count + 1
end
return count
end
-- 以下用於本地測試
--[[
f.countsigns = getcount
f.lastsign = getlastsign
f.maketable = maketable
]]
f.getcount = getcount
return f