// ==UserScript==
// @name shitaraba_option
// @namespace john1851
// @description shitaraba_option
// @include http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete*
// @exclude http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete_confirm
// @exclude http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete_finish
// @grant none
// @version 0.5.0.00
// ==/UserScript==
(function (_, count) {
_('content', function (board, threads) {
console.log('今日は' + count + '回目の起動の日です。おめでとうございます。');
board.click(function (type, isForward) {
board.draw(sort(threads, type, isForward));
});
board.draw(threads);
});
function align(array, name, isForward, action) {
if (isForward)
return array.sort(function (x, y) { return action(x[name], y[name]); });
return array.sort(function (x, y) { return action(y[name], x[name]); });
}
function sort(threads, type, isForward) {
if (['no', 'title', 'response', 'newResponse', 'vigor'].indexOf(type) < 0)
throw new RangeError('not support type: ' + type);
if (type === 'title') {
return align(threads, type, isForward, function (x, y) {
return x < y ? -1 : x > y ? 1 : 0;
});
}
return align(threads, type, isForward, function (x, y) {
return x - y;
});
}
})((function () {
function array(str) {
return new Function('return [' + str + '];')();
}
function project(array, name) {
return array.map(function (x) { return x[name]; });
}
function zip() {
var results = [],
arrays = [].slice.call(arguments, 0),
length = Math.max.apply(null, project(arrays, 'length'));
for (var i = 0; i < length; i++) {
results.push(project(arrays, i));
}
return results;
}
function format(date) {
var m = (date.getMonth() + 1) + '/',
d = date.getDate() + '/',
h = date.getHours() + ':',
min = date.getMinutes();
return m + d + h + min;
}
function interval(date) {
var sec = Math.floor(new Date() / 1000 - date / 1000),
min, hour, day, mon, times, unit, result;
min = Math.floor(sec / 60);
sec = sec % 60;
hour = Math.floor(min / 60);
min = min % 60;
day = Math.floor(hour / 24);
hour = hour % 24;
mon = Math.floor(day / 30);
day = day % 30;
times = [mon, day, hour, min, sec];
unit = ['ヶ月', '日', '時間', '分', '秒'];
result = '';
for (var i = 0; i < times.length; i++) if (times[i] !== 0) {
result += times[i] + unit[i];
if (i != times.length - 1)
result += times[i + 1] + unit[i + 1];
break;
}
return result + '前';
}
function $(tag, option) {
var children = [].slice.call(arguments, 2),
element = document.createElement(tag);
$.option(element, option || {});
children.forEach(function (x) { element.appendChild(x); });
return element;
}
$.attribute = function (element, attributes) {
Object.keys(attributes).filter(function (x) {
return ['klass', 'text'].indexOf(x) < 0;
}).forEach(function (x) {
element.setAttribute(x, attributes[x]);
});
return element;
};
$.option = function (element, option) {
if (option.klass)
element.classList.add(option.klass);
if (option.text)
element.textContent = option.text;
return $.attribute(element, option);
};
$.find = function (id, child) {
if (child)
return document.getElementById(id).getElementsByTagName(child)[0];
return document.getElementById(id);
};
$.clone = function (node, option) {
var element = node.cloneNode(true);
return $.option(element, option || {});
};
$.prepend = function (node, child) {
node.insertBefore(child, node.firstElementChild);
return node;
};
function threads(id) {
var table = $.find(id).getElementsByTagName('table')[0],
tr = table.getElementsByTagName('tr'),
th = tr[0].children;
return [].slice.call(tr, 1).map(function (x) {
return x.children;
}).map(function (x, i) {
var date = new Date(),
response = parseInt(x[3].textContent, 10),
url = x[2].children[1].textContent.split('/').pop();
date.setTime(parseInt(url, 10) * 1000);
return {
check: x[0].children[0].checked,
no: parseInt(x[1].textContent, 10),
title: x[2].children[0].textContent,
url: url,
date: date,
response: response,
vigor: (function () {
var timestamp = new Date() / 1000 - date / 1000,
vigor = response / (timestamp / (60 * 60 * 24));
if (response <= 1)
return 0;
else if (response <= 10)
return vigor * response / 10;
return vigor;
})()
};
});
}
function view() {
return $('tbody', {},
$('tr', {},
$('th', { klass: 'checkbox', text: '削除' }),
$('th', { klass: 'no', 'data-type': 'no', text: 'No.', style: 'cursor: pointer;' }),
$('th', { klass: 'title', 'data-type': 'title', text: 'タイトル', style: 'cursor: pointer;' }),
$('th', { klass: 'resNumber', 'data-type': 'response', text: 'レス数', style: 'cursor: pointer;' }),
$('th', { klass: 'resNumber', 'data-type': 'newResponse', text: '新レス数', style: 'cursor: pointer;' }),
$('th', { klass: 'resNumber', 'data-type': 'vigor', text: '勢い', style: 'cursor: pointer;' }),
$('th', { klass: 'individualDeletion' }),
$('th', { klass: 'infectionDeletion' })
)
);
}
view.row = function (thread, i) {
function checkbox(url) {
return $('td', { klass: 'checkbox' },
$('input', {
type: 'checkbox', name: 'key_' + url
})
);
}
function no(no) {
return $('td', { klass: 'no', text: no });
}
function title(title, url, date, isNew) {
var element = $('td', { klass: 'title' },
$('a', {
target: '_blank', text: title,
href: 'http://j...content-available-to-author-only...a.net/bbs/read.cgi/internet/' + url + '/1405372778/l50'
}),
$('p', {
klass: 'searchURL', text: format(date) + ' '
}, $('b', { text: interval(date) }))
);
if (isNew)
$.prepend(element, $('span', { text: '(New) ', style: 'color: #eb6101;' }));
return element;
}
function response(response) {
return $('td', { klass: 'resNumber', text: response });
}
function newResponse(newResponse, isNew) {
var element = $('td', { klass: 'resNumber' },
$('i', { text: '(' + newResponse + ')' })
);
if (isNew)
$.option(element.firstElementChild, { style: 'color: red;' });
else
$.option(element.firstElementChild, { style: 'color: blue;' });
return element;
}
function vigor(vigor) {
return $('td', { klass: 'resNumber', style: 'overflow: hidden;' },
$('b', {
text: (function () {
if (vigor < 1)
return vigor.toFixed(2);
else if (vigor < 100)
return vigor.toFixed(1);
else if (vigor < 1000)
return vigor.toFixed(0);
return '999+';
})()
})
);
}
function save(url) {
return $('td', { klass: 'individualDeletion' },
$('a', {
text: '倉庫へ送る',
href: './thread_delete_confirm?key=' + url + '&subcommand=save'
})
);
}
function remove(url) {
return $('td', { klass: 'infectionDeletion' },
$('a', {
text: '完全削除',
href: './thread_delete_confirm?key=' + url + '&subcommand=remove'
})
);
}
return $('tr', { klass: i % 2 === 0 ? 'even' : 'odd' },
checkbox(thread.url),
no(thread.no),
title(thread.title, thread.url, thread.date, thread.isNew),
response(thread.response),
newResponse(thread.newResponse, thread.isNew),
vigor(thread.vigor),
save(thread.url),
remove(thread.url)
);
};
function board(id) {
var table = $.find(id, 'table');
table.replaceChild(view(), table.firstElementChild);
return {
draw: function (threads) {
[].slice.call(table.firstElementChild.children, 1).forEach(function (x) {
table.firstElementChild.removeChild(x);
});
threads.forEach(function (x, i) {
table.firstElementChild.appendChild(view.row(x, i));
});
},
click: function (callback) {
[].slice.call(table.firstElementChild.firstElementChild.children, 1, -2).forEach(function (x) {
x.addEventListener('click', function () {
var type = x.dataset['type'],
value = sessionStorage[type],
isForward = typeof value === 'undefined' ? true : value === 'true';
callback(type, isForward);
sessionStorage[type] = !isForward;
}, false);
});
}
};
}
function autoCheck(id) {
var form = $.find(id, 'table').parentNode,
button = $('input', { type: 'button', value: '自動チェック' }),
tip = $('b', { text: '現在の総スレ数: ' + threads(id).length });
button.addEventListener('click', function () {
var elements = document.forms[0].elements,
value = sessionStorage['autoCheck'],
toggle = typeof value === 'undefined' ? true : value === 'true';
if (toggle)
{
zip(threads(id), elements).filter(function (x) {
return typeof x[0] !== 'undefined';
}).sort(function (x, y) {
return x[0].vigor - y[0].vigor;
}).slice(0, 30).forEach(function (x) {
x[1].checked = true;
});
}
else
{
for (var i = 0, length = elements.length; i < length; i++) {
elements[i].checked = false;
}
}
sessionStorage['autoCheck'] = !toggle;
}, false);
form.parentNode.insertBefore(button, form);
form.parentNode.insertBefore(tip, form);
}
return function (id, callback) {
var _threads = threads(id),
responses = array(localStorage['responses'] || project(_threads, 'response'));
responses = new Array(_threads.length - responses.length).concat(responses);
zip(_threads, responses).forEach(function (x) {
x[0].newResponse = x[0].response - (x[1] || 0);
x[0].isNew = typeof x[1] === 'undefined';
});
autoCheck(id);
callback(board(id), _threads);
localStorage['responses'] = project(_threads, 'response').toString();
localStorage['count'] = (parseInt(localStorage['count'], 10) || 1) + 1;
};
})(), parseInt(localStorage['count'], 10) || 1);
// ==UserScript==
// @name        shitaraba_option
// @namespace   john1851
// @description shitaraba_option
// @include     http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete*
// @exclude     http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete_confirm
// @exclude     http://c...content-available-to-author-only...a.net/internet/21187/config/data/thread_delete_finish
// @grant       none
// @version     0.5.0.00
// ==/UserScript==


