diff --git a/CHANGES.md b/CHANGES.md
index b847ed4970fe481ebfea621bc04ee3b2cf48141d..00e17cb6c1ef661861aa477843d67e6a15778c33 100644
--- a/CHANGES.md
+++ b/CHANGES.md
@@ -1,19 +1,60 @@
+### 4.0.17 (2015-04-19)
+
+[full changelog](https://github.com/SiCKRAGETV/SickRage/compare/v4.0.16...v4.0.17)
+
+* libnotify: Use gir-notify instead of pynotify.
+* Fix for xml_declaration unexpected keyword in python 2.6.
+* Fix UnboundLocalError exception can be thrown at piList.append(curQueueItem).
+* Fix Auto determine indexer when indexer tag not present in nfo.
+* Fix editShow: Select the default indexer language instead of forcing en.
+* Fix Catch emailnotify smtp error.
+* Fix Change log timeout errors to WARNING in requests provider clients.
+* Fix Ignore OSError when obtaining the size of a path in get_size.
+* New Add RSS to EZTV provider.
+* New SceneTime provider.
+* Fix iptorrent: Add missing import.
+* New Add Pause Button on displayShow page.
+* Fix Episode Thumbnail url when show is in DVD Order.
+* Fix Catch all errors from Kodi notify.
+* Fix Add user agent to RARBG when login.
+* New Use better logging for get_size exceptions.
+* New Added qbittorrent support.
+* Fix rarbg: remove urllib.quote.
+* Update PNotify to latest version.
+* Fix Plex Notification.
+* New added the option to use Plex Token vs Username / Password.
+* Fix PP Case Sensitive associated files check.
+* Fix unknown quality being accepted as a valid quality in PP.
+* Fix added notification of unsupported multiepisode torrent.
+* Fix proper: saveSearch broken for download propers.
+* New Add logline to show found associated files.
+* New Add a log line containing connecting IP on login.
+* Fix eztv provider.
+* Fix Title of None for Errors submitted through UI.
+* New Stay on Show Page after "Update Kodi".
+* Fix NoneType object has no attribute getOverview.
+* New  Manage Rolling Download on unpause from submenu button.
+* Fix provide a more detailed http code error.
+* Fix Show subtitle service unavailable as info.
+* New Column Selection for displayShow.
+* Fix Update IPTorrents search URL.
+
 ### 4.0.16 (2015-04-12)
 
 [full changelog](https://github.com/SiCKRAGETV/SickRage/compare/v4.0.15...v4.0.16)
 
-* 	New Feature: Added option to add a filter row on main show page (enabled in general settings - interface)
-* 	New Feature: added scheduling status page
-* 	Fixed EZTV rss blank result
-* 	Fixed RARBG tokend
-* 	Fixed Home page filter: change to allow to use parsed data for active(yes/no)
-* 	Fixed OldTPB: Check if the returned results ar proper|repack
-* 	Don't display paused show in backlogOverview
-* 	Trakt Sync by Episode not by Show
-* 	Added gzip setting in config.ini
-* 	Added TRAKTROLLING to filter in viewlog
-* 	Redone Scheduler
-* 	Replace fuzzy images on Add Show ( Add Trending) 
+* New Feature: Added option to add a filter row on main show page (enabled in general settings - interface)
+* New Feature: added scheduling status page
+* Fixed EZTV rss blank result
+* Fixed RARBG tokend
+* Fixed Home page filter: change to allow to use parsed data for active(yes/no)
+* Fixed OldTPB: Check if the returned results ar proper|repack
+* Don't display paused show in backlogOverview
+* Trakt Sync by Episode not by Show
+* Added gzip setting in config.ini
+* Added TRAKTROLLING to filter in viewlog
+* Redone Scheduler
+* Replace fuzzy images on Add Show ( Add Trending) 
 
 ### 4.0.15 (2015-04-05)
 
diff --git a/gui/slick/css/dark.css b/gui/slick/css/dark.css
index 95467041f27778b1f1ba2ff1e62d3ea9a5217582..6a320fef293220ca1dda1ca6b9e6114a7366c03b 100644
--- a/gui/slick/css/dark.css
+++ b/gui/slick/css/dark.css
@@ -1192,6 +1192,29 @@ span.snatched b {
 	opacity: 0.4;
 }
 
+.displayShowTable {
+	table-layout: auto;
+	width: 100%;
+	border-collapse: collapse;
+	border-spacing: 0;
+	text-align: center;
+	border: none;
+	empty-cells: show;
+	color: #000 !important;
+}
+
+.displayShowTable.display_show {
+	clear:both
+}
+
+.displayShowTable th.row-seasonheader {
+	border: none !important;
+	background-color: #222 !important;
+	color: #fff !important;
+	padding-top: 15px !important;
+	text-align: left !important;
+}
+  
 .sickbeardTable {
 	table-layout: auto;
 	width: 100%;
diff --git a/gui/slick/css/lib/pnotify.custom.min.css b/gui/slick/css/lib/pnotify.custom.min.css
index 1d0fa42eca7d01a07cc39d456075f7ce0c76c997..8f2d3eb74d4cf473f30609b7d6711947e16b47d1 100644
Binary files a/gui/slick/css/lib/pnotify.custom.min.css and b/gui/slick/css/lib/pnotify.custom.min.css differ
diff --git a/gui/slick/css/light.css b/gui/slick/css/light.css
index 8df3fe1d64aacf8d1fc65bcdca2ee36d32c58ce7..1c49d9714af641a0a414e4b1eb8d2fc6a20c2197 100644
--- a/gui/slick/css/light.css
+++ b/gui/slick/css/light.css
@@ -1167,6 +1167,29 @@ span.snatched b {
 	opacity: 0.4;
 }
 
+.displayShowTable {
+	table-layout: auto;
+	width: 100%;
+	border-collapse: collapse;
+	border-spacing: 0;
+	text-align: center;
+	border: none;
+	empty-cells: show;
+	color: #000 !important;
+}
+
+.displayShowTable.display_show {
+	clear:both
+}
+
+.displayShowTable th.row-seasonheader {
+	border: none !important;
+	background-color: #fff !important;
+	color: #000 !important; 
+	padding-top: 15px !important;
+	text-align: left !important;
+}
+
 .sickbeardTable {
 	table-layout: auto;
 	width: 100%;
diff --git a/gui/slick/css/style.css b/gui/slick/css/style.css
index f4d250dc5c8ffd73d8350ed3549768d7ded9d381..4216d0c4ac1585481bdbde9cac540c47e4bf9113 100644
--- a/gui/slick/css/style.css
+++ b/gui/slick/css/style.css
@@ -1193,6 +1193,29 @@ span.snatched b {
 	opacity: 0.4;
 }
 
+.displayShowTable {
+	table-layout: auto;
+	width: 100%;
+	border-collapse: collapse;
+	border-spacing: 0;
+	text-align: center;
+	border: none;
+	empty-cells: show;
+	color: #000 !important;
+}
+
+.displayShowTable.display_show {
+	clear:both
+}
+
+.displayShowTable th.row-seasonheader {
+	border: none !important;
+	background-color: #222 !important;
+	color: #fff !important;
+	padding-top: 15px !important;
+	text-align: left !important;
+}
+
 .sickbeardTable {
 	table-layout: auto;
 	width: 100%;
diff --git a/gui/slick/images/providers/scenetime.png b/gui/slick/images/providers/scenetime.png
new file mode 100644
index 0000000000000000000000000000000000000000..32e729817493ca9411fc78e698a8b9c37a081780
Binary files /dev/null and b/gui/slick/images/providers/scenetime.png differ
diff --git a/gui/slick/interfaces/default/config_general.tmpl b/gui/slick/interfaces/default/config_general.tmpl
index e4651ce30aa262b1cb2560b3fc16242296476a06..3948f456e21749e879ee65fb39d716018da2ba6f 100644
--- a/gui/slick/interfaces/default/config_general.tmpl
+++ b/gui/slick/interfaces/default/config_general.tmpl
@@ -299,20 +299,11 @@
 							<label for="filter_row">
 								<span class="component-title">Filter Row</span>
 								<span class="component-desc">
-									<input type="checkbox" name="filter_row" id="display_filesize" #if $sickbeard.FILTER_ROW == True then 'checked="checked"' else ''#/>
+									<input type="checkbox" name="filter_row" id="filter_row" #if $sickbeard.FILTER_ROW == True then 'checked="checked"' else ''#/>
 									<p>Add a filter row to the show display on the home page</p>
 								</span>
 							</label>
 						</div>                        
-						<div class="field-pair">
-							<label for="display_filesize">
-								<span class="component-title">Display Filesizes</span>
-								<span class="component-desc">
-									<input type="checkbox" name="display_filesize" id="display_filesize" #if $sickbeard.DISPLAY_FILESIZE == True then 'checked="checked"' else ''#/>
-									<p>display filesizes for downloaded episodes on show page</p>
-								</span>
-							</label>
-						</div>
 						<div class="field-pair">
 							<label for="coming_eps_missed_range">
 								<span class="component-title">Missed episodes range</span>
@@ -774,4 +765,4 @@
 //-->
 </script>
 
-#include $os.path.join($sickbeard.PROG_DIR, 'gui/slick/interfaces/default/inc_bottom.tmpl')
+#include $os.path.join($sickbeard.PROG_DIR, 'gui/slick/interfaces/default/inc_bottom.tmpl')
\ No newline at end of file
diff --git a/gui/slick/interfaces/default/config_notifications.tmpl b/gui/slick/interfaces/default/config_notifications.tmpl
index 3fb21751c7305d6e4c29eb30956b2c06d3784230..5b82eb6f80418f4d54a018a137acb373086d87f2 100644
--- a/gui/slick/interfaces/default/config_notifications.tmpl
+++ b/gui/slick/interfaces/default/config_notifications.tmpl
@@ -175,6 +175,73 @@
                         </div>
 
                         <div id="content_use_plex">
+                            <div class="field-pair">
+                                <label for="plex_server_token">
+                                    <span class="component-title">Plex Media Server Auth Token</span>
+                                    <input type="text" name="plex_server_token" id="plex_server_token" value="$sickbeard.PLEX_SERVER_TOKEN" class="form-control input-sm input250" />
+                                </label>
+                                <label>
+                                    <span class="component-title">&nbsp;</span>
+                                    <span class="component-desc">Auth Token used by plex</span>
+                                </label>
+                                <label>
+                                    <span class="component-title">&nbsp;</span>
+                                    <span class="component-desc">(<a href="<%= anon_url('https://support.plex.tv/hc/en-us/articles/204059436-Finding-your-account-token-X-Plex-Token') %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;"><u>Finding your account token</u></a>)</span>
+                                </label>
+                            </div>                        
+							<div class="component-group" style="padding: 0; min-height: 130px">
+								<div class="field-pair">
+									<label for="plex_username">
+										<span class="component-title">Server/client username</span>
+										<span class="component-desc">
+											<input type="text" name="plex_username" id="plex_username" value="$sickbeard.PLEX_USERNAME" class="form-control input-sm input250" />
+											<p>blank = no authentication</p>
+										</span>
+									</label>
+								</div>
+								<div class="field-pair">
+									<label for="plex_password">
+										<span class="component-title">Server/client password</span>
+										<span class="component-desc">
+											<input type="password" name="plex_password" id="plex_password" value="#echo '*' * len($sickbeard.PLEX_PASSWORD)#" class="form-control input-sm input250" />
+											<p>blank = no authentication</p>
+										</span>
+									</label>
+								</div>
+							</div>
+                        
+							<div class="component-group" style="padding: 0; min-height: 50px">
+								<div class="field-pair">
+									<label for="plex_update_library">
+										<span class="component-title">Update server library</span>
+										<span class="component-desc">
+											<input type="checkbox" class="enabler" name="plex_update_library" id="plex_update_library" #if $sickbeard.PLEX_UPDATE_LIBRARY then 'checked="checked" ' else ''#/>
+											<p>update Plex Media Server library when a download finishes</p>
+										</span>
+									</label>
+								</div>
+								<div id="content_plex_update_library">
+									<div class="field-pair">
+										<label for="plex_server_host">
+											<span class="component-title">Plex Media Server IP:Port</span>
+											<span class="component-desc">
+												<input type="text" name="plex_server_host" id="plex_server_host" value="<%= re.sub(r'\b,\b', ', ', sickbeard.PLEX_SERVER_HOST) %>" class="form-control input-sm input350" />
+												<div class="clear-left">
+													<p>one or more hosts running Plex Media Server<br />(eg. 192.168.1.1:32400, 192.168.1.2:32400)</p>
+												</div>
+											</span>
+										</label>
+									</div>
+
+									<div class="field-pair">
+										<div class="testNotification" id="testPMS-result">Click below to test Plex server(s)</div>
+										<input class="btn" type="button" value="Test Plex Server" id="testPMS" />
+										<input type="submit" class="config_submitter btn" value="Save Changes" />
+										<div class="clear-left">&nbsp;</div>
+									</div>
+								</div>
+							</div>
+                                    
                             <div class="field-pair">
                                 <label for="plex_notify_onsnatch">
                                     <span class="component-title">Notify on snatch</span>
@@ -202,78 +269,25 @@
                                     </span>
                                 </label>
                             </div>
-                            <div class="field-pair">
-                                <label for="plex_update_library">
-                                    <span class="component-title">Update library</span>
-                                    <span class="component-desc">
-                                        <input type="checkbox" name="plex_update_library" id="plex_update_library" #if $sickbeard.PLEX_UPDATE_LIBRARY then "checked=\"checked\"" else ""# />
-                                        <p>update Plex Media Server library when a download finishes ?</p>
-                                    </span>
-                                </label>
-                            </div>
-                            <div class="field-pair">
-                                <label for="plex_server_host">
-                                    <span class="component-title">Plex Media Server IP:Port</span>
-                                    <input type="text" name="plex_server_host" id="plex_server_host" value="$sickbeard.PLEX_SERVER_HOST" class="form-control input-sm input250" />
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">host running Plex Media Server (eg. 192.168.1.100:32400)</span>
-                                </label>
-                            </div>
-                            <div class="field-pair">
-                                <label for="plex_server_token">
-                                    <span class="component-title">Plex Media Server Auth Token</span>
-                                    <input type="text" name="plex_server_token" id="plex_server_token" value="$sickbeard.PLEX_SERVER_TOKEN" class="form-control input-sm input250" />
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">Auth Token used by plex</span>
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">(<a href="<%= anon_url('https://support.plex.tv/hc/en-us/articles/204059436-Finding-your-account-token-X-Plex-Token') %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;"><u>Finding your account token</u></a>)</span>
-                                </label>
-                            </div>
                             <div class="field-pair">
                                 <label for="plex_host">
                                     <span class="component-title">Plex Client IP:Port</span>
-                                    <input type="text" name="plex_host" id="plex_host" value="$sickbeard.PLEX_HOST" class="form-control input-sm input350" />
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">host running Plex Client (eg. 192.168.1.100:3000)</span>
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">(multiple host strings must be separated by commas)</span>
-                                </label>
-                            </div>
-                            <div class="field-pair">
-                                <label for="plex_username">
-                                    <span class="component-title">Plex Client username</span>
-                                    <input type="text" name="plex_username" id="plex_username" value="$sickbeard.PLEX_USERNAME" class="form-control input-sm input250" />
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">username for your Plex client API (blank for none)</span>
+                                    <span class="component-desc">
+                                        <input type="text" name="plex_host" id="plex_host" value="$sickbeard.PLEX_HOST" class="form-control input-sm input350" />
+                                        <div class="clear-left">
+                                            <p>one or more hosts running Plex client<br />(eg. 192.168.1.100:3000, 192.168.1.101:3000)</p>
+                                        </div>
+                                    </span>
                                 </label>
                             </div>
+
                             <div class="field-pair">
-                                <label for="plex_password">
-                                    <span class="component-title">Plex Client password</span>
-                                    <input type="password" name="plex_password" id="plex_password" value="$sickbeard.PLEX_PASSWORD" class="form-control input-sm input250" />
-                                </label>
-                                <label>
-                                    <span class="component-title">&nbsp;</span>
-                                    <span class="component-desc">password for your Plex client API (blank for none)</span>
-                                </label>
+                                <div class="testNotification" id="testPMC-result">Click below to test Plex client(s)</div>
+                                <input class="btn" type="button" value="Test Plex Client" id="testPMC" />
+                                <input type="submit" class="config_submitter btn" value="Save Changes" />
+                                <div class=clear-left><p>Note: some Plex clients <b class="boldest">do not</b> support notifications e.g. Plexapp for Samsung TVs</p></div>
                             </div>
-                            <div class="testNotification" id="testPLEX-result">Click below to test.</div>
-                            <input class="btn" type="button" value="Test Plex Client" id="testPLEX" />
-                            <input type="submit" class="config_submitter btn" value="Save Changes" />
                         </div><!-- /content_use_plex -->
-
                     </fieldset>
                 </div><!-- /plex component-group -->
 
diff --git a/gui/slick/interfaces/default/config_search.tmpl b/gui/slick/interfaces/default/config_search.tmpl
index 2c745ab0d6baa96d22c8737f44480376da7a329b..4b2a8648aac37e51ad2db537b06272dd66dd83a4 100755
--- a/gui/slick/interfaces/default/config_search.tmpl
+++ b/gui/slick/interfaces/default/config_search.tmpl
@@ -448,8 +448,8 @@
 								<span class="component-title">Send .torrent files to:</span>
 								<span class="component-desc">
 								<select name="torrent_method" id="torrent_method" class="form-control input-sm">
-#set $torrent_method_text = {'blackhole': "Black hole", 'utorrent': "uTorrent", 'transmission': "Transmission", 'deluge': "Deluge", 'download_station': "Synology DS", 'rtorrent': "rTorrent"}
-#for $curAction in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent'):
+#set $torrent_method_text = {'blackhole': "Black hole", 'utorrent': "uTorrent", 'transmission': "Transmission", 'deluge': "Deluge", 'download_station': "Synology DS", 'rtorrent': "rTorrent", 'qbittorrent': "qbittorrent"}
+#for $curAction in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent', 'qbittorrent'):
     #set $selected = $html_selected if $sickbeard.TORRENT_METHOD == $curAction else ''
 								<option value="$curAction"$selected>$torrent_method_text[$curAction]</option>
 #end for
diff --git a/gui/slick/interfaces/default/displayShow.tmpl b/gui/slick/interfaces/default/displayShow.tmpl
index 3ed05330b56b85fe151a1061e3bc978fb5767836..358389403416a46f0658d96a6e7e579b9c9443f9 100644
--- a/gui/slick/interfaces/default/displayShow.tmpl
+++ b/gui/slick/interfaces/default/displayShow.tmpl
@@ -7,6 +7,7 @@
 #import os.path, os
 #import datetime
 #import urllib
+#import ntpath
 
 #set global $title=$show.name
 ##set global $header = '<a></a>' % 
@@ -47,487 +48,533 @@
         });
     });
     #end raw
-	
-	\$.fn.generateStars = function() {
-		return this.each(function(i,e){\$(e).html(\$('<span/>').width(\$(e).text()*12));});
-	};
+    
+    \$.fn.generateStars = function() {
+        return this.each(function(i,e){\$(e).html(\$('<span/>').width(\$(e).text()*12));});
+    };
+
+    \$('.imdbstars').generateStars();
+
+
+
+            #if $show.is_anime:
+    \$("#animeTable").tablesorter({
+            #else:
+    \$("#showTable").tablesorter({
+            #end if
+        widgets: ['saveSort', 'stickyHeaders', 'columnSelector'],
+        widgetOptions : {
+            columnSelector_saveColumns: true,
+            columnSelector_layout : '<br/><label><input type="checkbox">{name}</label>',
+            columnSelector_mediaquery: false,
+            columnSelector_cssChecked : 'checked'
+            },
+        });    
 
-	\$('.imdbstars').generateStars();
-	
+    
+    
+    \$('#popover')
+        .popover({
+          placement: 'bottom',
+          html: true, // required if content has HTML
+          content: '<div id="popover-target"></div>'
+        })
+        // bootstrap popover event triggered when the popover opens
+        .on('shown.bs.popover', function () {
+                #if $show.is_anime:
+            \$.tablesorter.columnSelector.attachTo( \$('#animeTable'), '#popover-target');
+                #else:
+            \$.tablesorter.columnSelector.attachTo( \$('#showTable'), '#popover-target');                
+                #end if
+        });      
 });
 //-->
 </script>
 
-	<div class="pull-left form-inline">
-		Change Show:
-		<div class="navShow"><img id="prevShow" src="$sbRoot/images/prev.png" alt="&lt;&lt;" title="Prev Show" /></div>
-			<select id="pickShow" class="form-control form-control-inline input-sm">
-			#for $curShowList in $sortedShowLists:
-				#set $curShowType = $curShowList[0]
-				#set $curShowList = $curShowList[1]
-
-				#if len($sortedShowLists) > 1:
-					<optgroup label="$curShowType">
-				#end if
-					#for $curShow in $curShowList:
-					<option value="$curShow.indexerid" #if $curShow == $show then "selected=\"selected\"" else ""#>$curShow.name</option>
-					#end for
-				#if len($sortedShowLists) > 1:
-					</optgroup>
-				#end if
-			#end for
-			</select>
-		<div class="navShow"><img id="nextShow" src="$sbRoot/images/next.png" alt="&gt;&gt;" title="Next Show" /></div>
-	</div>
-
-	<div class="clearfix"></div>
-
-	<div id="showtitle" data-showname="$show.name">
-		<h1 class="title" id="scene_exception_$show.indexerid">$show.name</h1>
-	</div>
-
-	
-		#if $seasonResults:
-		##There is a special/season_0?##
-		#if int($seasonResults[-1]["season"]) == 0:
-			#set $season_special = 1
-		#else: 
-			#set $season_special = 0
-		#end if
-
-		#if not $sickbeard.DISPLAY_SHOW_SPECIALS and $season_special:
-			$seasonResults.pop(-1)
-		#end if
-
-		<span class="h2footer displayspecials pull-right">
-			#if $season_special:
-			Display Specials:
-				#if sickbeard.DISPLAY_SHOW_SPECIALS:
-					<a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Hide</a>
-				#else:
-					<a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Show</a>
-				#end if
-			#end if
-		</span>
-		
-		<div class="h2footer pull-right"> 
-			<span>
-			#if (len($seasonResults) > 14):
-				<select id="seasonJump" class="form-control input-sm" style="position: relative; top: -4px;">
-					<option value="jump">Jump to Season</option>
-				#for $seasonNum in $seasonResults:
-					<option value="#season-$seasonNum["season"]">#if int($seasonNum["season"]) == 0 then "Specials" else "Season " + str($seasonNum["season"])#</option>
-				#end for
-				</select>
-			#else:
-				Season:
-				#for $seasonNum in $seasonResults:
-					#if int($seasonNum["season"]) == 0:
-						<a href="#season-$seasonNum["season"]">Specials</a>
-					#else:
-						<a href="#season-$seasonNum["season"]">${str($seasonNum["season"])}</a>
-					#end if
-					#if $seasonNum != $seasonResults[-1]:
-						<span class="separator">|</span>
-					#end if
-				#end for
-			#end if
-			</span>
-
-		#end if
-		</div>
-	
-	<div class="clearfix"></div>
+    <div class="pull-left form-inline">
+        Change Show:
+        <div class="navShow"><img id="prevShow" src="$sbRoot/images/prev.png" alt="&lt;&lt;" title="Prev Show" /></div>
+            <select id="pickShow" class="form-control form-control-inline input-sm">
+            #for $curShowList in $sortedShowLists:
+                #set $curShowType = $curShowList[0]
+                #set $curShowList = $curShowList[1]
+
+                #if len($sortedShowLists) > 1:
+                    <optgroup label="$curShowType">
+                #end if
+                    #for $curShow in $curShowList:
+                    <option value="$curShow.indexerid" #if $curShow == $show then "selected=\"selected\"" else ""#>$curShow.name</option>
+                    #end for
+                #if len($sortedShowLists) > 1:
+                    </optgroup>
+                #end if
+            #end for
+            </select>
+        <div class="navShow"><img id="nextShow" src="$sbRoot/images/next.png" alt="&gt;&gt;" title="Next Show" /></div>
+    </div>
+
+    <div class="clearfix"></div>
+
+    <div id="showtitle" data-showname="$show.name">
+        <h1 class="title" id="scene_exception_$show.indexerid">$show.name</h1>
+    </div>
+
+    
+        #if $seasonResults:
+        ##There is a special/season_0?##
+        #if int($seasonResults[-1]["season"]) == 0:
+            #set $season_special = 1
+        #else: 
+            #set $season_special = 0
+        #end if
+
+        #if not $sickbeard.DISPLAY_SHOW_SPECIALS and $season_special:
+            $seasonResults.pop(-1)
+        #end if
+
+        <span class="h2footer displayspecials pull-right">
+            #if $season_special:
+            Display Specials:
+                #if sickbeard.DISPLAY_SHOW_SPECIALS:
+                    <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Hide</a>
+                #else:
+                    <a class="inner" href="$sbRoot/toggleDisplayShowSpecials/?show=$show.indexerid">Show</a>
+                #end if
+            #end if
+        </span>
+        
+        <div class="h2footer pull-right"> 
+            <span>
+            #if (len($seasonResults) > 14):
+                <select id="seasonJump" class="form-control input-sm" style="position: relative; top: -4px;">
+                    <option value="jump">Jump to Season</option>
+                #for $seasonNum in $seasonResults:
+                    <option value="#season-$seasonNum["season"]">#if int($seasonNum["season"]) == 0 then "Specials" else "Season " + str($seasonNum["season"])#</option>
+                #end for
+                </select>
+            #else:
+                Season:
+                #for $seasonNum in $seasonResults:
+                    #if int($seasonNum["season"]) == 0:
+                        <a href="#season-$seasonNum["season"]">Specials</a>
+                    #else:
+                        <a href="#season-$seasonNum["season"]">${str($seasonNum["season"])}</a>
+                    #end if
+                    #if $seasonNum != $seasonResults[-1]:
+                        <span class="separator">|</span>
+                    #end if
+                #end for
+            #end if
+            </span>
+
+        #end if
+        </div>
+    
+    <div class="clearfix"></div>
     
 #if $show_message:
-	<div class="alert alert-info">
-		$show_message
-	</div>
+    <div class="alert alert-info">
+        $show_message
+    </div>
 #end if
-	
-	<div id="container">
-		<div id="posterCol">
-			<a href="$sbRoot/showPoster/?show=$show.indexerid&amp;which=poster" rel="dialog" title="View Poster for $show.name"><img src="$sbRoot/showPoster/?show=$show.indexerid&amp;which=poster_thumb" class="tvshowImg" alt=""/></a>
-		</div>
-
-		<div id="showCol">
-		
-			<div id="showinfo">
+    
+    <div id="container">
+        <div id="posterCol">
+            <a href="$sbRoot/showPoster/?show=$show.indexerid&amp;which=poster" rel="dialog" title="View Poster for $show.name"><img src="$sbRoot/showPoster/?show=$show.indexerid&amp;which=poster_thumb" class="tvshowImg" alt=""/></a>
+        </div>
+
+        <div id="showCol">
+        
+            <div id="showinfo">
 #if 'rating' in $show.imdb_info:
     #set $rating_tip = str($show.imdb_info['rating']) + " / 10" + " Stars" + "<br />" + str($show.imdb_info['votes']) + " Votes"
-				<span class="imdbstars" qtip-content="$rating_tip">$show.imdb_info['rating']</span>
+                <span class="imdbstars" qtip-content="$rating_tip">$show.imdb_info['rating']</span>
 #end if
-				
+                
 #set $_show = $show
 #if not $show.imdbid
-				<span>($show.startyear) - $show.runtime minutes - </span>
+                <span>($show.startyear) - $show.runtime minutes - </span>
 #else
     #if 'country_codes' in $show.imdb_info:
         #for $country in $show.imdb_info['country_codes'].split('|')
-				<img src="$sbRoot/images/blank.png" class="country-flag flag-${$country}" width="16" height="11" style="margin-left: 3px; vertical-align:middle;" />
+                <img src="$sbRoot/images/blank.png" class="country-flag flag-${$country}" width="16" height="11" style="margin-left: 3px; vertical-align:middle;" />
         #end for
     #end if
     #if 'year' in $show.imdb_info:
-				<span>($show.imdb_info['year']) - $show.imdb_info['runtimes'] minutes - </span>
+                <span>($show.imdb_info['year']) - $show.imdb_info['runtimes'] minutes - </span>
     #end if
-				<a href="<%= anon_url('http://www.imdb.com/title/', _show.imdbid) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://www.imdb.com/title/$show.imdbid"><img alt="[imdb]" height="16" width="16" src="$sbRoot/images/imdb.png" style="margin-top: -1px; vertical-align:middle;"/></a>
+                <a href="<%= anon_url('http://www.imdb.com/title/', _show.imdbid) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://www.imdb.com/title/$show.imdbid"><img alt="[imdb]" height="16" width="16" src="$sbRoot/images/imdb.png" style="margin-top: -1px; vertical-align:middle;"/></a>
 #end if
-				<a href="<%= anon_url(sickbeard.indexerApi(_show.indexer).config['show_url'], _show.indexerid) %>" onclick="window.open(this.href, '_blank'); return false;" title="$sickbeard.indexerApi($show.indexer).config["show_url"]$show.indexerid"><img alt="$sickbeard.indexerApi($show.indexer).name" height="16" width="16" src="$sbRoot/images/$sickbeard.indexerApi($show.indexer).config["icon"] "style="margin-top: -1px; vertical-align:middle;"/></a>
+                <a href="<%= anon_url(sickbeard.indexerApi(_show.indexer).config['show_url'], _show.indexerid) %>" onclick="window.open(this.href, '_blank'); return false;" title="$sickbeard.indexerApi($show.indexer).config["show_url"]$show.indexerid"><img alt="$sickbeard.indexerApi($show.indexer).name" height="16" width="16" src="$sbRoot/images/$sickbeard.indexerApi($show.indexer).config["icon"] "style="margin-top: -1px; vertical-align:middle;"/></a>
 #if $xem_numbering or $xem_absolute_numbering:
-				<a href="<%= anon_url('http://thexem.de/search?q=', _show.name) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://thexem.de/search?q-$show.name"><img alt="[xem]" height="16" width="16" src="$sbRoot/images/xem.png" style="margin-top: -1px; vertical-align:middle;"/></a>
+                <a href="<%= anon_url('http://thexem.de/search?q=', _show.name) %>" rel="noreferrer" onclick="window.open(this.href, '_blank'); return false;" title="http://thexem.de/search?q-$show.name"><img alt="[xem]" height="16" width="16" src="$sbRoot/images/xem.png" style="margin-top: -1px; vertical-align:middle;"/></a>
 #end if
-			</div>
-
-			<div id="tags">
-				<ul class="tags">
-					#if not $show.imdbid
-					#if $show.genre:
-					#for $genre in $show.genre[1:-1].split('|')
-						<a href="<%= anon_url('http://trakt.tv/shows/popular/', genre.lower()) %>" target="_blank" title="View other popular $genre shows on trakt.tv."><li>$genre</li></a>
-					#end for
-					#end if
-					#end if
-					#if 'year' in $show.imdb_info:
-					#for $imdbgenre in $show.imdb_info['genres'].replace('Sci-Fi','Science-Fiction').split('|')
-						<a href="<%= anon_url('http://trakt.tv/shows/popular/', imdbgenre.lower()) %>" target="_blank" title="View other popular $imdbgenre shows on trakt.tv."><li>$imdbgenre</li></a>
-					#end for
-					#end if
-				</ul>
-			</div>
-		
-			<div id="summary">
-				<table class="summaryTable pull-left">
-				#set $anyQualities, $bestQualities = $Quality.splitQuality(int($show.quality))
-					<tr><td class="showLegend">Quality: </td><td>
-				#if $show.quality in $qualityPresets:
-					<span class="quality $qualityPresetStrings[$show.quality]">$qualityPresetStrings[$show.quality]</span>
-				#else:
-				#if $anyQualities:
-					<i>Initial:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(anyQualities)])%> #if $bestQualities then " </br> " else ""#
-				#end if
-				#if $bestQualities:
-					<i>Replace with:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(bestQualities)])%>
-				#end if
-				#end if
-
-				#if $show.network and $show.airs:
-					<tr><td class="showLegend">Originally Airs: </td><td>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""# on $show.network</td></tr>
-				#else if $show.network:
-					<tr><td class="showLegend">Originally Airs: </td><td>$show.network</td></tr>
-				#else if $show.airs:
-					<tr><td class="showLegend">Originally Airs: </td><td>>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""#</td></tr>
-				#end if
-					<tr><td class="showLegend">Show Status: </td><td>$show.status</td></tr>
-					<tr><td class="showLegend">Default EP Status: </td><td>$statusStrings[$show.default_ep_status]</td></tr>
-				#if $showLoc[1]:
-					<tr><td class="showLegend">Location: </td><td>$showLoc[0]</td></tr>
-				#else:
-					<tr><td class="showLegend"><span style="color: red;">Location: </span></td><td><span style="color: red;">$showLoc[0]</span> (dir is missing)</td></tr>
-				#end if
-					<tr><td class="showLegend">Scene Name:</td><td>#if $show.exceptions then $exceptions_string else $show.name#</td></tr>
-					
-				#if $show.rls_require_words:
-					<tr><td class="showLegend">Required Words: </td><td>#echo $show.rls_require_words#</td></tr>
-				#end if
-				#if $show.rls_ignore_words:
-					<tr><td class="showLegend">Ignored Words: </td><td>#echo $show.rls_ignore_words#</td></tr>
-				#end if
-				#if $bwl and $bwl.get_white_keywords_for("release_group"):
-					<tr><td class="showLegend">Wanted Group#if len($bwl.get_white_keywords_for("release_group"))>1 then "s" else ""#:</td>
-					<td>#echo ', '.join($bwl.get_white_keywords_for("release_group"))#</td>
-					</tr>
-				#end if
-				#if $bwl and $bwl.get_black_keywords_for("release_group"):
-					<tr><td class="showLegend">Unwanted Group#if len($bwl.get_black_keywords_for("release_group"))>1 then "s" else ""#:</td>
-					<td>#echo ', '.join($bwl.get_black_keywords_for("release_group"))#</td>
-					</tr>
-				#end if
-
-				<tr><td class="showLegend">Size:</td><td>$sickbeard.helpers.pretty_filesize(sickbeard.helpers.get_size($showLoc[0]))</td></tr>
-
-				</table>
-			
-				<table style="width:180px; float: right; vertical-align: middle; height: 100%;">
-					<tr><td class="showLegend">Info Language:</td><td><img src="$sbRoot/images/flags/${show.lang}.png" width="16" height="11" alt="$show.lang" title="$show.lang" /></td></tr>
-					#if $sickbeard.USE_SUBTITLES
-					<tr><td class="showLegend">Subtitles: </td><td><img src="$sbRoot/images/#if int($show.subtitles) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					#end if
-					<tr><td class="showLegend">Flat Folders: </td><td><img src="$sbRoot/images/#if $show.flatten_folders == 1 or $sickbeard.NAMING_FORCE_FOLDERS then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+            </div>
+
+            <div id="tags">
+                <ul class="tags">
+                    #if not $show.imdbid
+                    #if $show.genre:
+                    #for $genre in $show.genre[1:-1].split('|')
+                        <a href="<%= anon_url('http://trakt.tv/shows/popular/', genre.lower()) %>" target="_blank" title="View other popular $genre shows on trakt.tv."><li>$genre</li></a>
+                    #end for
+                    #end if
+                    #end if
+                    #if 'year' in $show.imdb_info:
+                    #for $imdbgenre in $show.imdb_info['genres'].replace('Sci-Fi','Science-Fiction').split('|')
+                        <a href="<%= anon_url('http://trakt.tv/shows/popular/', imdbgenre.lower()) %>" target="_blank" title="View other popular $imdbgenre shows on trakt.tv."><li>$imdbgenre</li></a>
+                    #end for
+                    #end if
+                </ul>
+            </div>
+        
+            <div id="summary">
+                <table class="summaryTable pull-left">
+                #set $anyQualities, $bestQualities = $Quality.splitQuality(int($show.quality))
+                    <tr><td class="showLegend">Quality: </td><td>
+                #if $show.quality in $qualityPresets:
+                    <span class="quality $qualityPresetStrings[$show.quality]">$qualityPresetStrings[$show.quality]</span>
+                #else:
+                #if $anyQualities:
+                    <i>Initial:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(anyQualities)])%> #if $bestQualities then " </br> " else ""#
+                #end if
+                #if $bestQualities:
+                    <i>Replace with:</i> <%=", ".join([Quality.qualityStrings[x] for x in sorted(bestQualities)])%>
+                #end if
+                #end if
+
+                #if $show.network and $show.airs:
+                    <tr><td class="showLegend">Originally Airs: </td><td>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""# on $show.network</td></tr>
+                #else if $show.network:
+                    <tr><td class="showLegend">Originally Airs: </td><td>$show.network</td></tr>
+                #else if $show.airs:
+                    <tr><td class="showLegend">Originally Airs: </td><td>>$show.airs #if not $network_timezones.test_timeformat($show.airs) then " <font color='#FF0000'><b>(invalid Timeformat)</b></font> " else ""#</td></tr>
+                #end if
+                    <tr><td class="showLegend">Show Status: </td><td>$show.status</td></tr>
+                    <tr><td class="showLegend">Default EP Status: </td><td>$statusStrings[$show.default_ep_status]</td></tr>
+                #if $showLoc[1]:
+                    <tr><td class="showLegend">Location: </td><td>$showLoc[0]</td></tr>
+                #else:
+                    <tr><td class="showLegend"><span style="color: red;">Location: </span></td><td><span style="color: red;">$showLoc[0]</span> (dir is missing)</td></tr>
+                #end if
+                    <tr><td class="showLegend">Scene Name:</td><td>#if $show.exceptions then $exceptions_string else $show.name#</td></tr>
+                    
+                #if $show.rls_require_words:
+                    <tr><td class="showLegend">Required Words: </td><td>#echo $show.rls_require_words#</td></tr>
+                #end if
+                #if $show.rls_ignore_words:
+                    <tr><td class="showLegend">Ignored Words: </td><td>#echo $show.rls_ignore_words#</td></tr>
+                #end if
+                #if $bwl and $bwl.get_white_keywords_for("release_group"):
+                    <tr><td class="showLegend">Wanted Group#if len($bwl.get_white_keywords_for("release_group"))>1 then "s" else ""#:</td>
+                    <td>#echo ', '.join($bwl.get_white_keywords_for("release_group"))#</td>
+                    </tr>
+                #end if
+                #if $bwl and $bwl.get_black_keywords_for("release_group"):
+                    <tr><td class="showLegend">Unwanted Group#if len($bwl.get_black_keywords_for("release_group"))>1 then "s" else ""#:</td>
+                    <td>#echo ', '.join($bwl.get_black_keywords_for("release_group"))#</td>
+                    </tr>
+                #end if
+
+                <tr><td class="showLegend">Size:</td><td>$sickbeard.helpers.pretty_filesize(sickbeard.helpers.get_size($showLoc[0]))</td></tr>
+
+                </table>
+            
+                <table style="width:180px; float: right; vertical-align: middle; height: 100%;">
+                    <tr><td class="showLegend">Info Language:</td><td><img src="$sbRoot/images/flags/${show.lang}.png" width="16" height="11" alt="$show.lang" title="$show.lang" /></td></tr>
+                    #if $sickbeard.USE_SUBTITLES
+                    <tr><td class="showLegend">Subtitles: </td><td><img src="$sbRoot/images/#if int($show.subtitles) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    #end if
+                    <tr><td class="showLegend">Flat Folders: </td><td><img src="$sbRoot/images/#if $show.flatten_folders == 1 or $sickbeard.NAMING_FORCE_FOLDERS then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
 					<tr><td class="showLegend">Paused: </td><td><img src="$sbRoot/images/#if int($show.paused) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					<tr><td class="showLegend">Air-by-Date: </td><td><img src="$sbRoot/images/#if int($show.air_by_date) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					<tr><td class="showLegend">Sports: </td><td><img src="$sbRoot/images/#if int($show.is_sports) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					<tr><td class="showLegend">Anime: </td><td><img src="$sbRoot/images/#if int($show.is_anime) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					<tr><td class="showLegend">DVD Order: </td><td><img src="$sbRoot/images/#if int($show.dvdorder) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					<tr><td class="showLegend">Scene Numbering: </td><td><img src="$sbRoot/images/#if int($show.scene) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					#if $anyQualities + $bestQualities
