MediaWiki:Gadget-ondemand-arbcomVoting.js: различия между версиями
Перейти к навигации
Перейти к поиску
Содержимое удалено Содержимое добавлено
Rubin16 (обсуждение | вклад) one more |
Putnik (обсуждение | вклад) поддержка тёмной темы |
||
(не показана 141 промежуточная версия 11 участников) | |||
Строка 1: | Строка 1: | ||
/* |
|||
(function(){ |
|||
* Скрипт для выборов в Арбитражный комитет (ВП:АК) |
|||
appendCSS('.cand-list { width: 25em } #voting-container ul { margin: 1.5em 0 } #voting-container ul li { width: 19em !important } #voting-help-final { margin: 3em 0 0; border-top: 1px solid #888} .cand-selected { background: #97B6E2 !important } #voting-limited-count { float: right; font-size: 300%; text-align: right; width: 1.5em; line-height: 100%; color: #ff5555 } #voting-limited-count.ok { color: #71C837; font-weight: bold }') |
|||
* |
|||
* Авторы: Kalan, Serhio Magpie, stjn |
|||
function en(e) { return encodeURIComponent(e) } |
|||
*/ |
|||
function el(e) { return document.createElement(e) } |
|||
function tstamp(t) { return !t.getUTCFullYear() ? null : // Safari + Chrome |
|||
(t.getUTCFullYear()+":0"+(t.getUTCMonth()+1)+":0"+t.getUTCDate()+":0"+ |
|||
t.getUTCHours() +":0"+ t.getUTCMinutes() +":0"+t.getUTCSeconds()) |
|||
.replace(/:0?(\d\d)/g, '$1') } |
|||
function ch(o) { for (var i in o) { return o[i] } } |
|||
var conf = { |
|||
'criteria': { |
|||
'count': 500, |
|||
'registration': '2012-08-04T23:59:60Z' |
|||
}, |
|||
'pagepath': 'Википедия:Выборы арбитров/Осень 2012 2/Голосование', |
|||
'talkpath': 'Википедия:Выборы арбитров/Осень 2012 2/Обсуждение/', |
|||
'votepath': 'Википедия:Выборы арбитров/Осень 2012 2/Голосование/', |
|||
'start': new Date('Jan 2 2013 00:00 +0000'), |
|||
'end': new Date('Jan 9 2013 00:00 +0000') |
|||
} |
|||
var ico = { |
|||
'up': '/upwiki/wikipedia/commons/thumb/', |
|||
'supp' : '2/2d/Support-gray.svg/39px-Support-gray.svg.png', |
|||
'suppinact' : '8/8d/Support-colored.svg/39px-Support-colored.svg.png', |
|||
'suppact' : '5/5b/Support-filled.svg/39px-Support-filled.svg.png', |
|||
'opp' : 'e/e7/Oppose-gray.svg/39px-Oppose-gray.svg.png', |
|||
'oppinact' : '0/06/Oppose-colored.svg/39px-Oppose-colored.svg.png', |
|||
'oppact' : '7/7d/Oppose-filled.svg/39px-Oppose-filled.svg.png' |
|||
} |
|||
var loc = { |
|||
'votebutton': 'Проголосовать', |
|||
'statuscriteria': 'Проверка критериев…', |
|||
'criteriafail': '<h3 class="voting-error">Вы не соответствуете критериям.</h3><p>Ваши голоса на этих выборах <b>не будут засчитаны</b>.</p><p>Впрочем, если вам просто любопытно, как работает скрипт, вы можете посмотреть на него без сохранения голосов.</p>', |
|||
'criteriafailbutton': 'Посмотреть', |
|||
'loadingvotes': 'Проверка имеющихся голосов…', |
|||
'draghere': 'Перетащите первого кандидата сюда', |
|||
'votinghelp': '<div class="voting-help"><!-- h4>Основная часть</h4 -->\ |
|||
<p>Расставьте голоса «за» и «против» рядом с именами тех кандидатов, относительно которых у вас сформировано мнение. Вы сможете дополнить или изменить выбор позже.</p>\ |
|||
</div>', |
|||
'etchelp1': '<div class="voting-help"><h4>Дополнительная часть</h4>\ |
|||
<p>Дополнительная часть состоит из нескольких опросников. Вы можете принять участие не во всех или вовсе проигнорировать этот блок, но лучше всего заполните все три. Это поможет улучшить процедуру выборов.</p></div>\ |
|||
<div class="voting-help"><h5>Метод Шульце</h5>\ |
|||
<ol>\ |
|||
<li>Перетащите какого-нибудь подходящего, на ваш взгляд, кандидата на первый уровень.</li>\ |
|||
<li>Перетаскивайте остальных кандидатов в этом же направлении. Кандидатов можно оставлять на существующих уровнях (отмечены цифрами) или на новых (места для их создания обозначены пунктиром). Во время перетаскивания пунктир показывает относительное расположение кандидата в списке.</li>\ |
|||
<li>Перед сохранением убедитесь в соответствии расстановки вашему мнению: более хорошие кандидаты выше, менее хорошие ниже, равные на одном уровне, совсем неподходящие в разделе «нет предпочтения». Если вы являетесь одним из кандидатов, <em>не оставляйте</em> себя в нижней секции.</li>\ |
|||
</ol>\ |
|||
</div>', |
|||
'etchelp2': '<div id="voting-etc-schulze"></div>\ |
|||
<div class="voting-help"><h5>Целевой набор резервных арбитров</h5>\ |
|||
<p>Оцените, насколько каждый кандидат пригоден на роль <strong>именно резервного арбитра</strong>. Если, по вашему мнению, кандидат будет хорош только как основной арбитр, <strong>не голосуйте за</strong>.</p>\ |
|||
</div>', |
|||
'etchelp3': '<div class="voting-help"><h5>Идеальный состав</h5>\ |
|||
<p>Выберите <strong>ровно пять</strong> арбитров из списка. Голос за большее или меньшее количество арбитров будет сохранён, но проигнорирован при подсчёте.</p>\ |
|||
</div>', |
|||
'etchelp4': '<div class="voting-help" id="voting-help-final"><p>До конца выборов вы сможете изменить своё мнение.</p>\ |
|||
<p>Если что-то осталось непонятным, можно <a target="_blank" href="/ruwiki/wiki/Обсуждение_Википедии:Выборы_арбитров/Весна_2012/Голосование">задать вопрос</a>.</p></div>', |
|||
'nopreference': 'нет предпочтения', |
|||
'newlevel': 'новый уровень', |
|||
'justbeforesave': 'Смело направляйте любые отзывы, предложения и сообщения об ошибках на <a href="/ruwiki/wiki/MediaWiki_talk:Script/Voting.js">страницу обсуждения</a>.', |
|||
'savebutton': 'Сохранить', |
|||
'saveprog': '<b>Не закрывайте страницу</b> до завершения сохранения.', |
|||
'summary': '\u200Bг\u200Bол\u200Bос\u200B', |
|||
'thankyou': '<h3>Спасибо за участие в выборах!</h3><p>Вы сможете изменить ваши голоса до конца голосования.</p><p>Только что отданные голоса можно посмотреть на <a href="/ruwiki/wiki/Special:Mycontributions">странице вклада</a>.</p>' |
|||
} |
|||
var cand = 'DonaldDuck·Drbug·Ignatus·Lazyhawk·putnik·Rave·TenBaseT'.split('·') |
|||
var wgAPIPath = wgServer + wgScriptPath + '/api.php?format=json&' |
|||
var votes = {} |
|||
var criteriaMatch |
|||
var saving = null |
|||
var aj = sajax_init_object() |
|||
var aj2 = sajax_init_object() |
|||
var token |
|||
// drag-n-drop |
|||
var coo |
|||
var cooD |
|||
var drag, dragGhost, dragItem, dragOriginal |
|||
var dragHere |
|||
function votingStart() { |
|||
importStylesheetURI('//ru.wikipedia.org/ruwiki/w/index.php?title=MediaWiki:Voting7.css&action=raw&ctype=text/css') |
|||
btn.disabled = true |
|||
btn.style.display = 'none' |
|||
status.innerHTML = loc.statuscriteria |
|||
aj2.onreadystatechange = votingStartContinue |
|||
aj2.open('GET', wgAPIPath + 'action=query&list=users&usprop=registration|editcount&prop=revisions&rvprop=content&ususers=' + |
|||
en(wgUserName) + '&titles=' + en(conf.votepath + '^'), true) |
|||
aj2.send('') |
|||
} |
|||
function votingStartContinue() { |
|||
if (aj2.readyState != 4) return |
|||
if (aj2.status != 200) { // temporary problems? |
|||
votingStart() |
|||
return |
|||
} |
|||
query = eval('(' + aj2.responseText + ')').query |
|||
userinfo = query.users[0] |
|||
if ((// a valid voter must be a non-anonymous user |
|||
userinfo.missing !== undefined || |
|||
// who is not blocked |
|||
userinfo.blockedby || |
|||
// whose editcount is at least conf.criteria.count |
|||
userinfo.editcount < conf.criteria.count || |
|||
// and who is registered no later than conf.criteria.registration |
|||
// "null" means "before 2005-12-29", user creation was not logged before then |
|||
(userinfo.registration !== null && userinfo.registration > conf.criteria.registration)) && |
|||
// exemptions |
|||
('\n' + ch(query.pages).revisions[0]['*'] + '\n').indexOf('\n* %\n'.replace('%', wgUserName)) == -1) |
|||
votingCriteriaFail() |
|||
else { |
|||
criteriaMatch = 1 |
|||
votingContinue() |
|||
} |
|||
} |
|||
function votingCriteriaFail() { |
|||
status.innerHTML = loc.criteriafail |
|||
btn.style.display = '' |
|||
btn.disabled = false |
|||
btn.value = loc.criteriafailbutton |
|||
btn.onclick = function(){criteriaMatch = 0; votingContinue()} |
|||
} |
|||
function votingContinue() { |
|||
status.innerHTML = loc.loadingvotes |
|||
btn.style.display = '' |
|||
btn.onclick = votingToken |
|||
btn.value = loc.savebutton |
|||
btn.disabled = true |
|||
// latest contributions in ns:4 + Schulze method page |
|||
aj.open('GET', wgAPIPath + 'action=query' + |
|||
'&list=usercontribs&ucnamespace=4&uclimit=500' + |
|||
'&ucuser=' + en(wgUserName) + |
|||
(tstamp(conf.start) ? '&ucend=' + tstamp(conf.start) : '') + |
|||
'&ucdir=older' + |
|||
'&fake=1&prop=revisions&rvprop=content' + |
|||
'&titles=' + en(conf.votepath+'$|') |
|||
+ en(conf.votepath+'%|') |
|||
+ en(conf.votepath+'=') , |
|||
true) |
|||
aj.onreadystatechange = votingDraw |
|||
aj.send('') |
|||
} |
|||
function votingDraw() { |
|||
if (aj.readyState != 4) return |
|||
if (aj.status != 200) { // temporary problems? |
|||
votingContinue() |
|||
return |
|||
} |
|||
btn.disabled = !criteriaMatch |
|||
status.innerHTML = '' |
|||
var query = eval('(' + aj.responseText + ')') |
|||
query = query.query |
|||
var co = query.usercontribs |
|||
/******* for (var k in query.pages) { |
|||
spl = query.pages[k].revisions[0]['*'].split('\n') |
|||
switch (query.pages[k].title.match(/(.)$/)[1]) { |
|||
case '$': |
|||
var sc = spl |
|||
break |
|||
case '%': |
|||
var reserve = spl |
|||
break |
|||
case '=': |
|||
var limited = spl |
|||
break |
|||
} |
|||
} *******/ |
|||
// retrieving "traditional" votes, according to contributions, only latest ones are valid |
|||
for (i=0; i<cand.length; i++) votes[cand[i]] = {'orig':0, 'value':0} |
|||
for (var i=co.length-1; i>=0; i--) { |
|||
var m = co[i].title.indexOf(conf.votepath) == 0 |
|||
if (m) m = co[i].title.match(/\/([+-])\/(.*?)$/) |
|||
if (m) votes[m[2]] = { 'orig' : m[1]=='+' ? +1 : -1, |
|||
'value': 0 } |
|||
} |
|||
// retrieving Schulze method page |
|||
// we assume that the bot will immediately revert not-well-formed edits |
|||
/******** for (i=0; i<sc.length; i++) { |
|||
if (sc[i].indexOf('* ' + wgUserName + ' | ') == 0) { |
|||
votes._s = {'reading': true} |
|||
var j = 0 |
|||
continue |
|||
} |
|||
if (!votes._s) continue |
|||
if (sc[i].indexOf('* ') == 0) votes._s.reading = undefined |
|||
if (votes._s.reading) { |
|||
j++ |
|||
var m = sc[i].match(/^\*(.) (.*)/) |
|||
m[2] = m[2].split(' | ') |
|||
votes._s[m[1]=='#'?j:0] = m[2] |
|||
} |
|||
} |
|||
if (!votes._s) { |
|||
votes._s = {0: cand} |
|||
} |
|||
if (!votes._s[1]) votes._s[1] = [] |
|||
if (!votes._s[0]) votes._s[0] = [] |
|||
if (votes._s.reading) votes._s.reading = undefined |
|||
// retrieving "reserve" voting |
|||
votes._r = {} |
|||
for (i = 0; i < cand.length; ++i) |
|||
votes._r[cand[i]] = { 'orig' : 0, 'value': 0 } |
|||
reading = false |
|||
for (i=0; i<reserve.length; i++) { |
|||
if (reserve[i].indexOf('* ' + wgUserName + ' | ') == 0) { |
|||
reading = true |
|||
continue |
|||
} |
|||
if (!votes._r) continue |
|||
if (reserve[i].indexOf('* ') == 0) reading = false |
|||
if (reading) { |
|||
m = reserve[i].match(/^\*: (.) (.*)$/) |
|||
votes._r[m[2]].orig = votes._r[m[2]].value = ((m[1] == '.') ? 0 : (m[1] == '+' ? 1 : -1)) |
|||
} |
|||
} |
|||
// retrieving "limited" voting |
|||
votes._l = {} |
|||
for (i = 0; i < cand.length; ++i) |
|||
votes._l[cand[i]] = { 'orig' : 0, 'value': 0 } |
|||
reading = false |
|||
for (i=0; i<limited.length; i++) { |
|||
if (limited[i].indexOf('* ' + wgUserName + ' | ') == 0) { |
|||
reading = true |
|||
continue |
|||
} |
|||
if (!votes._l) continue |
|||
if (limited[i].indexOf('* ') == 0) reading = false |
|||
if (reading) { |
|||
m = limited[i].match(/^\*: (.) (.*)$/) |
|||
votes._l[m[2]].orig = votes._l[m[2]].value = ((m[1] == '.') ? 0 : 1) |
|||
} |
|||
} |
|||
*******/ |
|||
// drawing |
|||
var div1 = el('div') |
|||
div1.id = 'voting-standard' |
|||
div1.innerHTML = loc.votinghelp |
|||
var tab = el('table') |
|||
var tr, img, td1, td2, td3, lin |
|||
for (i in votes) { |
|||
if (!i.match(/^_[srl]$/)) { |
|||
tr = el('tr') |
|||
td1 = el('td') |
|||
img = el('img') |
|||
img.alt = '+' |
|||
img.width = img.height = 39 |
|||
img.src = ico.up + (votes[i].orig == 1 ? ico.suppinact : ico.supp) |
|||
lin = el('a') |
|||
lin.href = '#' |
|||
lin.title = '+' |
|||
lin.appendChild(img) |
|||
td1.appendChild(lin) |
|||
td2 = el('td') |
|||
img = el('img') |
|||
img.alt = '−' |
|||
img.width = img.height = 39 |
|||
img.src = ico.up + (votes[i].orig == -1 ? ico.oppinact : ico.opp) |
|||
lin = el('a') |
|||
lin.href = '#' |
|||
lin.title = '-' |
|||
lin.appendChild(img) |
|||
td2.appendChild(lin) |
|||
td3 = el('td') |
|||
td3.className = 'talklink' |
|||
lin = el('a') |
|||
lin.href = wgServer + wgScript + '?title=' + encodeURI(conf.talkpath + i) |
|||
lin.target = '_blank' |
|||
lin.innerHTML = i |
|||
td3.appendChild(lin) |
|||
tr.appendChild(td1) |
|||
tr.appendChild(td2) |
|||
tr.appendChild(td3) |
|||
tab.appendChild(tr) |
|||
} |
|||
} |
|||
div1.appendChild(tab) |
|||
// IE has problems inserting the table |
|||
div1.innerHTML += '' |
|||
var imgs = div1.getElementsByTagName('img') |
|||
for (i=0; i<imgs.length; i++) |
|||
imgs[i].parentNode.onclick = oncl |
|||
/******* dragHere = el('span') |
|||
dragHere.id = 'voting-draghere' |
|||
dragHere.innerHTML = loc.draghere |
|||
var divetc = el('div') |
|||
divetc.id = 'voting-schulze' |
|||
var div2 = el('div') |
|||
div2.id = 'voting-etc-schulze' |
|||
div2.innerHTML = loc.etchelp1 |
|||
var ul = el('ul') |
|||
ul.id = 'schulze-list' |
|||
var gw, theid, can, first, level, spc |
|||
var c |
|||
for (i=1; i!=null; i==0 ? i=null : i++) { |
|||
gw = el('li') |
|||
gw.className = 'groundwork' |
|||
theid = el('div') |
|||
theid.className = 'li-id' |
|||
theid.innerHTML = !votes._s[i] ? i : (i==1 ? '' : i-1) + '…' + i |
|||
gw.appendChild(theid) |
|||
if (!votes._s[i]) i=0 |
|||
level = el('li') |
|||
level.className = 'level' |
|||
if (i==0) level.id = 'voting-no-preference' |
|||
if (i==1 && votes._s[i].join('!') == '') { |
|||
level.appendChild(dragHere) |
|||
} |
|||
theid = el('div') |
|||
theid.className = 'li-id' |
|||
theid.innerHTML = i==0 ? loc.nopreference : i |
|||
level.appendChild(theid) |
|||
for (c=0; c<votes._s[i].length; c++) { |
|||
can = el('a') |
|||
can.onclick = function() { return false } |
|||
can.onmousedown = boxMouseDown |
|||
can.innerHTML = votes._s[i][c] |
|||
can.className = 'cand' |
|||
level.appendChild(can) |
|||
} |
|||
level.appendChild(spacer()) |
|||
ul.appendChild(gw) |
|||
ul.appendChild(level) |
|||
} |
|||
div2.appendChild(ul) |
|||
divetc.appendChild(div2) |
|||
var div2 = el('div') |
|||
div2.id = 'voting-etc-reserve' |
|||
div2.innerHTML = loc.etchelp2 |
|||
var tab = el('table') |
|||
var tr, img, td1, td2, td3, lin |
|||
for (i in votes._r) { |
|||
tr = el('tr') |
|||
td1 = el('td') |
|||
img = el('img') |
|||
img.alt = '+' |
|||
img.width = img.height = 19 |
|||
img.src = ico.up + (votes._r[i].orig == 1 ? ico.suppinact : ico.supp) |
|||
lin = el('a') |
|||
lin.href = '#' |
|||
lin.title = '+' |
|||
lin.appendChild(img) |
|||
td1.appendChild(lin) |
|||
$(function () { |
|||
td2 = el('td') |
|||
var containerEl = document.getElementById( 'voting-container' ); |
|||
img = el('img') |
|||
if ( containerEl === null ) { |
|||
img.alt = '−' |
|||
mw.log.warn( 'arbcomVoting: Aborted because no voting container was detected.' ); |
|||
img.width = img.height = 19 |
|||
return; |
|||
img.src = ico.up + (votes._r[i].orig == -1 ? ico.oppinact : ico.opp) |
|||
} |
|||
lin = el('a') |
|||
lin.href = '#' |
|||
var wgUserName = mw.config.get( 'wgUserName' ); |
|||
lin.title = '-' |
|||
if ( wgUserName === null ) { |
|||
lin.appendChild(img) |
|||
mw.log.warn( 'arbcomVoting: Not logged in.' ); |
|||
td2.appendChild(lin) |
|||
return; |
|||
} |
|||
/******* COMMON ******* */ |
|||
function tstamp(t) { |
|||
return !t.getUTCFullYear() ? null : // Safari + Chrome |
|||
(t.getUTCFullYear()+":0"+(t.getUTCMonth()+1)+":0"+t.getUTCDate()+":0"+ |
|||
t.getUTCHours() +":0"+ t.getUTCMinutes() +":0"+t.getUTCSeconds()) |
|||
.replace(/:0?(\d\d)/g, '$1'); |
|||
} |
|||
function ch(o) { for (var i in o) { return o[i] } } |
|||
/******* VARIABLES *******/ |
|||
td3 = el('td') |
|||
td3.className = 'talklink' |
|||
lin = el('a') |
|||
lin.href = wgServer + wgScript + '?title=' + encodeURI(conf.talkpath + i) |
|||
lin.target = '_blank' |
|||
lin.innerHTML = i |
|||
var data = require( './ondemand-arbcomVoting.json' ); |
|||
td3.appendChild(lin) |
|||
var wgNamespaceNumber = mw.config.get( 'wgNamespaceNumber' ); |
|||
var votes = {}; |
|||
var saving = null; |
|||
var api; |
|||
var statusEl; |
|||
var btnEl; |
|||
/******* MAIN *******/ |
|||
tr.appendChild(td1) |
|||
function votingLength( date, days ) { |
|||
tr.appendChild(td2) |
|||
var result = new Date( date ); |
|||
tr.appendChild(td3) |
|||
result.setDate( result.getDate() + days ); |
|||
tab.appendChild(tr) |
|||
return result; |
|||
} |
|||
} |
|||
div2.appendChild(tab) |
|||
var imgs = div2.getElementsByTagName('img') |
|||
function votingStart() { |
|||
for (i=0; i<imgs.length; i++) { |
|||
$( '.voting-container' ).addClass( 'active' ); |
|||
imgs[i].src = imgs[i].src.replace(/39px/, '19px') |
|||
statusEl.textContent = data.strings.statuscriteria; |
|||
imgs[i].parentNode.onclick = onclr |
|||
btnEl.setDisabled(true); |
|||
} |
|||
btnEl.$element.css( 'display', 'none' ); |
|||
divetc.appendChild(div2) |
|||
api.get({ |
|||
'action': 'query', |
|||
var div2 = el('div') |
|||
'list': 'users', |
|||
div2.id = 'voting-etc-limited' |
|||
'usprop': ['registration', 'editcount'], |
|||
div2.innerHTML = loc.etchelp3 |
|||
'prop': 'revisions', |
|||
var count = 0 |
|||
'rvprop': 'content', |
|||
var hol = el('div') |
|||
'ususers': wgUserName, |
|||
hol.className = 'cand-list' |
|||
'titles': data.config.voteexceptionspath |
|||
for (i in votes._l) { |
|||
}).done(votingStartContinue); |
|||
one = el('a') |
|||
} |
|||
one.innerHTML = i |
|||
one.className = votes._l[i].orig ? 'cand cand-selected' : 'cand' |
|||
function votingStartContinue(d) { |
|||
count += votes._l[i].orig |
|||
query = d.query; |
|||
one.href = '#' |
|||
pages = query.pages; |
|||
one.onclick = oncll |
|||
userinfo = query.users[0]; |
|||
hol.appendChild(one) |
|||
if ((// a valid voter must be a non-anonymous user |
|||
} |
|||
typeof userinfo === 'undefined' || typeof userinfo.missing !== 'undefined' || |
|||
var cnt = el('div') |
|||
// who is not blocked |
|||
cnt.id = 'voting-limited-count' |
|||
userinfo.blockedby || |
|||
cnt.innerHTML = count |
|||
// whose editcount is at least data.config.criteria.count |
|||
if (count == 5) cnt.className = 'ok' |
|||
userinfo.editcount < data.config.criteria.count || |
|||
div2.appendChild(cnt) |
|||
// and who is registered no later than data.config.criteria.registration |
|||
div2.appendChild(hol) |
|||
// "null" means "before 2005-12-29", user creation was not logged before then |
|||
divetc.appendChild(div2) |
|||
(userinfo.registration !== null && userinfo.registration > data.config.criteria.registration)) && |
|||
// exemptions |
|||
('\n' + ch(pages).revisions[0]['*'] + '\n').indexOf('\n* %\n'.replace('%', wgUserName)) == -1 |
|||
var div2 = el('div') |
|||
) { |
|||
div2.innerHTML = loc.etchelp4 |
|||
votingCriteriaFail(); |
|||
} else { |
|||
divetc.appendChild(div2) |
|||
votingContinue(); |
|||
} |
|||
if (navigator.appName=='Microsoft Internet Explorer') { |
|||
} |
|||
// troubles with scopes |
|||
document.onmousemove = mouseMove |
|||
function votingCriteriaFail() { |
|||
document.onmouseup = mouseUp |
|||
statusEl.textContent = ''; |
|||
} else { |
|||
$( '#voting-msg-criteriafail' ).show(); |
|||
hookEvent('mousemove', mouseMove) |
|||
} |
|||
hookEvent('mouseup', mouseUp) |
|||
}*******/ |
|||
function votingContinue() { |
|||
statusEl.textContent = data.strings.loadingvotes; |
|||
status.innerHTML = loc.justbeforesave |
|||
// latest contributions in ns:4 |
|||
status.parentNode.insertBefore(div1, status) |
|||
api.get({ |
|||
//*******status.parentNode.insertBefore(divetc, status) |
|||
'action': 'query', |
|||
} |
|||
'list': 'usercontribs', |
|||
function votingToken() { |
|||
'ucnamespace': wgNamespaceNumber, |
|||
aj = sajax_init_object() |
|||
'uclimit': 500, |
|||
aj.onreadystatechange = votingTokenContinue |
|||
'ucuser': wgUserName, |
|||
aj.open('GET', wgAPIPath + 'action=query&prop=info&intoken=edit&titles=42', true) |
|||
'ucend': tstamp( data.config.start ), |
|||
aj.send('') |
|||
'ucdir': 'older' |
|||
} |
|||
}).done(votingDraw); |
|||
function votingTokenContinue() { |
|||
} |
|||
if (aj.readyState != 4) return |
|||
if (aj.readyState == 4 && aj.status != 200) { |
|||
function onClick( e ) { |
|||
votingToken() |
|||
e.preventDefault(); |
|||
return |
|||
var button = e.currentTarget; |
|||
} |
|||
var span = button.querySelector( 'span' ); |
|||
token = eval('(' + aj.responseText + ')') |
|||
var tr = button.parentNode.parentNode; |
|||
token = ch(token.query.pages).edittoken |
|||
var candidate = button.dataset.candidate; |
|||
votingSave() |
|||
var type = button.dataset.type; |
|||
} |
|||
var active = Boolean( button.getAttribute( 'aria-checked' ) ); |
|||
function votingSave() { |
|||
var option = data.options[ type ]; |
|||
if (saving == null) { |
|||
var div = el('div') |
|||
// Update other button first |
|||
div.id = 'voting-saving' |
|||
var otherButton = tr.querySelector( `.vote-button[aria-checked="true"]` ); |
|||
div.innerHTML = '<div id="voting-progress"><div id="voting-progress-progress" style="width:0%"> </div></div>' + loc.saveprog |
|||
var otherType = otherButton && otherButton.dataset.type; |
|||
status.parentNode.appendChild(div) |
|||
if ( otherButton ) { |
|||
btn.disabled = true |
|||
var otherVote = otherType === 'support' ? 1 : -1; |
|||
//document.getElementById('voting-schulze' ).style.visibility = 'hidden' |
|||
otherButton.setAttribute( 'aria-checked', false ); |
|||
document.getElementById('voting-standard').style.visibility = 'hidden' |
|||
} |
|||
saving = {'cursor': 0, 'pages': []} |
|||
// Change vote |
|||
var vote = type === 'support' ? 1 : -1; |
|||
for (var i in votes) { |
|||
var sameVote = votes[ candidate ].value === vote; |
|||
if (votes[i].value != 0 && votes[i].orig != votes[i].value) { |
|||
votes[ candidate ].value = ( sameVote ? 0 : vote ); |
|||
saving.pages[saving.cursor++] = {'text': '\n# [[user:' + wgUserName + '|' + wgUserName + ']] ~' + '~~' + '~~\n', |
|||
button.setAttribute( 'aria-checked', !sameVote ); |
|||
'page': conf.votepath + (votes[i].value==1?'+':'-') + '/' + i} |
|||
} |
|||
// Update table row |
|||
} |
|||
tr.className = !sameVote ? 'vote-' + type : ''; |
|||
/*******var text, text2, changed |
|||
} |
|||
var ms, mi |
|||
var li = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
function addVoteButton( candidate, type, previousVote ) { |
|||
var option = data.options[ type ]; |
|||
text = text2 = '\n* ' + wgUserName + ' | ~' + '~~' + '~~\n' |
|||
var didVote = previousVote === ( type === 'support' ? 1 : -1 ); |
|||
for (i=0; i<li.length; i++) { |
|||
var votedOpposite = previousVote === ( type === 'support' ? -1 : 1 ); |
|||
ms = li[i].getElementsByTagName('a') |
|||
var voteType = votedOpposite ? 'changevote' : 'novote'; |
|||
if (ms.length) { |
|||
mi = [] |
|||
var button = document.createElement( 'button' ); |
|||
for (var j=0; j<ms.length; j++) mi[mi.length] = ms[j].innerHTML |
|||
button.className = 'vote-button skin-invert'; |
|||
text += (li[i].id == 'voting-no-preference' ? '*: ' : '*# ') + mi.join(' | ') + '\n' |
|||
button.setAttribute( 'role', 'switch' ); |
|||
} |
|||
button.setAttribute( 'type', 'button' ); |
|||
} |
|||
button.setAttribute( 'data-candidate', candidate ); |
|||
button.setAttribute( 'data-type', type ); |
|||
for (i=1; i!=null; i==0 ? i=null : i++) { |
|||
if ( didVote ) { |
|||
if (!votes._s[i]) i=0 |
|||
button.setAttribute( 'data-voted', true ); |
|||
if (votes._s[i].join('.')) text2 += (i==0 ? '*: ' : '*# ') + votes._s[i].join(' | ') + '\n' |
|||
} |
|||
} |
|||
button.setAttribute( 'aria-checked', didVote ); |
|||
button.addEventListener( 'click', onClick ); |
|||
if (text != text2) { |
|||
saving.pages[saving.cursor++] = {'page': conf.votepath + '$', |
|||
var span = document.createElement( 'span' ); |
|||
'text': text} |
|||
span.textContent = option[ voteType ]; |
|||
} |
|||
if ( votedOpposite ) { |
|||
span.textContent = span.textContent.replace( '%candidate%', candidate ); |
|||
text = '\n* ' + wgUserName + ' | ~' + '~~' + '~~\n' |
|||
} |
|||
changed = false |
|||
button.title = span.textContent; |
|||
for (var i in votes._l) { |
|||
text += '*: ' + (votes._l[i].value ? '+' : '.') + ' ' + i + '\n' |
|||
button.appendChild( span ); |
|||
if (votes._l[i].orig != votes._l[i].value) |
|||
return button; |
|||
changed = true |
|||
} |
|||
} |
|||
if (changed) { |
|||
function votingDraw(d) { |
|||
saving.pages[saving.cursor++] = {'page': conf.votepath + '=', |
|||
statusEl.textContent = ''; |
|||
'text': text} |
|||
btnEl.setLabel(data.strings.savebutton); |
|||
} |
|||
btnEl.setDisabled(false); |
|||
btnEl.off('click').on('click', votingSave); |
|||
text = '\n* ' + wgUserName + ' | ~' + '~~' + '~~\n' |
|||
btnEl.$element.css( 'display', '' ); |
|||
changed = false |
|||
var vd = {99: '-', 100: '.', 101: '+'} |
|||
var co = ( d.query && d.query.usercontribs ) || []; |
|||
for (var i in votes._r) { |
|||
text += '*: ' + vd[votes._r[i].value + 100] + ' ' + i + '\n' |
|||
// retrieving votes, according to contributions, only latest ones are valid |
|||
if (votes._l[i].orig != votes._l[i].value) |
|||
for ( var i = 0; i < data.candidates.length; i++ ) { |
|||
changed = true |
|||
} |
votes[ data.candidates [ i ] ] = { 'orig': 0, 'value': 0 }; |
||
} |
|||
if (changed) { |
|||
for ( var i = co.length - 1; i >= 0; i-- ) { |
|||
saving.pages[saving.cursor++] = {'page': conf.votepath + '%', |
|||
var match = false; |
|||
'text': text} |
|||
if ( co[ i ].title.startsWith( data.config.votepath ) ) { |
|||
}*******/ |
|||
match = co[ i ].title.match( /\/([+-])\/(.*?)$/ ); |
|||
} |
|||
saving.cursor = -5 |
|||
if ( match ) { |
|||
if (saving.pages.length) |
|||
var hasSummary = co[ i ].comment === data.strings.summary; |
|||
votingSave() |
|||
if ( hasSummary ) { |
|||
else |
|||
votes[ match[ 2 ] ] = { |
|||
document.getElementById('voting-container').innerHTML = loc.thankyou |
|||
'orig': match[ 1 ] === '+' ? +1 : -1, |
|||
} else { |
|||
'value': 0 |
|||
if (saving.cursor != -5 && aj.readyState != 4) return |
|||
}; |
|||
if (aj.readyState == 4 && aj.status != 200) { |
|||
} |
|||
votingSave() |
|||
} |
|||
return |
|||
} |
|||
} |
|||
if (saving.cursor == -5) saving.cursor = 0 |
|||
var div1 = document.createElement( 'div' ); |
|||
if (saving.cursor == -1) return |
|||
div1.id = 'voting-standard'; |
|||
var help = document.createElement( 'p' ); |
|||
document.getElementById('voting-progress-progress').style.width = 100*(saving.cursor+1)/saving.pages.length + '%' |
|||
help.textContent = data.strings.votinghelp; |
|||
aj = sajax_init_object() |
|||
div1.appendChild( help ); |
|||
aj.open('POST', wgAPIPath + 'action=edit¬minor=1&unwatch=1&token=' + en(token) + '&summary=' + en(loc.summary) + |
|||
var tab = document.createElement( 'table' ); |
|||
'&title=' + en(saving.pages[saving.cursor].page) + |
|||
tab.className = 'voting-table'; |
|||
'&appendtext=' + en(saving.pages[saving.cursor].text), true) |
|||
aj.onreadystatechange = votingSave |
|||
var caption = document.createElement( 'caption' ); |
|||
aj.send('') |
|||
caption.textContent = data.strings.votebutton; |
|||
saving.cursor++ |
|||
tab.appendChild( caption ); |
|||
if (!saving.pages[saving.cursor]) { |
|||
saving.cursor = -1 |
|||
for ( var candidate in votes ) { |
|||
document.getElementById('voting-container').innerHTML = loc.thankyou |
|||
if ( candidate.match( /^_[srl]$/ ) ) { |
|||
} |
|||
continue; |
|||
} |
|||
} |
|||
} |
|||
var tr = document.createElement( 'tr' ); |
|||
// onclick() for round buttons |
|||
var th = document.createElement( 'th' ); |
|||
function oncl() { |
|||
th.setAttribute( 'scope', 'row' ); |
|||
var imgs = this.parentNode.parentNode.getElementsByTagName('img') |
|||
var lin = document.createElement('a'); |
|||
var link = this.parentNode.parentNode.getElementsByTagName('a')[2] |
|||
lin.href = mw.util.getUrl( data.config.talkpath + candidate ); |
|||
var ca = link.innerHTML |
|||
lin.target = '_blank'; |
|||
var ti = this.title |
|||
lin.textContent = candidate; |
|||
var vo = ti=='+'?1:-1 |
|||
lin.title = data.strings.questionsTooltip.replace( '%candidate%', candidate ); |
|||
votes[ca].value = (votes[ca].value==vo) ? 0 : vo |
|||
th.appendChild( lin ); |
|||
tr.appendChild( th ); |
|||
imgs[0].src = ico.up + (votes[ca].value== 1 ?ico.suppact:(votes[ca].orig== 1 ?ico.suppinact:ico.supp)) |
|||
imgs[1].src = ico.up + (votes[ca].value==-1 ?ico.oppact :(votes[ca].orig==-1 ?ico.oppinact :ico.opp)) |
|||
// Support button |
|||
var td1 = document.createElement( 'td' ); |
|||
link.className = ['opp', '', 'supp'][votes[ca].value + 1] |
|||
td1.appendChild( addVoteButton( candidate, 'support', votes[ candidate ].orig ) ); |
|||
tr.appendChild( td1 ); |
|||
return false |
|||
} |
|||
// Oppose button |
|||
var td2 = document.createElement( 'td' ); |
|||
function onclr() { |
|||
td2.appendChild( addVoteButton( candidate, 'oppose', votes[ candidate ].orig ) ); |
|||
var imgs = this.parentNode.parentNode.getElementsByTagName('img') |
|||
tr.appendChild( td2 ); |
|||
var link = this.parentNode.parentNode.getElementsByTagName('a')[2] |
|||
var ca = link.innerHTML |
|||
tab.appendChild( tr ); |
|||
var ti = this.title |
|||
} |
|||
var vo = ti=='+'?1:-1 |
|||
votes._r[ca].value = (votes._r[ca].value==vo) ? 0 : vo |
|||
div1.appendChild(tab); |
|||
imgs[0].src = ico.up + (votes._r[ca].value== 1 ?ico.suppact:(votes._r[ca].orig== 1 ?ico.suppinact:ico.supp)).replace(/39px/, '19px') |
|||
statusEl.textContent = ''; |
|||
imgs[1].src = ico.up + (votes._r[ca].value==-1 ?ico.oppact :(votes._r[ca].orig==-1 ?ico.oppinact :ico.opp)).replace(/39px/, '19px') |
|||
$( '#voting-msg-justbeforesave' ).show(); |
|||
statusEl.parentNode.insertBefore(div1, statusEl); |
|||
link.className = ['opp', '', 'supp'][votes._r[ca].value + 1] |
|||
} |
|||
return false |
|||
function showThankYou() { |
|||
} |
|||
containerEl.innerHTML = ''; |
|||
$( '.voting-msg' ).hide(); |
|||
function oncll() { |
|||
$( '#voting-msg-thankyou' ).show(); |
|||
if (this.className == 'cand') |
|||
} |
|||
this.className = 'cand cand-selected' |
|||
else |
|||
function votingSave() { |
|||
this.className = 'cand' |
|||
if (!saving) { |
|||
var div = document.createElement('div'); |
|||
var ca = this.innerHTML |
|||
div.className = 'voting-saving'; |
|||
div.textContent = data.strings.saveprog; |
|||
div.insertAdjacentHTML( 'afterbegin', '<div class="voting-progress"><div class="voting-progress-progress" style="width:0%"> </div></div>' ); |
|||
var count = 0 |
|||
statusEl.parentNode.appendChild( div ); |
|||
for (var i in votes._l) |
|||
btnEl.setDisabled( true ); |
|||
count += votes._l[i].value |
|||
$( '#voting-standard' ).hide(); |
|||
document.getElementById('voting-limited-count').innerHTML = count |
|||
saving = { 'cursor': 0, 'pages': [] }; |
|||
document.getElementById('voting-limited-count').className = (count==5)?'ok':'' |
|||
for (var i in votes) { |
|||
return false |
|||
if (votes[i].value != 0 && votes[i].orig != votes[i].value) { |
|||
} |
|||
saving.pages[saving.cursor++] = { |
|||
'text': '\n# [[user:' + wgUserName + '|' + wgUserName + ']] ~' + '~~' + '~~\n', |
|||
'page': data.config.votepath + (votes[i].value==1?'+':'-') + '/' + i |
|||
// used by drag-n-drop |
|||
}; |
|||
} |
|||
function disableSelection(el, g){ |
|||
} |
|||
// doesn't work for Opera, thus onmousemove drops selection ranges |
|||
el.onselectstart = g ? function(){ return false } : null |
|||
saving.cursor = -5; |
|||
el.unselectable = g ? "on" : "off" |
|||
if (saving.pages.length) { |
|||
el.style.MozUserSelect = g ? "none" : "" |
|||
votingSave(); |
|||
} |
|||
} else { |
|||
function absolutePosition(el) { |
|||
showThankYou(); |
|||
q = el |
|||
} |
|||
co = {'x': 0, 'y': 0} |
|||
} else { |
|||
while (q.offsetParent){ |
|||
if (saving.cursor === -5) { |
|||
co.x += q.offsetLeft |
|||
saving.cursor = 0; |
|||
co.y += q.offsetTop |
|||
} |
|||
q = q.offsetParent |
|||
if (saving.cursor === -1) { |
|||
} |
|||
return; |
|||
} |
|||
} |
|||
function coords(e) { |
|||
document.querySelector( '.voting-progress-progress' ).style.width = 100*(saving.cursor+1)/saving.pages.length + '%'; |
|||
if (e.pageX !== undefined) |
|||
return {'x': e.pageX, 'y': e.pageY} |
|||
api.postWithToken('edit', { |
|||
else // IE (buggy for v7 while zoomed, but who cares?) |
|||
'action': 'edit', |
|||
return { |
|||
'notminor': 1, |
|||
'x': e.clientX + document.documentElement.scrollLeft, |
|||
'unwatch': 1, |
|||
'y': e.clientY + document.documentElement.scrollTop |
|||
'assert': 'user', |
|||
} |
|||
'summary': data.strings.summary, |
|||
} |
|||
'title': saving.pages[saving.cursor].page, |
|||
function boxMouseDown() { |
|||
'appendtext': saving.pages[saving.cursor].text |
|||
if (dragHere && dragHere.parentNode) dragHere.parentNode.removeChild(dragHere) |
|||
}).done(votingSave); |
|||
saving.cursor++; |
|||
dragGhost = this |
|||
if (!saving.pages[saving.cursor]) { |
|||
dragGhost.id = 'voting-ghost' |
|||
saving.cursor = -1; |
|||
dragOriginal = dragItem = this.parentNode |
|||
showThankYou(); |
|||
dragItem.className += ' dragitem' |
|||
} |
|||
drag = document.createElement('a') |
|||
} |
|||
drag.innerHTML = this.innerHTML |
|||
} |
|||
drag.id = 'voting-drag' |
|||
drag.className = 'cand' |
|||
mw.loader.using(['mediawiki.api', 'mediawiki.util', 'oojs', 'oojs-ui']).done(function () { |
|||
api = new mw.Api(); |
|||
cooD = absolutePosition(dragGhost) |
|||
cooD.x -= coo.x |
|||
// Prepare dates |
|||
cooD.y -= coo.y |
|||
data.config.start = new Date(data.config.start); |
|||
mouseMove({'pageX': coo.x, 'pageY': coo.y}) |
|||
if ( typeof data.config.votelength !== 'undefined' ) { |
|||
data.config.end = votingLength( data.config.start, data.config.votelength ); |
|||
document.body.appendChild(drag) |
|||
} else { |
|||
data.config.end = new Date( data.config.end ); |
|||
disableSelection(document.body, true) |
|||
} |
|||
disableSelection(drag, true) |
|||
// Prepare pages paths |
|||
} |
|||
data.config.pagepath = data.config.pagepath.replace('%config.path%', data.config.path); |
|||
function mouseMove(e) { |
|||
data.config.talkpagepath = mw.util.getUrl( data.config.talkpagepath.replace('%config.path%', data.config.path) ); |
|||
coo = coords(e || window.event) |
|||
data.config.talkpath = data.config.talkpath.replace('%config.path%', data.config.path); |
|||
if (!drag) return |
|||
data.config.votepath = data.config.votepath.replace('%config.path%', data.config.path); |
|||
if (document.defaultView) // Opera: removing selection |
|||
data.config.voteexceptionspath = data.config.voteexceptionspath.replace('%config.path%', data.config.path); |
|||
document.defaultView.getSelection().removeAllRanges() |
|||
// Init |
|||
if ( containerEl ) { |
|||
drag.style.left = (coo.x + cooD.x) + "px" |
|||
containerEl.innerHTML = ''; |
|||
drag.style.top = (coo.y + cooD.y) + "px" |
|||
statusEl = document.createElement('div'); |
|||
var lis = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
statusEl.className = 'voting-status voting-msg'; |
|||
var dragCurrent |
|||
containerEl.appendChild( statusEl ); |
|||
for (i=0; i<lis.length; i++) { |
|||
$( '.voting-msg a' ).attr( 'target', '_blank' ); |
|||
pos = absolutePosition(lis[i]) |
|||
if (pos.x <= coo.x && |
|||
btnEl = new OO.ui.ButtonWidget( { |
|||
pos.y <= coo.y && |
|||
label: data.strings.votebutton, |
|||
pos.x + lis[i].offsetWidth > coo.x && |
|||
flags: ['primary', 'progressive'], |
|||
pos.y + lis[i].offsetHeight > coo.y) { |
|||
id: 'voting-button' |
|||
dragCurrent = lis[i] |
|||
} ); |
|||
} |
|||
btnEl.on('click', votingStart); |
|||
} |
|||
containerEl.appendChild(btnEl.$element.get(0)); |
|||
if (!dragCurrent) { // pointer outside the whole list, returning ghost to its original location |
|||
} |
|||
dragCurrent = dragOriginal |
|||
} ); |
|||
} |
|||
}); |
|||
if (dragCurrent != dragItem) { // changing target level |
|||
dragItem.className = dragItem.className.replace(' dragitem', '') |
|||
d = dragItem.getElementsByTagName('div')[0] |
|||
d.innerHTML = d.innerHTML.replace(/^([^:]+).*/, '$1') |
|||
dragItem.removeChild(dragGhost) |
|||
dragItem = dragCurrent |
|||
dragItem.className += ' dragitem' |
|||
if (dragItem.className.match(/groundwork/)) { |
|||
dragItem.getElementsByTagName('div')[0].innerHTML += ': ' + loc.newlevel |
|||
dragItem.appendChild(dragGhost) |
|||
} |
|||
else { // not an empty level, so inserting dragGhost in an appropriate place |
|||
var bef |
|||
var ch = dragItem.getElementsByTagName('a') |
|||
var spacer = dragItem.getElementsByTagName('span')[0] |
|||
if (ch.length == 0) { |
|||
bef = spacer |
|||
} else { |
|||
bef = ch[0] |
|||
for (i=0; i<ch.length; i++) |
|||
if (ch[i].innerHTML < dragGhost.innerHTML) |
|||
bef = ch[i+1] || spacer |
|||
} |
|||
dragItem.insertBefore(dragGhost, bef) |
|||
} |
|||
} |
|||
} |
|||
function spacer() { |
|||
var el = document.createElement('span') |
|||
el.className = 'spacer' |
|||
el.innerHTML = ' ' |
|||
return el |
|||
} |
|||
function mouseUp() { |
|||
if (!dragGhost || !drag || !dragItem) return |
|||
dragItem.className = dragItem.className.replace(' dragitem', '') |
|||
dragItem = null |
|||
dragGhost.id = '' |
|||
dragGhost = null |
|||
disableSelection(document.body, false) |
|||
disableSelection(drag, false) |
|||
document.body.removeChild(drag) |
|||
drag = null |
|||
// arranging structures |
|||
function groundwork() { |
|||
var el = document.createElement('li') |
|||
el.className = 'groundwork' |
|||
el.innerHTML = '<div class="li-id">!</div>' |
|||
return el |
|||
} |
|||
// creating groundworks around any newly created level |
|||
var lis = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
for (i=0; i<lis.length-1; i++) |
|||
if (lis[i].className == 'groundwork' && lis[i].getElementsByTagName('a').length > 0) { |
|||
lis[i].className = 'level' |
|||
lis[i].appendChild(spacer()) |
|||
lis[i].parentNode.insertBefore(groundwork(), lis[i].nextSibling) |
|||
lis[i].parentNode.insertBefore(groundwork(), lis[i]) |
|||
} |
|||
// destroying empty levels, except when there's only one |
|||
var lis = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
if (lis.length > 4) |
|||
for (i=0; i<lis.length-1; i++) |
|||
if (lis[i].className == 'level' && lis[i].getElementsByTagName('a').length == 0) |
|||
lis[i].parentNode.removeChild(lis[i]) |
|||
// blowing up inevitable duplicate groundworks |
|||
var lis = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
for (i=1; i<lis.length; i++) { |
|||
if (lis[i].className == 'groundwork' && lis[i-1].className == 'groundwork') |
|||
lis[i].parentNode.removeChild(lis[i]) |
|||
} |
|||
var lis = document.getElementById('schulze-list').getElementsByTagName('li') |
|||
// "drag first candidate here" label |
|||
if (lis[1].getElementsByTagName('a').length == 0) |
|||
lis[1].insertBefore(dragHere, lis[1].getElementsByTagName('span')[0]) |
|||
// renumbering |
|||
var j = 0 |
|||
for (i=0; i<lis.length-1; i++) { |
|||
lis[i].getElementsByTagName('div')[0].innerHTML = |
|||
lis[i].className == 'groundwork' |
|||
? (j==0 ? '…'+(++j) : (i==lis.length-2 ? ++j : j+'…'+(++j))) : j |
|||
} |
|||
} |
|||
document.getElementById('voting-container').innerHTML = '' |
|||
var btn = el('input') |
|||
btn.type = 'button' |
|||
btn.id = 'voting-button' |
|||
btn.value = loc.votebutton |
|||
btn.onclick = votingStart |
|||
var status = el('div') |
|||
status.id = 'voting-status' |
|||
document.getElementById('voting-container').appendChild(status) |
|||
document.getElementById('voting-container').appendChild(btn) |
|||
})() |
Текущая версия от 13:25, 1 августа 2024
/*
* Скрипт для выборов в Арбитражный комитет (ВП:АК)
*
* Авторы: Kalan, Serhio Magpie, stjn
*/
$(function () {
var containerEl = document.getElementById( 'voting-container' );
if ( containerEl === null ) {
mw.log.warn( 'arbcomVoting: Aborted because no voting container was detected.' );
return;
}
var wgUserName = mw.config.get( 'wgUserName' );
if ( wgUserName === null ) {
mw.log.warn( 'arbcomVoting: Not logged in.' );
return;
}
/******* COMMON ******* */
function tstamp(t) {
return !t.getUTCFullYear() ? null : // Safari + Chrome
(t.getUTCFullYear()+":0"+(t.getUTCMonth()+1)+":0"+t.getUTCDate()+":0"+
t.getUTCHours() +":0"+ t.getUTCMinutes() +":0"+t.getUTCSeconds())
.replace(/:0?(\d\d)/g, '$1');
}
function ch(o) { for (var i in o) { return o[i] } }
/******* VARIABLES *******/
var data = require( './ondemand-arbcomVoting.json' );
var wgNamespaceNumber = mw.config.get( 'wgNamespaceNumber' );
var votes = {};
var saving = null;
var api;
var statusEl;
var btnEl;
/******* MAIN *******/
function votingLength( date, days ) {
var result = new Date( date );
result.setDate( result.getDate() + days );
return result;
}
function votingStart() {
$( '.voting-container' ).addClass( 'active' );
statusEl.textContent = data.strings.statuscriteria;
btnEl.setDisabled(true);
btnEl.$element.css( 'display', 'none' );
api.get({
'action': 'query',
'list': 'users',
'usprop': ['registration', 'editcount'],
'prop': 'revisions',
'rvprop': 'content',
'ususers': wgUserName,
'titles': data.config.voteexceptionspath
}).done(votingStartContinue);
}
function votingStartContinue(d) {
query = d.query;
pages = query.pages;
userinfo = query.users[0];
if ((// a valid voter must be a non-anonymous user
typeof userinfo === 'undefined' || typeof userinfo.missing !== 'undefined' ||
// who is not blocked
userinfo.blockedby ||
// whose editcount is at least data.config.criteria.count
userinfo.editcount < data.config.criteria.count ||
// and who is registered no later than data.config.criteria.registration
// "null" means "before 2005-12-29", user creation was not logged before then
(userinfo.registration !== null && userinfo.registration > data.config.criteria.registration)) &&
// exemptions
('\n' + ch(pages).revisions[0]['*'] + '\n').indexOf('\n* %\n'.replace('%', wgUserName)) == -1
) {
votingCriteriaFail();
} else {
votingContinue();
}
}
function votingCriteriaFail() {
statusEl.textContent = '';
$( '#voting-msg-criteriafail' ).show();
}
function votingContinue() {
statusEl.textContent = data.strings.loadingvotes;
// latest contributions in ns:4
api.get({
'action': 'query',
'list': 'usercontribs',
'ucnamespace': wgNamespaceNumber,
'uclimit': 500,
'ucuser': wgUserName,
'ucend': tstamp( data.config.start ),
'ucdir': 'older'
}).done(votingDraw);
}
function onClick( e ) {
e.preventDefault();
var button = e.currentTarget;
var span = button.querySelector( 'span' );
var tr = button.parentNode.parentNode;
var candidate = button.dataset.candidate;
var type = button.dataset.type;
var active = Boolean( button.getAttribute( 'aria-checked' ) );
var option = data.options[ type ];
// Update other button first
var otherButton = tr.querySelector( `.vote-button[aria-checked="true"]` );
var otherType = otherButton && otherButton.dataset.type;
if ( otherButton ) {
var otherVote = otherType === 'support' ? 1 : -1;
otherButton.setAttribute( 'aria-checked', false );
}
// Change vote
var vote = type === 'support' ? 1 : -1;
var sameVote = votes[ candidate ].value === vote;
votes[ candidate ].value = ( sameVote ? 0 : vote );
button.setAttribute( 'aria-checked', !sameVote );
// Update table row
tr.className = !sameVote ? 'vote-' + type : '';
}
function addVoteButton( candidate, type, previousVote ) {
var option = data.options[ type ];
var didVote = previousVote === ( type === 'support' ? 1 : -1 );
var votedOpposite = previousVote === ( type === 'support' ? -1 : 1 );
var voteType = votedOpposite ? 'changevote' : 'novote';
var button = document.createElement( 'button' );
button.className = 'vote-button skin-invert';
button.setAttribute( 'role', 'switch' );
button.setAttribute( 'type', 'button' );
button.setAttribute( 'data-candidate', candidate );
button.setAttribute( 'data-type', type );
if ( didVote ) {
button.setAttribute( 'data-voted', true );
}
button.setAttribute( 'aria-checked', didVote );
button.addEventListener( 'click', onClick );
var span = document.createElement( 'span' );
span.textContent = option[ voteType ];
if ( votedOpposite ) {
span.textContent = span.textContent.replace( '%candidate%', candidate );
}
button.title = span.textContent;
button.appendChild( span );
return button;
}
function votingDraw(d) {
statusEl.textContent = '';
btnEl.setLabel(data.strings.savebutton);
btnEl.setDisabled(false);
btnEl.off('click').on('click', votingSave);
btnEl.$element.css( 'display', '' );
var co = ( d.query && d.query.usercontribs ) || [];
// retrieving votes, according to contributions, only latest ones are valid
for ( var i = 0; i < data.candidates.length; i++ ) {
votes[ data.candidates [ i ] ] = { 'orig': 0, 'value': 0 };
}
for ( var i = co.length - 1; i >= 0; i-- ) {
var match = false;
if ( co[ i ].title.startsWith( data.config.votepath ) ) {
match = co[ i ].title.match( /\/([+-])\/(.*?)$/ );
}
if ( match ) {
var hasSummary = co[ i ].comment === data.strings.summary;
if ( hasSummary ) {
votes[ match[ 2 ] ] = {
'orig': match[ 1 ] === '+' ? +1 : -1,
'value': 0
};
}
}
}
var div1 = document.createElement( 'div' );
div1.id = 'voting-standard';
var help = document.createElement( 'p' );
help.textContent = data.strings.votinghelp;
div1.appendChild( help );
var tab = document.createElement( 'table' );
tab.className = 'voting-table';
var caption = document.createElement( 'caption' );
caption.textContent = data.strings.votebutton;
tab.appendChild( caption );
for ( var candidate in votes ) {
if ( candidate.match( /^_[srl]$/ ) ) {
continue;
}
var tr = document.createElement( 'tr' );
var th = document.createElement( 'th' );
th.setAttribute( 'scope', 'row' );
var lin = document.createElement('a');
lin.href = mw.util.getUrl( data.config.talkpath + candidate );
lin.target = '_blank';
lin.textContent = candidate;
lin.title = data.strings.questionsTooltip.replace( '%candidate%', candidate );
th.appendChild( lin );
tr.appendChild( th );
// Support button
var td1 = document.createElement( 'td' );
td1.appendChild( addVoteButton( candidate, 'support', votes[ candidate ].orig ) );
tr.appendChild( td1 );
// Oppose button
var td2 = document.createElement( 'td' );
td2.appendChild( addVoteButton( candidate, 'oppose', votes[ candidate ].orig ) );
tr.appendChild( td2 );
tab.appendChild( tr );
}
div1.appendChild(tab);
statusEl.textContent = '';
$( '#voting-msg-justbeforesave' ).show();
statusEl.parentNode.insertBefore(div1, statusEl);
}
function showThankYou() {
containerEl.innerHTML = '';
$( '.voting-msg' ).hide();
$( '#voting-msg-thankyou' ).show();
}
function votingSave() {
if (!saving) {
var div = document.createElement('div');
div.className = 'voting-saving';
div.textContent = data.strings.saveprog;
div.insertAdjacentHTML( 'afterbegin', '<div class="voting-progress"><div class="voting-progress-progress" style="width:0%"> </div></div>' );
statusEl.parentNode.appendChild( div );
btnEl.setDisabled( true );
$( '#voting-standard' ).hide();
saving = { 'cursor': 0, 'pages': [] };
for (var i in votes) {
if (votes[i].value != 0 && votes[i].orig != votes[i].value) {
saving.pages[saving.cursor++] = {
'text': '\n# [[user:' + wgUserName + '|' + wgUserName + ']] ~' + '~~' + '~~\n',
'page': data.config.votepath + (votes[i].value==1?'+':'-') + '/' + i
};
}
}
saving.cursor = -5;
if (saving.pages.length) {
votingSave();
} else {
showThankYou();
}
} else {
if (saving.cursor === -5) {
saving.cursor = 0;
}
if (saving.cursor === -1) {
return;
}
document.querySelector( '.voting-progress-progress' ).style.width = 100*(saving.cursor+1)/saving.pages.length + '%';
api.postWithToken('edit', {
'action': 'edit',
'notminor': 1,
'unwatch': 1,
'assert': 'user',
'summary': data.strings.summary,
'title': saving.pages[saving.cursor].page,
'appendtext': saving.pages[saving.cursor].text
}).done(votingSave);
saving.cursor++;
if (!saving.pages[saving.cursor]) {
saving.cursor = -1;
showThankYou();
}
}
}
mw.loader.using(['mediawiki.api', 'mediawiki.util', 'oojs', 'oojs-ui']).done(function () {
api = new mw.Api();
// Prepare dates
data.config.start = new Date(data.config.start);
if ( typeof data.config.votelength !== 'undefined' ) {
data.config.end = votingLength( data.config.start, data.config.votelength );
} else {
data.config.end = new Date( data.config.end );
}
// Prepare pages paths
data.config.pagepath = data.config.pagepath.replace('%config.path%', data.config.path);
data.config.talkpagepath = mw.util.getUrl( data.config.talkpagepath.replace('%config.path%', data.config.path) );
data.config.talkpath = data.config.talkpath.replace('%config.path%', data.config.path);
data.config.votepath = data.config.votepath.replace('%config.path%', data.config.path);
data.config.voteexceptionspath = data.config.voteexceptionspath.replace('%config.path%', data.config.path);
// Init
if ( containerEl ) {
containerEl.innerHTML = '';
statusEl = document.createElement('div');
statusEl.className = 'voting-status voting-msg';
containerEl.appendChild( statusEl );
$( '.voting-msg a' ).attr( 'target', '_blank' );
btnEl = new OO.ui.ButtonWidget( {
label: data.strings.votebutton,
flags: ['primary', 'progressive'],
id: 'voting-button'
} );
btnEl.on('click', votingStart);
containerEl.appendChild(btnEl.$element.get(0));
}
} );
});