Linkify = init: -> return if g.VIEW is 'catalog' or not Conf['Linkify'] @regString = if Conf['Allow False Positives'] ///( \b( [a-z]+:// | [a-z]{3,}\.[-a-z0-9]+\.[a-z] | [-a-z0-9]+\.[a-z] | [0-9]+\.[0-9]+\.[0-9]+\.[0-9] | [a-z]{3,}:[a-z0-9?] | [^\s@]+@[a-z0-9.-]+\.[a-z0-9] ) [^\s'"]+ )///gi else /(((magnet|mailto)\:|(www\.)|(news|(ht|f)tp(s?))\:\/\/){1}\S+)/gi if Conf['Comment Expansion'] ExpandComment.callbacks.push @node if Conf['Title Link'] $.sync 'CachedTitles', Linkify.titleSync Post::callbacks.push name: 'Linkify' cb: @node node: -> if @isClone and Conf['Embedding'] for embedder in $$ '.embedder', @nodes.comment $.on embedder, "click", Linkify.cb.toggle return snapshot = $.X './/text()', @nodes.comment i = -1 len = snapshot.snapshotLength while ++i < len node = snapshot.snapshotItem i data = node.data if Linkify.regString.test data Linkify.regString.lastIndex = 0 Linkify.gatherLinks node, @ return unless Conf['Embedding'] or Conf['Link Title'] for range in @nodes.links if data = Linkify.services range Linkify.embed data if Conf['Embedding'] Linkify.title data if Conf['Link Title'] return gatherLinks: (node, post) -> {data} = node len = data.length links = [] while (match = Linkify.regString.exec data) {index} = match link = match[0] len2 = index + link.length break if len - len2 is 0 range = document.createRange(); range.setStart node, index range.setEnd node, len2 links.push range if match Linkify.seek match, node, post for range in links.reverse() Linkify.makeLink range, post return seek: (match, node, post) -> {index} = match link = match[0] range = document.createRange() range.setStart node, index while (next = node.nextSibling) and next.nodeName isnt 'BR' node = next data = node.data if result = /[\s'"]/.exec data range.setEnd node, result.index if range.collapsed if node.nodeName is 'WBR' node = node.previousSibling range.setEnd node, node.length Linkify.makeLink range, post makeLink: (range, post) -> link = range.toString() link = if link.contains ':' link else ( if link.contains '@' 'mailto:' else 'http://' ) + link a = $.el 'a', className: 'linkify' rel: 'nofollow noreferrer' target: '_blank' href: link range.surroundContents a post.nodes.links.push a return services: (link) -> href = link.href for key, type of Linkify.types continue unless match = type.regExp.exec href return [key, match[1], match[2], link] return embed: (data) -> [key, uid, options, link] = data embed = $.el 'a', name: uid option: options className: 'embedder' href: 'javascript:;' textContent: '(embed)' embed.dataset.service = key embed.dataset.originalurl = link.href $.addClass link, "#{embed.dataset.service}" $.on embed, 'click', Linkify.cb.toggle $.after link, [$.tn(' '), embed] title: (data) -> [key, uid, options, link] = data return unless service = Linkify.types[key].title titles = Conf['CachedTitles'] if title = titles[uid] link.textContent = title[0] if Conf['Embedding'] link.nextElementSibling.dataset.title = title[0] else try $.cache service.api(uid), -> title = Linkify.cb.title.apply @, [data] catch err link.innerHTML = "[#{key}] Title Link Blocked (are you using NoScript?)" return if title titles[uid] = [title, Date.now()] $.set 'CachedTitles', titles titleSync: (value) -> Conf['CachedTitles'] = value cb: toggle: -> # We setup the link to be replaced by the embedded video embed = @previousElementSibling # Unembed. el = unless @className.contains "embedded" Linkify.cb.embed @ else Linkify.cb.unembed @ $.replace embed, el $.toggleClass @, 'embedded' embed: (a) -> # We create an element to embed el = (type = Linkify.types[a.dataset.service]).el.call a # Set style values. el.style.cssText = if style = type.style style else "border: 0; width: 640px; height: 390px" a.textContent = '(unembed)' return el unembed: (a) -> # Recreate the original link. el = $.el 'a', rel: 'nofollow noreferrer' target: 'blank' className: 'linkify' href: url = a.dataset.originalurl textContent: a.dataset.title or url a.textContent = '(embed)' $.addClass el, "#{a.dataset.service}" return el title: (data) -> [key, uid, options, link] = data service = Linkify.types[key].title link.textContent = switch @status when 200, 304 text = "#{service.text.call @}" if Conf['Embedding'] link.nextElementSibling.dataset.title = text text when 404 "[#{key}] Not Found" when 403 "[#{key}] Forbidden or Private" else "[#{key}] #{@status}'d" types: YouTube: regExp: /.*(?:youtu.be\/|youtube.*v=|youtube.*\/embed\/|youtube.*\/v\/|youtube.*videos\/)([^#\&\?]*)\??(t\=.*)?/ el: -> $.el 'iframe', src: "//www.youtube.com/embed/#{@name}#{if @option then '#' + @option else ''}?wmode=opaque" title: api: (uid) -> "https://gdata.youtube.com/feeds/api/videos/#{uid}?alt=json&fields=title/text(),yt:noembed,app:control/yt:state/@reasonCode" text: -> JSON.parse(@responseText).entry.title.$t Vocaroo: regExp: /.*(?:vocaroo.com\/)([^#\&\?]*).*/ style: 'border: 0; width: 150px; height: 45px;' el: -> $.el 'object', innerHTML: "" Vimeo: regExp: /.*(?:vimeo.com\/)([^#\&\?]*).*/ el: -> $.el 'iframe', src: "//player.vimeo.com/video/#{@name}?wmode=opaque" title: api: (uid) -> "https://vimeo.com/api/oembed.json?url=http://vimeo.com/#{uid}" text: -> JSON.parse(@responseText).title LiveLeak: regExp: /.*(?:liveleak.com\/view.+i=)([0-9a-z_]+)/ el: -> $.el 'object', innerHTML: "" audio: regExp: /(.*\.(mp3|ogg|wav))$/ el: -> $.el 'audio', controls: 'controls' preload: 'auto' src: @name image: regExp: /(http|www).*\.(gif|png|jpg|jpeg|bmp)$/ style: 'border: 0; width: auto; height: auto;' el: -> $.el 'div', innerHTML: "" SoundCloud: regExp: /.*(?:soundcloud.com\/|snd.sc\/)([^#\&\?]*).*/ style: 'height: auto; width: 500px; display: inline-block;' el: -> div = $.el 'div', className: "soundcloud" name: "soundcloud" $.ajax( "//soundcloud.com/oembed?show_artwork=false&&maxwidth=500px&show_comments=false&format=json&url=https://www.soundcloud.com/#{@name}" div: div onloadend: -> @div.innerHTML = JSON.parse(@responseText).html false) div title: api: (uid) -> "//soundcloud.com/oembed?show_artwork=false&&maxwidth=500px&show_comments=false&format=json&url=https://www.soundcloud.com/#{uid}" text: -> JSON.parse(@responseText).title pastebin: regExp: /.*(?:pastebin.com\/(?!u\/))([^#\&\?]*).*/ el: -> div = $.el 'iframe', src: "http://pastebin.com/embed_iframe.php?i=#{@name}" gist: regExp: /.*(?:gist.github.com.*\/)([^\/][^\/]*)$/ el: -> div = $.el 'iframe', # Github doesn't allow embedding straight from the site, so we use an external site to bypass that. src: "http://www.purplegene.com/script?url=https://gist.github.com/#{@name}.js" title: api: (uid) -> "https://api.github.com/gists/#{uid}" text: -> response = JSON.parse(@responseText).files return file for file of response when response.hasOwnProperty file InstallGentoo: regExp: /.*(?:paste.installgentoo.com\/view\/)([0-9a-z_]+)/ el: -> $.el 'iframe', src: "http://paste.installgentoo.com/view/embed/#{@name}"