-					<tr><td class="showLegend">Archive First Match: </td><td><img src="$sbRoot/images/#if int($show.archive_firstmatch) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
-					#end if
-				</table>
-			</div>
-		</div>
-	</div>
-		
-	<div class="clearfix"></div>
-
-	<div class="pull-left" >
-		Change selected episodes to:</br>
-		<select id="statusSelect" class="form-control form-control-inline input-sm">
-			#for $curStatus in [$WANTED, $SKIPPED, $ARCHIVED, $IGNORED, $FAILED] + sorted($Quality.DOWNLOADED):
-			#if $curStatus == $DOWNLOADED:
-			#continue
-			#end if
-			<option value="$curStatus">$statusStrings[$curStatus]</option>
-			#end for
-		</select>
-		<input type="hidden" id="showID" value="$show.indexerid" />
-		<input type="hidden" id="indexer" value="$show.indexer" />
-		<input class="btn btn-inline" type="button" id="changeStatus" value="Go" />
-	</div>
-
-	</br>
-
-	<div class="pull-right clearfix" id="checkboxControls">
-		<div style="padding-bottom: 5px;">
-			<label for="wanted"><span class="wanted"><input type="checkbox" id="wanted" checked="checked" /> Wanted: <b>$epCounts[$Overview.WANTED]</b></span></label>
-			<label for="qual"><span class="qual"><input type="checkbox" id="qual" checked="checked" /> Low Quality: <b>$epCounts[$Overview.QUAL]</b></span></label>
-			<label for="good"><span class="good"><input type="checkbox" id="good" checked="checked" /> Downloaded: <b>$epCounts[$Overview.GOOD]</b></span></label>
-			<label for="skipped"><span class="skipped"><input type="checkbox" id="skipped" checked="checked" /> Skipped: <b>$epCounts[$Overview.SKIPPED]</b></span></label>
-			<label for="snatched"><span class="snatched"><input type="checkbox" id="snatched" checked="checked" /> Snatched: <b>$epCounts[$Overview.SNATCHED]</b></span></label>
-		</div>
-		
-		<div class="pull-right" >
-			<button class="btn btn-xs seriesCheck">Select Filtered Episodes</button> 
-			<button class="btn btn-xs clearAll">Clear All</button>
-		</div>
-	</div>
-<br />	
-
-<table class="sickbeardTable display_show" cellspacing="0" border="0" cellpadding="0">
+                    <tr><td class="showLegend">Air-by-Date: </td><td><img src="$sbRoot/images/#if int($show.air_by_date) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    <tr><td class="showLegend">Sports: </td><td><img src="$sbRoot/images/#if int($show.is_sports) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    <tr><td class="showLegend">Anime: </td><td><img src="$sbRoot/images/#if int($show.is_anime) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    <tr><td class="showLegend">DVD Order: </td><td><img src="$sbRoot/images/#if int($show.dvdorder) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    <tr><td class="showLegend">Scene Numbering: </td><td><img src="$sbRoot/images/#if int($show.scene) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    #if $anyQualities + $bestQualities
+                    <tr><td class="showLegend">Archive First Match: </td><td><img src="$sbRoot/images/#if int($show.archive_firstmatch) == 1 then "yes16.png\" alt=\"Y" else "no16.png\" alt=\"N"#" width="16" height="16" /></td></tr>
+                    #end if
+                </table>
+            </div>
+        </div>
+    </div>
+        
+    <div class="clearfix"></div>
+
+    <div class="pull-left" >
+        Change selected episodes to:</br>
+        <select id="statusSelect" class="form-control form-control-inline input-sm">
+            #for $curStatus in [$WANTED, $SKIPPED, $ARCHIVED, $IGNORED, $FAILED] + sorted($Quality.DOWNLOADED):
+            #if $curStatus == $DOWNLOADED:
+            #continue
+            #end if
+            <option value="$curStatus">$statusStrings[$curStatus]</option>
+            #end for
+        </select>
+        <input type="hidden" id="showID" value="$show.indexerid" />
+        <input type="hidden" id="indexer" value="$show.indexer" />
+        <input class="btn btn-inline" type="button" id="changeStatus" value="Go" />
+    </div>
 