(function (_, count) {
	_('content', function (board, threads) {
		console.log('今日は' + count + '回目の起動の日です。おめでとうございます。');

		board.click(function (type, isForward) {
			board.draw(sort(threads, type, isForward));
		});
		board.draw(threads);
	});

	function align(array, name, isForward, action) {
		if (isForward)
			return array.sort(function (x, y) { return action(x[name], y[name]); });
		return array.sort(function (x, y) { return action(y[name], x[name]); });
	}
	function sort(threads, type, isForward) {
		if (['no', 'title', 'response', 'newResponse', 'vigor'].indexOf(type) < 0)
			throw new RangeError('not support type: ' + type);

		if (type === 'title') {
			return align(threads, type, isForward, function (x, y) {
				return x < y ? -1 : x > y ? 1 : 0;
			});
		}
		return align(threads, type, isForward, function (x, y) {
			return x - y;
		});
	}
})((function () {
	function array(str) {
		return new Function('return [' + str + '];')();
	}
	function project(array, name) {
		return array.map(function (x) { return x[name]; });
	}
	function zip() {
		var results = [],
			arrays = [].slice.call(arguments, 0),
			length = Math.max.apply(null, project(arrays, 'length'));
		for (var i = 0; i < length; i++) {
			results.push(project(arrays, i));
		}
		return results;
	}
	function format(date) {
		var m = (date.getMonth() + 1) + '/',
			d = date.getDate() + '/',
			h = date.getHours() + ':',
			min = date.getMinutes();
		return m + d + h + min;
	}
	function interval(date) {
		var sec = Math.floor(new Date() / 1000 - date / 1000),
			min, hour, day, mon, times, unit, result;
		min = Math.floor(sec / 60);
		sec = sec % 60;
		hour = Math.floor(min / 60);
		min = min % 60;
		day = Math.floor(hour / 24);
		hour = hour % 24;
		mon = Math.floor(day / 30);
		day = day % 30;

		times = [mon, day, hour, min, sec];
		unit = ['ヶ月', '日', '時間', '分', '秒'];
		result = '';
		for (var i = 0; i < times.length; i++) if (times[i] !== 0) {
			result += times[i] + unit[i];
			if (i != times.length - 1)
				result += times[i + 1] + unit[i + 1];
			break;
		}
		return result + '前';
	}

	function $(tag, option) {
		var children = [].slice.call(arguments, 2),
			element = document.createElement(tag);
		$.option(element, option || {});
		children.forEach(function (x) { element.appendChild(x); });
		return element;
	}
	$.attribute = function (element, attributes) {
		Object.keys(attributes).filter(function (x) {
			return ['klass', 'text'].indexOf(x) < 0;
		}).forEach(function (x) {
			element.setAttribute(x, attributes[x]);
		});
		return element;
	};
	$.option = function (element, option) {
		if (option.klass)
			element.classList.add(option.klass);
		if (option.text)
			element.textContent = option.text;
		return $.attribute(element, option);
	};
	$.find = function (id, child) {
		if (child)
			return document.getElementById(id).getElementsByTagName(child)[0];
		return document.getElementById(id);
	};
	$.clone = function (node, option) {
		var element = node.cloneNode(true);
		return $.option(element, option || {});
	};
	$.prepend = function (node, child) {
		node.insertBefore(child, node.firstElementChild);
		return node;
	};

	function threads(id) {
		var table = $.find(id).getElementsByTagName('table')[0],
			tr = table.getElementsByTagName('tr'),
			th = tr[0].children;

		return [].slice.call(tr, 1).map(function (x) {
			return x.children;
		}).map(function (x, i) {
			var date = new Date(),
				response = parseInt(x[3].textContent, 10),
				url = x[2].children[1].textContent.split('/').pop();
			date.setTime(parseInt(url, 10) * 1000);
			return {
				check: x[0].children[0].checked,
				no: parseInt(x[1].textContent, 10),
				title: x[2].children[0].textContent,
				url: url,
				date: date,
				response: response,
				vigor: (function () {
					var timestamp = new Date() / 1000 - date / 1000,
						vigor = response / (timestamp / (60 * 60 * 24));
					if (response <= 1)
						return 0;
					else if (response <= 10)
						return vigor * response / 10;
					return vigor;
				})()
			};
		});
	}

	function view() {
		return $('tbody', {},
			$('tr', {},
				$('th', { klass: 'checkbox', text: '削除' }),
				$('th', { klass: 'no', 'data-type': 'no', text: 'No.', style: 'cursor: pointer;' }),
				$('th', { klass: 'title', 'data-type': 'title', text: 'タイトル', style: 'cursor: pointer;' }),
				$('th', { klass: 'resNumber', 'data-type': 'response', text: 'レス数', style: 'cursor: pointer;' }),
				$('th', { klass: 'resNumber', 'data-type': 'newResponse', text: '新レス数', style: 'cursor: pointer;' }),
				$('th', { klass: 'resNumber', 'data-type': 'vigor', text: '勢い', style: 'cursor: pointer;' }),
				$('th', { klass: 'individualDeletion' }),
				$('th', { klass: 'infectionDeletion' })
			)
		);
	}
	view.row = function (thread, i) {
		function checkbox(url) {
			return $('td', { klass: 'checkbox' },
				$('input', {
					type: 'checkbox', name: 'key_' + url
				})
			);
		}
		function no(no) {
			return $('td', { klass: 'no', text: no });
		}
		function title(title, url, date, isNew) {
			var element = $('td', { klass: 'title' },
				$('a', {
					target: '_blank', text: title,
					href: 'http://j...content-available-to-author-only...a.net/bbs/read.cgi/internet/' + url + '/1405372778/l50'
				}),
				$('p', {
					klass: 'searchURL', text: format(date) + ' '
				}, $('b', { text: interval(date) }))
			);
			if (isNew)
				$.prepend(element, $('span', { text: '(New) ', style: 'color: #eb6101;' }));
			return element;
		}
		function response(response) {
			return $('td', { klass: 'resNumber', text: response });
		}
		function newResponse(newResponse, isNew) {
			var element = $('td', { klass: 'resNumber' },
				$('i', { text: '(' + newResponse + ')' })
			);
			if (isNew)
				$.option(element.firstElementChild, { style: 'color: red;' });
			else
				$.option(element.firstElementChild, { style: 'color: blue;' });
			return element;
		}
		function vigor(vigor) {
			return $('td', { klass: 'resNumber', style: 'overflow: hidden;' },
				$('b', {
					text: (function () {
						if (vigor < 1)
							return vigor.toFixed(2);
						else if (vigor < 100)
							return vigor.toFixed(1);
						else if (vigor < 1000)
							return vigor.toFixed(0);
						return '999+';
					})()
				})
			);
		}
		function save(url) {
			return $('td', { klass: 'individualDeletion' },
				$('a', {
					text: '倉庫へ送る',
					href: './thread_delete_confirm?key=' + url + '&subcommand=save'
				})
			);
		}
		function remove(url) {
			return $('td', { klass: 'infectionDeletion' },
				$('a', {
					text: '完全削除',
					href: './thread_delete_confirm?key=' + url + '&subcommand=remove'
				})
			);
		}

		return $('tr', { klass: i % 2 === 0 ? 'even' : 'odd' },
			checkbox(thread.url),
			no(thread.no),
			title(thread.title, thread.url, thread.date, thread.isNew),
			response(thread.response),
			newResponse(thread.newResponse, thread.isNew),
			vigor(thread.vigor),
			save(thread.url),
			remove(thread.url)
		);
	};

	function board(id) {
		var table = $.find(id, 'table');
		table.replaceChild(view(), table.firstElementChild);
		return {
			draw: function (threads) {
				[].slice.call(table.firstElementChild.children, 1).forEach(function (x) {
					table.firstElementChild.removeChild(x);
				});
				threads.forEach(function (x, i) {
					table.firstElementChild.appendChild(view.row(x, i));
				});
			},
			click: function (callback) {
				[].slice.call(table.firstElementChild.firstElementChild.children, 1, -2).forEach(function (x) {
					x.addEventListener('click', function () {
						var type = x.dataset['type'],
							value = sessionStorage[type],
							isForward = typeof value === 'undefined' ? true : value === 'true';
						callback(type, isForward);
						sessionStorage[type] = !isForward;
					}, false);
				});
			}
		};
	}

	function autoCheck(id) {
		var form = $.find(id, 'table').parentNode,
			button = $('input', { type: 'button', value: '自動チェック' }),
			tip = $('b', { text: '現在の総スレ数: ' + threads(id).length });

		button.addEventListener('click', function () {
			var elements = document.forms[0].elements,
				value = sessionStorage['autoCheck'],
				toggle = typeof value === 'undefined' ? true : value === 'true';
			if (toggle)
			{
				zip(threads(id), elements).filter(function (x) {
					return typeof x[0] !== 'undefined';
				}).sort(function (x, y) {
					return x[0].vigor - y[0].vigor;
				}).slice(0, 30).forEach(function (x) {
					x[1].checked = true;
				});
			}
			else
			{
				for (var i = 0, length = elements.length; i < length; i++) {
					elements[i].checked = false;
				}
			}
			sessionStorage['autoCheck'] = !toggle;
		}, false);
		form.parentNode.insertBefore(button, form);
		form.parentNode.insertBefore(tip, form);
	}

	return function (id, callback) {
		var _threads = threads(id),
			responses = array(localStorage['responses'] || project(_threads, 'response'));
		responses = new Array(_threads.length - responses.length).concat(responses);

		zip(_threads, responses).forEach(function (x) {
			x[0].newResponse = x[0].response - (x[1] || 0);
			x[0].isNew = typeof x[1] === 'undefined';
		});

		autoCheck(id);
		callback(board(id), _threads);

		localStorage['responses'] = project(_threads, 'response').toString();
		localStorage['count'] = (parseInt(localStorage['count'], 10) || 1) + 1;
	};
})(), parseInt(localStorage['count'], 10) || 1);
