diff --git a/couchpotato/core/helpers/request.py b/couchpotato/core/helpers/request.py index 6ffb7d5..09ee6b8 100644 --- a/couchpotato/core/helpers/request.py +++ b/couchpotato/core/helpers/request.py @@ -2,6 +2,7 @@ from couchpotato.core.helpers.variable import natcmp from flask.globals import current_app from flask.helpers import json from libs.werkzeug.urls import url_decode +from urllib import unquote_plus import flask import re @@ -24,7 +25,7 @@ def getParams(): for item in nested: if item is nested[-1]: - current[item] = value + current[item] = unquote_plus(value) else: try: current[item] @@ -33,7 +34,7 @@ def getParams(): current = current[item] else: - temp[param] = value + temp[param] = unquote_plus(value) return dictToList(temp) @@ -54,7 +55,10 @@ def dictToList(params): return new def getParam(attr, default = None): - return getattr(flask.request, 'args').get(attr, default) + try: + return unquote_plus(getattr(flask.request, 'args').get(attr, default)) + except: + return None def padded_jsonify(callback, *args, **kwargs): content = str(callback) + '(' + json.dumps(dict(*args, **kwargs)) + ')' diff --git a/couchpotato/core/notifications/core/static/notification.js b/couchpotato/core/notifications/core/static/notification.js index f4026d4..be42b06 100644 --- a/couchpotato/core/notifications/core/static/notification.js +++ b/couchpotato/core/notifications/core/static/notification.js @@ -9,6 +9,7 @@ var NotificationBase = new Class({ // Listener App.addEvent('load', self.startInterval.bind(self)); + App.addEvent('unload', self.stopTimer.bind(self)); self.addEvent('notification', self.notify.bind(self)) // Add test buttons to settings page diff --git a/couchpotato/core/plugins/updater/static/updater.js b/couchpotato/core/plugins/updater/static/updater.js index f7268b2..623bacf 100644 --- a/couchpotato/core/plugins/updater/static/updater.js +++ b/couchpotato/core/plugins/updater/static/updater.js @@ -4,6 +4,10 @@ var UpdaterBase = new Class({ var self = this; App.addEvent('load', self.info.bind(self, 1000)) + App.addEvent('unload', function(){ + if(self.timer) + clearTimeout(self.timer); + }); }, info: function(timeout){ diff --git a/couchpotato/core/plugins/userscript/main.py b/couchpotato/core/plugins/userscript/main.py index 188790f..34ab6b5 100644 --- a/couchpotato/core/plugins/userscript/main.py +++ b/couchpotato/core/plugins/userscript/main.py @@ -1,12 +1,12 @@ from couchpotato import index from couchpotato.api import addApiView -from couchpotato.core.event import fireEvent +from couchpotato.core.event import fireEvent, addEvent from couchpotato.core.helpers.request import getParam, jsonified from couchpotato.core.helpers.variable import isDict from couchpotato.core.logger import CPLog from couchpotato.core.plugins.base import Plugin from flask.globals import request -from flask.helpers import url_for +from flask.helpers import url_for, make_response log = CPLog(__name__) @@ -18,20 +18,26 @@ class Userscript(Plugin): addApiView('userscript', self.iFrame) addApiView('userscript.add_via_url', self.getViaUrl) + addEvent('userscript.get_version', self.getVersion) + def getExtension(self): params = { 'includes': fireEvent('userscript.get_includes', merge = True), 'excludes': fireEvent('userscript.get_excludes', merge = True), 'version': self.getVersion(), - 'host': '%s%suserscript/' % (request.host_url.rstrip('/'), url_for('api.index')), + 'api': '%suserscript/' % url_for('api.index').lstrip('/'), + 'host': request.host_url, } - return self.renderTemplate(__file__, 'template.js', **params) + response = make_response(self.renderTemplate(__file__, 'template.js', **params)) + response.headers['Content-Type'] = 'text/javascript' + return response + return def getVersion(self): - versions = fireEvent('userscript.get_version') + versions = fireEvent('userscript.get_provider_version') version = 0 for v in versions: diff --git a/couchpotato/core/plugins/userscript/static/userscript.js b/couchpotato/core/plugins/userscript/static/userscript.js index c03016d..a904bce 100644 --- a/couchpotato/core/plugins/userscript/static/userscript.js +++ b/couchpotato/core/plugins/userscript/static/userscript.js @@ -7,7 +7,7 @@ Page.Userscript = new Class({ options: { 'onOpened': function(){ - App.stopLoadTimer(); + App.fireEvent('unload'); App.getBlock('header').hide(); } }, @@ -43,4 +43,71 @@ Page.Userscript = new Class({ } -}); \ No newline at end of file +}); + +var UserscriptSettingTab = new Class({ + + tab: '', + content: '', + + initialize: function(){ + var self = this; + + App.addEvent('load', self.addSettings.bind(self)) + + }, + + addSettings: function(){ + var self = this; + + self.settings = App.getPage('Settings') + self.settings.addEvent('create', function(){ + var tab = self.settings.createTab('userscript', { + 'label': 'Userscript', + 'name': 'userscript' + }); + + self.tab = tab.tab; + self.content = tab.content; + + self.createUserscript(); + + }); + + }, + + createUserscript: function(){ + var self = this; + + + self.settings.createGroup({ + 'label': 'Install the Userscript' + }).inject(self.content).adopt( + new Element('a', { + 'text': 'Install userscript', + 'href': Api.createUrl('userscript.get')+'?couchpotato.user.js', + 'normalhref': true + }) + ); + + } + +}); + +window.addEvent('domready', function(){ + new UserscriptSettingTab(); +}); + +window.addEvent('load', function(){ + var your_version = $(document.body).get('data-userscript_version') + latest_version = App.getOption('userscript_version') + key = 'cp_version_check', + checked_already = Cookie.read(key); + + if(your_version < latest_version && checked_already < latest_version){ + if(confirm("Update to the latest Userscript?\nYour version: " + your_version + ', new version: ' + latest_version )){ + document.location = Api.getOption('url')+'userscript.get/?couchpotato.user.js'; + } + Cookie.write(key, latest_version, {duration: 100}); + } +}); diff --git a/couchpotato/core/plugins/userscript/template.js b/couchpotato/core/plugins/userscript/template.js index 7c1b0c2..7dbd4ea 100644 --- a/couchpotato/core/plugins/userscript/template.js +++ b/couchpotato/core/plugins/userscript/template.js @@ -1,16 +1,20 @@ // ==UserScript== // @name CouchPotato UserScript // @description Add movies like a real CouchPotato +// @version {{version}} // @include {{host}}* {% for include in includes %} // @include {{include}}{% endfor %} {% for exclude in excludes %} // @exclude {{exclude}}{% endfor %} +// @exclude {{host}}{{api.rstrip('/')}}* + // ==/UserScript== var version = {{version}}, - cpLocation = '{{host}}'; + host = '{{host}}'; + api = '{{api}}'; function create() { switch (arguments.length) { @@ -40,7 +44,7 @@ function create() { if (typeof GM_addStyle == 'undefined'){ GM_addStyle = function(css) { - var head = document.getElementsByTagName('head')[0], + var head = document.getElementsByTagName('head')[0], style = document.createElement('style'); if (!head) return; @@ -53,53 +57,56 @@ if (typeof GM_addStyle == 'undefined'){ // Styles GM_addStyle('\ - #cpPopup { opacity: 0.5; width:200px; font-family: "Helvetica Neue", Helvetica, Arial, Geneva, sans-serif; -moz-border-radius-topleft: 6px; -moz-border-radius-topright: 6px; -webkit-border-top-left-radius: 6px; -webkit-border-top-right-radius: 6px; -moz-box-shadow: 0 0 20px rgba(0,0,0,0.5); -webkit-box-shadow: 0 0 20px rgba(0,0,0,0.5); position:fixed; z-index:9999; bottom:0; right:0; font-size:15px; margin: 0 20px; display: block; background:#f5f5f5; } \ - #cpPopup:hover { opacity: 1; } \ - #cpPopup a#addTo { cursor:pointer; text-align:center; text-decoration:none; color: #000; display:block; padding:15px 0 10px; } \ - #cpPopup a#closeBtn { cursor:pointer; float: right; padding:10px; } \ - #cpPopup a img { vertical-align: middle; } \ - #cpPopup a:hover { color:#000; } \ - #cpPopup iframe{ background:#f5f5f5; margin:6px; height:70px; width:188px; overflow:hidden; border:none; } \ + #cp_popup { font-family: "Helvetica Neue", Helvetica, Arial, Geneva, sans-serif; -moz-border-radius-topleft: 6px; -moz-border-radius-bottomleft: 6px; -webkit-border-top-left-radius: 6px; -webkit-border-bottom-left-radius: 6px; -moz-box-shadow: 0 0 20px rgba(0,0,0,0.5); -webkit-box-shadow: 0 0 20px rgba(0,0,0,0.5); position:fixed; z-index:9999; bottom:0; right:0; font-size:15px; margin: 20px 0; display: block; background:#4E5969; } \ + #cp_popup:hover { } \ + #cp_popup a#add_to { cursor:pointer; text-align:center; text-decoration:none; color: #000; display:block; padding:5px 0 5px 5px; } \ + #cp_popup a#close_button { cursor:pointer; float: right; padding:120px 10px 10px; } \ + #cp_popup a img { vertical-align: middle; } \ + #cp_popup a:hover { color:#000; } \ + #cp_popup iframe{ background:#4E5969; margin:6px 0 2px 6px; height:140px; width:450px; overflow:hidden; border:none; } \ '); -var movieImg = ''; -var closeImg = ''; +var cp_icon = ''; +var close_img = ''; var osd = function(){ var navbar, newElement; - var iFrame = create('iframe', { - 'src': cpLocation + "userscript.add_via_url/?url=" + escape(document.location), + var iframe = create('iframe', { + 'src': host + api + "?url=" + escape(document.location.href), 'frameborder': 0, 'scrolling': 'no' }); - var addToText = ''; - var popupId = 'cpPopup'; - var popup = create('div', { - 'id': popupId, - 'innerHTML': addToText + 'id': 'cp_popup' }); - var addButton = create('a', { - 'innerHTML': 'Add to CouchPotato', - 'id': 'addTo', + var add_button = create('a', { + 'innerHTML': '', + 'id': 'add_to', 'onclick': function(){ popup.innerHTML = ''; popup.appendChild(create('a', { - 'innerHTML': '', - 'id': 'closeBtn', + 'innerHTML': '', + 'id': 'close_button', 'onclick': function(){ popup.innerHTML = ''; - popup.appendChild(addButton); + popup.appendChild(add_button); } })); - popup.appendChild(iFrame) + popup.appendChild(iframe) } }); - popup.appendChild(addButton); + popup.appendChild(add_button); document.body.parentNode.insertBefore(popup, document.body); }; -osd(); \ No newline at end of file +var setVersion = function(){ + document.body.setAttribute('data-userscript_version', version) +}; + +if(document.location.href.indexOf(host) == -1) + osd(); +else + setVersion(); \ No newline at end of file diff --git a/couchpotato/core/providers/nzb/x264/main.py b/couchpotato/core/providers/nzb/x264/main.py index aba7f64..c8eb1cb 100644 --- a/couchpotato/core/providers/nzb/x264/main.py +++ b/couchpotato/core/providers/nzb/x264/main.py @@ -60,8 +60,6 @@ class X264(NZBProvider): 'check_nzb': False, } - print new['name'] - new['score'] = fireEvent('score.calculate', new, movie, single = True) is_correct_movie = fireEvent('searcher.correct_movie', nzb = new, movie = movie, quality = quality, diff --git a/couchpotato/core/providers/userscript/base.py b/couchpotato/core/providers/userscript/base.py index ec7693f..e23a540 100644 --- a/couchpotato/core/providers/userscript/base.py +++ b/couchpotato/core/providers/userscript/base.py @@ -17,8 +17,7 @@ class UserscriptBase(Plugin): def __init__(self): addEvent('userscript.get_includes', self.getInclude) addEvent('userscript.get_excludes', self.getExclude) - addEvent('userscript.get_version', self.getVersion) - + addEvent('userscript.get_provider_version', self.getVersion) addEvent('userscript.get_movie_via_url', self.belongsTo) def search(self, name, year = None): diff --git a/couchpotato/core/providers/userscript/imdb/main.py b/couchpotato/core/providers/userscript/imdb/main.py index 06f6f16..5e5ae91 100644 --- a/couchpotato/core/providers/userscript/imdb/main.py +++ b/couchpotato/core/providers/userscript/imdb/main.py @@ -1,7 +1,6 @@ -from beautifulsoup import BeautifulSoup from couchpotato.core.event import fireEvent +from couchpotato.core.helpers.variable import getImdb from couchpotato.core.providers.userscript.base import UserscriptBase -import re class IMDB(UserscriptBase): @@ -9,16 +8,4 @@ class IMDB(UserscriptBase): includes = ['http*://*.imdb.com/title/tt*', 'http*://imdb.com/title/tt*'] def getMovie(self, url): - - data = self.urlopen(url) - - html = BeautifulSoup(data) - headers = html.findAll('h5') - - # Don't add TV show - for head in headers: - if 'seasons' in head.lower(): - return 'IMDB url is a TV Show' - - identifier = re.search('(?Ptt[0-9{7}]+)', url).group('id') - return fireEvent('movie.info', identifier = identifier, merge = True) + return fireEvent('movie.info', identifier = getImdb(url), merge = True) diff --git a/couchpotato/static/scripts/couchpotato.js b/couchpotato/static/scripts/couchpotato.js index 2c5a02c..e1649ef 100644 --- a/couchpotato/static/scripts/couchpotato.js +++ b/couchpotato/static/scripts/couchpotato.js @@ -29,7 +29,7 @@ var CouchPotato = new Class({ else self.openPage(window.location.pathname); - self.c.addEvent('click:relay(a:not([target=_blank]))', self.pushState.bind(self)); + self.c.addEvent('click:relay(a:not([target=_blank]):not([normalhref=true]))', self.pushState.bind(self)); }, getOption: function(name){ @@ -72,16 +72,9 @@ var CouchPotato = new Class({ $(pg).inject(self.content); }); - self.load_timer = (function(){ - self.fireEvent('load'); - }).delay(1000); + self.fireEvent('load'); }, - - stopLoadTimer: function(){ - if(this.load_timer) - clearInterval(this.load_timer); - }, openPage: function(url) { var self = this; diff --git a/couchpotato/templates/_desktop.html b/couchpotato/templates/_desktop.html index 463ab18..95fb1cd 100644 --- a/couchpotato/templates/_desktop.html +++ b/couchpotato/templates/_desktop.html @@ -18,7 +18,7 @@ - + @@ -66,7 +66,8 @@ 'args': '{{ env.get('args') }}', 'options': '{{ env.get('options') }}', 'app_dir': '{{ env.get('app_dir') }}', - 'data_dir': '{{ env.get('data_dir') }}' + 'data_dir': '{{ env.get('data_dir') }}', + 'userscript_version': {{ fireEvent('userscript.get_version', single = True)|safe }} }); //Wizard.start.delay(100, Wizard);