+    </br>
+
+    <div class="pull-right clearfix" id="checkboxControls">
+        <div style="padding-bottom: 5px;">
+            <label for="wanted"><span class="wanted"><input type="checkbox" id="wanted" checked="checked" /> Wanted: <b>$epCounts[$Overview.WANTED]</b></span></label>
+            <label for="qual"><span class="qual"><input type="checkbox" id="qual" checked="checked" /> Low Quality: <b>$epCounts[$Overview.QUAL]</b></span></label>
+            <label for="good"><span class="good"><input type="checkbox" id="good" checked="checked" /> Downloaded: <b>$epCounts[$Overview.GOOD]</b></span></label>
+            <label for="skipped"><span class="skipped"><input type="checkbox" id="skipped" checked="checked" /> Skipped: <b>$epCounts[$Overview.SKIPPED]</b></span></label>
+            <label for="snatched"><span class="snatched"><input type="checkbox" id="snatched" checked="checked" /> Snatched: <b>$epCounts[$Overview.SNATCHED]</b></span></label>
+        </div>
+        
+        <button id="popover" type="button" class="btn btn-xs">Select Columns</button>
+        <div class="pull-right" >
+            <button class="btn btn-xs seriesCheck">Select Filtered Episodes</button> 
+            <button class="btn btn-xs clearAll">Clear All</button>
+        </div>
+    </div>
+<br />    
+<br />
+<br />
+
+<table #if not $show.is_anime then "id=\"showTable\"" else "id=\"animeTable\""# class="displayShowTable display_show" cellspacing="0" border="0" cellpadding="0">
 #set $curSeason = -1
 #set $odd = 0
+    #for $epResult in $sqlResults:
+        #set $epStr = str($epResult["season"]) + "x" + str($epResult["episode"])
+        #if not $epStr in $epCats:
+            #continue
+        #end if
+        #if not $sickbeard.DISPLAY_SHOW_SPECIALS and int($epResult["season"]) == 0:
+            #continue
+        #end if    
+        #set $scene = False
+        #set $scene_anime = False
+        #if not $show.air_by_date and not $show.is_sports and not $show.is_anime and $show.is_scene:
+            #set $scene = True
+        #elif not $show.air_by_date and not $show.is_sports and $show.is_anime and $show.is_scene:
+            #set $scene_anime = True
+        #end if
+        #set ($dfltSeas, $dfltEpis, $dfltAbsolute) = (0, 0, 0)
+        #if (epResult["season"], epResult["episode"]) in $xem_numbering:
+            #set ($dfltSeas, $dfltEpis) = $xem_numbering[(epResult["season"], epResult["episode"])]
+        #end if
+        #if epResult["absolute_number"] in $xem_absolute_numbering:
+            #set $dfltAbsolute = $xem_absolute_numbering[epResult["absolute_number"]]
+        #end if
+
+        #if epResult["absolute_number"] in $scene_absolute_numbering:
+            #set $scAbsolute = $scene_absolute_numbering[epResult["absolute_number"]]
+            #set $dfltAbsNumbering = False
+        #else
+            #set $scAbsolute = $dfltAbsolute
+            #set $dfltAbsNumbering = True
+        #end if
 
-	#for $epResult in $sqlResults:
-		#set $epStr = str($epResult["season"]) + "x" + str($epResult["episode"])
-		#if not $epStr in $epCats:
-			#continue
-		#end if
-
-		#if not $sickbeard.DISPLAY_SHOW_SPECIALS and int($epResult["season"]) == 0:
-			#continue
-		#end if	
-
-		#set $scene = False
-		#set $scene_anime = False
-		#if not $show.air_by_date and not $show.is_sports and not $show.is_anime and $show.is_scene:
-			#set $scene = True
-		#elif not $show.air_by_date and not $show.is_sports and $show.is_anime and $show.is_scene:
-			#set $scene_anime = True
-		#end if
-
-		#set ($dfltSeas, $dfltEpis, $dfltAbsolute) = (0, 0, 0)
-
-		#if (epResult["season"], epResult["episode"]) in $xem_numbering:
-			#set ($dfltSeas, $dfltEpis) = $xem_numbering[(epResult["season"], epResult["episode"])]
-		#end if
-
-		#if epResult["absolute_number"] in $xem_absolute_numbering:
-			#set $dfltAbsolute = $xem_absolute_numbering[epResult["absolute_number"]]
-		#end if
-
-		#if epResult["absolute_number"] in $scene_absolute_numbering:
-			#set $scAbsolute = $scene_absolute_numbering[epResult["absolute_number"]]
-			#set $dfltAbsNumbering = False
-		#else
-			#set $scAbsolute = $dfltAbsolute
-			#set $dfltAbsNumbering = True
-		#end if
-
-		#if (epResult["season"], epResult["episode"]) in $scene_numbering:
-			#set ($scSeas, $scEpis) = $scene_numbering[(epResult["season"], epResult["episode"])]
-			#set $dfltEpNumbering = False
-		#else
-			#set ($scSeas, $scEpis) = ($dfltSeas, $dfltEpis)
-			#set $dfltEpNumbering = True
-		#end if
-
-		#if int($epResult["season"]) != $curSeason:
-		    <tr>
-				<th class="row-seasonheader" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th>
-			</tr>
-			
-			<tr id="season-$epResult["season"]-cols" class="seasoncols">
-				<th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th>
-				<th class="col-metadata">NFO</th>
-				<th class="col-metadata">TBN</th>
-				<th class="col-ep">Episode</th>
-			#if $show.is_anime:
-				<th class="col-ep">Absolute</th>
-			#end if
-			#if $scene:
-				<th class="col-ep">Scene</th>
-			#end if
-			#if $scene_anime:
-				<th class="col-ep">Scene Absolute</th>
-			#end if
-				<th class="col-name"
-			#if ($sickbeard.DISPLAY_FILESIZE == True):
-			style="min-width: 190px"
-			#end if
-			>Name</th>
-			#if ($sickbeard.DISPLAY_FILESIZE == True):
-				<th class="col-ep">Size</th>
-			#end if
-				<th class="col-airdate">Airdate</th>
-			#if $sickbeard.DOWNLOAD_URL
-				<th class="col-ep">Download</th>
-			#end if
-			#if $sickbeard.USE_SUBTITLES and $show.subtitles:
-				<th class="col-subtitles">Subtitles</th>
-			#end if
-				<th class="col-status">Status</th>
-				<th class="col-search">Search</th>
-			</tr>
-			#set $curSeason = int($epResult["season"])
-		#end if    
-
-		#set $epLoc = $epResult["location"]
-		
-		<tr class="$Overview.overviewStrings[$epCats[$epStr]] season-$curSeason seasonstyle">
-		
-		<td class="col-checkbox">
-		
-		#if int($epResult["status"]) != $UNAIRED
-			<input type="checkbox" class="epCheck" id="<%=str(epResult["season"])+'x'+str(epResult["episode"])%>" name="<%=str(epResult["season"]) +"x"+str(epResult["episode"]) %>" />
-		#end if
-		</td>
-		
-		<td align="center"><img src="$sbRoot/images/#if $epResult["hasnfo"] == 1 then "nfo.gif\" alt=\"Y" else "nfo-no.gif\" alt=\"N"#" width="23" height="11" /></td>
-		
-		<td align="center"><img src="$sbRoot/images/#if $epResult["hastbn"] == 1 then "tbn.gif\" alt=\"Y" else "tbn-no.gif\" alt=\"N"#" width="23" height="11" /></td>
-		
-		<td align="center">
-			#if $epLoc and $show._location and $epLoc.lower().startswith($show._location.lower()):
-				#set $epLoc = $epLoc[len($show._location)+1:]
-			#elif $epLoc and (not $epLoc.lower().startswith($show._location.lower()) or not $show._location):
-				#set $epLoc = $epLoc
-			#end if
-		
-			#if $epLoc != "" and $epLoc != None:
-				<span title="$epLoc" class="addQTip">$epResult["episode"]</span>
-			#else
-				$epResult["episode"]
-			#end if
-		</td>
-		
-		#if $show.is_anime:
-			<td align="center">$epResult["absolute_number"]</td>
-		#end if
-
-		#if $scene:
-			<td align="center">
-				<input type="text" placeholder="<%=str(dfltSeas) + 'x' + str(dfltEpis)%>" size="6" maxlength="8"
-					class="sceneSeasonXEpisode form-control input-scene" data-for-season="$epResult["season"]" data-for-episode="$epResult["episode"]"
-					id="sceneSeasonXEpisode_$show.indexerid<%="_"+str(epResult["season"])+"_"+str(epResult["episode"])%>"
-					title="Change the value here if scene numbering differs from the indexer episode numbering"
-					#if $dfltEpNumbering:
-						value=""
-					#else
-						value="<%=str(scSeas) + 'x' + str(scEpis)%>"
-					#end if
-						style="padding: 0; text-align: center; max-width: 60px;" />
-			</td>
-		#elif $scene_anime:
-			<td align="center">
-				<input type="text" placeholder="<%=str(dfltAbsolute)%>" size="6" maxlength="8"
-					class="sceneAbsolute form-control input-scene" data-for-absolute="$epResult["absolute_number"]"
-					id="sceneAbsolute_$show.indexerid<%="_"+str(epResult["absolute_number"])%>"
-					title="Change the value here if scene absolute numbering differs from the indexer absolute numbering"
-					#if $dfltAbsNumbering:
-						value=""
-					#else
-						value="<%=str(scAbsolute)%>"
-					#end if
-						style="padding: 0; text-align: center; max-width: 60px;" />
-			</td>
-		#end if
-
-		<td class="col-name">
-			#if $epResult["description"] != "" and $epResult["description"] != None:
-				<img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfo" alt="" id="plot_info_$show.indexerid<%="_" + str(epResult["season"]) + "_" + str(epResult["episode"])%>" />
-			#else:
-				<img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfoNone" alt="" />
-			#end if
-			$epResult["name"]
-		</td>
-		
-		#if ($sickbeard.DISPLAY_FILESIZE == True):
-		<td class="col-ep">
-			#if $epResult["file_size"]:
-				#set $file_size = $sickbeard.helpers.pretty_filesize($epResult["file_size"])
-				$file_size
-			#end if
-		</td>
-		#end if
-		<td class="col-airdate">
-			<span class="${fuzzydate}">#if int($epResult['airdate']) == 1 then 'never' else $sbdatetime.sbdatetime.sbfdate($sbdatetime.sbdatetime.convert_to_setting($network_timezones.parse_date_time($epResult['airdate'],$show.airs,$show.network)))#</span>
-		</td>
-
-		#if $sickbeard.DOWNLOAD_URL and $epResult['location']
-		<td>
-			#set $filename = $epResult['location']
-		  	#for $rootDir in $sickbeard.ROOT_DIRS.split('|')
-		   		#if $rootDir.startswith('/')
-		   			#set $filename = $filename.replace($rootDir, "")
-		   		#end if
-		   	#end for
-			#set $filename = $sickbeard.DOWNLOAD_URL + $urllib.quote($filename.encode('utf8'))
-			<center><a href="$filename">Download</a></center>
-		</td>
-		#elif $sickbeard.DOWNLOAD_URL
-			<td></td>
-		#end if
-		
-		#if $sickbeard.USE_SUBTITLES and $show.subtitles:
-			<td class="col-subtitles" align="center">
-        #if $epResult["subtitles"]:
-            #for $sub_lang in subliminal.language.language_list([x.strip() for x in $epResult["subtitles"].split(',') if x != ""]):
-                #if sub_lang.alpha2 != ""
-                    <img src="$sbRoot/images/flags/${sub_lang.alpha2}.png" width="16" height="11" alt="${sub_lang}" />
+        #if (epResult["season"], epResult["episode"]) in $scene_numbering:
+            #set ($scSeas, $scEpis) = $scene_numbering[(epResult["season"], epResult["episode"])]
+            #set $dfltEpNumbering = False
+        #else
+            #set ($scSeas, $scEpis) = ($dfltSeas, $dfltEpis)
+            #set $dfltEpNumbering = True
+        #end if
+
+        #set $epLoc = $epResult["location"]       
+
+        #if int($epResult["season"]) != $curSeason:
+            #if $curSeason == -1:
+    <thead>
+        <tr class="seasoncols" style="display:none;">
+                <th data-sorter="false" data-priority="critical" class="col-checkbox"><input type="checkbox" class="seasonCheck"/></th>
+                <th data-sorter="false" class="col-metadata">NFO</th>
+                <th data-sorter="false" class="col-metadata">TBN</th>
+                <th data-sorter="false" class="col-ep">Episode</th>
+                <th data-sorter="false" #if not $show.is_anime then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Absolute</th>
+                <th data-sorter="false" #if not $scene then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Scene</th>
+                <th data-sorter="false" #if not $scene_anime then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Scene Absolute</th>
+                <th data-sorter="false" class="col-name">Name</th>
+                <th data-sorter="false" class="col-name columnSelector-false">File Name</th>
+                <th data-sorter="false" class="col-ep columnSelector-false">Size</th>
+                <th data-sorter="false" class="col-airdate">Airdate</th>
+                <th data-sorter="false" #if not $sickbeard.DOWNLOAD_URL then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Download</th>
+                <th data-sorter="false" #if not $sickbeard.USE_SUBTITLES then "class=\"col-ep columnSelector-false\"" else "class=\"col-ep\""#>Subtitles</th>
+                <th data-sorter="false" class="col-status">Status</th>
+                <th data-sorter="false" class="col-search">Search</th>        
+        </tr>
+    </thead>            
+    <tbody class="tablesorter-no-sort">      
+        <tr>
+            <th class="row-seasonheader displayShowTable" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th>
+        </tr>            
+        <tr id="season-$epResult["season"]-cols" class="seasoncols">
+            <th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th>
+            <th class="col-metadata">NFO</th>
+            <th class="col-metadata">TBN</th>
+            <th class="col-ep">Episode</th>
+            <th class="col-ep">Absolute</th>
+            <th class="col-ep">Scene</th>
+            <th class="col-ep">Scene Absolute</th>
+            <th class="col-name">Name</th>
+            <th class="col-name">File Name</th>
+            <th class="col-ep">Size</th>
+            <th class="col-airdate">Airdate</th>
+            <th class="col-ep">Download</th>
+            <th class="col-ep">Subtitles</th>
+            <th class="col-status">Status</th>
+            <th class="col-search">Search</th>        
+        </tr>
+            #else:
+    </tbody>
+    <tbody class="tablesorter-no-sort">       
+        <tr>
+            <th class="row-seasonheader displayShowTable" colspan="13" style="width: auto;"><h3><a name="season-$epResult["season"]"></a>#if int($epResult["season"]) == 0 then "Specials" else "Season " + str($epResult["season"])#</h3></th>
+        </tr>            
+        <tr id="season-$epResult["season"]-cols" class="seasoncols">
+            <th class="col-checkbox"><input type="checkbox" class="seasonCheck" id="$epResult["season"]" /></th>
+            <th class="col-metadata">NFO</th>
+            <th class="col-metadata">TBN</th>
+            <th class="col-ep">Episode</th>
+            <th class="col-ep">Absolute</th>
+            <th class="col-ep">Scene</th>
+            <th class="col-ep">Scene Absolute</th>
+            <th class="col-name">Name</th>
+            <th class="col-name">File Name</th>
+            <th class="col-ep">Size</th>
+            <th class="col-airdate">Airdate</th>
+            <th class="col-ep">Download</th>
+            <th class="col-ep">Subtitles</th>
+            <th class="col-status">Status</th>
+            <th class="col-search">Search</th> 
+        </tr>                    
+            #end if
+    </tbody>
+    <tbody>
+            #set $curSeason = int($epResult["season"])
+        #end if
+        #set $epLoc = $epResult["location"]            
+        <tr class="$Overview.overviewStrings[$epCats[$epStr]] season-$curSeason seasonstyle">
+            <td class="col-checkbox">    
+                #if int($epResult["status"]) != $UNAIRED
+                    <input type="checkbox" class="epCheck" id="<%=str(epResult["season"])+'x'+str(epResult["episode"])%>" name="<%=str(epResult["season"]) +"x"+str(epResult["episode"]) %>" />
+                #end if
+            </td>    
+            <td align="center"><img src="$sbRoot/images/#if $epResult["hasnfo"] == 1 then "nfo.gif\" alt=\"Y" else "nfo-no.gif\" alt=\"N"#" width="23" height="11" /></td>    
+            <td align="center"><img src="$sbRoot/images/#if $epResult["hastbn"] == 1 then "tbn.gif\" alt=\"Y" else "tbn-no.gif\" alt=\"N"#" width="23" height="11" /></td>
+            <td align="center">
+                #if $epLoc and $show._location and $epLoc.lower().startswith($show._location.lower()):
+                    #set $epLoc = $epLoc[len($show._location)+1:]
+                #elif $epLoc and (not $epLoc.lower().startswith($show._location.lower()) or not $show._location):
+                    #set $epLoc = $epLoc
+                #end if
+            
+                #if $epLoc != "" and $epLoc != None:
+                    <span title="$epLoc" class="addQTip">$epResult["episode"]</span>
                 #else
