2042 lines
60 KiB
CoffeeScript
2042 lines
60 KiB
CoffeeScript
# TODO
|
|
# option to skip post form directly to contents on first page,
|
|
# like what happens when using thread nav to go to next page
|
|
# (floating) qr no-quote button?
|
|
# updater cache hacks
|
|
|
|
# XXX chrome can't into `{log} = console`
|
|
if console?
|
|
log = (arg) ->
|
|
console.log arg
|
|
|
|
config =
|
|
main:
|
|
monitor:
|
|
'Thread Updater': [true, 'Update threads']
|
|
'Unread Count': [true, 'Show unread post count in tab title']
|
|
'Post in Title': [true, 'Show the op\'s post in the tab title']
|
|
'Thread Watcher': [true, 'Bookmark threads']
|
|
'Auto Watch': [true, 'Automatically watch threads that you start']
|
|
'Auto Watch Reply': [false, 'Automatically watch threads that you reply to']
|
|
img:
|
|
'Image Auto-Gif': [false, 'Animate gif thumbnails']
|
|
'Image Expansion': [true, 'Expand images']
|
|
'Image Hover': [false, 'Show full image on mouseover']
|
|
'Image Preloading': [false, 'Preload Images']
|
|
'Sauce': [true, 'Add sauce to images']
|
|
post:
|
|
'Cooldown': [false, 'Prevent \'flood detected\' errors (buggy)']
|
|
'Quick Reply': [true, 'Reply without leaving the page']
|
|
'Persistent QR': [false, 'Quick reply won\'t disappear after posting. Only in replies.']
|
|
quote:
|
|
'Quote Backlinks': [false, 'Add quote backlinks']
|
|
'Quote Inline': [false, 'Show quoted post inline on quote click']
|
|
'Quote Preview': [false, 'Show quote content on hover']
|
|
hide:
|
|
'Reply Hiding': [true, 'Hide single replies']
|
|
'Thread Hiding': [true, 'Hide entire threads']
|
|
'Show Stubs': [true, 'Of hidden threads / replies']
|
|
misc:
|
|
'404 Redirect': [true, 'Redirect dead threads']
|
|
'Anonymize': [false, 'Make everybody anonymous']
|
|
'Keybinds': [false, 'Binds actions to keys']
|
|
'Localize Time': [true, 'Show times based on your timezone']
|
|
'Report Button': [true, 'Add report buttons']
|
|
'Comment Expansion': [true, 'Expand too long comments']
|
|
'Thread Expansion': [true, 'View all replies']
|
|
'Thread Navigation': [true, 'Navigate to previous / next thread']
|
|
textarea:
|
|
flavors: [
|
|
'http://regex.info/exif.cgi?url='
|
|
'http://iqdb.org/?url='
|
|
'http://tineye.com/search?url='
|
|
'#http://saucenao.com/search.php?db=999&url='
|
|
].join '\n'
|
|
updater:
|
|
checkbox:
|
|
'Verbose': [true, 'Show countdown timer, new post count']
|
|
'Auto Update': [false, 'Automatically fetch new posts']
|
|
'Interval': 30
|
|
|
|
# FIXME this is fucking horrible
|
|
# flatten the config
|
|
_config = {}
|
|
((parent, obj) ->
|
|
if obj.length #array
|
|
if typeof obj[0] is 'boolean'
|
|
_config[parent] = obj[0]
|
|
else
|
|
_config[parent] = obj
|
|
else if typeof obj is 'object'
|
|
for key, val of obj
|
|
arguments.callee key, val
|
|
else #constant
|
|
_config[parent] = obj
|
|
) null, config
|
|
|
|
ui =
|
|
dialog: (id, position, html) ->
|
|
el = document.createElement 'div'
|
|
el.className = 'reply dialog'
|
|
el.innerHTML = html
|
|
el.id = id
|
|
{left, top} = position
|
|
left = localStorage["#{id}Left"] ? left
|
|
top = localStorage["#{id}Top"] ? top
|
|
if left then el.style.left = left else el.style.right = '0px'
|
|
if top then el.style.top = top else el.style.bottom = '0px'
|
|
el.querySelector('div.move').addEventListener 'mousedown', ui.dragstart, false
|
|
el.querySelector('div.move a[name=close]')?.addEventListener 'click',
|
|
(-> el.parentNode.removeChild(el)), true
|
|
el
|
|
dragstart: (e) ->
|
|
#prevent text selection
|
|
e.preventDefault()
|
|
ui.el = el = @parentNode
|
|
d = document
|
|
d.addEventListener 'mousemove', ui.drag, true
|
|
d.addEventListener 'mouseup', ui.dragend, true
|
|
#distance from pointer to el edge is constant; calculate it here.
|
|
# XXX opera reports el.offsetLeft / el.offsetTop as 0
|
|
rect = el.getBoundingClientRect()
|
|
ui.dx = e.clientX - rect.left
|
|
ui.dy = e.clientY - rect.top
|
|
#factor out el from document dimensions
|
|
ui.width = document.body.clientWidth - el.offsetWidth
|
|
ui.height = document.body.clientHeight - el.offsetHeight
|
|
drag: (e) ->
|
|
e.preventDefault()
|
|
{el} = ui
|
|
left = e.clientX - ui.dx
|
|
if left < 10 then left = '0px'
|
|
else if ui.width - left < 10 then left = ''
|
|
right = if left then '' else '0px'
|
|
el.style.left = left
|
|
el.style.right = right
|
|
top = e.clientY - ui.dy
|
|
if top < 10 then top = '0px'
|
|
else if ui.height - top < 10 then top = ''
|
|
bottom = if top then '' else '0px'
|
|
el.style.top = top
|
|
el.style.bottom = bottom
|
|
dragend: ->
|
|
#{id} = {el} = ui
|
|
#equivalent to
|
|
#{id} = ui; {el} = ui
|
|
{el} = ui
|
|
{id} = el
|
|
localStorage["#{id}Left"] = el.style.left
|
|
localStorage["#{id}Top"] = el.style.top
|
|
d = document
|
|
d.removeEventListener 'mousemove', ui.drag, true
|
|
d.removeEventListener 'mouseup', ui.dragend, true
|
|
hover: (e) ->
|
|
{clientX, clientY} = e
|
|
{el} = ui
|
|
height = el.offsetHeight
|
|
|
|
top = clientY - 120
|
|
bot = top + height
|
|
el.style.top =
|
|
if ui.winHeight < height or top < 0
|
|
'0px'
|
|
else if bot > ui.winHeight
|
|
ui.winHeight - height + 'px'
|
|
else
|
|
top + 'px'
|
|
el.style.left = clientX + 45
|
|
hoverend: (e) ->
|
|
$.hide ui.el
|
|
|
|
#convenience
|
|
d = document
|
|
g = null #globals
|
|
|
|
#utility
|
|
$ = (selector, root=d.body) ->
|
|
root.querySelector selector
|
|
|
|
$.extend = (object, properties) ->
|
|
for key, val of properties
|
|
object[key] = val
|
|
object
|
|
|
|
$.extend $,
|
|
globalEval: (code) ->
|
|
script = $.el 'script',
|
|
textContent: "(#{code})()"
|
|
$.append d.head, script
|
|
$.rm script
|
|
get: (url, cb) ->
|
|
r = new XMLHttpRequest()
|
|
r.onload = cb
|
|
r.open 'get', url, true
|
|
r.send()
|
|
r
|
|
cb:
|
|
checked: ->
|
|
$.setValue @name, @checked
|
|
value: ->
|
|
$.setValue @name, @value
|
|
addStyle: (css) ->
|
|
style = document.createElement 'style'
|
|
style.type = 'text/css'
|
|
style.textContent = css
|
|
$.append d.head, style
|
|
config: (name) ->
|
|
$.getValue name, _config[name]
|
|
zeroPad: (n) ->
|
|
if n < 10 then '0' + n else n
|
|
x: (path, root=d.body) ->
|
|
d.evaluate(path, root, null, XPathResult.ANY_UNORDERED_NODE_TYPE, null).
|
|
singleNodeValue
|
|
tn: (s) ->
|
|
d.createTextNode s
|
|
replace: (root, el) ->
|
|
root.parentNode.replaceChild el, root
|
|
hide: (el) ->
|
|
el.style.display = 'none'
|
|
show: (el) ->
|
|
el.style.display = ''
|
|
addClass: (el, className) ->
|
|
el.className += ' ' + className
|
|
removeClass: (el, className) ->
|
|
el.className = el.className.replace ' ' + className, ''
|
|
rm: (el) ->
|
|
el.parentNode.removeChild el
|
|
append: (parent, children...) ->
|
|
for child in children
|
|
parent.appendChild child
|
|
prepend: (parent, child) ->
|
|
parent.insertBefore child, parent.firstChild
|
|
after: (root, el) ->
|
|
root.parentNode.insertBefore el, root.nextSibling
|
|
before: (root, el) ->
|
|
root.parentNode.insertBefore el, root
|
|
el: (tag, properties) ->
|
|
el = d.createElement tag
|
|
$.extend el, properties if properties
|
|
el
|
|
bind: (el, eventType, handler) ->
|
|
el.addEventListener eventType, handler, true
|
|
unbind: (el, eventType, handler) ->
|
|
el.removeEventListener eventType, handler, true
|
|
isDST: ->
|
|
# XXX this should check for DST in NY
|
|
###
|
|
http://en.wikipedia.org/wiki/Daylight_saving_time_in_the_United_States
|
|
Since 2007, daylight saving time starts on the second Sunday of March
|
|
and ends on the first Sunday of November, with all time changes taking
|
|
place at 2:00 AM (0200) local time.
|
|
###
|
|
|
|
date = new Date()
|
|
month = date.getMonth()
|
|
|
|
#this is the easy part
|
|
if month < 2 or 10 < month
|
|
return false
|
|
if 2 < month < 10
|
|
return true
|
|
|
|
# (sunday's date) = (today's date) - (number of days past sunday)
|
|
# date is not zero-indexed
|
|
sunday = date.getDate() - date.getDay()
|
|
|
|
if month is 2
|
|
#before second sunday
|
|
if sunday < 8
|
|
return false
|
|
|
|
#during second sunday
|
|
if sunday < 15 and date.getDay() is 0
|
|
if date.getHour() < 1
|
|
return false
|
|
return true
|
|
|
|
#after second sunday
|
|
return true
|
|
|
|
if month is 10
|
|
# before first sunday
|
|
if sunday < 1
|
|
return true
|
|
|
|
# during first sunday
|
|
if sunday < 8 and date.getDay() is 0
|
|
if date.getHour() < 1
|
|
return true
|
|
return false
|
|
|
|
#after first sunday
|
|
return false
|
|
|
|
if GM_deleteValue?
|
|
$.extend $,
|
|
deleteValue: (name) ->
|
|
name = NAMESPACE + name
|
|
GM_deleteValue name
|
|
getValue: (name, defaultValue) ->
|
|
name = NAMESPACE + name
|
|
if value = GM_getValue name
|
|
JSON.parse value
|
|
else
|
|
defaultValue
|
|
openInTab: (url) ->
|
|
GM_openInTab url
|
|
setValue: (name, value) ->
|
|
name = NAMESPACE + name
|
|
GM_setValue name, JSON.stringify value
|
|
else
|
|
$.extend $,
|
|
deleteValue: (name) ->
|
|
name = NAMESPACE + name
|
|
delete localStorage[name]
|
|
getValue: (name, defaultValue) ->
|
|
name = NAMESPACE + name
|
|
if value = localStorage[name]
|
|
JSON.parse value
|
|
else
|
|
defaultValue
|
|
openInTab: (url) ->
|
|
window.open url, "_blank"
|
|
setValue: (name, value) ->
|
|
name = NAMESPACE + name
|
|
localStorage[name] = JSON.stringify value
|
|
|
|
# XXX opera cannot into Object.keys
|
|
if not Object.keys
|
|
Object.keys = (o) ->
|
|
key for key in o
|
|
|
|
$$ = (selector, root=d.body) ->
|
|
Array::slice.call root.querySelectorAll selector
|
|
|
|
#funks
|
|
expandComment =
|
|
init: ->
|
|
for a in $$ 'span.abbr a'
|
|
$.bind a, 'click', expandComment.expand
|
|
expand: (e) ->
|
|
e.preventDefault()
|
|
[_, threadID, replyID] = @href.match /(\d+)#(\d+)/
|
|
@textContent = "Loading #{replyID}..."
|
|
if req = g.requests[threadID]
|
|
if req.readyState is 4
|
|
expandComment.parse req, this, threadID, replyID
|
|
else
|
|
a = this
|
|
g.requests[threadID] = $.get @href, (-> expandComment.parse this, a, threadID, replyID)
|
|
parse: (req, a, threadID, replyID) ->
|
|
if req.status isnt 200
|
|
a.textContent = "#{req.status} #{req.statusText}"
|
|
return
|
|
|
|
body = $.el 'body',
|
|
innerHTML: req.responseText
|
|
|
|
if threadID is replyID #OP
|
|
bq = $ 'blockquote', body
|
|
else
|
|
#css selectors don't like ids starting with numbers,
|
|
# getElementById only works for root document.
|
|
for reply in $$ 'td[id]', body
|
|
if reply.id == replyID
|
|
bq = $ 'blockquote', reply
|
|
break
|
|
$.replace a.parentNode.parentNode, bq
|
|
|
|
expandThread =
|
|
init: ->
|
|
for span in $$ 'span.omittedposts'
|
|
a = $.el 'a',
|
|
textContent: "+ #{span.textContent}"
|
|
className: 'omittedposts'
|
|
$.bind a, 'click', expandThread.cb.toggle
|
|
$.replace span, a
|
|
|
|
cb:
|
|
toggle: (e) ->
|
|
thread = @parentNode
|
|
expandThread.toggle thread
|
|
|
|
toggle: (thread) ->
|
|
threadID = thread.firstChild.id
|
|
a = $ 'a.omittedposts', thread
|
|
|
|
switch a.textContent[0]
|
|
when '+'
|
|
a.textContent = a.textContent.replace '+', 'X Loading...'
|
|
if req = g.requests[threadID]
|
|
if req.readyState is 4
|
|
expandThread.parse req, thread, a
|
|
else
|
|
g.requests[threadID] = $.get "res/#{threadID}", (-> expandThread.parse this, thread, a)
|
|
|
|
when 'X'
|
|
a.textContent = a.textContent.replace 'X Loading...', '+'
|
|
g.requests[id].abort()
|
|
|
|
when '-'
|
|
a.textContent = a.textContent.replace '-', '+'
|
|
#goddamit moot
|
|
num = if g.BOARD is 'b' then 3 else 5
|
|
table = $.x "following::br[@clear][1]/preceding::table[#{num}]", a
|
|
while (prev = table.previousSibling) and (prev.nodeName is 'TABLE')
|
|
$.rm prev
|
|
|
|
parse: (req, thread, a) ->
|
|
if req.status isnt 200
|
|
a.textContent = "#{req.status} #{req.statusText}"
|
|
$.unbind a, 'click', expandThread.cb.toggle
|
|
return
|
|
|
|
a.textContent = a.textContent.replace 'X Loading...', '-'
|
|
|
|
# eat everything, then replace with fresh full posts
|
|
while (next = a.nextSibling) and not next.clear #br[clear]
|
|
$.rm next
|
|
br = next
|
|
|
|
body = $.el 'body',
|
|
innerHTML: req.responseText
|
|
|
|
tables = $$ 'form[name=delform] table', body
|
|
tables.pop()
|
|
for table in tables
|
|
$.before br, table
|
|
|
|
replyHiding =
|
|
init: ->
|
|
g.callbacks.push replyHiding.cb.node
|
|
|
|
cb:
|
|
hide: (e) ->
|
|
reply = @parentNode.nextSibling
|
|
replyHiding.hide reply
|
|
|
|
node: (root) ->
|
|
return unless dd = $ 'td.doubledash', root
|
|
dd.className = 'replyhider'
|
|
a = $.el 'a',
|
|
textContent: '[ - ]'
|
|
$.bind a, 'click', replyHiding.cb.hide
|
|
$.replace dd.firstChild, a
|
|
|
|
reply = dd.nextSibling
|
|
id = reply.id
|
|
if id of g.hiddenReplies
|
|
replyHiding.hide reply
|
|
|
|
show: (e) ->
|
|
div = @parentNode
|
|
table = div.nextSibling
|
|
replyHiding.show table
|
|
|
|
$.rm div
|
|
|
|
hide: (reply) ->
|
|
table = reply.parentNode.parentNode.parentNode
|
|
$.hide table
|
|
|
|
if $.config 'Show Stubs'
|
|
name = $('span.commentpostername', reply).textContent
|
|
trip = $('span.postertrip', reply)?.textContent or ''
|
|
a = $.el 'a',
|
|
textContent: "[ + ] #{name} #{trip}"
|
|
$.bind a, 'click', replyHiding.cb.show
|
|
|
|
div = $.el 'div'
|
|
$.append div, a
|
|
$.before table, div
|
|
|
|
id = reply.id
|
|
g.hiddenReplies[id] = Date.now()
|
|
$.setValue "hiddenReplies/#{g.BOARD}/", g.hiddenReplies
|
|
|
|
show: (table) ->
|
|
$.show table
|
|
|
|
id = $('td[id]', table).id
|
|
delete g.hiddenReplies[id]
|
|
$.setValue "hiddenReplies/#{g.BOARD}/", g.hiddenReplies
|
|
|
|
keybinds =
|
|
init: ->
|
|
$.bind d, 'keydown', keybinds.cb.keydown
|
|
$.bind d, 'keypress', keybinds.cb.keypress
|
|
|
|
cb:
|
|
keydown: (e) ->
|
|
if d.activeElement.nodeName in ['TEXTAREA', 'INPUT']
|
|
keybinds.mode = keybinds.insert
|
|
else
|
|
keybinds.mode = keybinds.normal
|
|
|
|
kc = e.keyCode
|
|
if 65 <= kc <= 90 #A-Z
|
|
key = String.fromCharCode kc
|
|
if !e.shiftKey
|
|
key = key.toLowerCase()
|
|
if e.ctrlKey then key = '^' + key
|
|
else
|
|
if kc is 27
|
|
key = '<Esc>'
|
|
else if 48 <= kc <= 57 #0-9
|
|
key = String.fromCharCode kc
|
|
keybinds.key = key
|
|
|
|
keypress: (e) ->
|
|
keybinds.mode e
|
|
|
|
insert: (e) ->
|
|
switch keybinds.key
|
|
when '<Esc>'
|
|
e.preventDefault()
|
|
$.rm $ '#qr'
|
|
when '^s'
|
|
ta = d.activeElement
|
|
return unless ta.nodeName is 'TEXTAREA'
|
|
|
|
e.preventDefault()
|
|
|
|
value = ta.value
|
|
selStart = ta.selectionStart
|
|
selEnd = ta.selectionEnd
|
|
|
|
valStart = value[0...selStart] + '[spoiler]'
|
|
valMid = value[selStart...selEnd]
|
|
valEnd = '[/spoiler]' + value[selEnd..]
|
|
|
|
ta.value = valStart + valMid + valEnd
|
|
range = valStart.length + valMid.length
|
|
ta.setSelectionRange range, range
|
|
|
|
normal: (e) ->
|
|
thread = nav.getThread()
|
|
switch keybinds.key
|
|
when '0'
|
|
window.location = "/#{g.BOARD}/#0"
|
|
when 'I'
|
|
keybinds.qr thread
|
|
when 'J'
|
|
keybinds.hl.next thread
|
|
when 'K'
|
|
keybinds.hl.prev thread
|
|
when 'M'
|
|
keybinds.img thread, true
|
|
when 'O'
|
|
keybinds.open thread
|
|
when 'i'
|
|
keybinds.qr thread, true
|
|
when 'm'
|
|
keybinds.img thread
|
|
when 'n'
|
|
nav.next()
|
|
when 'o'
|
|
keybinds.open thread, true
|
|
when 'p'
|
|
nav.prev()
|
|
when 'u'
|
|
updater.update()
|
|
when 'w'
|
|
watcher.toggle thread
|
|
when 'x'
|
|
threadHiding.toggle thread
|
|
|
|
img: (thread, all) ->
|
|
if all
|
|
$("#imageExpand").click()
|
|
else
|
|
root = $('td.replyhl', thread) or thread
|
|
thumb = $ 'img[md5]', root
|
|
imgExpand.toggle thumb.parentNode
|
|
|
|
qr: (thread, quote) ->
|
|
unless qrLink = $ 'td.replyhl span[id] a:not(:first-child)', thread
|
|
qrLink = $ "span[id^=nothread] a:not(:first-child)", thread
|
|
|
|
if quote
|
|
qr.quote qrLink
|
|
else
|
|
unless $ '#qr'
|
|
qr.dialog qrLink
|
|
$('#qr textarea').focus()
|
|
|
|
open: (thread, tab) ->
|
|
id = thread.firstChild.id
|
|
url = "http://boards.4chan.org/#{g.BOARD}/res/#{id}"
|
|
if tab
|
|
$.openInTab url
|
|
else
|
|
location.href = url
|
|
|
|
hl:
|
|
next: (thread) ->
|
|
if td = $ 'td.replyhl', thread
|
|
td.className = 'reply'
|
|
rect = td.getBoundingClientRect()
|
|
if rect.top > 0 and rect.bottom < d.body.clientHeight #you're fully visible
|
|
next = $.x 'following::td[@class="reply"]', td
|
|
rect = next.getBoundingClientRect()
|
|
if rect.top > 0 and rect.bottom < d.body.clientHeight #and so is the next
|
|
next.className = 'replyhl'
|
|
return
|
|
|
|
replies = $$ 'td.reply', thread
|
|
for reply in replies
|
|
top = reply.getBoundingClientRect().top
|
|
if top > 0
|
|
reply.className = 'replyhl'
|
|
return
|
|
|
|
prev: (thread) ->
|
|
if td = $ 'td.replyhl', thread
|
|
td.className = 'reply'
|
|
rect = td.getBoundingClientRect()
|
|
if rect.top > 0 and rect.bottom < d.body.clientHeight #you're fully visible
|
|
prev = $.x 'preceding::td[@class="reply"][1]', td
|
|
rect = prev.getBoundingClientRect()
|
|
if rect.top > 0 and rect.bottom < d.body.clientHeight #and so is the prev
|
|
prev.className = 'replyhl'
|
|
return
|
|
|
|
replies = $$ 'td.reply', thread
|
|
replies.reverse()
|
|
height = d.body.clientHeight
|
|
for reply in replies
|
|
bot = reply.getBoundingClientRect().bottom
|
|
if bot < height
|
|
reply.className = 'replyhl'
|
|
return
|
|
|
|
nav =
|
|
# ◀ ▶
|
|
init: ->
|
|
span = $.el 'span',
|
|
id: 'navlinks'
|
|
prev = $.el 'a',
|
|
textContent: '▲'
|
|
next = $.el 'a',
|
|
textContent: '▼'
|
|
|
|
$.bind prev, 'click', nav.prev
|
|
$.bind next, 'click', nav.next
|
|
|
|
$.append span, prev, $.tn(' '), next
|
|
$.append d.body, span
|
|
|
|
prev: ->
|
|
nav.scroll -1
|
|
|
|
next: ->
|
|
nav.scroll +1
|
|
|
|
threads: []
|
|
|
|
getThread: (full) ->
|
|
nav.threads = $$ 'div.thread:not([style])'
|
|
for thread, i in nav.threads
|
|
rect = thread.getBoundingClientRect()
|
|
{bottom} = rect
|
|
if bottom > 0 #we have not scrolled past
|
|
if full
|
|
return [thread, i, rect]
|
|
return thread
|
|
return null
|
|
|
|
scroll: (delta) ->
|
|
[thread, i, rect] = nav.getThread true
|
|
{top} = rect
|
|
|
|
#unless we're not at the beginning of the current thread
|
|
# (and thus wanting to move to beginning)
|
|
# or we're above the first thread and don't want to skip it
|
|
unless (delta is -1 and Math.ceil(top) < 0) or (delta is +1 and top > 1)
|
|
i += delta
|
|
|
|
if i is -1
|
|
if g.PAGENUM is 0
|
|
window.scrollTo 0, 0
|
|
else
|
|
window.location = "#{g.PAGENUM - 1}#0"
|
|
return
|
|
if delta is +1
|
|
# if we're at the last thread, or we're at the bottom of the page.
|
|
# kind of hackish, what we really need to do is make nav.getThread smarter.
|
|
if i is nav.threads.length or (innerHeight + pageYOffset == d.body.scrollHeight)
|
|
if $ 'table.pages input[value="Next"]'
|
|
window.location = "#{g.PAGENUM + 1}#0"
|
|
return
|
|
#TODO sfx
|
|
|
|
{top} = nav.threads[i].getBoundingClientRect()
|
|
window.scrollBy 0, top
|
|
|
|
options =
|
|
init: ->
|
|
home = $ '#navtopr a'
|
|
a = $.el 'a',
|
|
textContent: '4chan X'
|
|
$.bind a, 'click', options.toggle
|
|
$.replace home, a
|
|
home = $ '#navbotr a'
|
|
a = $.el 'a',
|
|
textContent: '4chan X'
|
|
$.bind a, 'click', options.toggle
|
|
$.replace home, a
|
|
|
|
toggle: ->
|
|
if dialog = $ '#options'
|
|
$.rm dialog
|
|
else
|
|
options.dialog()
|
|
|
|
dialog: ->
|
|
hiddenThreads = $.getValue "hiddenThreads/#{g.BOARD}/", {}
|
|
hiddenNum = Object.keys(g.hiddenReplies).length + Object.keys(hiddenThreads).length
|
|
html = "
|
|
<div class=move>Options <a name=close>X</a></div>
|
|
<hr>
|
|
<div class=column><ul id=monitor><li>Monitoring</li></ul><ul id=img><li>Imaging</li></ul></div>
|
|
<div class=column><ul id=post><li>Posting</li></ul><ul id=quote><li>Quoting</li></ul><ul id=hide><li>Hiding</li></ul></div>
|
|
<div class=column><ul id=misc><li>Enhancing</li></ul></div>
|
|
<br clear=left>
|
|
<hr>
|
|
<div id=floaty>
|
|
<div><input type=button value='hidden: #{hiddenNum}'></div>
|
|
<div><a name=flavors>Sauce flavors</a></div>
|
|
</div>
|
|
<div id=credits>
|
|
<div><a href=://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=2DBVZBUAM4DHC&lc=US&item_name=Aeosynth¤cy_code=USD&bn=PP%2dDonationsBF%3abtn_donate_LG%2egif%3aNonHosted><img alt=Donate src=></a></div>
|
|
<div><a href=http://chat.now.im/x/aeos>support throd</a> | <a href=https://github.com/aeosynth/4chan-x/issues>github</a> | <a href=http://userscripts.org/scripts/show/51412>uso</a></div>
|
|
</div>
|
|
<div><textarea style='display: none;' name=flavors>#{$.config 'flavors'}</textarea></div>
|
|
"
|
|
|
|
dialog = ui.dialog 'options', top: '25%', left: '25%', html
|
|
options.append config.main.monitor, $('#monitor', dialog)
|
|
options.append config.main.img, $('#img', dialog)
|
|
options.append config.main.post, $('#post', dialog)
|
|
options.append config.main.quote, $('#quote', dialog)
|
|
options.append config.main.hide, $('#hide', dialog)
|
|
options.append config.main.misc, $('#misc', dialog)
|
|
for input in $$ 'input[type=checkbox]', dialog
|
|
$.bind input, 'click', $.cb.checked
|
|
$.bind $('input[type=button]', dialog), 'click', options.cb.clearHidden
|
|
$.bind $('a[name=flavors]', dialog), 'click', options.flavors
|
|
$.bind $('textarea', dialog), 'change', $.cb.value
|
|
$.append d.body, dialog
|
|
|
|
append: (conf, id) ->
|
|
for name of conf
|
|
title = conf[name][1]
|
|
checked = if $.config name then "checked" else ""
|
|
li = $.el 'li',
|
|
innerHTML: "<label title='#{title}'><input name='#{name}' #{checked} type=checkbox>#{name}</label>"
|
|
$.append id, li
|
|
|
|
flavors: ->
|
|
ta = $ '#options textarea'
|
|
if ta.style.display then $.show ta else $.hide ta
|
|
|
|
cb:
|
|
clearHidden: (e) ->
|
|
#'hidden' might be misleading; it's the number of IDs we're *looking* for,
|
|
# not the number of posts actually hidden on the page.
|
|
$.deleteValue "hiddenReplies/#{g.BOARD}/"
|
|
$.deleteValue "hiddenThreads/#{g.BOARD}/"
|
|
@value = "hidden: 0"
|
|
g.hiddenReplies = {}
|
|
|
|
qr =
|
|
init: ->
|
|
g.callbacks.push qr.cb.node
|
|
iframe = $.el 'iframe',
|
|
name: 'iframe'
|
|
$.append d.body, iframe
|
|
$.bind window, 'message', qr.cb.message
|
|
|
|
#hack - nuke id so it doesn't grab focus when reloading
|
|
$('#recaptcha_response_field').id = ''
|
|
|
|
autohide:
|
|
set: ->
|
|
$('#qr input[title=autohide]:not(:checked)')?.click()
|
|
unset: ->
|
|
$('#qr input[title=autohide]:checked')?.click()
|
|
|
|
cb:
|
|
autohide: (e) ->
|
|
dialog = $ '#qr'
|
|
if @checked
|
|
$.addClass dialog, 'auto'
|
|
else
|
|
$.removeClass dialog, 'auto'
|
|
|
|
message: (e) ->
|
|
{data} = e
|
|
dialog = $ '#qr'
|
|
if data # error message
|
|
$('#error').textContent = data
|
|
qr.autohide.unset()
|
|
else # success
|
|
if dialog
|
|
if $.config 'Persistent QR'
|
|
qr.refresh dialog
|
|
else
|
|
$.rm dialog
|
|
if $.config 'Cooldown'
|
|
qr.cooldown true
|
|
|
|
Recaptcha.reload()
|
|
$('iframe[name=iframe]').src = 'about:blank'
|
|
|
|
node: (root) ->
|
|
quote = $ 'a.quotejs:not(:first-child)', root
|
|
$.bind quote, 'click', qr.cb.quote
|
|
|
|
submit: (e) ->
|
|
form = this
|
|
isQR = form.parentNode.id == 'qr'
|
|
|
|
if $.config('Auto Watch Reply') and $.config('Thread Watcher')
|
|
if g.REPLY and $('img.favicon').src is Favicon.empty
|
|
watcher.watch null, g.THREAD_ID
|
|
else
|
|
id = $('input[name=resto]').value
|
|
op = d.getElementById id
|
|
if $('img.favicon', op).src is Favicon.empty
|
|
watcher.watch op, id
|
|
|
|
if isQR
|
|
$('#error').textContent = ''
|
|
|
|
if $.config 'Cooldown'
|
|
# check if we've posted on this board in another tab
|
|
if qr.cooldown()
|
|
e.preventDefault()
|
|
alert 'Stop posting so often!'
|
|
|
|
if isQR
|
|
$('#error').textContent = 'Stop posting so often!'
|
|
|
|
return
|
|
|
|
qr.sage = $('input[name=email]', form).value == 'sage'
|
|
if isQR
|
|
qr.autohide.set()
|
|
|
|
quote: (e) ->
|
|
e.preventDefault()
|
|
qr.quote this
|
|
|
|
quote: (link) ->
|
|
if dialog = $ '#qr'
|
|
qr.autohide.unset()
|
|
else
|
|
dialog = qr.dialog link
|
|
|
|
id = link.textContent
|
|
text = ">>#{id}\n"
|
|
|
|
selection = window.getSelection()
|
|
if s = selection.toString()
|
|
selectionID = $.x('preceding::input[@type="checkbox"][1]', selection.anchorNode)?.name
|
|
if selectionID == id
|
|
text += ">#{s}\n"
|
|
|
|
ta = $ 'textarea', dialog
|
|
ta.focus()
|
|
ta.value += text
|
|
|
|
refresh: (dialog) ->
|
|
$('textarea', dialog).value = ''
|
|
$('input[name=recaptcha_response_field]', dialog).value = ''
|
|
# XXX file.value = '' doesn't work in opera
|
|
f = $('input[type=file]', dialog).parentNode
|
|
f.innerHTML = f.innerHTML
|
|
|
|
cooldown: (restart) ->
|
|
now = Date.now()
|
|
|
|
if restart
|
|
duration = if qr.sage then 60 else 30
|
|
qr.cooldownStart duration
|
|
$.setValue "#{g.BOARD}/cooldown", now + duration * 1000
|
|
return
|
|
|
|
end = $.getValue "#{g.BOARD}/cooldown", 0
|
|
if now < end
|
|
duration = Math.ceil (end - now) / 1000
|
|
qr.cooldownStart duration
|
|
return true
|
|
|
|
cooldownStart: (duration) ->
|
|
submits = $$ '#com_submit'
|
|
for submit in submits
|
|
submit.value = duration
|
|
submit.disabled = true
|
|
qr.cooldownIntervalID = window.setInterval qr.cooldownCB, 1000
|
|
qr.duration = duration
|
|
|
|
cooldownCB: ->
|
|
qr.duration = qr.duration - 1
|
|
|
|
submits = $$ '#com_submit'
|
|
for submit in submits
|
|
if qr.duration == 0
|
|
submit.disabled = false
|
|
submit.value = 'Submit'
|
|
else
|
|
submit.value = qr.duration
|
|
|
|
if qr.duration == 0
|
|
clearInterval qr.cooldownIntervalID
|
|
|
|
dialog: (link) ->
|
|
#maybe should be global
|
|
MAX_FILE_SIZE = $('input[name="MAX_FILE_SIZE"]').value
|
|
#FIXME inlined cross-thread quotes
|
|
THREAD_ID = g.THREAD_ID or $.x('ancestor::div[@class="thread"]/div', link).id
|
|
challenge = $('input[name=recaptcha_challenge_field]').value
|
|
src = "http://www.google.com/recaptcha/api/image?c=#{challenge}"
|
|
name = $('input[name=name]').value
|
|
mail = $('input[name=email]').value
|
|
pass = $('input[name=pwd]').value
|
|
html = "
|
|
<div class=move>
|
|
<input class=inputtext type=text name=name placeholder=Name form=qr_form value='#{name}'>
|
|
Quick Reply
|
|
<input type=checkbox id=autohide title=autohide>
|
|
<a name=close title=close>X</a>
|
|
</div>
|
|
<form name=post action=http://sys.4chan.org/#{g.BOARD}/post method=POST enctype=multipart/form-data target=iframe id=qr_form>
|
|
<input type=hidden name=MAX_FILE_SIZE value=#{MAX_FILE_SIZE}>
|
|
<input type=hidden name=resto value=#{THREAD_ID}>
|
|
<input type=hidden name=recaptcha_challenge_field value=#{challenge}>
|
|
<div><input class=inputtext type=text name=email placeholder=E-mail value='#{mail}'></div>
|
|
<div><input class=inputtext type=text name=sub placeholder=Subject><input type=submit value=Submit id=com_submit></div>
|
|
<div><textarea class=inputtext name=com placeholder=Comment></textarea></div>
|
|
<div><img src=#{src}></div>
|
|
<div><input class=inputtext type=text name=recaptcha_response_field placeholder=Verification required autocomplete=off></div>
|
|
<div><input type=file name=upfile></div>
|
|
<div><input class=inputtext type=password name=pwd maxlength=8 placeholder=Password value='#{pass}'><input type=hidden name=mode value=regist></div>
|
|
</form>
|
|
<div id=error class=error></div>
|
|
"
|
|
dialog = ui.dialog 'qr', top: '0px', left: '0px', html
|
|
|
|
$.bind $('input[name=name]', dialog), 'mousedown', (e) -> e.stopPropagation()
|
|
$.bind $('#autohide', dialog), 'click', qr.cb.autohide
|
|
$.bind $('img', dialog), 'click', Recaptcha.reload
|
|
|
|
if $ '.postarea label'
|
|
spoiler = $.el 'label',
|
|
innerHTML: " [<input type=checkbox name=spoiler>Spoiler Image?]"
|
|
$.after $('input[name=email]', dialog), spoiler
|
|
|
|
$.bind $('form', dialog), 'submit', qr.cb.submit
|
|
$.bind $('input[name=recaptcha_response_field]', dialog), 'keydown', Recaptcha.listener
|
|
|
|
$.append d.body, dialog
|
|
|
|
dialog
|
|
|
|
persist: ->
|
|
$.append d.body, qr.dialog()
|
|
qr.autohide.set()
|
|
|
|
sys: ->
|
|
if recaptcha = $ '#recaptcha_response_field' #post reporting
|
|
$.bind recaptcha, 'keydown', Recaptcha.listener
|
|
return
|
|
|
|
c = $('b').lastChild
|
|
if c.nodeType is 8 #comment node
|
|
[_, thread, id] = c.textContent.match(/thread:(\d+),no:(\d+)/)
|
|
if thread is '0'
|
|
[_, board] = $('meta', d).content.match(/4chan.org\/(\w+)\//)
|
|
window.location = "http://boards.4chan.org/#{board}/res/#{id}#watch"
|
|
return
|
|
|
|
###
|
|
http://code.google.com/p/chromium/issues/detail?id=20773
|
|
Let content scripts see other frames (instead of them being undefined)
|
|
|
|
To access the parent, we have to break out of the sandbox and evaluate
|
|
in the global context.
|
|
###
|
|
$.globalEval ->
|
|
data = document.querySelector('table font b')?.firstChild.textContent or ''
|
|
parent.postMessage data, '*'
|
|
|
|
threading =
|
|
init: ->
|
|
# don't thread image controls
|
|
node = $ 'form[name=delform] > *:not([id])'
|
|
# don't confuse other scripts *cough*/b/ackwash*cough*
|
|
# gotta have a named function to unbind.
|
|
$.bind d, 'DOMNodeInserted', threading.stopPropagation
|
|
threading.thread node
|
|
$.unbind d, 'DOMNodeInserted', threading.stopPropagation
|
|
|
|
op: (node) ->
|
|
op = $.el 'div',
|
|
className: 'op'
|
|
$.before node, op
|
|
while node.nodeName isnt 'BLOCKQUOTE'
|
|
$.append op, node
|
|
node = op.nextSibling
|
|
$.append op, node #add the blockquote
|
|
op.id = $('input[name]', op).name
|
|
op
|
|
|
|
thread: (node) ->
|
|
node = threading.op node
|
|
|
|
return if g.REPLY
|
|
|
|
div = $.el 'div',
|
|
className: 'thread'
|
|
$.before node, div
|
|
|
|
while node.nodeName isnt 'HR'
|
|
$.append div, node
|
|
node = div.nextSibling
|
|
|
|
node = node.nextElementSibling #skip text node
|
|
#{N,}SFW
|
|
unless node.align or node.nodeName is 'CENTER'
|
|
threading.thread node
|
|
|
|
stopPropagation: (e) ->
|
|
e.stopPropagation()
|
|
|
|
threadHiding =
|
|
init: ->
|
|
hiddenThreads = $.getValue "hiddenThreads/#{g.BOARD}/", {}
|
|
for thread in $$ 'div.thread'
|
|
op = thread.firstChild
|
|
a = $.el 'a',
|
|
textContent: '[ - ]'
|
|
$.bind a, 'click', threadHiding.cb.hide
|
|
$.prepend op, a
|
|
|
|
if op.id of hiddenThreads
|
|
threadHiding.hideHide thread
|
|
|
|
cb:
|
|
hide: (e) ->
|
|
thread = @parentNode.parentNode
|
|
threadHiding.hide thread
|
|
show: (e) ->
|
|
thread = @parentNode.parentNode
|
|
threadHiding.show thread
|
|
|
|
toggle: (thread) ->
|
|
if thread.className.indexOf('stub') != -1 or thread.style.display is 'none'
|
|
threadHiding.show thread
|
|
else
|
|
threadHiding.hide thread
|
|
|
|
hide: (thread) ->
|
|
threadHiding.hideHide thread
|
|
|
|
id = thread.firstChild.id
|
|
|
|
hiddenThreads = $.getValue "hiddenThreads/#{g.BOARD}/", {}
|
|
hiddenThreads[id] = Date.now()
|
|
$.setValue "hiddenThreads/#{g.BOARD}/", hiddenThreads
|
|
|
|
hideHide: (thread) ->
|
|
if $.config 'Show Stubs'
|
|
if span = $ '.omittedposts', thread
|
|
num = Number span.textContent.match(/\d+/)[0]
|
|
else
|
|
num = 0
|
|
num += $$('table', thread).length
|
|
text = if num is 1 then "1 reply" else "#{num} replies"
|
|
name = $('span.postername', thread).textContent
|
|
trip = $('span.postername + span.postertrip', thread)?.textContent || ''
|
|
|
|
a = $.el 'a',
|
|
textContent: "[ + ] #{name}#{trip} (#{text})"
|
|
$.bind a, 'click', threadHiding.cb.show
|
|
|
|
div = $.el 'div',
|
|
className: 'block'
|
|
|
|
$.append div, a
|
|
$.append thread, div
|
|
$.addClass thread, 'stub'
|
|
else
|
|
$.hide thread
|
|
$.hide thread.nextSibling
|
|
|
|
show: (thread) ->
|
|
$.rm $ 'div.block', thread
|
|
$.removeClass thread, 'stub'
|
|
$.show thread
|
|
$.show thread.nextSibling
|
|
|
|
id = thread.firstChild.id
|
|
|
|
hiddenThreads = $.getValue "hiddenThreads/#{g.BOARD}/", {}
|
|
delete hiddenThreads[id]
|
|
$.setValue "hiddenThreads/#{g.BOARD}/", hiddenThreads
|
|
|
|
updater =
|
|
init: ->
|
|
html = "<div class=move><span id=count></span> <span id=timer>-#{$.config 'Interval'}</span></div>"
|
|
conf = config.updater.checkbox
|
|
for name of conf
|
|
title = conf[name][1]
|
|
checked = if $.config name then "checked" else ""
|
|
html += "<div><label title=\"#{title}\">#{name}<input name=\"#{name}\" #{checked} type=checkbox></label></div>"
|
|
|
|
name = 'Auto Update This'
|
|
title = 'Controls whether *this* thread auotmatically updates or not'
|
|
checked = if $.config 'Auto Update' then 'checked' else ''
|
|
html += "<div><label title=\"#{title}\">#{name}<input name=\"#{name}\" #{checked} type=checkbox></label></div>"
|
|
|
|
html += "<div><label>Interval (s)<input name=Interval value=#{$.config 'Interval'} type=text></label></div>"
|
|
html += "<div><input value=\"Update Now\" type=button></div>"
|
|
|
|
dialog = ui.dialog 'updater', bottom: '0px', right: '0px', html
|
|
|
|
for input in $$ 'input[type=checkbox]', dialog
|
|
$.bind input, 'click', $.cb.checked
|
|
$.bind $('input[type=text]', dialog), 'change', $.cb.value
|
|
|
|
verbose = $ 'input[name=\"Verbose\"]', dialog
|
|
autoUpT = $ 'input[name=\"Auto Update This\"]', dialog
|
|
interva = $ 'input[name=\"Interval\"]', dialog
|
|
updNow = $ 'input[type=button]', dialog
|
|
$.bind verbose, 'click', updater.cb.verbose
|
|
$.bind autoUpT, 'click', updater.cb.autoUpdate
|
|
$.bind updNow, 'click', updater.updateNow
|
|
|
|
$.append d.body, dialog
|
|
|
|
updater.cb.verbose.call verbose
|
|
updater.cb.autoUpdate.call autoUpT
|
|
|
|
cb:
|
|
verbose: (e) ->
|
|
count = $ '#count'
|
|
timer = $ '#timer'
|
|
if @checked
|
|
count.textContent = '+0'
|
|
$.show timer
|
|
else
|
|
$.extend count,
|
|
className: ''
|
|
textContent: 'Thread Updater'
|
|
$.hide timer
|
|
autoUpdate: (e) ->
|
|
if @checked
|
|
updater.intervalID = window.setInterval updater.timeout, 1000
|
|
else
|
|
window.clearInterval updater.intervalID
|
|
update: (e) ->
|
|
count = $ '#count'
|
|
timer = $ '#timer'
|
|
|
|
if @status is 404
|
|
count.textContent = 404
|
|
count.className = 'error'
|
|
timer.textContent = ''
|
|
clearInterval updater.intervalID
|
|
for input in $$ 'input[type=submit]'
|
|
input.disabled = true
|
|
input.value = 404
|
|
d.title = d.title.match(/.+- /)[0] + 404
|
|
g.dead = true
|
|
Favicon.update()
|
|
return
|
|
|
|
br = $ 'br[clear]'
|
|
id = Number $('td[id]', br.previousElementSibling)?.id or 0
|
|
|
|
arr = []
|
|
body = $.el 'body',
|
|
innerHTML: @responseText
|
|
replies = $$ 'td[id]', body
|
|
while (reply = replies.pop()) and (reply.id > id)
|
|
arr.push reply.parentNode.parentNode.parentNode #table
|
|
|
|
timer.textContent = '-' + $.config 'Interval'
|
|
if $.config 'Verbose'
|
|
count.textContent = '+' + arr.length
|
|
if arr.length is 0
|
|
count.className = ''
|
|
else
|
|
count.className = 'new'
|
|
|
|
#XXX add replies in correct order so /b/acklinks resolve
|
|
while reply = arr.pop()
|
|
$.before br, reply
|
|
|
|
timeout: ->
|
|
timer = $ '#timer'
|
|
n = Number timer.textContent
|
|
n += 1
|
|
|
|
if n == 10
|
|
count = $ '#count'
|
|
count.textContent = 'retry'
|
|
count.className = ''
|
|
n = 0
|
|
|
|
timer.textContent = n
|
|
|
|
if n == 0
|
|
updater.update()
|
|
|
|
updateNow: ->
|
|
$('#timer').textContent = 0
|
|
updater.update()
|
|
|
|
update: ->
|
|
updater.request?.abort()
|
|
url = location.href + '?' + Date.now() # fool the cache
|
|
cb = updater.cb.update
|
|
updater.request = $.get url, cb
|
|
|
|
watcher =
|
|
init: ->
|
|
html = '<div class=move>Thread Watcher</div>'
|
|
dialog = ui.dialog 'watcher', top: '50px', left: '0px', html
|
|
$.append d.body, dialog
|
|
|
|
#add watch buttons
|
|
inputs = $$ 'form > input[value=delete], div.thread > input[value=delete]'
|
|
for input in inputs
|
|
favicon = $.el 'img',
|
|
className: 'favicon'
|
|
$.bind favicon, 'click', watcher.cb.toggle
|
|
$.before input, favicon
|
|
|
|
#populate watcher, display watch buttons
|
|
watcher.refresh $.getValue 'watched', {}
|
|
|
|
setInterval (->
|
|
if watcher.lastUpdated < $.getValue 'watcher.lastUpdated', 0
|
|
watcher.refresh $.getValue 'watched', {}
|
|
), 1000
|
|
|
|
refresh: (watched) ->
|
|
dialog = $ '#watcher'
|
|
for div in $$ 'div:not(.move)', dialog
|
|
$.rm div
|
|
for board of watched
|
|
for id, props of watched[board]
|
|
div = $.el 'div'
|
|
x = $.el 'a',
|
|
textContent: 'X'
|
|
$.bind x, 'click', watcher.cb.x
|
|
link = $.el 'a', props
|
|
|
|
$.append div, x, $.tn(' '), link
|
|
$.append dialog, div
|
|
|
|
watchedBoard = watched[g.BOARD] or {}
|
|
for favicon in $$ 'img.favicon'
|
|
id = favicon.nextSibling.name
|
|
if id of watchedBoard
|
|
favicon.src = Favicon.default
|
|
else
|
|
favicon.src = Favicon.empty
|
|
watcher.lastUpdated = Date.now()
|
|
|
|
cb:
|
|
toggle: (e) ->
|
|
watcher.toggle @parentNode
|
|
x: (e) ->
|
|
[board, _, id] = @nextElementSibling
|
|
.getAttribute('href').substring(1).split('/')
|
|
watcher.unwatch board, id
|
|
|
|
toggle: (thread) ->
|
|
favicon = $ 'img.favicon', thread
|
|
id = favicon.nextSibling.name
|
|
if favicon.src == Favicon.empty
|
|
watcher.watch thread, id
|
|
else # favicon.src == Favicon.default
|
|
watcher.unwatch g.BOARD, id
|
|
|
|
unwatch: (board, id) ->
|
|
watched = $.getValue 'watched', {}
|
|
delete watched[board][id]
|
|
$.setValue 'watched', watched
|
|
|
|
watcher.refresh watched
|
|
$.setValue 'watcher.lastUpdated', Date.now()
|
|
|
|
watch: (thread, id) ->
|
|
tc = $('span.filetitle', thread).textContent or $('blockquote', thread).textContent
|
|
props =
|
|
textContent: "/#{g.BOARD}/ - #{tc[...25]}"
|
|
href: "/#{g.BOARD}/res/#{id}"
|
|
|
|
watched = $.getValue 'watched', {}
|
|
watched[g.BOARD] or= {}
|
|
watched[g.BOARD][id] = props
|
|
$.setValue 'watched', watched
|
|
|
|
watcher.refresh watched
|
|
$.setValue 'watcher.lastUpdated', Date.now()
|
|
|
|
anonymize =
|
|
init: ->
|
|
g.callbacks.push anonymize.cb.node
|
|
cb:
|
|
node: (root) ->
|
|
name = $ 'span.commentpostername, span.postername', root
|
|
name.textContent = 'Anonymous'
|
|
if trip = $ 'span.postertrip', root
|
|
if trip.parentNode.nodeName is 'A'
|
|
$.rm trip.parentNode
|
|
else
|
|
$.rm trip
|
|
|
|
sauce =
|
|
init: ->
|
|
g.callbacks.push sauce.cb.node
|
|
cb:
|
|
node: (root) ->
|
|
return if root.className is 'inline'
|
|
prefixes = (s for s in ($.config('flavors').split '\n') when s[0] != '#')
|
|
names = (prefix.match(/(\w+)\./)[1] for prefix in prefixes)
|
|
if span = $ 'span.filesize', root
|
|
suffix = $('a', span).href
|
|
for prefix, i in prefixes
|
|
link = $.el 'a',
|
|
textContent: names[i]
|
|
href: prefix + suffix
|
|
$.append span, $.tn(' '), link
|
|
|
|
localize =
|
|
init: ->
|
|
g.callbacks.push (root) ->
|
|
for span in $$ 'span[id^=no]', root
|
|
s = span.previousSibling
|
|
[_, month, day, year, hour, min_sec] =
|
|
s.textContent.match /(\d+)\/(\d+)\/(\d+)\(\w+\)(\d+):(\S+)/
|
|
year = "20#{year}"
|
|
month -= 1 #months start at 0
|
|
hour = g.chanOffset + Number hour
|
|
date = new Date year, month, day, hour
|
|
year = date.getFullYear() - 2000
|
|
month = $.zeroPad date.getMonth() + 1
|
|
day = $.zeroPad date.getDate()
|
|
hour = $.zeroPad date.getHours()
|
|
dotw = [
|
|
'Sun'
|
|
'Mon'
|
|
'Tue'
|
|
'Wed'
|
|
'Thu'
|
|
'Fri'
|
|
'Sat'
|
|
][date.getDay()]
|
|
s.textContent = " #{month}/#{day}/#{year}(#{dotw})#{hour}:#{min_sec} "
|
|
|
|
titlePost =
|
|
init: ->
|
|
if tc = $('span.filetitle').textContent or $('blockquote').textContent
|
|
d.title = "/#{g.BOARD}/ - #{tc}"
|
|
|
|
quoteBacklink =
|
|
init: ->
|
|
g.callbacks.push quoteBacklink.node
|
|
node: (root) ->
|
|
return if root.className is 'inline'
|
|
#better coffee-script way of doing this?
|
|
id = root.id or $('td[id]', root).id
|
|
quotes = {}
|
|
tid = g.THREAD_ID or root.parentNode.firstChild.id
|
|
for quote in $$ 'a.quotelink', root
|
|
continue unless qid = quote.hash[1..]
|
|
#don't backlink the op
|
|
continue if qid == tid
|
|
#duplicate quotes get overwritten
|
|
quotes[qid] = quote
|
|
for qid, quote of quotes
|
|
continue unless el = d.getElementById qid
|
|
link = $.el 'a',
|
|
href: '#'+id
|
|
className: 'backlink'
|
|
textContent: '>>'+id
|
|
if $.config 'Quote Preview'
|
|
$.bind link, 'mouseover', quotePreview.mouseover
|
|
$.bind link, 'mousemove', ui.hover
|
|
$.bind link, 'mouseout', ui.hoverend
|
|
if $.config 'Quote Inline'
|
|
$.bind link, 'click', quoteInline.toggle
|
|
$.before $('td > br, blockquote', el), link
|
|
|
|
quoteInline =
|
|
init: ->
|
|
g.callbacks.push quoteInline.node
|
|
node: (root) ->
|
|
for quote in $$ 'a.quotelink, a.backlink', root
|
|
quote.removeAttribute 'onclick'
|
|
$.bind quote, 'click', quoteInline.toggle
|
|
toggle: (e) ->
|
|
e.preventDefault()
|
|
return unless id = @hash[1..]
|
|
root = $.x 'ancestor::td[1]', this
|
|
if td = $ "#i#{id}", root
|
|
$.rm $.x 'ancestor::table[1]', td
|
|
if @className is 'backlink'
|
|
$.show $.x 'ancestor::table[1]', d.getElementById id
|
|
return
|
|
inline = $.el 'table',
|
|
className: 'inline'
|
|
innerHTML: "<tbody><tr><td class=reply id=i#{id}></td></tr></tbody>"
|
|
td = $ 'td', inline
|
|
if el = d.getElementById id
|
|
td.innerHTML = el.innerHTML
|
|
else
|
|
td.innerHTML = "Loading #{id}..."
|
|
# or ... is for index page new posts.
|
|
# FIXME x-thread quotes
|
|
threadID = @pathname.split('/').pop() or $.x('ancestor::div[@class="thread"]/div', this).id
|
|
if req = g.requests[threadID]
|
|
if req.readyState is 4
|
|
quoteInline.parse req, id, threadID, inline
|
|
else
|
|
#FIXME need an array of callbacks
|
|
g.requests[threadID] = $.get @href, (-> quoteInline.parse this, id, threadID, inline)
|
|
if @className is 'backlink'
|
|
$.after $('td > br:first-of-type, td > a:last-of-type', @parentNode), inline
|
|
$.hide $.x 'ancestor::table[1]', el
|
|
else
|
|
$.after @parentNode, inline
|
|
parse: (req, id, threadID, inline) ->
|
|
if req.status isnt 200
|
|
inline.innerHTML = "#{req.status} #{req.statusText}"
|
|
return
|
|
|
|
clone = inline.cloneNode true
|
|
body = $.el 'body',
|
|
innerHTML: req.responseText
|
|
if id == threadID #OP
|
|
op = threading.op $ 'form[name=delform] > *', body
|
|
html = op.innerHTML
|
|
else
|
|
for reply in $$ 'td.reply', body
|
|
if reply.id == id
|
|
html = reply.innerHTML
|
|
break
|
|
$('td', clone).innerHTML = html
|
|
$.replace inline, clone
|
|
|
|
quotePreview =
|
|
init: ->
|
|
g.callbacks.push quotePreview.node
|
|
preview = $.el 'div',
|
|
id: 'qp'
|
|
className: 'replyhl'
|
|
$.hide preview
|
|
$.append d.body, preview
|
|
node: (root) ->
|
|
for quote in $$ 'a.quotelink, a.backlink', root
|
|
$.bind quote, 'mouseover', quotePreview.mouseover
|
|
$.bind quote, 'mousemove', ui.hover
|
|
$.bind quote, 'mouseout', ui.hoverend
|
|
mouseover: (e) ->
|
|
return unless id = @hash[1..]
|
|
qp = $ '#qp'
|
|
if el = d.getElementById id
|
|
qp.innerHTML = el.innerHTML
|
|
else
|
|
qp.innerHTML = "Loading #{id}..."
|
|
threadID = @pathname.split('/').pop()
|
|
if req = g.requests[threadID]
|
|
if req.readyState is 4
|
|
quotePreview.parse req, id, threadID
|
|
else
|
|
g.requests[threadID] = $.get @href, (-> quotePreview.parse this, id, threadID)
|
|
$.show qp
|
|
ui.el = qp
|
|
ui.winHeight = d.body.clientHeight
|
|
parse: (req, id, threadID) ->
|
|
qp = $ '#qp'
|
|
return unless qp.innerHTML is "Loading #{id}..."
|
|
|
|
if req.status isnt 200
|
|
qp.innerHTML = "#{req.status} #{req.statusText}"
|
|
return
|
|
|
|
body = $.el 'body',
|
|
innerHTML: req.responseText
|
|
if id == threadID #OP
|
|
op = threading.op $ 'form[name=delform] > *', body
|
|
html = op.innerHTML
|
|
else
|
|
for reply in $$ 'td.reply', body
|
|
if reply.id == id
|
|
html = reply.innerHTML
|
|
break
|
|
qp.innerHTML = html
|
|
|
|
reportButton =
|
|
init: ->
|
|
g.callbacks.push reportButton.cb.node
|
|
cb:
|
|
node: (root) ->
|
|
if not a = $ 'a.reportbutton', root
|
|
span = $ 'span[id^=no]', root
|
|
a = $.el 'a',
|
|
className: 'reportbutton'
|
|
innerHTML: '[ ! ]'
|
|
$.after span, a
|
|
$.after span, $.tn(' ')
|
|
$.bind a, 'click', reportButton.cb.report
|
|
report: (e) ->
|
|
reportButton.report this
|
|
report: (target) ->
|
|
input = $.x('preceding-sibling::input[1]', target)
|
|
input.click()
|
|
$('input[value="Report"]').click()
|
|
input.click()
|
|
|
|
unread =
|
|
init: ->
|
|
unread.replies = []
|
|
d.title = '(0) ' + d.title
|
|
$.bind window, 'scroll', unread.cb.scroll
|
|
g.callbacks.push unread.cb.node
|
|
|
|
cb:
|
|
node: (root) ->
|
|
return if root.className is 'inline'
|
|
unread.replies.push root
|
|
unread.updateTitle()
|
|
Favicon.update()
|
|
|
|
scroll: (e) ->
|
|
height = d.body.clientHeight
|
|
for reply, i in unread.replies
|
|
{bottom} = reply.getBoundingClientRect()
|
|
if bottom > height #post is not completely read
|
|
break
|
|
return if i is 0
|
|
|
|
unread.replies = unread.replies[i..]
|
|
unread.updateTitle()
|
|
if unread.replies.length is 0
|
|
Favicon.update()
|
|
|
|
updateTitle: ->
|
|
d.title = d.title.replace /\d+/, unread.replies.length
|
|
|
|
Favicon =
|
|
dead: ''
|
|
deadHalo: ''
|
|
default: $('link[rel="shortcut icon"]', d.head)?.href or '' #no favicon in `post successful` page
|
|
empty: ''
|
|
haloSFW: ''
|
|
haloNSFW: ''
|
|
|
|
update: ->
|
|
l = unread.replies.length
|
|
if g.dead
|
|
if l > 0
|
|
href = Favicon.deadHalo
|
|
else
|
|
href = Favicon.dead
|
|
else
|
|
if l > 0
|
|
href = Favicon.halo
|
|
else
|
|
href = Favicon.default
|
|
|
|
#XXX `favicon.href = href` doesn't work on Firefox
|
|
favicon = $ 'link[rel="shortcut icon"]', d.head
|
|
clone = favicon.cloneNode true
|
|
clone.href = href
|
|
$.replace favicon, clone
|
|
|
|
redirect = ->
|
|
switch g.BOARD
|
|
when 'a', 'g', 'lit', 'sci', 'tv'
|
|
url = "http://green-oval.net/cgi-board.pl/#{g.BOARD}/thread/#{g.THREAD_ID}"
|
|
when 'jp', 'm', 'tg'
|
|
url = "http://archive.easymodo.net/cgi-board.pl/#{g.BOARD}/thread/#{g.THREAD_ID}"
|
|
when '3', 'adv', 'an', 'ck', 'co', 'fa', 'fit', 'int', 'k', 'mu', 'n', 'o', 'p', 'po', 'soc', 'sp', 'toy', 'trv', 'v', 'vp', 'x'
|
|
url = "http://archive.no-ip.org/#{g.BOARD}/thread/#{g.THREAD_ID}"
|
|
else
|
|
url = "http://boards.4chan.org/#{g.BOARD}"
|
|
location.href = url
|
|
|
|
Recaptcha =
|
|
init: ->
|
|
#hack to tab from comment straight to recaptcha
|
|
for el in $$ '#recaptcha_table a'
|
|
el.tabIndex = 1
|
|
recaptcha = $ '#recaptcha_response_field'
|
|
$.bind recaptcha, 'keydown', Recaptcha.listener
|
|
listener: (e) ->
|
|
if e.keyCode is 8 and @value is ''
|
|
Recaptcha.reload()
|
|
reload: ->
|
|
window.location = 'javascript:Recaptcha.reload()'
|
|
|
|
nodeInserted = (e) ->
|
|
{target} = e
|
|
if target.nodeName is 'TABLE'
|
|
for callback in g.callbacks
|
|
callback target
|
|
else if target.id is 'recaptcha_challenge_field' and dialog = $ '#qr'
|
|
$('img', dialog).src = "http://www.google.com/recaptcha/api/image?c=" + target.value
|
|
$('input[name=recaptcha_challenge_field]', dialog).value = target.value
|
|
|
|
imageHover =
|
|
init: ->
|
|
img = $.el 'img', id: 'iHover'
|
|
$.hide img
|
|
$.append d.body, img
|
|
g.callbacks.push imageHover.cb.node
|
|
cb:
|
|
node: (root) ->
|
|
return unless thumb = $ 'img[md5]', root
|
|
$.bind thumb, 'mouseover', imageHover.cb.mouseover
|
|
$.bind thumb, 'mousemove', ui.hover
|
|
$.bind thumb, 'mouseout', ui.hoverend
|
|
mouseover: (e) ->
|
|
el = $ '#iHover'
|
|
el.src = null
|
|
el.src = @parentNode.href
|
|
$.show el
|
|
ui.el = el
|
|
ui.winHeight = d.body.clientHeight
|
|
|
|
imgPreloading =
|
|
init: ->
|
|
g.callbacks.push (root) ->
|
|
thumbs = $$ 'img[md5]', root
|
|
for thumb in thumbs
|
|
parent = thumb.parentNode
|
|
el = $.el 'img', src: parent.href
|
|
|
|
imgGif =
|
|
init: ->
|
|
g.callbacks.push (root) ->
|
|
thumbs = $$ 'img[md5]', root
|
|
for thumb in thumbs
|
|
src = thumb.parentNode.href
|
|
if /gif$/.test src
|
|
thumb.src = src
|
|
|
|
imgExpand =
|
|
init: ->
|
|
g.callbacks.push imgExpand.cb.node
|
|
imgExpand.dialog()
|
|
|
|
cb:
|
|
node: (root) ->
|
|
return unless thumb = $ 'img[md5]', root
|
|
a = thumb.parentNode
|
|
$.bind a, 'click', imgExpand.cb.toggle
|
|
if imgExpand.on then imgExpand.toggle a
|
|
toggle: (e) ->
|
|
return if e.shiftKey or e.altKey or e.ctrlKey or e.button isnt 0
|
|
e.preventDefault()
|
|
imgExpand.toggle this
|
|
all: (e) ->
|
|
thumbs = $$ 'img[md5]'
|
|
imgExpand.on = @checked
|
|
imgExpand.foo()
|
|
if imgExpand.on #expand
|
|
for thumb in thumbs
|
|
unless thumb.style.display #thumbnail hidden, image already expanded
|
|
imgExpand.expand thumb
|
|
else #contract
|
|
for thumb in thumbs
|
|
if thumb.style.display #thumbnail hidden - unhide it
|
|
imgExpand.contract thumb
|
|
typeChange: (e) ->
|
|
imgExpand.foo()
|
|
for img in $$ 'img[md5] + img'
|
|
imgExpand.resize img
|
|
|
|
toggle: (a) ->
|
|
thumb = a.firstChild
|
|
imgExpand.foo()
|
|
if thumb.style.display
|
|
imgExpand.contract thumb
|
|
else
|
|
imgExpand.expand thumb
|
|
|
|
contract: (thumb) ->
|
|
$.show thumb
|
|
$.rm thumb.nextSibling
|
|
|
|
expand: (thumb) ->
|
|
$.hide thumb
|
|
a = thumb.parentNode
|
|
img = $.el 'img',
|
|
src: a.href
|
|
a.appendChild img
|
|
|
|
imgExpand.resize img
|
|
|
|
foo: ->
|
|
formWidth = $('form[name=delform]').getBoundingClientRect().width
|
|
if td = $('td.reply')
|
|
table = td.parentNode.parentNode.parentNode
|
|
left = td.getBoundingClientRect().left - table.getBoundingClientRect().left
|
|
crap = td.getBoundingClientRect().width - parseInt(getComputedStyle(td).width)
|
|
|
|
imgExpand.maxWidthOP = formWidth
|
|
imgExpand.maxWidthReply = formWidth - left - crap
|
|
imgExpand.maxHeight = d.body.clientHeight
|
|
imgExpand.type = $('#imageType').value
|
|
|
|
resize: (img) ->
|
|
{maxWidthOP, maxWidthReply, maxHeight, type} = imgExpand
|
|
[_, imgWidth, imgHeight] = $
|
|
.x("preceding::span[@class][1]/text()[2]", img)
|
|
.textContent.match(/(\d+)x(\d+)/)
|
|
imgWidth = Number imgWidth
|
|
imgHeight = Number imgHeight
|
|
|
|
if img.parentNode.parentNode.nodeName == 'TD'
|
|
maxWidth = maxWidthReply
|
|
else
|
|
maxWidth = maxWidthOP
|
|
|
|
switch type
|
|
when 'full'
|
|
img.removeAttribute 'style'
|
|
when 'fit width'
|
|
if imgWidth > maxWidth
|
|
img.style.width = maxWidth
|
|
when 'fit screen'
|
|
ratio = Math.min maxWidth/imgWidth, maxHeight/imgHeight
|
|
if ratio < 1
|
|
img.style.width = Math.floor ratio * imgWidth
|
|
|
|
dialog: ->
|
|
controls = $.el 'div',
|
|
id: 'imgControls'
|
|
innerHTML:
|
|
"<select id=imageType name=imageType><option>full</option><option>fit width</option><option>fit screen</option></select>
|
|
<label>Expand Images<input type=checkbox id=imageExpand></label>"
|
|
imageType = $.getValue 'imageType', 'full'
|
|
for option in $$ 'option', controls
|
|
if option.textContent is imageType
|
|
option.selected = true
|
|
break
|
|
$.bind $('select', controls), 'change', $.cb.value
|
|
$.bind $('select', controls), 'change', imgExpand.cb.typeChange
|
|
$.bind $('input', controls), 'click', imgExpand.cb.all
|
|
|
|
delform = $ 'form[name=delform]'
|
|
$.prepend delform, controls
|
|
|
|
#main
|
|
NAMESPACE = 'AEOS.4chan_x.'
|
|
g =
|
|
requests: {}
|
|
callbacks: []
|
|
|
|
main =
|
|
init: ->
|
|
pathname = location.pathname.substring(1).split('/')
|
|
[g.BOARD, temp] = pathname
|
|
if temp is 'res'
|
|
g.REPLY = temp
|
|
g.THREAD_ID = pathname[2]
|
|
else
|
|
g.PAGENUM = parseInt(temp) || 0
|
|
|
|
if location.hostname is 'sys.4chan.org'
|
|
qr.sys()
|
|
return
|
|
if $.config('404 Redirect') and d.title is '4chan - 404' and /^\d+$/.test g.THREAD_ID
|
|
redirect()
|
|
return
|
|
if not $ '#navtopr'
|
|
return
|
|
|
|
Favicon.halo = if /ws/.test Favicon.default then Favicon.haloSFW else Favicon.haloNSFW
|
|
$('link[rel="shortcut icon"]', d.head).setAttribute 'type', 'image/x-icon'
|
|
g.hiddenReplies = $.getValue "hiddenReplies/#{g.BOARD}/", {}
|
|
tzOffset = (new Date()).getTimezoneOffset() / 60
|
|
# GMT -8 is given as +480; would GMT +8 be -480 ?
|
|
g.chanOffset = 5 - tzOffset# 4chan = EST = GMT -5
|
|
if $.isDST() then g.chanOffset -= 1
|
|
|
|
lastChecked = $.getValue 'lastChecked', 0
|
|
now = Date.now()
|
|
DAY = 1000 * 60 * 60 * 24
|
|
if lastChecked < now - 1*DAY
|
|
cutoff = now - 7*DAY
|
|
hiddenThreads = $.getValue "hiddenThreads/#{g.BOARD}/", {}
|
|
|
|
for id, timestamp of hiddenThreads
|
|
if timestamp < cutoff
|
|
delete hiddenThreads[id]
|
|
|
|
for id, timestamp of g.hiddenReplies
|
|
if timestamp < cutoff
|
|
delete g.hiddenReplies[id]
|
|
|
|
$.setValue "hiddenThreads/#{g.BOARD}/", hiddenThreads
|
|
$.setValue "hiddenReplies/#{g.BOARD}/", g.hiddenReplies
|
|
$.setValue 'lastChecked', now
|
|
|
|
$.addStyle main.css
|
|
|
|
Recaptcha.init()
|
|
|
|
$.bind $('form[name=post]'), 'submit', qr.cb.submit
|
|
|
|
#major features
|
|
if $.config 'Cooldown'
|
|
qr.cooldown()
|
|
|
|
if $.config 'Image Expansion'
|
|
imgExpand.init()
|
|
|
|
if $.config 'Image Auto-Gif'
|
|
imgGif.init()
|
|
|
|
if $.config 'Localize Time'
|
|
localize.init()
|
|
|
|
if $.config 'Sauce'
|
|
sauce.init()
|
|
|
|
if $.config 'Anonymize'
|
|
anonymize.init()
|
|
|
|
if $.config 'Image Hover'
|
|
imageHover.init()
|
|
|
|
if $.config 'Reply Hiding'
|
|
replyHiding.init()
|
|
|
|
if $.config 'Quick Reply'
|
|
qr.init()
|
|
|
|
if $.config 'Report Button'
|
|
reportButton.init()
|
|
|
|
if $.config 'Quote Backlinks'
|
|
quoteBacklink.init()
|
|
|
|
if $.config 'Quote Inline'
|
|
quoteInline.init()
|
|
|
|
if $.config 'Quote Preview'
|
|
quotePreview.init()
|
|
|
|
if $.config 'Thread Watcher'
|
|
watcher.init()
|
|
|
|
if $.config 'Keybinds'
|
|
keybinds.init()
|
|
|
|
threading.init()
|
|
|
|
if g.REPLY
|
|
if $.config 'Thread Updater'
|
|
updater.init()
|
|
|
|
if $.config 'Image Preloading'
|
|
imgPreloading.init()
|
|
|
|
if $.config('Quick Reply') and $.config 'Persistent QR'
|
|
qr.persist()
|
|
|
|
if $.config 'Post in Title'
|
|
titlePost.init()
|
|
|
|
if $.config 'Unread Count'
|
|
unread.init()
|
|
|
|
if $.config('Auto Watch') and $.config('Thread Watcher') and
|
|
location.hash is '#watch' and $('img.favicon').src is Favicon.empty
|
|
watcher.watch null, g.THREAD_ID
|
|
|
|
else #not reply
|
|
if $.config 'Thread Hiding'
|
|
threadHiding.init()
|
|
|
|
if $.config 'Thread Navigation'
|
|
nav.init()
|
|
|
|
if $.config 'Thread Expansion'
|
|
expandThread.init()
|
|
|
|
if $.config 'Comment Expansion'
|
|
expandComment.init()
|
|
|
|
for op in $$ 'div.op'
|
|
for callback in g.callbacks
|
|
callback op
|
|
for reply in $$ 'td[id]'
|
|
table = reply.parentNode.parentNode.parentNode
|
|
for callback in g.callbacks
|
|
callback table
|
|
$.bind d.body, 'DOMNodeInserted', nodeInserted
|
|
options.init()
|
|
|
|
css: '
|
|
/* dialog styling */
|
|
div.dialog {
|
|
border: 1px solid;
|
|
}
|
|
div.dialog > div.move {
|
|
cursor: move;
|
|
}
|
|
label, a, .favicon, #qr img {
|
|
cursor: pointer;
|
|
}
|
|
|
|
.new {
|
|
background: lime;
|
|
}
|
|
.error {
|
|
color: red;
|
|
}
|
|
td.replyhider {
|
|
vertical-align: top;
|
|
}
|
|
|
|
div.thread.stub > *:not(.block) {
|
|
display: none;
|
|
}
|
|
|
|
img[md5] + img {
|
|
float: left;
|
|
}
|
|
iframe {
|
|
display: none;
|
|
}
|
|
|
|
#qp, #iHover {
|
|
position: fixed;
|
|
}
|
|
|
|
#iHover {
|
|
max-height: 100%;
|
|
}
|
|
|
|
#navlinks {
|
|
position: fixed;
|
|
top: 25px;
|
|
right: 5px;
|
|
}
|
|
#navlinks {
|
|
font-size: 16px;
|
|
}
|
|
|
|
#options {
|
|
position: fixed;
|
|
padding: 5px;
|
|
}
|
|
#options .move, #credits {
|
|
text-align: right;
|
|
}
|
|
.column {
|
|
float: left;
|
|
margin: 0 10px;
|
|
}
|
|
#options ul {
|
|
list-style: none;
|
|
margin: 0;
|
|
padding: 0;
|
|
}
|
|
#options li:first-child {
|
|
text-decoration: underline;
|
|
}
|
|
#floaty {
|
|
float: left;
|
|
}
|
|
#options textarea {
|
|
height: 100px;
|
|
width: 100%;
|
|
}
|
|
|
|
#qr {
|
|
position: fixed;
|
|
}
|
|
#qr > div.move {
|
|
text-align: right;
|
|
}
|
|
#qr input[name=name] {
|
|
float: left;
|
|
}
|
|
#qr_form {
|
|
clear: left;
|
|
}
|
|
#qr_form, #qr #com_submit, #qr input[name=upfile] {
|
|
margin: 0;
|
|
}
|
|
#qr textarea {
|
|
width: 100%;
|
|
height: 120px;
|
|
}
|
|
#qr.auto:not(:hover) > form {
|
|
height: 0;
|
|
overflow: hidden;
|
|
}
|
|
/* http://stackoverflow.com/questions/2610497/change-an-inputs-html5-placeholder-color-with-css */
|
|
#qr input::-webkit-input-placeholder {
|
|
color: grey;
|
|
}
|
|
#qr input:-moz-placeholder {
|
|
color: grey;
|
|
}
|
|
/* qr reCAPTCHA */
|
|
#qr img {
|
|
border: 1px solid #AAA;
|
|
}
|
|
|
|
#updater {
|
|
position: fixed;
|
|
text-align: right;
|
|
}
|
|
#updater input[type=text] {
|
|
width: 50px;
|
|
}
|
|
#updater:not(:hover) {
|
|
border: none;
|
|
background: transparent;
|
|
}
|
|
#updater:not(:hover) > div:not(.move) {
|
|
display: none;
|
|
}
|
|
|
|
#watcher {
|
|
position: absolute;
|
|
}
|
|
#watcher > div {
|
|
padding-right: 5px;
|
|
padding-left: 5px;
|
|
}
|
|
#watcher > div.move {
|
|
text-decoration: underline;
|
|
padding-top: 5px;
|
|
}
|
|
#watcher > div:last-child {
|
|
padding-bottom: 5px;
|
|
}
|
|
|
|
#qp {
|
|
border: 1px solid;
|
|
}
|
|
#qp input {
|
|
display: none;
|
|
}
|
|
'
|
|
|
|
main.init()
|