-                    <img src="$sbRoot/images/flags/unknown.png" width="16" height="11" alt="Unknown" />
+                    $epResult["episode"]
                 #end if
-            #end for
-        #end if
-			</td>
-		#end if
-		
-		#set $curStatus, $curQuality = $Quality.splitCompositeStatus(int($epResult["status"]))
-			#if $curQuality != Quality.NONE:   
-				<td class="col-status">$statusStrings[$curStatus] <span class="quality $Quality.qualityStrings[$curQuality].replace("720p","HD720p").replace("1080p","HD1080p").replace("RawHD TV", "RawHD").replace("HD TV", "HD720p")">$Quality.qualityStrings[$curQuality]</span></td>
-			#else:    
-				<td class="col-status">$statusStrings[$curStatus]</td>
-			#end if
-		
-		<td class="col-search">
-			#if int($epResult["season"]) != 0:
-			#if ( int($epResult["status"]) in $Quality.SNATCHED or int($epResult["status"]) in $Quality.DOWNLOADED ) and $sickbeard.USE_FAILED_DOWNLOADS:
-				<a class="epRetry" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="retryEpisode?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" height="16" alt="retry" title="Retry Download" /></a>
-			#else:
-				<a class="epSearch" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="searchEpisode?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" width="16" height="16" alt="search" title="Manual Search" /></a>
-			#end if
-			#end if
-			
-			#if $sickbeard.USE_SUBTITLES and $show.subtitles and len(set(str($epResult["subtitles"]).split(',')).intersection(set($subtitles.wantedLanguages()))) < len($subtitles.wantedLanguages()) and $epResult["location"]
-				<a class="epSubtitlesSearch" href="searchEpisodeSubtitles?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/closed_captioning.png" height="16" alt="search subtitles" title="Search Subtitles" /></a>
-			#end if
-		</td>
-	</tr>
-
-	#end for
-	
+            </td>
+            <td align="center">$epResult["absolute_number"]</td>
+            <td align="center">
+                <input type="text" placeholder="<%=str(dfltSeas) + 'x' + str(dfltEpis)%>" size="6" maxlength="8"
+                    class="sceneSeasonXEpisode form-control input-scene" data-for-season="$epResult["season"]" data-for-episode="$epResult["episode"]"
+                    id="sceneSeasonXEpisode_$show.indexerid<%="_"+str(epResult["season"])+"_"+str(epResult["episode"])%>"
+                    title="Change the value here if scene numbering differs from the indexer episode numbering"
+                    #if $dfltEpNumbering:
+                        value=""
+                    #else
+                        value="<%=str(scSeas) + 'x' + str(scEpis)%>"
+                    #end if
+                        style="padding: 0; text-align: center; max-width: 60px;" />
+            </td>
+            <td align="center">
+                <input type="text" placeholder="<%=str(dfltAbsolute)%>" size="6" maxlength="8"
+                    class="sceneAbsolute form-control input-scene" data-for-absolute="$epResult["absolute_number"]"
+                    id="sceneAbsolute_$show.indexerid<%="_"+str(epResult["absolute_number"])%>"
+                    title="Change the value here if scene absolute numbering differs from the indexer absolute numbering"
+                    #if $dfltAbsNumbering:
+                        value=""
+                    #else
+                        value="<%=str(scAbsolute)%>"
+                    #end if
+                        style="padding: 0; text-align: center; max-width: 60px;" />
+            </td>
+            <td class="col-name">
+            #if $epResult["description"] != "" and $epResult["description"] != None:
+                <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfo" alt="" id="plot_info_$show.indexerid<%="_" + str(epResult["season"]) + "_" + str(epResult["episode"])%>" />
+            #else:
+                <img src="$sbRoot/images/info32.png" width="16" height="16" class="plotInfoNone" alt="" />
+            #end if
+            $epResult["name"]
+            </td>
+            <td class="col-name]">
+                #if $epResult['location']
+                    #set $filename = $epResult['location']
+                    #for $rootDir in $sickbeard.ROOT_DIRS.split('|')
+                        #if $rootDir.startswith('/')
+                            #set $filename = ntpath.basename($filename)
+                        #end if
+                    #end for
+                    $filename
+                #end if    
+            </td>
+            <td class="col-ep">
+                #if $epResult["file_size"]:
+                    #set $file_size = $sickbeard.helpers.pretty_filesize($epResult["file_size"])
+                    $file_size
+                #end if
+            </td>
+            <td class="col-airdate">
+                <span class="${fuzzydate}">#if int($epResult['airdate']) == 1 then 'never' else $sbdatetime.sbdatetime.sbfdate($sbdatetime.sbdatetime.convert_to_setting($network_timezones.parse_date_time($epResult['airdate'],$show.airs,$show.network)))#</span>
+            </td>        
+            <td>
+                #if $sickbeard.DOWNLOAD_URL and $epResult['location']
+                    #if $epResult['location']
+                        #set $filename = $epResult['location']
+                        #for $rootDir in $sickbeard.ROOT_DIRS.split('|')
+                            #if $rootDir.startswith('/')
+                                #set $filename = $filename.replace($rootDir, "")
+                            #end if
+                        #end for
+                        #set $filename = $sickbeard.DOWNLOAD_URL + $urllib.quote($filename.encode('utf8'))
+                <center><a href="$filename">Download</a></center>
+                    #end if
+                #end if
+            </td>
+            <td class="col-subtitles" align="center">
+            #if $epResult["subtitles"]:
+                #for $sub_lang in subliminal.language.language_list([x.strip() for x in $epResult["subtitles"].split(',') if x != ""]):
+                    #if sub_lang.alpha2 != ""
+                        <img src="$sbRoot/images/flags/${sub_lang.alpha2}.png" width="16" height="11" alt="${sub_lang}" />
+                    #else
+                        <img src="$sbRoot/images/flags/unknown.png" width="16" height="11" alt="Unknown" />
+                    #end if
+                #end for
+            #end if
+            </td>
+            #set $curStatus, $curQuality = $Quality.splitCompositeStatus(int($epResult["status"]))
+                #if $curQuality != Quality.NONE:   
+                    <td class="col-status">$statusStrings[$curStatus] <span class="quality $Quality.qualityStrings[$curQuality].replace("720p","HD720p").replace("1080p","HD1080p").replace("RawHD TV", "RawHD").replace("HD TV", "HD720p")">$Quality.qualityStrings[$curQuality]</span></td>
+                #else:    
+                    <td class="col-status">$statusStrings[$curStatus]</td>
+                #end if
+            <td class="col-search">
+                #if int($epResult["season"]) != 0:
+                    #if ( int($epResult["status"]) in $Quality.SNATCHED or int($epResult["status"]) in $Quality.DOWNLOADED ) and $sickbeard.USE_FAILED_DOWNLOADS:
+                        <a class="epRetry" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="retryEpisode?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" height="16" alt="retry" title="Retry Download" /></a>
+                    #else:
+                        <a class="epSearch" id="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" name="#echo $str($show.indexerid)+'x'+$str(epResult["season"])+'x'+$str(epResult["episode"])#" href="searchEpisode?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/search16.png" width="16" height="16" alt="search" title="Manual Search" /></a>
+                    #end if
+                #end if
+                #if $sickbeard.USE_SUBTITLES and $show.subtitles and len(set(str($epResult["subtitles"]).split(',')).intersection(set($subtitles.wantedLanguages()))) < len($subtitles.wantedLanguages()) and $epResult["location"]
+                    <a class="epSubtitlesSearch" href="searchEpisodeSubtitles?show=$show.indexerid&amp;season=$epResult["season"]&amp;episode=$epResult["episode"]"><img src="$sbRoot/images/closed_captioning.png" height="16" alt="search subtitles" title="Search Subtitles" /></a>
+                #end if
+            </td>
+        </tr>      
+    #end for
+    </tbody>
 </table>
-
+    
 <!--Begin - Bootstrap Modal-->
 
 <div id="manualSearchModalFailed" class="modal fade">
diff --git a/gui/slick/interfaces/default/editShow.tmpl b/gui/slick/interfaces/default/editShow.tmpl
index aafd072732e54d27d7e9909c90c3b68af3dd2403..ff40985259a683ceebc192deca44800e4d3e1b2d 100644
--- a/gui/slick/interfaces/default/editShow.tmpl
+++ b/gui/slick/interfaces/default/editShow.tmpl
@@ -78,7 +78,7 @@
 
 <b>Info Language:</b><br />
 (this will only affect the language of the retrieved metadata file contents and episode filenames)<br />
-<select name="indexerLang" id="indexerLangSelect" class="form-control form-control-inline input-sm bfh-languages" data-language="en" data-available="#echo ','.join($sickbeard.indexerApi().config['valid_languages'])#"></select><br />
+<select name="indexerLang" id="indexerLangSelect" class="form-control form-control-inline input-sm bfh-languages" data-language="#echo $sickbeard.INDEXER_DEFAULT_LANGUAGE#" data-available="#echo ','.join($sickbeard.indexerApi().config['valid_languages'])#"></select><br />
 <br />
 
 <b>Flatten files (no folders): </b> <input type="checkbox" name="flatten_folders" #if $show.flatten_folders == 1 and not $sickbeard.NAMING_FORCE_FOLDERS then "checked=\"checked\"" else ""# #if $sickbeard.NAMING_FORCE_FOLDERS then "disabled=\"disabled\"" else ""#/><br />
diff --git a/gui/slick/interfaces/default/inc_top.tmpl b/gui/slick/interfaces/default/inc_top.tmpl
index 2bc4e16b18687a751426fdf66885b47f573b24da..f46ad2d63d7d2eb66fd1eacb81262034c3cb4cef 100644
--- a/gui/slick/interfaces/default/inc_top.tmpl
+++ b/gui/slick/interfaces/default/inc_top.tmpl
@@ -119,6 +119,9 @@
 				\$("#SubMenu a:contains('Notification')").addClass('btn').html('<span class="ui-icon ui-icon-note pull-left"></span> Notifications');
 				\$("#SubMenu a:contains('Update show in KODI')").addClass('btn').html('<span class="submenu-icon-kodi pull-left"></span> Update show in KODI');
 				\$("#SubMenu a[href$='/home/updateKODI/']").addClass('btn').html('<span class="submenu-icon-kodi pull-left"></span> Update KODI');
+                \$("#SubMenu a:contains('Pause')").addClass('btn').html('<span class="ui-icon ui-icon-pause pull-left"></span> Pause');
+                \$("#SubMenu a:contains('Resume')").addClass('btn').html('<span class="ui-icon ui-icon-play pull-left"></span> Resume');                
+                
 			}
 		
 			\$(document).ready(function() {
diff --git a/gui/slick/js/ajaxNotifications.js b/gui/slick/js/ajaxNotifications.js
index 92b2fc22b853b4d2c880e66d7743424e30d410af..f1b38d7cdfab4198322c67ad140574a0332e5ab2 100644
--- a/gui/slick/js/ajaxNotifications.js
+++ b/gui/slick/js/ajaxNotifications.js
@@ -1,31 +1,42 @@
-var message_url = sbRoot + '/ui/get_messages';
+var message_url = sbRoot + '/ui/get_messages',
+	test = !1;
 
+PNotify.prototype.options.addclass = 'stack-bottomright';
+PNotify.prototype.options.buttons.closer_hover = !1;
+PNotify.prototype.options.delay = 5000;
+PNotify.prototype.options.desktop = {desktop: !0, icon: ''};
+PNotify.prototype.options.hide = !0;
+PNotify.prototype.options.history = !1;
+PNotify.prototype.options.shadow = !1;
+PNotify.prototype.options.stack = {dir1: 'up', dir2: 'left', firstpos1: 25, firstpos2: 25};
 PNotify.prototype.options.styling = 'jqueryui';
-PNotify.prototype.options.buttons.closer_hover = false;
-PNotify.prototype.options.delay = 4000;
 PNotify.prototype.options.width = '340px';
-PNotify.prototype.options.shadow = false;
-PNotify.prototype.options.addclass = 'stack-bottomright';
-PNotify.prototype.options.stack = {'dir1': 'up', 'dir2': 'left', 'firstpos1': 25, 'firstpos2': 25};
+PNotify.desktop.permission();
+
+function displayPNotify(type, title, message) {
+	var notification = new PNotify({
+		type: type, title: title,
+		text: message.replace(/<br[\s\/]*(?:\s[^>]*)?>/ig, "\n")
+			.replace(/<[\/]?b(?:\s[^>]*)?>/ig, '*')
+			.replace(/<i(?:\s[^>]*)?>/ig, '[').replace(/<[\/]i>/ig, ']')
+			.replace(/<(?:[\/]?ul|\/li)(?:\s[^>]*)?>/ig, '').replace(/<li(?:\s[^>]*)?>/ig, "\n" + '* ')
+	});
+}
 
 function check_notifications() {
-    if(document.visibilityState == 'visible') {
-        $.getJSON(message_url, function(data){
-            $.each(data, function(name,data){
-                new PNotify({
-                    type: data.type,
-                    hide: data.type == 'notice',
-                    title: data.title,
-                    text: data.message,
-                    history: false
-                });
-            });
-        });
-    }
-    
-    setTimeout(check_notifications, 3000)
+	if ('visible' == document.visibilityState) {
+		$.getJSON(message_url, function (data) {
+			$.each(data, function (name, data) {
+				displayPNotify(data.type, data.title, data.message)
+			});
+		});
+	}
+	setTimeout(check_notifications, 3000)
 }
 
 $(document).ready(function(){
-    check_notifications();
+	check_notifications();
+	if (test) {
+		displayPNotify('notice', 'test', 'test<br/><i class="test-class">hello <b>world</b></i><ul><li>item 1</li><li>item 2</li></ul>');
+	}
 });
\ No newline at end of file
diff --git a/gui/slick/js/configNotifications.js b/gui/slick/js/configNotifications.js
index 2d88287e9bd77a970b7a28648c6e81ef214ac399..f0b52ea97cb035e6755c8f8a574122d57dc43961 100644
--- a/gui/slick/js/configNotifications.js
+++ b/gui/slick/js/configNotifications.js
@@ -56,24 +56,44 @@ $(document).ready(function(){
             });
     });
 
-    $('#testPLEX').click(function () {
-        var plex_host = $.trim($('#plex_host').val());
-        var plex_username = $.trim($('#plex_username').val());
-        var plex_password = $.trim($('#plex_password').val());
-        if (!plex_host) {
-            $('#testPLEX-result').html('Please fill out the necessary fields above.');
+	$('#testPMC').click(function () {
+		var plex_host = $.trim($('#plex_host').val());
+		var plex_username = $.trim($('#plex_username').val());
+		var plex_password = $.trim($('#plex_password').val());
+		if (!plex_host) {
+			$('#testPMC-result').html('Please fill out the necessary fields above.');
 			$('#plex_host').addClass('warning');
-            return;
-        }
-        $('#plex_host').removeClass('warning');
+			return;
+		}
+		$('#plex_host').removeClass('warning');
 		$(this).prop('disabled', true);
-        $('#testPLEX-result').html(loading);
-        $.get(sbRoot + '/home/testPLEX', {'host': plex_host, 'username': plex_username, 'password': plex_password})
-            .done(function (data) {
-                $('#testPLEX-result').html(data);
-                $('#testPLEX').prop('disabled', false);
-            });
-    });
+		$('#testPMC-result').html(loading);
+		$.get(sbRoot + '/home/testPMC', {'host': plex_host, 'username': plex_username, 'password': plex_password})
+			.done(function (data) {
+				$('#testPMC-result').html(data);
+				$('#testPMC').prop('disabled', false);
+			});
+	});
+
+	$('#testPMS').click(function () {
+		var plex_server_host = $.trim($('#plex_server_host').val());
+		var plex_username = $.trim($('#plex_username').val());
+		var plex_password = $.trim($('#plex_password').val());
+        var plex_server_token = $.trim($('#plex_server_token').val());
+		if (!plex_server_host) {
+			$('#testPMS-result').html('Please fill out the necessary fields above.');
+			$('#plex_server_host').addClass('warning');
+			return;
+		}
+		$('#plex_server_host').removeClass('warning');
+		$(this).prop('disabled', true);
+		$('#testPMS-result').html(loading);
+		$.get(sbRoot + '/home/testPMS', {'host': plex_server_host, 'username': plex_username, 'password': plex_password, 'plex_server_token': plex_server_token})
+			.done(function (data) {
+				$('#testPMS-result').html(data);
+				$('#testPMS').prop('disabled', false);
+			});
+	});
 
     $('#testBoxcar').click(function() {
 		var boxcar_username = $.trim($('#boxcar_username').val());
diff --git a/gui/slick/js/configSearch.js b/gui/slick/js/configSearch.js
index f586a488c832b21043c42adbc7921a3f95083776..1840bd68d60d86c72343173fdca9a51afde7f8bb 100644
--- a/gui/slick/js/configSearch.js
+++ b/gui/slick/js/configSearch.js
@@ -150,6 +150,12 @@ $(document).ready(function(){
                 $(torrent_verify_rtorrent).show();
                 $(torrent_auth_type_option).show();
                 //$('#directory_title').text(client + directory);
+            } else if ('qbittorrent' == selectedProvider){
+                client = 'qbittorrent';
+                $(torrent_path_option).hide();
+                $(torrent_label_option).hide();
+                $(torrent_label_anime_option).hide();
+                $('#host_desc_torrent').text('URL to your qbittorrent client (e.g. http://localhost:8080)');
             }
             $('#host_title').text(client + host);
             $('#username_title').text(client + username);
diff --git a/gui/slick/js/displayShow.js b/gui/slick/js/displayShow.js
index 4825c2cbcc45420f9fd67a2aa8199a504fd0c04f..fb13af78ae2b75e31d81470b27313683f018dd9f 100644
--- a/gui/slick/js/displayShow.js
+++ b/gui/slick/js/displayShow.js
@@ -259,5 +259,5 @@ $(document).ready(function () {
             sceneAbsolute = m[1];
         }
         setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute);
-    });
+    });   
 });
diff --git a/gui/slick/js/lib/pnotify.custom.min.js b/gui/slick/js/lib/pnotify.custom.min.js
index 704c85ae41ad58fb3d4bd2142a226ec6683d8b94..f927d92712aa970d3baeabde56c84448bb888a33 100644
Binary files a/gui/slick/js/lib/pnotify.custom.min.js and b/gui/slick/js/lib/pnotify.custom.min.js differ
diff --git a/sickbeard/__init__.py b/sickbeard/__init__.py
index 2a9b9e0900c2db4036ad9e7cc7d570173352e8a3..f20e8ab68c229664d0557e75c5ddc01f0e539ef9 100755
--- a/sickbeard/__init__.py
+++ b/sickbeard/__init__.py
@@ -38,7 +38,7 @@ from sickbeard import providers, metadata, config, webserveInit
 from sickbeard.providers.generic import GenericProvider
 from providers import ezrss, btn, newznab, womble, thepiratebay, oldpiratebay, torrentleech, kat, iptorrents, \
     omgwtfnzbs, scc, hdtorrents, torrentday, hdbits, hounddawgs, nextgen, speedcd, nyaatorrents, animenzb, torrentbytes, animezb, \
-    freshontv, morethantv, bitsoup, t411, tokyotoshokan, shazbat, rarbg, alpharatio, tntvillage, binsearch, eztv
+    freshontv, morethantv, bitsoup, t411, tokyotoshokan, shazbat, rarbg, alpharatio, tntvillage, binsearch, eztv, scenetime
 from sickbeard.config import CheckSection, check_setting_int, check_setting_str, check_setting_float, ConfigMigrator, \
     naming_ep_type
 from sickbeard import searchBacklog, showUpdater, versionChecker, properFinder, autoPostProcesser, \
@@ -482,7 +482,6 @@ COMING_EPS_LAYOUT = None
 COMING_EPS_DISPLAY_PAUSED = False
 COMING_EPS_SORT = None
 COMING_EPS_MISSED_RANGE = None
-DISPLAY_FILESIZE = False
 FUZZY_DATING = False
 TRIM_ZERO = False
 DATE_PRESET = None
@@ -802,7 +801,7 @@ def initialize(consoleLogging=True):
             NZB_METHOD = 'blackhole'
 
         TORRENT_METHOD = check_setting_str(CFG, 'General', 'torrent_method', 'blackhole')
-        if TORRENT_METHOD not in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent'):
+        if TORRENT_METHOD not in ('blackhole', 'utorrent', 'transmission', 'deluge', 'download_station', 'rtorrent', 'qbittorrent'):
             TORRENT_METHOD = 'blackhole'
 
         DOWNLOAD_PROPERS = bool(check_setting_int(CFG, 'General', 'download_propers', 1))
@@ -1123,7 +1122,6 @@ def initialize(consoleLogging=True):
         COMING_EPS_DISPLAY_PAUSED = bool(check_setting_int(CFG, 'GUI', 'coming_eps_display_paused', 0))
         COMING_EPS_SORT = check_setting_str(CFG, 'GUI', 'coming_eps_sort', 'date')
         COMING_EPS_MISSED_RANGE = check_setting_int(CFG, 'GUI', 'coming_eps_missed_range', 7)
-        DISPLAY_FILESIZE = bool(check_setting_int(CFG, 'GUI', 'display_filesize', 0))
         FUZZY_DATING = bool(check_setting_int(CFG, 'GUI', 'fuzzy_dating', 0))
         TRIM_ZERO = bool(check_setting_int(CFG, 'GUI', 'trim_zero', 0))
         DATE_PRESET = check_setting_str(CFG, 'GUI', 'date_preset', '%x')
@@ -2040,7 +2038,6 @@ def save_config():
     new_config['GUI']['coming_eps_display_paused'] = int(COMING_EPS_DISPLAY_PAUSED)
     new_config['GUI']['coming_eps_sort'] = COMING_EPS_SORT
     new_config['GUI']['coming_eps_missed_range'] = int(COMING_EPS_MISSED_RANGE)
-    new_config['GUI']['display_filesize'] = int(DISPLAY_FILESIZE)
     new_config['GUI']['fuzzy_dating'] = int(FUZZY_DATING)
     new_config['GUI']['trim_zero'] = int(TRIM_ZERO)
     new_config['GUI']['date_preset'] = DATE_PRESET
diff --git a/sickbeard/classes.py b/sickbeard/classes.py
index 92bbc23e11cb067beee83b1757eaf503fa10c8f1..ec8fce996f5191dcae3f9ae1936893d66a37dcf8 100644
--- a/sickbeard/classes.py
+++ b/sickbeard/classes.py
@@ -276,6 +276,6 @@ class UIError():
     """
 
     def __init__(self, message):
-        self.title = sys.exc_info()[-2]
+        self.title = sys.exc_info()[-2] or message
         self.message = message
         self.time = datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
\ No newline at end of file
diff --git a/sickbeard/clients/__init__.py b/sickbeard/clients/__init__.py
index 84646ac32955d2beb27b92ecf9ab13d609be43fc..432efc797127ae729371bfd42f03f4b63d00f4f8 100644
--- a/sickbeard/clients/__init__.py
+++ b/sickbeard/clients/__init__.py
@@ -20,7 +20,8 @@ __all__ = ['utorrent',
            'transmission',
            'deluge',
            'download_station',
-           'rtorrent'
+           'rtorrent',
+           'qbittorrent'
 ]
 
 import sickbeard
@@ -29,13 +30,28 @@ from os import sys
 
 # Mapping error status codes to official W3C names
 http_error_code = {
+    100: 'Continue',
+    101: 'Switching Protocols',
+    102: 'Processing',
+    200: 'OK',
+    201: 'Created',
+    202: 'Accepted',
+    203: 'Non-Authoritative Information',
+    204: 'No Content',
+    205: 'Reset Content',
+    206: 'Partial Content',
+    207: 'Multi-Status',
+    208: 'Already Reported',
+    226: 'IM Used',
     300: 'Multiple Choices',
     301: 'Moved Permanently',
     302: 'Found',
     303: 'See Other',
     304: 'Not Modified',
     305: 'Use Proxy',
+    306: 'Switch Proxy',
     307: 'Temporary Redirect',
+    308: 'Permanent Redirect',
     400: 'Bad Request',
     401: 'Unauthorized',
     402: 'Payment Required',
@@ -54,13 +70,45 @@ http_error_code = {
     415: 'Unsupported Media Type',
     416: 'Requested Range Not Satisfiable',
     417: 'Expectation Failed',
+    418: 'Im a teapot',
+    419: 'Authentication Timeout',
+    420: 'Enhance Your Calm',
+    422: 'Unprocessable Entity',
+    423: 'Locked',
+    424: 'Failed Dependency',
+    426: 'Upgrade Required',
+    428: 'Precondition Required',
+    429: 'Too Many Requests',
+    431: 'Request Header Fields Too Large',
+    440: 'Login Timeout',
+    444: 'No Response',
+    449: 'Retry With',
+    450: 'Blocked by Windows Parental Controls',
+    451: 'Redirect',
+    451: 'Unavailable For Legal Reasons',
+    494: 'Request Header Too Large',
+    495: 'Cert Error',
+    496: 'No Cert',
+    497: 'HTTP to HTTPS',
+    498: 'Token expired/invalid',
+    499: 'Client Closed Request',
+    499: 'Token required',
     500: 'Internal Server Error',
     501: 'Not Implemented',
     502: 'Bad Gateway',
     503: 'Service Unavailable',
     504: 'Gateway Timeout',
     505: 'HTTP Version Not Supported',
-    524: 'Request to host timedout waiting for reply back'
+    506: 'Variant Also Negotiates',
+    507: 'Insufficient Storage',
+    508: 'Loop Detected',
+    509: 'Bandwidth Limit Exceeded',
+    510: 'Not Extended',
+    511: 'Network Authentication Required',
+    522: 'Cloudfare Connection timed out',
+    524: 'Request to host timedout waiting for reply back',
+    598: 'Network read timeout error',
+    599: 'Network connect timeout error '
 }
 
 default_host = {'utorrent': 'http://localhost:8000',
@@ -68,6 +116,7 @@ default_host = {'utorrent': 'http://localhost:8000',
                 'deluge': 'http://localhost:8112',
                 'download_station': 'http://localhost:5000',
                 'rtorrent': 'scgi://localhost:5000',
+                'qbittorrent': 'http://localhost:8080'
 }
 
 
@@ -82,4 +131,4 @@ def getClientIstance(name):
     module = getClientModule(name)
     className = module.api.__class__.__name__
 
-    return getattr(module, className)
\ No newline at end of file
+    return getattr(module, className)
diff --git a/sickbeard/clients/generic.py b/sickbeard/clients/generic.py
index 7940ab17da62cc70cf64a1271facf9acc9b40d5b..3dfc58dde954a368363ced1c354d0eae3e2d755d 100644
--- a/sickbeard/clients/generic.py
+++ b/sickbeard/clients/generic.py
@@ -61,7 +61,7 @@ class GenericClient(object):
             logger.log(self.name + u': Invalid HTTP Request ' + str(e), logger.ERROR)
             return False
         except requests.exceptions.Timeout, e:
-            logger.log(self.name + u': Connection Timeout ' + str(e), logger.ERROR)
+            logger.log(self.name + u': Connection Timeout ' + str(e), logger.WARNING)
             return False
         except Exception, e:
             logger.log(self.name + u': Unknown exception raised when send torrent to ' + self.name + ': ' + str(e),
diff --git a/sickbeard/clients/qbittorrent.py b/sickbeard/clients/qbittorrent.py
new file mode 100644
index 0000000000000000000000000000000000000000..5923c274cc75b298107b5a839f074d978a0447aa
--- /dev/null
+++ b/sickbeard/clients/qbittorrent.py
@@ -0,0 +1,73 @@
+# Author: Mr_Orange <mr_orange@hotmail.it>
+# URL: http://code.google.com/p/sickbeard/
+#
+# This file is part of SickRage.
+#
+# SickRage is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# SickRage is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#  GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with SickRage.  If not, see <http://www.gnu.org/licenses/>.
+
+import sickbeard
+from sickbeard import logger
+from sickbeard.clients.generic import GenericClient
+from lib import requests
+from lib.requests.auth import HTTPDigestAuth
+
+class qbittorrentAPI(GenericClient):
+    def __init__(self, host=None, username=None, password=None):
+
+        super(qbittorrentAPI, self).__init__('qbittorrent', host, username, password)
+
+        self.url = self.host
+        self.session.auth = HTTPDigestAuth(self.username, self.password);
+
+    def _get_auth(self):
+
+        try:
+            self.response = self.session.get(self.host, verify=False)
+            self.auth = self.response.content
+        except:
+            return None
+
+        return self.auth if not self.response.status_code == 404 else None
+
+    def _add_torrent_uri(self, result):
+
+        self.url = self.host+'command/download'
+        data = {'urls': result.url}
+        return self._request(method='post', data=data)
+
+    def _add_torrent_file(self, result):
+
+        self.url = self.host+'command/upload'
+        files = {'torrents': (result.name + '.torrent', result.content)}
+        return self._request(method='post', files=files)
+
+    def _set_torrent_priority(self, result):
+
+        self.url = self.host+'command/decreasePrio '
+        if result.priority == 1:
+            self.url = self.host+'command/increasePrio'
+
+        data = {'hashes': result.hash}
+        return self._request(method='post', data=data)
+
+    def _set_torrent_pause(self, result):
+        
+        self.url = self.host+'command/resume'
+        if sickbeard.TORRENT_PAUSED:
+            self.url = self.host+'command/pause'
+
+        data = {'hash': result.hash}
+        return self._request(method='post', data=data)
+
+api = qbittorrentAPI()
\ No newline at end of file
diff --git a/sickbeard/config.py b/sickbeard/config.py
index 94413e32fcfe7ced4e738f164806859fd4961723..9a25153026e0aa6043c53cf136c9640ac9d9d325 100644
--- a/sickbeard/config.py
+++ b/sickbeard/config.py
@@ -230,7 +230,7 @@ def change_DOWNLOAD_PROPERS(download_propers):
     else:
         sickbeard.properFinderScheduler.enable = False
         sickbeard.traktCheckerScheduler.silent = True
-        logger.log(u"Waiting for the PROPERFINDER thread to exit", logger.INFO)
+        logger.log(u"Stopping PROPERFINDER thread", logger.INFO)
 
 def change_USE_TRAKT(use_trakt):
     use_trakt = checkbox_to_value(use_trakt)
diff --git a/sickbeard/encodingKludge.py b/sickbeard/encodingKludge.py
index ffef1d2b348d73e7b5f1ad007567646bce6a9838..fadb28d0ff178d05a0f080d2936eeab2f9fc4a7c 100644
--- a/sickbeard/encodingKludge.py
+++ b/sickbeard/encodingKludge.py
@@ -20,6 +20,21 @@ import os
 import chardet
 import sickbeard
 
+def fixStupidEncodings(x, silent=False):
+    if type(x) == str:
+        try:
+            return x.decode(sickbeard.SYS_ENCODING)
+        except UnicodeDecodeError:
+            logger.log(u"Unable to decode value: " + repr(x), logger.ERROR)
+            return None
+    elif type(x) == unicode:
+        return x
+    else:
+        logger.log(
+            u"Unknown value passed in, ignoring it: " + str(type(x)) + " (" + repr(x) + ":" + repr(type(x)) + ")",
+            logger.DEBUG if silent else logger.ERROR)
+        return None
+        
 def _toUnicode(x):
     try:
         if not isinstance(x, unicode):
diff --git a/sickbeard/helpers.py b/sickbeard/helpers.py
index e3451266735199e79a47db046c9c5364637c2908..1d0d041d9e4fe6701834ba2b11dcc5b52de07fae 100644
--- a/sickbeard/helpers.py
+++ b/sickbeard/helpers.py
@@ -1037,6 +1037,9 @@ def validateShow(show, season=None, episode=None):
         if indexer_lang and not indexer_lang == sickbeard.INDEXER_DEFAULT_LANGUAGE:
             lINDEXER_API_PARMS['language'] = indexer_lang
 
+        if show.dvdorder != 0:
+            lINDEXER_API_PARMS['dvdorder'] = True
+            
         t = sickbeard.indexerApi(show.indexer).indexer(**lINDEXER_API_PARMS)
         if season is None and episode is None:
             return t
@@ -1247,6 +1250,16 @@ def _getTempDir():
 
     return os.path.join(tempfile.gettempdir(), "sickrage-%s" % (uid))
 
+def codeDescription(status_code):
+    """
+    Returns the description of the URL error code
+    """
+    if status_code in clients.http_error_code:
+        return clients.http_error_code[status_code]
+    else:
+        logger.log(u"Unknown error code. Please submit an issue", logger.WARNING)
+        return 'unknown'
+
 def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=None, json=False, proxyGlypeProxySSLwarning=None):
     """
     Returns a byte-string retrieved from the url provider.
@@ -1283,7 +1296,7 @@ def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=Non
 
         if not resp.ok:
             logger.log(u"Requested url " + url + " returned status code is " + str(
-                resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG)
+                resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG)
             return
 
         if proxyGlypeProxySSLwarning is not None:
@@ -1292,7 +1305,7 @@ def getURL(url, post_data=None, params=None, headers={}, timeout=30, session=Non
 
                 if not resp.ok:
                     logger.log(u"GlypeProxySSLwarning: Requested url " + url + " returned status code is " + str(
-                        resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG)
+                        resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG)
                     return
 
     except requests.exceptions.HTTPError, e:
@@ -1334,9 +1347,10 @@ def download_file(url, filename, session=None):
 
     try:
         resp = session.get(url)
+            
         if not resp.ok:
             logger.log(u"Requested url " + url + " returned status code is " + str(
-                resp.status_code) + ': ' + clients.http_error_code[resp.status_code], logger.DEBUG)
+                resp.status_code) + ': ' + codeDescription(resp.status_code), logger.DEBUG)
             return False
 
         with open(filename, 'wb') as fp:
@@ -1409,7 +1423,11 @@ def get_size(start_path='.'):
     for dirpath, dirnames, filenames in ek.ek(os.walk, start_path):
         for f in filenames:
             fp = ek.ek(os.path.join, dirpath, f)
-            total_size += ek.ek(os.path.getsize, fp)
+            try:
+                total_size += ek.ek(os.path.getsize, fp)
+            except OSError as e:
+                logger.log('Unable to get size for file {filePath}. Error msg is: {errorMsg}'.format(filePath=fp, errorMsg=str(e)), logger.ERROR)
+                logger.log(traceback.format_exc(), logger.DEBUG)
     return total_size
 
 def generateApiKey():
@@ -1537,4 +1555,4 @@ def pretty_time_delta(seconds):
     elif minutes > 0:
         return '%s%02dm%02ds' % (sign_string, minutes, seconds)
     else:
-        return '%s%02ds' % (sign_string, seconds)
\ No newline at end of file
+        return '%s%02ds' % (sign_string, seconds)
diff --git a/sickbeard/metadata/generic.py b/sickbeard/metadata/generic.py
index bd14875d003131092c3a2e624490ddc5851aa351..098b770424f5aacceb5e96a38d26e3942fee850e 100644
--- a/sickbeard/metadata/generic.py
+++ b/sickbeard/metadata/generic.py
@@ -951,11 +951,6 @@ class GenericMetadata():
 
             name = showXML.findtext('title')
 
-            try:
-                indexer = int(showXML.findtext('indexer'))
-            except:
-                indexer = None
-
             if showXML.findtext('tvdbid') != None:
                 indexer_id = int(showXML.findtext('tvdbid'))
             elif showXML.findtext('id') != None:
@@ -968,6 +963,18 @@ class GenericMetadata():
                 logger.log(u"Invalid Indexer ID (" + str(indexer_id) + "), not using metadata file", logger.WARNING)
                 return empty_return
 
+            indexer = None
+            if showXML.findtext('indexer') != None:
+                indexer = int(showXML.findtext('indexer'))
+            elif showXML.find('episodeguide/url') != None:
+                epg_url = showXML.findtext('episodeguide/url').lower()
+                if str(indexer_id) in epg_url:
+                    if 'thetvdb.com' in epg_url:
+                        indexer = 1
+                    elif 'tvrage' in epg_url:
+                        indexer = 2
+
+
         except Exception, e:
             logger.log(
                 u"There was an error parsing your existing metadata file: '" + metadata_path + "' error: " + ex(e),
diff --git a/sickbeard/metadata/kodi_12plus.py b/sickbeard/metadata/kodi_12plus.py
index 1dba0d1730678ff7d099e39ee961a235ac39b91d..02d5b8735086229133f65fd4d269a9913b309cf2 100644
--- a/sickbeard/metadata/kodi_12plus.py
+++ b/sickbeard/metadata/kodi_12plus.py
@@ -153,11 +153,8 @@ class KODI_12PlusMetadata(generic.GenericMetadata):
 
         episodeguide = etree.SubElement(tv_node, "episodeguide")
         episodeguideurl = etree.SubElement(episodeguide, "url")
-        episodeguideurl2 = etree.SubElement(tv_node, "episodeguideurl")
         if getattr(myShow, 'id', None) is not None:
-            showurl = sickbeard.indexerApi(show_obj.indexer).config['base_url'] + str(myShow["id"]) + '/all/en.zip'
-            episodeguideurl.text = showurl
-            episodeguideurl2.text = showurl
+            episodeguideurl.text = sickbeard.indexerApi(show_obj.indexer).config['base_url'] + str(myShow["id"]) + '/all/en.zip'
 
         mpaa = etree.SubElement(tv_node, "mpaa")
         if getattr(myShow, 'contentrating', None) is not None:
diff --git a/sickbeard/metadata/mede8er.py b/sickbeard/metadata/mede8er.py
index 0fe31761a6395f2d933b4807ef5de940d3940778..4db891493c761c5f7d000d188d776f564807e4ab 100644
--- a/sickbeard/metadata/mede8er.py
+++ b/sickbeard/metadata/mede8er.py
@@ -390,7 +390,7 @@ class Mede8erMetadata(mediabrowser.MediaBrowserMetadata):
 
             nfo_file = ek.ek(open, nfo_file_path, 'w')
 
-            data.write(nfo_file, encoding="utf-8", xml_declaration=True)
+            data.write(nfo_file, encoding="UTF-8")
             nfo_file.close()
             helpers.chmodAsParent(nfo_file_path)
         except IOError, e:
@@ -435,7 +435,7 @@ class Mede8erMetadata(mediabrowser.MediaBrowserMetadata):
 
             nfo_file = ek.ek(open, nfo_file_path, 'w')
 
-            data.write(nfo_file, encoding="utf-8", xml_declaration = True)
+            data.write(nfo_file, encoding="UTF-8")
             nfo_file.close()
             helpers.chmodAsParent(nfo_file_path)
         except IOError, e:
diff --git a/sickbeard/notifiers/emailnotify.py b/sickbeard/notifiers/emailnotify.py
index 02255ac4da3584884880032ba7ad2f6590516193..9b21d29bd8f2390873a646cbdb36f111105aa814 100644
--- a/sickbeard/notifiers/emailnotify.py
+++ b/sickbeard/notifiers/emailnotify.py
@@ -190,7 +190,13 @@ class EmailNotifier:
     def _sendmail(self, host, port, smtp_from, use_tls, user, pwd, to, msg, smtpDebug=False):
         logger.log('HOST: %s; PORT: %s; FROM: %s, TLS: %s, USER: %s, PWD: %s, TO: %s' % (
             host, port, smtp_from, use_tls, user, pwd, to), logger.DEBUG)
-        srv = smtplib.SMTP(host, int(port))
+        try:
+            srv = smtplib.SMTP(host, int(port))
+        except Exception as e:
+            logger.log(u"Exception generated while sending e-mail: " + str(e), logger.ERROR)
+            logger.log(traceback.format_exc(), logger.DEBUG)
+            return False
+            
         if smtpDebug:
             srv.set_debuglevel(1)
         try:
diff --git a/sickbeard/notifiers/kodi.py b/sickbeard/notifiers/kodi.py
index 7b4bd3c6e991d6a92ef28be450fea5e815b427e6..f7e3b91689452f78eda1c28309c7026d444c2734 100644
--- a/sickbeard/notifiers/kodi.py
+++ b/sickbeard/notifiers/kodi.py
@@ -247,8 +247,8 @@ class KODINotifier:
             logger.log(u"KODI HTTP response: " + result.replace('\n', ''), logger.DEBUG)
             return result
 
-        except (urllib2.URLError, IOError), e:
-            logger.log(u"Warning: Couldn't contact KODI HTTP at " + ek.ss(url) + " " + ex(e),
+        except Exception as e:
+            logger.log(u"Warning: Couldn't contact KODI HTTP at " + ek.ss(url) + " " + str(e),
                        logger.WARNING)
             return False
 
diff --git a/sickbeard/notifiers/libnotify.py b/sickbeard/notifiers/libnotify.py
index 9984bdce0901f84b6197c1f61371ce1237102a10..d4dc7953598ddf0851d7db3b3ecc36f84fc475c1 100644
--- a/sickbeard/notifiers/libnotify.py
+++ b/sickbeard/notifiers/libnotify.py
@@ -29,10 +29,11 @@ def diagnose():
     user-readable message indicating possible issues.
     '''
     try:
-        import pynotify  #@UnusedImport
+        from gi.repository import Notify #@UnusedImport
     except ImportError:
-        return (u"<p>Error: pynotify isn't installed.  On Ubuntu/Debian, install the "
-                u"<a href=\"apt:python-notify\">python-notify</a> package.")
+        return (u"<p>Error: gir-notify isn't installed. On Ubuntu/Debian, install the "
+                u"<a href=\"apt:gir1.2-notify-0.7\">gir1.2-notify-0.7</a> or "
+                u"<a href=\"apt:gir1.0-notify-0.4\">gir1.0-notify-0.4</a> package.")
     if 'DISPLAY' not in os.environ and 'DBUS_SESSION_BUS_ADDRESS' not in os.environ:
         return (u"<p>Error: Environment variables DISPLAY and DBUS_SESSION_BUS_ADDRESS "
                 u"aren't set.  libnotify will only work when you run SickRage "
@@ -58,26 +59,26 @@ def diagnose():
 
 class LibnotifyNotifier:
     def __init__(self):
-        self.pynotify = None
+        self.Notify = None
         self.gobject = None
 
-    def init_pynotify(self):
-        if self.pynotify is not None:
+    def init_notify(self):
+        if self.Notify is not None:
             return True
         try:
-            import pynotify
+            from gi.repository import Notify
         except ImportError:
-            logger.log(u"Unable to import pynotify. libnotify notifications won't work.", logger.ERROR)
+            logger.log(u"Unable to import Notify from gi.repository. libnotify notifications won't work.", logger.ERROR)
             return False
         try:
             import gobject
         except ImportError:
             logger.log(u"Unable to import gobject. We can't catch a GError in display.", logger.ERROR)
             return False
-        if not pynotify.init('SickRage'):
-            logger.log(u"Initialization of pynotify failed. libnotify notifications won't work.", logger.ERROR)
+        if not Notify.init('SickRage'):
+            logger.log(u"Initialization of Notify failed. libnotify notifications won't work.", logger.ERROR)
             return False
-        self.pynotify = pynotify
+        self.Notify = Notify
         self.gobject = gobject
         return True
 
@@ -105,18 +106,17 @@ class LibnotifyNotifier:
     def _notify(self, title, message, force=False):
         if not sickbeard.USE_LIBNOTIFY and not force:
             return False
-        if not self.init_pynotify():
+        if not self.init_notify():
             return False
 
         # Can't make this a global constant because PROG_DIR isn't available
         # when the module is imported.
-        icon_path = os.path.join(sickbeard.PROG_DIR, "data/images/sickbeard_touch_icon.png")
-        icon_uri = 'file://' + os.path.abspath(icon_path)
+        icon_path = os.path.join(sickbeard.PROG_DIR, 'gui', 'slick', 'images', 'ico', 'favicon-120.png')
 
         # If the session bus can't be acquired here a bunch of warning messages
         # will be printed but the call to show() will still return True.
         # pynotify doesn't seem too keen on error handling.
-        n = self.pynotify.Notification(title, message, icon_uri)
+        n = self.Notify.Notification.new(title, message, icon_path)
         try:
             return n.show()
         except self.gobject.GError:
diff --git a/sickbeard/notifiers/plex.py b/sickbeard/notifiers/plex.py
index 735469a93dd8b386b21bc30050846788f0164ee3..a878801854f85826193b8d74a59ced4060c9d7a7 100644
--- a/sickbeard/notifiers/plex.py
+++ b/sickbeard/notifiers/plex.py
@@ -19,41 +19,122 @@
 import urllib
 import urllib2
 import base64
+import re
 
 import sickbeard
 
 from sickbeard import logger
 from sickbeard import common
 from sickbeard.exceptions import ex
-from sickbeard import encodingKludge as ek
+from sickbeard.encodingKludge import fixStupidEncodings
 
-from sickbeard.notifiers.kodi import KODINotifier
+try:
+    import xml.etree.cElementTree as etree
+except ImportError:
+    import elementtree.ElementTree as etree
 
-# TODO: switch over to using ElementTree
-from xml.dom import minidom
 
+class PLEXNotifier:
+
+    def _send_to_plex(self, command, host, username=None, password=None):
+        """Handles communication to Plex hosts via HTTP API
+
+        Args:
+            command: Dictionary of field/data pairs, encoded via urllib and passed to the legacy xbmcCmds HTTP API
+            host: Plex host:port
+            username: Plex API username
+            password: Plex API password
+
+        Returns:
+            Returns 'OK' for successful commands or False if there was an error
+
+        """
 
-class PLEXNotifier(KODINotifier):
-    def _notify_pmc(self, message, title="SickRage", host=None, username=None, password=None, force=False):
         # fill in omitted parameters
-        if not host:
-            if sickbeard.PLEX_HOST:
-                host = sickbeard.PLEX_HOST  # Use the default Plex host
-            else:
-                logger.log(u"No Plex host specified, check your settings", logger.DEBUG)
-                return False
         if not username:
             username = sickbeard.PLEX_USERNAME
         if not password:
             password = sickbeard.PLEX_PASSWORD
 
+        if not host:
+            logger.log(u'PLEX: No host specified, check your settings', logger.ERROR)
+            return False
+
+        for key in command:
+            if type(command[key]) == unicode:
+                command[key] = command[key].encode('utf-8')
+
+        enc_command = urllib.urlencode(command)
+        logger.log(u'PLEX: Encoded API command: ' + enc_command, logger.DEBUG)
+
+        url = 'http://%s/xbmcCmds/xbmcHttp/?%s' % (host, enc_command)
+        try:
+            req = urllib2.Request(url)
+            # if we have a password, use authentication
+            if password:
+                base64string = base64.encodestring('%s:%s' % (username, password))[:-1]
+                authheader = 'Basic %s' % base64string
+                req.add_header('Authorization', authheader)
+                logger.log(u'PLEX: Contacting (with auth header) via url: ' + url, logger.DEBUG)
+            else:
+                logger.log(u'PLEX: Contacting via url: ' + url, logger.DEBUG)
+
+            response = urllib2.urlopen(req)
+
+            result = response.read().decode(sickbeard.SYS_ENCODING)
+            response.close()
+
+            logger.log(u'PLEX: HTTP response: ' + result.replace('\n', ''), logger.DEBUG)
+            # could return result response = re.compile('<html><li>(.+\w)</html>').findall(result)
+            return 'OK'
+
+        except (urllib2.URLError, IOError), e:
+            logger.log(u'PLEX: Warning: Couldn\'t contact Plex at ' + fixStupidEncodings(url) + ' ' + ex(e), logger.WARNING)
+            return False
+
+    def _notify_pmc(self, message, title='SickRage', host=None, username=None, password=None, force=False):
+        """Internal wrapper for the notify_snatch and notify_download functions
+
+        Args:
+            message: Message body of the notice to send
+            title: Title of the notice to send
+            host: Plex Media Client(s) host:port
+            username: Plex username
+            password: Plex password
+            force: Used for the Test method to override config safety checks
+
+        Returns:
+            Returns a list results in the format of host:ip:result
+            The result will either be 'OK' or False, this is used to be parsed by the calling function.
+
+        """
+
         # suppress notifications if the notifier is disabled but the notify options are checked
         if not sickbeard.USE_PLEX and not force:
-            logger.log("Notification for Plex not enabled, skipping this notification", logger.DEBUG)
             return False
 
-        return self._notify_kodi(message=message, title=title, host=host, username=username, password=password,
-                                 force=True)
+        # fill in omitted parameters
+        if not host:
+            host = sickbeard.PLEX_HOST
+        if not username:
+            username = sickbeard.PLEX_USERNAME
+        if not password:
+            password = sickbeard.PLEX_PASSWORD
+
+        result = ''
+        for curHost in [x.strip() for x in host.split(',')]:
+            logger.log(u'PLEX: Sending notification to \'%s\' - %s' % (curHost, message), logger.DEBUG)
+
+            command = {'command': 'ExecBuiltIn', 'parameter': 'Notification(%s,%s)' % (title.encode('utf-8'), message.encode('utf-8'))}
+            notify_result = self._send_to_plex(command, curHost, username, password)
+            if notify_result:
+                result += '%s:%s' % (curHost, str(notify_result))
+
+        return result
+
+##############################################################################
+# Public functions
+##############################################################################
 
     def notify_snatch(self, ep_name):
         if sickbeard.PLEX_NOTIFY_ONSNATCH:
@@ -65,62 +146,128 @@ class PLEXNotifier(KODINotifier):
 
     def notify_subtitle_download(self, ep_name, lang):
         if sickbeard.PLEX_NOTIFY_ONSUBTITLEDOWNLOAD:
-            self._notify_pmc(ep_name + ": " + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD])
+            self._notify_pmc(ep_name + ': ' + lang, common.notifyStrings[common.NOTIFY_SUBTITLE_DOWNLOAD])
 
-    def notify_git_update(self, new_version="??"):
+    def notify_git_update(self, new_version='??'):
         if sickbeard.USE_PLEX:
             update_text = common.notifyStrings[common.NOTIFY_GIT_UPDATE_TEXT]
             title = common.notifyStrings[common.NOTIFY_GIT_UPDATE]
             self._notify_pmc(update_text + new_version, title)
 
-    def test_notify(self, host, username, password):
-        return self._notify_pmc("Testing Plex notifications from SickRage", "Test Notification", host, username,
-                                password, force=True)
+    def test_notify_pmc(self, host, username, password):
+        return self._notify_pmc('This is a test notification from SickRage', 'Test Notification', host, username, password, force=True)
+
+    def test_notify_pms(self, host, username, password, plex_server_token):
+        return self.update_library(host=host, username=username, password=password, plex_server_token=plex_server_token, force=False)
 
-    def update_library(self, showName=None):
+    def update_library(self, ep_obj=None, host=None, username=None, password=None, plex_server_token=None, force=True):
         """Handles updating the Plex Media Server host via HTTP API
 
         Plex Media Server currently only supports updating the whole video library and not a specific path.
 
         Returns:
-            Returns True or False
+            Returns None for no issue, else a string of host with connection issues
 
         """
 
         if sickbeard.USE_PLEX and sickbeard.PLEX_UPDATE_LIBRARY:
+
             if not sickbeard.PLEX_SERVER_HOST:
-                logger.log(u"No Plex Media Server host specified, check your settings", logger.DEBUG)
+                logger.log(u'PLEX: No Plex Media Server host specified, check your settings', logger.DEBUG)
                 return False
 
-            logger.log(u"Updating library for the Plex Media Server host: " + sickbeard.PLEX_SERVER_HOST,
-                       logger.INFO)
+            if not host:
+                host = sickbeard.PLEX_SERVER_HOST
+            if not username:
+                username = sickbeard.PLEX_USERNAME
+            if not password:
+                password = sickbeard.PLEX_PASSWORD
+                
+            if not plex_server_token:
+                token = sickbeard.PLEX_SERVER_TOKEN
+            
+            # if username and password were provided, fetch the auth token from plex.tv
+            token_arg = ''
+            if plex_server_token:
+                token_arg = '?X-Plex-Token=' + sickbeard.PLEX_SERVER_TOKEN            
+            elif username and password:
+                logger.log(u'PLEX: fetching plex.tv credentials for user: ' + username, logger.DEBUG)
+                req = urllib2.Request('https://plex.tv/users/sign_in.xml', data='')
+                authheader = 'Basic %s' % base64.encodestring('%s:%s' % (username, password))[:-1]
+                req.add_header('Authorization', authheader)
+                req.add_header('X-Plex-Device-Name', 'SickRage')
+                req.add_header('X-Plex-Product', 'SickRage Notifier')
+                req.add_header('X-Plex-Client-Identifier', sickbeard.CUR_COMMIT_HASH)
+                req.add_header('X-Plex-Version', '1.0')
 
-            url = "http://%s/library/sections" % sickbeard.PLEX_SERVER_HOST
-            if sickbeard.PLEX_SERVER_TOKEN:
-                url += "/?X-Plex-Token=" + sickbeard.PLEX_SERVER_TOKEN
-            try:
-                xml_sections = minidom.parse(urllib.urlopen(url))
-            except IOError, e:
-                logger.log(u"Error while trying to contact Plex Media Server: " + ex(e), logger.ERROR)
-                return False
+                try:
+                    response = urllib2.urlopen(req)
+                    auth_tree = etree.parse(response)
+                    token = auth_tree.findall('.//authentication-token')[0].text
+                    token_arg = '?X-Plex-Token=' + token
 
-            sections = xml_sections.getElementsByTagName('Directory')
-            if not sections:
-                logger.log(u"Plex Media Server not running on: " + sickbeard.PLEX_SERVER_HOST, logger.INFO)
-                return False
+                except urllib2.URLError as e:
+                    logger.log(u'PLEX: Error fetching credentials from from plex.tv for user %s: %s' % (username, ex(e)), logger.DEBUG)
+
+                except (ValueError, IndexError) as e:
+                    logger.log(u'PLEX: Error parsing plex.tv response: ' + ex(e), logger.DEBUG)
+                
+            file_location = '' if None is ep_obj else ep_obj.location
+            host_list = [x.strip() for x in host.split(',')]
+            hosts_all = {}
+            hosts_match = {}
+            hosts_failed = []
+            for cur_host in host_list:
+
+                url = 'http://%s/library/sections%s' % (cur_host, token_arg)
+                try:
+                    xml_tree = etree.parse(urllib.urlopen(url))
+                    media_container = xml_tree.getroot()
+                except IOError, e:
+                    logger.log(u'PLEX: Error while trying to contact Plex Media Server: ' + ex(e), logger.ERROR)
+                    hosts_failed.append(cur_host)
+                    continue
+
+                sections = media_container.findall('.//Directory')
+                if not sections:
+                    logger.log(u'PLEX: Plex Media Server not running on: ' + cur_host, logger.DEBUG)
+                    hosts_failed.append(cur_host)
+                    continue
 
-            for s in sections:
-                if s.getAttribute('type') == "show":
-                    url = "http://%s/library/sections/%s/refresh" % (sickbeard.PLEX_SERVER_HOST, s.getAttribute('key'))
-                    if sickbeard.PLEX_SERVER_TOKEN:
-                        url += "/?X-Plex-Token=" + sickbeard.PLEX_SERVER_TOKEN
-                    try:
-                        urllib.urlopen(url)
-                    except Exception, e:
-                        logger.log(u"Error updating library section for Plex Media Server: " + ex(e), logger.ERROR)
-                        return False
+                for section in sections:
+                    if 'show' == section.attrib['type']:
 
-            return True
+                        keyed_host = [(str(section.attrib['key']), cur_host)]
+                        hosts_all.update(keyed_host)
+                        if not file_location:
+                            continue
+
+                        for section_location in section.findall('.//Location'):
+                            section_path = re.sub(r'[/\\]+', '/', section_location.attrib['path'].lower())
+                            section_path = re.sub(r'^(.{,2})[/\\]', '', section_path)
+                            location_path = re.sub(r'[/\\]+', '/', file_location.lower())
+                            location_path = re.sub(r'^(.{,2})[/\\]', '', location_path)
+
+                            if section_path in location_path:
+                                hosts_match.update(keyed_host)
+
+            hosts_try = (hosts_all.copy(), hosts_match.copy())[len(hosts_match)]
+            host_list = []
+            for section_key, cur_host in hosts_try.items():
+
+                url = 'http://%s/library/sections/%s/refresh%s' % (cur_host, section_key, token_arg)
+                try:
+                    force and urllib.urlopen(url)
+                    host_list.append(cur_host)
+                except Exception, e:
+                    logger.log(u'PLEX: Error updating library section for Plex Media Server: ' + ex(e), logger.ERROR)
+                    hosts_failed.append(cur_host)
+
+            if len(hosts_match):
+                logger.log(u'PLEX: Updating hosts where TV section paths match the downloaded show: ' + ', '.join(set(host_list)), logger.DEBUG)
+            else:
+                logger.log(u'PLEX: Updating all hosts with TV sections: ' + ', '.join(set(host_list)), logger.DEBUG)
 
+            return (', '.join(set(hosts_failed)), None)[not len(hosts_failed)]
 
 notifier = PLEXNotifier
diff --git a/sickbeard/postProcessor.py b/sickbeard/postProcessor.py
index 173c47a2a7670793288b17c61055e5c083f3f16c..eede71fff326a0564381b4863b39490c26215555 100644
--- a/sickbeard/postProcessor.py
+++ b/sickbeard/postProcessor.py
@@ -183,10 +183,19 @@ class PostProcessor(object):
         # don't confuse glob with chars we didn't mean to use
         base_name = re.sub(r'[\[\]\*\?]', r'[\g<0>]', base_name)
         
-        if subfolders:
-            filelist = ek.ek(recursive_glob, ek.ek(os.path.dirname, file_path),  base_name + '*')
-        else:
-            filelist = ek.ek(glob.glob, base_name + '*')
+        if subfolders: # subfolders are only checked in show folder, so names will always be exactly alike
+            filelist = ek.ek(recursive_glob, ek.ek(os.path.dirname, file_path),  base_name + '*') # just create the list of all files starting with the basename
+        else: # this is called when PP, so we need to do the filename check case-insensitive
+            filelist = []
+            checklist = ek.ek(glob.glob, ek.ek(os.path.join, ek.ek(os.path.dirname, file_path), '*')) # get a list of all the files in the folder
+            for filefound in checklist: # loop through all the files in the folder, and check if they are the same name even when the cases don't match
+                file_name = filefound.rpartition('.')[0]
+                if not base_name_only:
+                    file_name = file_name + '.'
+                if file_name.lower() == base_name.lower(): # if there's no difference in the filename add it to the filelist
+                    filelist.append(filefound) 
+             
+                             
         for associated_file_path in filelist:
             # only add associated to list
             if associated_file_path == file_path:
@@ -201,7 +210,12 @@ class PostProcessor(object):
 
             if ek.ek(os.path.isfile, associated_file_path):
                 file_path_list.append(associated_file_path)
-
+        
+        if file_path_list:
+            self._log(u"Found the following associated files: " + str(file_path_list), logger.DEBUG)
+        else:
+            self._log(u"No associated files were during this pass", logger.DEBUG)
+            
         return file_path_list
 
     def _delete(self, file_path, associated_files=False):
@@ -841,7 +855,7 @@ class PostProcessor(object):
         old_ep_status, old_ep_quality = common.Quality.splitCompositeStatus(ep_obj.status)
 
         # get the quality of the episode we're processing
-        if quality:
+        if quality and not common.Quality.qualityStrings[quality] == 'Unknown':
             self._log(u"Snatch history had a quality in it, using that: " + common.Quality.qualityStrings[quality],
                       logger.DEBUG)
             new_ep_quality = quality
diff --git a/sickbeard/providers/__init__.py b/sickbeard/providers/__init__.py
index 02f150c0bf4b584a01ea64ad2913fe6384c33b96..453a34eb3b0836e4791abb12158d961789ea2ca1 100755
--- a/sickbeard/providers/__init__.py
+++ b/sickbeard/providers/__init__.py
@@ -47,6 +47,7 @@ __all__ = ['ezrss',
            'tntvillage',
            'binsearch',
            'eztv',
+           'scenetime',
 ]
 
 import sickbeard
diff --git a/sickbeard/providers/eztv.py b/sickbeard/providers/eztv.py
index 909725c5daca12bfabd39607f701cbf221df3180..22a32f4eb3376fb9c8235b2ebcc34ec7424fcddc 100644
--- a/sickbeard/providers/eztv.py
+++ b/sickbeard/providers/eztv.py
@@ -16,7 +16,6 @@
 #
 # You should have received a copy of the GNU General Public License
 # along with Sick Beard.  If not, see <http://www.gnu.org/licenses/>.
-
 import traceback
 import re, datetime
 
@@ -26,6 +25,7 @@ from sickbeard import classes
 from sickbeard import helpers
 from sickbeard import logger, tvcache, db
 from sickbeard.common import Quality
+from sickbeard.bs4_parser import BS4Parser
 
 class EZTVProvider(generic.TorrentProvider):
 
@@ -39,8 +39,9 @@ class EZTVProvider(generic.TorrentProvider):
         self.cache = EZTVCache(self)
 
         self.urls = {
-            'base_url': 'http://eztvapi.re/',
-            'show': 'http://eztvapi.re/show/%s',
+            'base_url': 'https://eztv.ch/',
+            'rss': 'https://eztv.ch/',
+            'episode': 'http://eztvapi.re/show/%s',
         }
 
         self.url = self.urls['base_url']
@@ -68,12 +69,15 @@ class EZTVProvider(generic.TorrentProvider):
         return [search_string]
 
     def getQuality(self, item, anime=False):
-        if item.get('quality') == "480p":
-            return Quality.SDTV
-        elif item.get('quality') == "720p":
-            return Quality.HDWEBDL
-        elif item.get('quality') == "1080p":
-            return Quality.FULLHDWEBDL
+        if 'quality' in item:
+            if item.get('quality') == "480p":
+                return Quality.SDTV
+            elif item.get('quality') == "720p":
+                return Quality.HDWEBDL
+            elif item.get('quality') == "1080p":
+                return Quality.FULLHDWEBDL
+            else:
+                return Quality.sceneQuality(item.get('title'), anime)
         else:
             return Quality.sceneQuality(item.get('title'), anime)
 
@@ -84,49 +88,83 @@ class EZTVProvider(generic.TorrentProvider):
 
         for mode in search_params.keys():
 
-            if mode != 'Episode':
-                logger.log(u"" + self.name + " does not accept " + mode + " mode", logger.DEBUG)
-                return results
+            if mode == 'RSS':
+                for search_string in search_params[mode]:
+                    searchURL = self.urls['rss']
+                    logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG)
 
-            for search_string in search_params[mode]:
+                    HTML = self.getURL(searchURL)
+                    if not HTML:
+                        logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG)
+                        return results
 
-                searchURL = self.urls['show'] % (search_string['imdb_id'])
-                logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG)
+                    try:
+                        with BS4Parser(HTML, features=["html5lib", "permissive"]) as parsedHTML:
+                            resultsTable = parsedHTML.find_all('tr', attrs={'name': 'hover', 'class': 'header_brd'})
 
-                try:
-                    parsedJSON = self.getURL(searchURL, json=True)
-                except ValueError as e:
-                    parsedJSON = None
+                            if not resultsTable:
+                                logger.log(u"The Data returned from " + self.name + " do not contains any torrent",
+                                           logger.DEBUG)
+                                continue
 
-                if not parsedJSON:
-                    logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG)
-                    return results
+                            for entries in resultsTable:
+                                title = entries.find('a', attrs={'class': 'epinfo'}).contents[0]
+                                link = entries.find('a', attrs={'class': 'magnet'}).get('href')
 
-            try:
-                for episode in parsedJSON['episodes']:
-                    if int(episode.get('season')) == search_string.get('season') and \
-                       int(episode.get('episode')) == search_string.get('episode'):
+                                item = {
+                                    'title': title,
+                                    'link': link,
+                                }
 
-                        for quality in episode['torrents'].keys():
-                            link = episode['torrents'][quality]['url']
-                            title = re.search('&dn=(.*?)&', link).group(1)
+                                items[mode].append(item)
 
-                            item = {
-                                'title': title,
-                                'link': link,
-                                'quality': quality
-                            }
+                    except Exception, e:
+                        logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(),
+                                    logger.ERROR)
 
-                            # re.search in case of PROPER|REPACK. In other cases
-                            # add_string is empty, so condition is met.
-                            if re.search(search_string.get('add_string'), title):
-                                items[mode].append(item)
+            elif mode == 'Episode':
+                for search_string in search_params[mode]:
+                    searchURL = self.urls['episode'] % (search_string['imdb_id'])
+                    logger.log(u"" + self.name + " search page URL: " + searchURL, logger.DEBUG)
+
+                    try:
+                        parsedJSON = self.getURL(searchURL, json=True)
+                    except ValueError as e:
+                        parsedJSON = None
+
+                    if not parsedJSON:
+                        logger.log(u"" + self.name + " could not retrieve page URL:" + searchURL, logger.DEBUG)
+                        return results
 
-                        break
+                    try:
+                        for episode in parsedJSON['episodes']:
+                            if int(episode.get('season')) == search_string.get('season') and \
+                               int(episode.get('episode')) == search_string.get('episode'):
 
-            except Exception, e:
-                logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(),
-                            logger.ERROR)
+                                for quality in episode['torrents'].keys():
+                                    link = episode['torrents'][quality]['url']
+                                    title = re.search('&dn=(.*?)&', link).group(1)
+
+                                    item = {
+                                        'title': title,
+                                        'link': link,
+                                        'quality': quality
+                                    }
+
+                                    # re.search in case of PROPER|REPACK. In other cases
+                                    # add_string is empty, so condition is met.
+                                    if 'add_string' in search_string and  re.search(search_string.get('add_string'), title):
+                                        items[mode].append(item)
+
+                                break
+
+                    except Exception, e:
+                        logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(),
+                                    logger.ERROR)
+
+            else:
+                logger.log(u"" + self.name + " does not accept " + mode + " mode", logger.DEBUG)
+                return results
 
             results += items[mode]
 
diff --git a/sickbeard/providers/iptorrents.py b/sickbeard/providers/iptorrents.py
index 1af7e8ca8008e926371ea5fde8efaffc9990c2c6..28a8f3c1038d54d0de6a39f96a20a62738811b53 100644
--- a/sickbeard/providers/iptorrents.py
+++ b/sickbeard/providers/iptorrents.py
@@ -24,6 +24,7 @@ import itertools
 
 import sickbeard
 import generic
+from sickbeard.common import MULTI_EP_RESULT, SEASON_RESULT
 from sickbeard.common import Quality
 from sickbeard import logger
 from sickbeard import tvcache
@@ -59,12 +60,12 @@ class IPTorrentsProvider(generic.TorrentProvider):
 
         self.urls = {'base_url': 'https://iptorrents.eu',
                 'login': 'https://iptorrents.eu/torrents/',
-                'search': 'https://iptorrents.eu/torrents/?%s%s&q=%s&qf=ti',
+                'search': 'https://iptorrents.eu/t?%s%s&q=%s&qf=#torrents',
         }
 
         self.url = self.urls['base_url']
 
-        self.categorie = 'l73=1&l78=1&l66=1&l65=1&l79=1&l5=1&l4=1'
+        self.categorie = 'l73='
 
     def isEnabled(self):
         return self.enabled
diff --git a/sickbeard/providers/rarbg.py b/sickbeard/providers/rarbg.py
index ad8e63d6a7e7302c4b8b8d859b82f48ded7c395f..c56a3ccc672cdc691632c35a66fd7bb5312fc01c 100644
--- a/sickbeard/providers/rarbg.py
+++ b/sickbeard/providers/rarbg.py
@@ -30,7 +30,7 @@ from lib import requests
 from lib.requests import exceptions
 
 import sickbeard
-from sickbeard.common import Quality
+from sickbeard.common import Quality, USER_AGENT
 from sickbeard import logger
 from sickbeard import tvcache
 from sickbeard import show_name_helpers
@@ -62,13 +62,13 @@ class RarbgProvider(generic.TorrentProvider):
         self.token = None
         self.tokenExpireDate = None
 
-        self.urls = {'url': 'https://rarbg.com',
-                     'token': 'https://torrentapi.org/pubapi.php?get_token=get_token&format=json',
-                     'listing': 'https://torrentapi.org/pubapi.php?mode=list',
-                     'search': 'https://torrentapi.org/pubapi.php?mode=search&search_string={search_string}',
-                     'search_tvdb': 'https://torrentapi.org/pubapi.php?mode=search&search_tvdb={tvdb}&search_string={search_string}',
-                     'search_tvrage': 'https://torrentapi.org/pubapi.php?mode=search&search_tvrage={tvrage}&search_string={search_string}',
-                     'api_spec': 'https://rarbg.com/pubapi/apidocs.txt',
+        self.urls = {'url': u'https://rarbg.com',
+                     'token': u'https://torrentapi.org/pubapi.php?get_token=get_token&format=json',
+                     'listing': u'https://torrentapi.org/pubapi.php?mode=list',
+                     'search': u'https://torrentapi.org/pubapi.php?mode=search&search_string={search_string}',
+                     'search_tvdb': u'https://torrentapi.org/pubapi.php?mode=search&search_tvdb={tvdb}&search_string={search_string}',
+                     'search_tvrage': u'https://torrentapi.org/pubapi.php?mode=search&search_tvrage={tvrage}&search_string={search_string}',
+                     'api_spec': u'https://rarbg.com/pubapi/apidocs.txt',
                      }
 
         self.url = self.urls['listing']
@@ -92,6 +92,8 @@ class RarbgProvider(generic.TorrentProvider):
         self.next_request = datetime.datetime.now()
 
         self.cache = RarbgCache(self)
+        
+        self.headers = {'User-Agent': USER_AGENT}
 
     def isEnabled(self):
         return self.enabled
@@ -107,7 +109,7 @@ class RarbgProvider(generic.TorrentProvider):
         resp_json = None
 
         try:
-            response = self.session.get(self.urls['token'], timeout=30, verify=False)
+            response = self.session.get(self.urls['token'], timeout=30, verify=False, headers=self.headers)
             response.raise_for_status()
             resp_json = response.json()
         except RequestException as e:
@@ -204,18 +206,18 @@ class RarbgProvider(generic.TorrentProvider):
                     searchURL = self.urls['listing'] + self.defaultOptions
                 elif mode == 'Season':
                     if ep_indexer == INDEXER_TVDB:
-                        searchURL = self.urls['search_tvdb'].format(search_string=urllib.quote(search_string), tvdb=ep_indexerid) + self.defaultOptions
+                        searchURL = self.urls['search_tvdb'].format(search_string=search_string, tvdb=ep_indexerid) + self.defaultOptions
                     elif ep_indexer == INDEXER_TVRAGE:
-                        searchURL = self.urls['search_tvrage'].format(search_string=urllib.quote(search_string), tvrage=ep_indexerid) + self.defaultOptions
+                        searchURL = self.urls['search_tvrage'].format(search_string=search_string, tvrage=ep_indexerid) + self.defaultOptions
                     else:
-                        searchURL = self.urls['search'].format(search_string=urllib.quote(search_string)) + self.defaultOptions
+                        searchURL = self.urls['search'].format(search_string=search_string) + self.defaultOptions
                 elif mode == 'Episode':
                     if ep_indexer == INDEXER_TVDB:
-                        searchURL = self.urls['search_tvdb'].format(search_string=urllib.quote(search_string), tvdb=ep_indexerid) + self.defaultOptions
+                        searchURL = self.urls['search_tvdb'].format(search_string=search_string, tvdb=ep_indexerid) + self.defaultOptions
                     elif ep_indexer == INDEXER_TVRAGE:
-                        searchURL = self.urls['search_tvrage'].format(search_string=urllib.quote(search_string), tvrage=ep_indexerid) + self.defaultOptions
+                        searchURL = self.urls['search_tvrage'].format(search_string=search_string, tvrage=ep_indexerid) + self.defaultOptions
                     else:
-                        searchURL = self.urls['search'].format(search_string=urllib.quote(search_string)) + self.defaultOptions
+                        searchURL = self.urls['search'].format(search_string=search_string) + self.defaultOptions
                 else:
                     logger.log(u'{name} invalid search mode:{mode}'.format(name=self.name, mode=mode), logger.ERROR)
 
diff --git a/sickbeard/providers/scenetime.py b/sickbeard/providers/scenetime.py
new file mode 100644
index 0000000000000000000000000000000000000000..c3aaca9d341022e5f457e145904e382f3f3f0970
--- /dev/null
+++ b/sickbeard/providers/scenetime.py
@@ -0,0 +1,285 @@
+# Author: Idan Gutman
+# URL: http://code.google.com/p/sickbeard/
+#
+# This file is part of SickRage.
+#
+# SickRage is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# SickRage is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with SickRage.  If not, see <http://www.gnu.org/licenses/>.
+
+import re
+import traceback
+import datetime
+import urlparse
+import sickbeard
+import generic
+import urllib
+from sickbeard.common import Quality
+from sickbeard import logger
+from sickbeard import tvcache
+from sickbeard import db
+from sickbeard import classes
+from sickbeard import helpers
+from sickbeard import show_name_helpers
+from sickbeard.exceptions import ex
+from sickbeard import clients
+from lib import requests
+from lib.requests import exceptions
+from sickbeard.bs4_parser import BS4Parser
+from lib.unidecode import unidecode
+from sickbeard.helpers import sanitizeSceneName
+
+
+class SceneTimeProvider(generic.TorrentProvider):
+
+    def __init__(self):
+
+        generic.TorrentProvider.__init__(self, "SceneTime")
+
+        self.supportsBacklog = True
+
+        self.enabled = False
+        self.username = None
+        self.password = None
+        self.ratio = None
+        self.minseed = None
+        self.minleech = None
+
+        self.cache = SceneTimeCache(self)
+
+        self.urls = {'base_url': 'https://www.scenetime.com',
+                'login': 'https://www.scenetime.com/takelogin.php',
+                'detail': 'https://www.scenetime.com/details.php?id=%s',
+                'search': 'https://www.scenetime.com/browse.php?search=%s%s',
+                'download': 'https://www.scenetime.com/download.php/%s/%s',
+                }
+
+        self.url = self.urls['base_url']
+
+        self.categories = "&c2=1&c43=13&c9=1&c63=1&c77=1&c79=1&c100=1&c101=1"
+
+    def isEnabled(self):
+        return self.enabled
+
+    def imageName(self):
+        return 'scenetime.png'
+
+    def getQuality(self, item, anime=False):
+
+        quality = Quality.sceneQuality(item[0], anime)
+        return quality
+
+    def _doLogin(self):
+
+        login_params = {'username': self.username,
+                        'password': self.password
+        }
+
+        self.session = requests.Session()
+
+        try:
+            response = self.session.post(self.urls['login'], data=login_params, timeout=30, verify=False)
+        except (requests.exceptions.ConnectionError, requests.exceptions.HTTPError), e:
+            logger.log(u'Unable to connect to ' + self.name + ' provider: ' + ex(e), logger.ERROR)
+            return False
+
+        if re.search('Username or password incorrect', response.text):
+            logger.log(u'Invalid username or password for ' + self.name + ' Check your settings', logger.ERROR)
+            return False
+
+        return True
+
+    def _get_season_search_strings(self, ep_obj):
+
+        search_string = {'Season': []}
+        for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
+            if ep_obj.show.air_by_date or ep_obj.show.sports:
+                ep_string = show_name + '.' + str(ep_obj.airdate).split('-')[0]
+            elif ep_obj.show.anime:
+                ep_string = show_name + '.' + "%d" % ep_obj.scene_absolute_number
+            else:
+                ep_string = show_name + '.S%02d' % int(ep_obj.scene_season)  #1) showName SXX
+
+            search_string['Season'].append(ep_string)
+
+        return [search_string]
+
+    def _get_episode_search_strings(self, ep_obj, add_string=''):
+
+        search_string = {'Episode': []}
+
+        if not ep_obj:
+            return []
+
+        if self.show.air_by_date:
+            for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
+                ep_string = sanitizeSceneName(show_name) + ' ' + \
+                            str(ep_obj.airdate).replace('-', '|')
+                search_string['Episode'].append(ep_string)
+        elif self.show.sports:
+            for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
+                ep_string = sanitizeSceneName(show_name) + ' ' + \
+                            str(ep_obj.airdate).replace('-', '|') + '|' + \
+                            ep_obj.airdate.strftime('%b')
+                search_string['Episode'].append(ep_string)
+        elif self.show.anime:
+            for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
+                ep_string = sanitizeSceneName(show_name) + ' ' + \
+                            "%i" % int(ep_obj.scene_absolute_number)
+                search_string['Episode'].append(ep_string)
+        else:
+            for show_name in set(show_name_helpers.allPossibleShowNames(self.show)):
+                ep_string = show_name_helpers.sanitizeSceneName(show_name) + ' ' + \
+                            sickbeard.config.naming_ep_type[2] % {'seasonnumber': ep_obj.scene_season,
+                                                                  'episodenumber': ep_obj.scene_episode} + ' %s' % add_string
+
+                search_string['Episode'].append(re.sub('\s+', ' ', ep_string))
+
+        return [search_string]
+
+    def _doSearch(self, search_params, search_mode='eponly', epcount=0, age=0, epObj=None):
+
+        results = []
+        items = {'Season': [], 'Episode': [], 'RSS': []}
+
+        if not self._doLogin():
+            return results
+
+        for mode in search_params.keys():
+            for search_string in search_params[mode]:
+
+                if isinstance(search_string, unicode):
+                    search_string = unidecode(search_string)
+
+                searchURL = self.urls['search'] % (urllib.quote(search_string), self.categories)
+
+                logger.log(u"Search string: " + searchURL, logger.DEBUG)
+
+                data = self.getURL(searchURL)
+                if not data:
+                    continue
+
+                try:
+                    with BS4Parser(data, features=["html5lib", "permissive"]) as html:
+                        torrent_table = html.select("#torrenttable table");
+                        torrent_rows = torrent_table[0].select("tr") if torrent_table else []
+
+                        #Continue only if one Release is found
+                        if len(torrent_rows) < 2:
+                            logger.log(u"The Data returned from %s does not contain any torrent links" % self.name,
+                                       logger.DEBUG)
+                            continue
+
+                        for result in torrent_rows[1:]:
+                            cells = result.find_all('td')
+
+                            link = cells[1].find('a');
+
+                            full_id = link['href'].replace('details.php?id=', '')
+                            torrent_id = full_id.split("&")[0]
+
+                            try:
+                                title = link.contents[0].get_text()
+
+                                filename = "%s.torrent" % title.replace(" ", ".") 
+                                
+                                download_url = self.urls['download'] % (torrent_id, filename)
+                              
+                                id = int(torrent_id)
+                                seeders = int(cells[6].get_text())
+                                leechers = int(cells[7].get_text())
+                            except (AttributeError, TypeError):
+                                continue
+
+                            #Filter unseeded torrent
+                            if mode != 'RSS' and (seeders < self.minseed or leechers < self.minleech):
+                                continue
+
+                            if not title or not download_url:
+                                continue
+
+                            item = title, download_url, id, seeders, leechers
+                            logger.log(u"Found result: " + title.replace(' ','.') + " (" + searchURL + ")", logger.DEBUG)
+
+                            items[mode].append(item)
+
+                except Exception, e:
+                    logger.log(u"Failed parsing " + self.name + " Traceback: " + traceback.format_exc(), logger.ERROR)
+
+            #For each search mode sort all the items by seeders
+            items[mode].sort(key=lambda tup: tup[3], reverse=True)
+
+            results += items[mode]
+
+        return results
+
+    def _get_title_and_url(self, item):
+
+        title, url, id, seeders, leechers = item
+
+        if title:
+            title = u'' + title
+            title = title.replace(' ', '.')
+            title = self._clean_title_from_provider(title)
+
+        if url:
+            url = str(url).replace('&amp;', '&')
+
+        return (title, url)
+
+    def findPropers(self, search_date=datetime.datetime.today()):
+
+        results = []
+
+        myDB = db.DBConnection()
+        sqlResults = myDB.select(
+            'SELECT s.show_name, e.showid, e.season, e.episode, e.status, e.airdate FROM tv_episodes AS e' +
+            ' INNER JOIN tv_shows AS s ON (e.showid = s.indexer_id)' +
+            ' WHERE e.airdate >= ' + str(search_date.toordinal()) +
+            ' AND (e.status IN (' + ','.join([str(x) for x in Quality.DOWNLOADED]) + ')' +
+            ' OR (e.status IN (' + ','.join([str(x) for x in Quality.SNATCHED]) + ')))'
+        )
+
+        if not sqlResults:
+            return []
+
+        for sqlshow in sqlResults:
+            self.show = helpers.findCertainShow(sickbeard.showList, int(sqlshow["showid"]))
+            if self.show:
+                curEp = self.show.getEpisode(int(sqlshow["season"]), int(sqlshow["episode"]))
+
+                searchString = self._get_episode_search_strings(curEp, add_string='PROPER|REPACK')
+
+                for item in self._doSearch(searchString[0]):
+                    title, url = self._get_title_and_url(item)
+                    results.append(classes.Proper(title, url, datetime.datetime.today(), self.show))
+
+        return results
+
+    def seedRatio(self):
+        return self.ratio
+
+
+class SceneTimeCache(tvcache.TVCache):
+    def __init__(self, provider):
+
+        tvcache.TVCache.__init__(self, provider)
+
+        # only poll SceneTime every 20 minutes max
+        self.minTime = 20
+
+    def _getRSSData(self):
+        search_params = {'RSS': ['']}
+        return {'entries': self.provider._doSearch(search_params)}
+
+
+provider = SceneTimeProvider()
diff --git a/sickbeard/showUpdater.py b/sickbeard/showUpdater.py
index a5dd91550afc7e3e46dcbc20acd76b68c0813ea4..6ceeefcd69bf3ab467dd4bdb32aafc1305b5e348 100644
--- a/sickbeard/showUpdater.py
+++ b/sickbeard/showUpdater.py
@@ -78,16 +78,14 @@ class ShowUpdater():
                 # if should_update returns True (not 'Ended') or show is selected stale 'Ended' then update, otherwise just refresh
                 if curShow.should_update(update_date=update_date) or curShow.indexerid in stale_should_update:
                     try:
-                        curQueueItem = sickbeard.showQueueScheduler.action.updateShow(curShow, True)  # @UndefinedVariable
+                        piList.append(sickbeard.showQueueScheduler.action.updateShow(curShow, True))  # @UndefinedVariable
                     except exceptions.CantUpdateException as e:
                         logger.log("Unable to update show: {0}".format(str(e)),logger.DEBUG)
                 else:
                     logger.log(
                         u"Not updating episodes for show " + curShow.name + " because it's marked as ended and last/next episode is not within the grace period.",
                         logger.DEBUG)
-                    curQueueItem = sickbeard.showQueueScheduler.action.refreshShow(curShow, True)  # @UndefinedVariable
-
-                piList.append(curQueueItem)
+                    piList.append(sickbeard.showQueueScheduler.action.refreshShow(curShow, True))  # @UndefinedVariable
 
             except (exceptions.CantUpdateException, exceptions.CantRefreshException), e:
                 logger.log(u"Automatic update failed: " + ex(e), logger.ERROR)
diff --git a/sickbeard/tv.py b/sickbeard/tv.py
index 512893fadf542c5852713d3cee4850e229b7e837..128a2eba4f8d6166a18107e544b71d6ee617784d 100644
--- a/sickbeard/tv.py
+++ b/sickbeard/tv.py
@@ -1437,7 +1437,8 @@ class TVEpisode(object):
                     for subtitle in subtitles.get(video):
                         added_subtitles.append(subtitle.language.alpha2)
                         helpers.chmodAsParent(subtitle.path)
-
+        except ServiceError as e:
+            logger.log("Service is unavailable: {0}".format(str(e)), logger.INFO)
         except Exception as e:
             logger.log("Error occurred when downloading subtitles: " + str(e), logger.ERROR)
             return
diff --git a/sickbeard/tvcache.py b/sickbeard/tvcache.py
index 4d642472ac861aea7fe68983df916331b43dc3b3..3e2804cba01b83d6181ddcdda6d9b1a3af9bd3a9 100644
--- a/sickbeard/tvcache.py
+++ b/sickbeard/tvcache.py
@@ -299,7 +299,7 @@ class TVCache():
 
     def searchCache(self, episode, manualSearch=False, downCurQuality=False):
         neededEps = self.findNeededEpisodes(episode, manualSearch, downCurQuality)
-        return neededEps[episode] if len(neededEps) > 0 else []
+        return neededEps[episode] if episode in neededEps else []
 
     def listPropers(self, date=None, delimiter="."):
         myDB = self._getDB()
diff --git a/sickbeard/webserve.py b/sickbeard/webserve.py
index d373be2228159b237f5f089f881168b13c157556..fc8bfcbd6e3139723669b6623655d828a33ec094 100644
--- a/sickbeard/webserve.py
+++ b/sickbeard/webserve.py
@@ -287,6 +287,9 @@ class LoginHandler(BaseHandler):
         if api_key:
             remember_me = int(self.get_argument('remember_me', default=0) or 0)
             self.set_secure_cookie('sickrage_user', api_key, expires_days=30 if remember_me > 0 else None)
+            logger.log('User logged into the SickRage web interface from IP: ' + self.request.remote_ip, logger.INFO)
+        else:
+            logger.log('User attempted a failed login to the SickRage web interface from IP: ' + self.request.remote_ip, logger.WARNING)    
 
         self.redirect('/home/')
 
@@ -891,21 +894,44 @@ class Home(WebRoot):
         return finalResult
 
 
-    def testPLEX(self, host=None, username=None, password=None):
-        # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store')
+    def testPMC(self, host=None, username=None, password=None):
+        self.set_header('Cache-Control', 'max-age=0,no-cache,no-store')
+
+        if None is not password and set('*') == set(password):
+            password = sickbeard.PLEX_PASSWORD
 
         finalResult = ''
-        for curHost in [x.strip() for x in host.split(",")]:
-            curResult = notifiers.plex_notifier.test_notify(urllib.unquote_plus(curHost), username, password)
-            if len(curResult.split(":")) > 2 and 'OK' in curResult.split(":")[2]:
-                finalResult += "Test Plex notice sent successfully to " + urllib.unquote_plus(curHost)
+        for curHost in [x.strip() for x in host.split(',')]:
+            curResult = notifiers.plex_notifier.test_notify_pmc(urllib.unquote_plus(curHost), username, password)
+            if len(curResult.split(':')) > 2 and 'OK' in curResult.split(':')[2]:
+                finalResult += 'Successful test notice sent to Plex client ... ' + urllib.unquote_plus(curHost)
             else:
-                finalResult += "Test Plex notice failed to " + urllib.unquote_plus(curHost)
-            finalResult += "<br />\n"
+                finalResult += 'Test failed for Plex client ... ' + urllib.unquote_plus(curHost)
+            finalResult += '<br />' + '\n'
+
+        ui.notifications.message('Tested Plex client(s): ', urllib.unquote_plus(host.replace(',', ', ')))
 
         return finalResult
 
+    def testPMS(self, host=None, username=None, password=None, plex_server_token=None):
+        self.set_header('Cache-Control', 'max-age=0,no-cache,no-store')
 
+        if None is not password and set('*') == set(password):
+            password = sickbeard.PLEX_PASSWORD
+
+        finalResult = ''
+
+        curResult = notifiers.plex_notifier.test_notify_pms(urllib.unquote_plus(host), username, password, plex_server_token)
+        if None is curResult:
+            finalResult += 'Successful test of Plex server(s) ... ' + urllib.unquote_plus(host.replace(',', ', '))
+        else:
+            finalResult += 'Test failed for Plex server(s) ... ' + urllib.unquote_plus(curResult.replace(',', ', '))
+        finalResult += '<br />' + '\n'
+
+        ui.notifications.message('Tested Plex Media Server host(s): ', urllib.unquote_plus(host.replace(',', ', ')))
+
+        return finalResult
+        
     def testLibnotify(self):
         # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store')
 
@@ -960,8 +986,7 @@ class Home(WebRoot):
         else:
             return '{"message": "Unable to find NMJ Database at location: %(dbloc)s. Is the right location selected and PCH running?", "database": ""}' % {
                 "dbloc": dbloc}
-
-
+   
     def testTrakt(self, username=None, password=None, disable_ssl=None, blacklist_name=None):
         # self.set_header('Cache-Control', 'max-age=0,no-cache,no-store')
         if disable_ssl == 'true':
@@ -1183,14 +1208,18 @@ class Home(WebRoot):
 
         if not sickbeard.showQueueScheduler.action.isBeingAdded(showObj):
             if not sickbeard.showQueueScheduler.action.isBeingUpdated(showObj):
+                if showObj.paused:
+                    t.submenu.append({'title': 'Resume', 'path': 'home/togglePause?show=%d' % showObj.indexerid})
+                else:
+                    t.submenu.append({'title': 'Pause', 'path': 'home/togglePause?show=%d' % showObj.indexerid})
+                    
                 t.submenu.append(
                     {'title': 'Remove', 'path': 'home/deleteShow?show=%d' % showObj.indexerid, 'confirm': True})
                 t.submenu.append({'title': 'Re-scan files', 'path': 'home/refreshShow?show=%d' % showObj.indexerid})
                 t.submenu.append(
                     {'title': 'Force Full Update', 'path': 'home/updateShow?show=%d&amp;force=1' % showObj.indexerid})
                 t.submenu.append({'title': 'Update show in KODI',
-                                  'path': 'home/updateKODI?showName=%s' % urllib.quote_plus(
-                                      showObj.name.encode('utf-8')), 'requires': self.haveKODI})
+                                  'path': 'home/updateKODI?show=%d' % showObj.indexerid, 'requires': self.haveKODI})
                 t.submenu.append({'title': 'Preview Rename', 'path': 'home/testRename?show=%d' % showObj.indexerid})
                 if sickbeard.USE_SUBTITLES and not sickbeard.showQueueScheduler.action.isBeingSubtitled(
                         showObj) and showObj.subtitles:
@@ -1516,7 +1545,7 @@ class Home(WebRoot):
         if not paused and (sickbeard.TRAKT_USE_ROLLING_DOWNLOAD and sickbeard.USE_TRAKT):
             # Checking if trakt and rolling_download are enable because updateWantedList()
             # doesn't do the distinction between a failuire and being not activated(Return false)
-            if not sickbeard.traktRollingScheduler.action.updateWantedList():
+            if not sickbeard.traktRollingScheduler.action.updateWantedList(showObj.indexerid):
                 errors.append("Unable to force an update on wanted episode")
 
         if do_update_scene_numbering:
@@ -1536,6 +1565,31 @@ class Home(WebRoot):
         return self.redirect("/home/displayShow?show=" + show)
 
 
+    def togglePause(self, show=None):
+        if show is None:
+            return self._genericMessage("Error", "Invalid show ID")
+        
+        showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(show))
+
+        if showObj is None:
+            return self._genericMessage("Error", "Unable to find the specified show")
+
+        if showObj.paused:
+            showObj.paused = 0           
+        else:
+            showObj.paused = 1
+
+        showObj.saveToDB()
+
+        if not showObj.paused and sickbeard.TRAKT_USE_ROLLING_DOWNLOAD and sickbeard.USE_TRAKT:
+            # Checking if trakt and rolling_download are enable because updateWantedList()
+            # doesn't do the distinction between a failuire and being not activated(Return false)
+            if not sickbeard.traktRollingScheduler.action.updateWantedList(showObj.indexerid):
+                errors.append("Unable to force an update on wanted episode")
+
+        ui.notifications.message('<b>%s</b> has been %s' % (showObj.name,('resumed', 'paused')[showObj.paused]))
+        return self.redirect("/home/displayShow?show=" + show)
+        
     def deleteShow(self, show=None, full=0):
 
         if show is None:
@@ -1628,8 +1682,14 @@ class Home(WebRoot):
         return self.redirect("/home/displayShow?show=" + str(showObj.indexerid))
 
 
-    def updateKODI(self, showName=None):
-
+    def updateKODI(self, show=None):        
+        showName=None
+        
+        if show:
+            showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(show))
+            if showObj:
+                showName=urllib.quote_plus(showObj.name.encode('utf-8'))
+            
         # only send update to first host in the list -- workaround for kodi sql backend users
         if sickbeard.KODI_UPDATE_ONLYFIRST:
             # only send update to first host in the list -- workaround for kodi sql backend users
@@ -1641,8 +1701,11 @@ class Home(WebRoot):
             ui.notifications.message("Library update command sent to KODI host(s): " + host)
         else:
             ui.notifications.error("Unable to contact one or more KODI host(s): " + host)
-        return self.redirect('/home/')
-
+            
+        if showObj:
+            return self.redirect('/home/displayShow?show=' + str(showObj.indexerid))
+        else:
+            return self.redirect('/home/')
 
     def updatePLEX(self):
         if notifiers.plex_notifier.update_library():
@@ -1914,7 +1977,11 @@ class Home(WebRoot):
         def getEpisodes(searchThread, searchstatus):
             results = []
             showObj = sickbeard.helpers.findCertainShow(sickbeard.showList, int(searchThread.show.indexerid))
-
+            
+            if not showObj:
+                logger.log('No Show Object found for show with indexerID: ' + searchThread.show.indexerid, logger.ERROR)
+                return results
+            
             if isinstance(searchThread, sickbeard.search_queue.ManualSearchQueueItem):
                 results.append({'show': searchThread.show.indexerid,
                                 'episode': searchThread.segment.episode,
@@ -3643,7 +3710,7 @@ class ConfigGeneral(Config):
                     handle_reverse_proxy=None, sort_article=None, auto_update=None, notify_on_update=None,
                     proxy_setting=None, proxy_indexers=None, anon_redirect=None, git_path=None, git_remote=None,
                     calendar_unprotected=None, debug=None, no_restart=None, coming_eps_missed_range=None,
-                    display_filesize=None, filter_row=None, fuzzy_dating=None, trim_zero=None, date_preset=None, date_preset_na=None, time_preset=None,
+                    filter_row=None, fuzzy_dating=None, trim_zero=None, date_preset=None, date_preset_na=None, time_preset=None,
                     indexer_timeout=None, download_url=None, rootDir=None, theme_name=None,
                     git_reset=None, git_username=None, git_password=None, git_autoissues=None):
 
@@ -3694,7 +3761,6 @@ class ConfigGeneral(Config):
         sickbeard.WEB_USERNAME = web_username
         sickbeard.WEB_PASSWORD = web_password
 
-        sickbeard.DISPLAY_FILESIZE = config.checkbox_to_value(display_filesize)
         sickbeard.FILTER_ROW = config.checkbox_to_value(filter_row)
         sickbeard.FUZZY_DATING = config.checkbox_to_value(fuzzy_dating)
         sickbeard.TRIM_ZERO = config.checkbox_to_value(trim_zero)
@@ -3854,8 +3920,7 @@ class ConfigSearch(Config):
 
         sickbeard.RANDOMIZE_PROVIDERS = config.checkbox_to_value(randomize_providers)
 
-        sickbeard.DOWNLOAD_PROPERS = config.checkbox_to_value(download_propers)
-        config.change_DOWNLOAD_PROPERS(sickbeard.DOWNLOAD_PROPERS)
+        config.change_DOWNLOAD_PROPERS(download_propers)
 
         sickbeard.CHECK_PROPERS_INTERVAL = check_propers_interval
 
@@ -4984,10 +5049,6 @@ class ErrorLogs(WebRoot):
                 with ek.ek(codecs.open, *[logger.logFile + "." + str(i), 'r', 'utf-8']) as f:
                         data += Get_Data(minLevel, f.readlines(), len(data), regex, logFilter, logSearch, maxLines)
 
-        
-
-               
-
         result = "".join(data)
 
         t.logLines = result