Greasy Fork is available in English.

Barefoot Essentials for GOG.com

Enhances the GOG.com website

اعتبارا من 04-11-2014. شاهد أحدث إصدار.

// ==UserScript==
// @name           Barefoot Essentials for GOG.com
// @namespace      http://userscripts.org/users/274735
// @description    Enhances the GOG.com website
// @include        https://www.gog.com/*
// @include        http://www.gog.com/*
// @require        https://ajax.googleapis.com/ajax/libs/jquery/1.11.0/jquery.min.js
// @version        2.1.9
// @grant          GM_getValue
// @grant          GM_setValue
// @grant          GM_deleteValue
// @grant          GM_xmlhttpRequest
// ==/UserScript==

/*-- constants --*/
var version = '2.1.9'
var default_prev_version = '2.0'	// On first use, all versions after this will be shown in the chanelog
var branch = 'Barefoot_Monkey/GreaseMonkey'	// author/platform. If you fork this then please change this as appropriate. Please also change the @namespace to your own.

var config = {
	'Changelog': [
	],
	'Navigation bar': [
		{ type: 'choice', options: ['normal', 'new and colourful', 'new with old logo', 'old-school'], def: 'new and colourful', key: 'navbar-style', label: 'Navigation bar style' },
		{ type: 'choice', options: ['fixed (normal)', 'absolute'], def: 'fixed (normal)', key: 'navbar-position', label: 'Navigation bar position' },
		{ type: 'choice', options: ['normal', 'red', 'green'], def: 'red', key: 'navbar-notification-style', label: 'Notification style' },
		{ type: 'range', min: 0, max: 1, step: 0.1, def: 1, key: 'navbar-opacity', label: 'Fadeout opacity' },
		{ type: 'multibool', options: { 'games': true, 'movies': true}, key:'navbar-show-sections', label:'Show sections in navigation bar'},
		{ type: 'multibool', options: { 'game updates': true, 'forum replies': true, 'new messages': true }, key:'navbar-show-alerts', label:'Show notifications on the navigation bar'},
		// { type: 'choice', options: ['last visited', 'list', 'shelf'], def: 'last visited', key: 'navbar-library-default', label: 'Default library view' },
		{ type: 'bool', def: true, key: 'navbar-library-links', label: 'Direct links to your game list and shelf in the menu' },
		{ type: 'choice', options: ['top', 'bottom'], def: 'top', key: 'navbar-essentials-position', label: 'Position of "Essentials" menu item' },
	],
	'Forums': [
		{ type: 'bool', def: true, key: 'forum-username-link', label: 'make usernames link to GOGWiki' },
		{ type: 'bool', def: true, key: 'forum-avatar-zoom', label: 'click on avatars to view at original size' },
		{ type: 'bool', def: true, key: 'forum-title-settings', label: 'click on own title to change forum settings' },
		{ type: 'bool', def: true, key: 'forum-move-edit-note', label: 'restyle "post edited" note on edited posts' },
		{ type: 'choice', options: ['normal', 'distinct', 'clear'], def: 'distinct', key: 'forum-quotation-style', label: 'quotation style' },
		{ type: 'bool', def: true, key: 'forum-post-preview', label: 'preview on post/reply window' },
		{ type: 'bool', def: true, key: 'forum-quick-post', label: 'enable quick post' },
		{ type: 'bool', def: true, key: 'forum-hide-spoilers', label: 'hide spoilers' },
		{ type: 'bool', def: true, key: 'forum-bold-text', label: 'enhanced bold text' },
		{ type: 'choice', options: ['leave as they are', 'group and collapse', 'group and expand', 'group below other topics', 'hide'], def: 'group below other topics', key: 'forum-group-giveaways', label: 'giveaway topics (page refresh required)' },
		{ type: 'bool', def: false, key: 'forum-show-hover-elements', label: 'Always show on-hover elements in forum posts', comment: 'Online/Offline status, PM button, and post link number' },
		{ type: 'bool', def: false, key: 'forum-collapsible-footer', label: 'Collapsible footer in forum topics' },
		{ type: 'bool', def: false, key: 'bugfix-collapsible-footer', label: 'Smaller spacer before the footer', comment: 'Corrects the large space which appears above the footer in some browsers. Leave this unchecked if you do not experience this problem.' },
	],
	'Misc': [
		{ type: 'choice', options: ['light'], def: 'light', key: 'BE-style', label: 'Barefoot Essentials style' },
		{ type: 'choice', options: ['current favicon', 'old favicon'], def: 'old favicon', key: 'favicon', label: 'Favicon' },
		{ type: 'bool', def: true, key: 'gamecard-show-descriptions', label: 'Automatically expand game and movie descriptions' },
		{ type: 'bool', def: true, key: 'gamecard-gogwiki-link', label: 'Gamecards: Show GOGWiki link' },
		{ type: 'bool', def: true, key: 'library-always-show-count', label: 'Always show total games/movies in library' },
		{ type: 'bool', def: false, key: 'catalogue-hide-owned', label: 'Hide owned games and movies in catalogue' },
		{ type: 'bool', def: true, key: 'catalogue-show-hide-owned-toggle', label: 'Add "hide owned" toggle switch to catalogues' },
	],
	'Share on GOGWiki': [
		// { type: 'multibool', options: { 'games': true, 'movies': true, 'wishlist': true, 'birth date': true}, key:'share-on-gogwiki', label:'Send following information to GOGWiki when syncinc'},
	]
}

var changelog = [
	{
		version: '2.1.9',
		date: '2014-11-04',
		changes: [
			"New feature: collapsible footer (default: off)",
			"New feature: Fix the huge space that appears before the footer in forum topics on some browsers (default: off)",
			"Fixed: \"hide owned games and movies in catalogue\" not working when filtering games by genre",
			"Fixed: \"Click here to sync with GOGWiki again\" not working reliably",
			"Account menu now get a \"gifts\" button alongside the \"shelf\" and \"list\" buttons. The blank icon is temporary - I'll add an icon later.",
		]
	},
	{
		version: '2.1.8',
		date: '2014-10-04',
		changes: [
			"New feature: toggle visibility of owned games/movies directly from the catalogue",
			"Restored feature: Hide specific navigation bar notifications (new/updated games, forum replies, new private messages)",
		]
	},
	{
		version: '2.1.7',
		date: '2014-09-26',
		changes: [
			"New feature: option to hide owned games and movies from the catalogue",
			"New feature: option to always show Online/Offline status, PM button, and post link number in forum topics",
		]
	},
	{
		version: '2.1.6',
		date: '2014-09-25',
		changes: [
			"Improved styling for absolute navigation bar position",
			"Fixed old-school navigation bar style when logged out"
		]
	},
	{
		version: '2.1.5',
		date: '2014-09-20',
		changes: [
			"New feature: Navigation bar position, which allows you to pin the navigation bar to the top of the page",
			"Compatiblity with Violent Monkey and Tampermonkey restored",
		]
	},
	{
		version: '2.1.4',
		date: '2014-09-17',
		changes: [
			"Fixed incorrect URLs on account menu shelf/list icons",
			"Restored feature: always show total games/movies in library",
		]
	},
	{
		version: '2.1.3',
		date: '2014-09-17',
		changes: [
			"Quotes are properly styled inside post previews",
			"Post previews no longer obstruct the button to attach images to posts",
		]
	},
	{
		version: '2.1.2',
		date: '2014-09-17',
		changes: [
			"Fixed: forum enhancements not being applied when the URL points to any specific post",
			"Fixed: account menu library icons not appearing in HTTPS pages",
			"\"Automatically expand descriptions\" feature now applies to both movie and game cards",
		]
	},
	{
		version: '2.1.1',
		date: '2014-09-17',
		changes: [
			'Fixed: In the settings, checkboxes initially set to default instead of actual setting',
			'Feature restored: extra-bold bold text in forums',
			'Feature restored: quick reply',
			'Feature restored: spoilers',
			'New option for green account notifications in the navigation bar',
			'Position of the "Essentials" account menu item can now be either "top" or "bottom"',
			'Fixed: duplicating the summary paragraph when expanding gamecard descriptions',
		]
	},
	{
		version: '2.1',
		date: '2014-09-16',
		changes: [
			'Compatible with 2014 GOG website redesign.',
			'COLOUR',
			'Complete rewrite of the Barefoot Essentials UI. I think it looks better than before, but there will be additional themes for this too.',
			'*All* features are now optional, and can be toggled without requiring a page refresh.',
			'The navigation bar links "GAMES" and "MOVIES" can be hidden if you so desire.',
			'The old favicon is back!',
			'Alternate navigation bar styles.',
			'Direct links to your shelf and game list in the account menu are now presented as compact icons.',
			'Descriptions on gamecards can be automatically expanded.',
			'The feature to suppress notification for new/updated games, forum replies or messages is temporarily removed. I\'ll re-add this when I can.',
			'The feature to force your library to display your game count even when there are no games new/updates is also temporarily removed',
		]
	},
	{
		version: '2.0',
		date: '2014-05-11',
		changes: [
			'New feature: Discrete "Post edited" notes in forum (enabled by default).',
			'Keyboard shortcuts are now displayed next to the quick post area.',
			'The configuration now uses HTML 5 input boxes. What this means is that mouse-users can edit numbers without needing to reach for their keyboards.',
			'Fixed minor css issue from previous version, which had been causing the Edit / Add new post page to be slightly larger than the window displaying it.'
		]
	},
	{
		version: '1.11',
		date: '2014-03-27',
		changes: [
			'In Reply/New Post/New Topic popup windows, the preview area now occupies all available space instead of having a fixed height.'
		]
	},
	{
		version: '1.10',
		date: '2013-12-11',
		changes: [
			"New feature: Ability to group together giveaway threads on forums.",
			"(1.10.1) The experimental \"Always display shelf/list game count \" feature should now be working correctly. This option requires browser features added in Firefox 14, Chrome 26 and Opera 15, and will have no effect on browsers older than that.",
			"(1.10.2) Fixed: game count on shelf was incorrectly rounding up to the nearest 5 when \"Always display shelf/list game count \" is enabled.",
			"(1.10.3) Syncing to GOGWiki now takes into account the sort-order selection in your wishlist",
			"(1.10.4) In the forums you can now click on your forum title to go to your settings page.",
			"(1.10.5) Fixed issue with some users not being able to sync since GOGWiki's PHP update.",
			"(1.10.6) Added option to share your birthday on GOGWiki. If you have this option enabled when syncing then you will be listed on the wiki's new Special:Birthdays page when it is your birthday.",
			'(1.10.6) "Always display shelf/list game count" is no longer marked as experimental.'
		]
	},
	{
		version: '1.9',
		date: '2013-10-27',
		changes: [
			"Options added to ignore game updates, private messages and/or forum replies.",
			"(All that this does is prevent the \"ignored\" updates from counting toward the number in the alert displayed next to \"My Account\" on the top bar. You will still be able to see the updates in the dropdown menu.)",
			"(1.9.1) Wiki links in game cards now correctly handle titles containing \"en\" dashes",
			"(1.9.2) Fixed: bug causing wishlists sent to gogwiki to be limited to 50 items",
			"(1.9.3) Fixed: notifications not being correctly hidden on pages with delayed top navigation bar",
			"(1.9.4) New feature: On the game list and shelf pages, always display the counter indicating the total games and the number of games new/updated. This feature is off by default and marked as experimental for now because I haven't yet had an opportunity to test when there are updates."
		]
	},
	{
		version: '1.8.1',
		date: '2013-10-09',
		changes: [
			"Cursors to indicate that avatars are clickable",
			'"Full-size" avatars now have a limit of 420 pixels in height, and their z-index is raised so that they don\'t go underneath the bottom bookend thingy on the forum',
			"Now clicking on an avatar will also shrink any currently-enlarged avatars back to normal size"
		]
	},
	{
		version: '1.8',
		date: '2013-10-08',
		changes: [
			"Click on an avatar in the forum to view it full-size",
			"3 forum quote styles are available to choose between",
			"Blank lines in and around spoilers are now trimmed",
			"Deeply-nested spoilers now have significantly less padding"
		]
	},
	{
		version: '1.7.1',
		date: '2013-08-22',
		changes: [
			"Fixed: wishlist images on the wiki.",
			"Fixed: wiki links appear on game pages again.",
			"Thanks to adambiser for the patch to handle the recent changes to gog.com"
		]
	},
	{
		version: '1.7',
		date: '2013-07-21',
		changes: [
			"Spoilers! Hotkey Ctrl+L",
			"Fixed: promo link on gamecards used to be unclickable during promotions."
		]
	},
	{
		version: '1.6',
		date: '2013-05-23',
		changes: [
			"Restyle of forum quotes is now optional.",
			"Fixed: wrong preview text colour in new/edit post window when using light forum style."
		]
	},
	{
		version: '1.5',
		date: '2013-05-10',
		changes: [
			"Shortcut keys from quick posts are now also supported in regular post/edit windows",
			"Quick-posting a reply to another post now properly causes a forum reply notification. Due to technical limitations, only the first quote in a quick post generates a notification.",
			"Submit quick post button is now disabled during post, and has a new \"disabled\" style.",
			"Varius CSS changes. Some fixes to changes in light theme, quoted text in dark theme is now closer to its regular colour, and the preview in post/reply window has been reorganised.",
		]
	},
	{
		version: '1.4',
		date: '2013-04-07',
		changes: [
			"This changelog now automatically appears once after each update so that you know when new features are available. Don't worry - I'll add an option to disable this in a future version in case you'd rather not have it keep appearing.",
			'Quick reply added.',
			"New shortcut: ctrl+space to jump to quick post.",
			"New shortcuts: ctrl+I, ctrl+B, ctrl+U and ctrl+Y in quick post to add [i], [b], [u] and [url] tags. (Y is for hYperlink - all the more obvious letters conflicted with some-or-other browser hotkey.)",
			"New shortcut: ctrl+enter in quick post to submit your post.",
			'Style change to quotes and bold text in forum posts. This change will become optional once in the next version.',
			'New, improved parser for generating better previews. Now handles quotes and nested tags, and points out missing/incorrect closing tags.',
			"Improvements to changelog.",
		]
	},
	{
		version: '1.3',
		date: '2013-02-27',
		changes: [
			'Brand new UI system added, along with this changelog. This looks better, and also allow some functions to be moved out of the over-long "My Account" menu.',
			'Username links are now coloured appropriately when using the light colour scheme',
			'Wishlist is now sent to GOGWiki in alphabetical order',
			'Default settings for hiding the navigation bar have been changed (I found that the sliding effect gets old)',
			'Navigation bar no longer hides when you move the mouse away while typing in the search box',
			'Wiki links no longer disappear, due to better handling of elements that appear only after pageload (thanks, adambiser)',
			'(1.3.2) Live preview when writing a forum post.',
			'(1.3.3) Fixed wiki link on KKnD2 gamecard.',
			'(1.3.5) Inline "Quick Post" feature added.',
			'(1.3.5.2) Changed textbox shrinking behaviour on "Quick Post" to avoid the post button having to be clicked twice.',
			'(1.3.5.3) Fixed ajax regression in wiki sync.'
		]
	}
]


/*-- global variables - don't judge me --*/
var forum_skin = null
var gog_sync_element = null
var sync_status_account = null
var sync_status_shelf = null
var sync_status_list = null
var sync_status_wishlist = null
var sync_status_progress = null
var sync_status_start = null
var sync_status_restart = null
var sync_status_send = null
var sync_status_output = null


/*-- utility functions --*/
function to_css(rules) {
	var text = ''
	for (var i = 0; i+1 < rules.length; i += 2) {
		var selectors = rules[i], declarations = rules[i+1]
		text += selectors + '{'
		for (var j in declarations) {
			text += declarations[j] + ';'
		}
		text += '}'
	}
	return text
}

function cmpVersion(a, b) {
	var i, cmp, len, re = /(\.0)+[^\.]*$/;
	a = (a + '').replace(re, '').split('.');
	b = (b + '').replace(re, '').split('.');
	len = Math.min(a.length, b.length);
	for( i = 0; i < len; i++ ) {
	    cmp = parseInt(a[i], 10) - parseInt(b[i], 10);
	    if( cmp !== 0 ) {
	        return cmp;
	    }
	}
	return a.length - b.length;
}

var popup = {
	show: function(section) {
		var popup = $('aside.BE-popup')
		if (popup.length == 0) {
			popup = $('<aside class="BE-popup"><div>')

			var nav = $('<nav>').appendTo(popup)
			var navlist = $('<ul>').appendTo(nav)

			// close button
			$('<li>')
			.text('Close')
			.click(function() { $('.BE-popup').remove() } )
			.prependTo(navlist)

			// dynamic sections
			for (var section_name in config) {
				$('<li>').appendTo(navlist).text(section_name).click(this.show_section.bind(this, section_name))
			}

			popup.appendTo(document.body)
		}

		this.show_section(section)
	},
	checkbox_change_event: function(key, subkey, e) {
		var setting = settings.get(key)
		setting[subkey] = e.target.checked
		settings.onchange(key)
	},
	change_event: function(key, e) {
		if (e.target.type == 'checkbox') settings.set(key, e.target.checked)
		else settings.set(key, e.target.value)
	},
	show_section: function(section) {
		var popup = $('aside.BE-popup')
		var root = popup.find('>div')
		root.empty()

		popup.find('>nav>ul>li').removeClass('active').each(function() {
			if (this.textContent == section) $(this).addClass('active')
		})

		$('<h1>').text('Barefoot Essentials - '+section).appendTo(root)

		switch (section) {
			case 'Share on GOGWiki':
				root.append(gog_sync_element)
				break
			case 'Changelog': {
				var old_versions = $('<div class="BE-older-changes">').hide()

				changelog.forEach(function(entry) {
					var p = $('<h2>')
					.text("Version " + entry.version)
					.append($('<small>').text(" - released on " + entry.date))
					
					var list = $('<ul>').addClass('BE-changelog')
					
					entry.changes.forEach(function(change) {
						$('<li>').html(change).appendTo(list)
					})
					
					var entry 
					if (entry.version == version || cmpVersion(last_BE_version, entry.version) < 0) {
						root.append(p, list)
					} else {
						old_versions.append(p, list)
					}
				})
				if (old_versions.children().length > 0) {
					var older = $('<p>').append($('<a>').html('older changes&hellip;').click(function() { old_versions.toggle() }))
					root.append(older, old_versions)
				}
				break;
			}
			default:
				var fields = config[section]
				for (var i in fields) {
					var field = fields[i]
					var p = $('<p>')
					$('<label>').text(field.label).appendTo(p)

					switch (field.type) {
						case 'range': {

							var e = $('<input type="range">')
							.attr('min', field.min)
							.attr('max', field.max)
							.attr('step', field.step)
							.val(settings.get(field.key))
							.appendTo(p)

							e.on('input', this.change_event.bind(this, field.key))

							break
						}
						case 'multibool': {
							var group = $('<div class="BE-multibool">')

							var value = settings.get(field.key)
							for (var option in field.options) {
								$('<label>')
								.text(option)
								.prepend(
									$('<input type="checkbox">')
									.prop('checked', value[option])
									.on('change', this.checkbox_change_event.bind(this, field.key, option))
								)
								.appendTo(group)
							}
							group.appendTo(p)
							break
						}
						case 'bool': {
							$('<input type="checkbox">')
							.prop('checked', settings.get(field.key))
							.on('change', this.change_event.bind(this, field.key))
							.appendTo(p)
							break
						}
						case 'choice': {

							var select = $('<select>')
							for (var i in field.options) {
								$('<option>')
								.text(field.options[i])
								.appendTo(select)
							}
							select.val(settings.get(field.key))
							select.appendTo(p)

							select.on('change', this.change_event.bind(this, field.key))

							break
						}
					}

					if (field.comment !== undefined) {
						$('<small>').text(field.comment).appendTo(p)
					}

					p.appendTo(root)
				}
		}

		root.focus()
	}
}



var settings = {
	get: function(key) {
		var setting = this.settings[key]
		if (setting) return setting.value
		else return undefined
	},
	set: function(key, value) {
		var setting = this.settings[key]

		if (setting) {
			if (setting.value != value) {
				setting.value = value
				this.save()

				for (var i in setting.onchange) {
					setting.onchange[i](value)
				}
			}
		}
	},
	save: function() {
		var saved_settings = {}
		for (var key in this.settings) {
			saved_settings[key] = this.settings[key].value
		}
		GM_setValue('settings', JSON.stringify(saved_settings))
	},
	onchange: function(key, callback) {
		var setting = this.settings[key]
		if (setting) {
			if (callback) {
				setting.onchange.push(callback)
				callback(setting.value)
			} else {
				this.save()

				for (var i in setting.onchange) {
					var callback = setting.onchange[i]
					callback(setting.value)
				}
			}
		}
	},

	initialise: function(initial_values, done) {

		var saved_settings = {}
		try {
			var s = GM_getValue('settings')
			if (s !== undefined) saved_settings = JSON.parse(s)
		} catch (exception) {
			console.log(exception)
			GM_deleteValue('settings')
		}

		for (var section_name in initial_values) {
			for (var i in initial_values[section_name]) {
				var item = initial_values[section_name][i]

				var setting = {
					onchange: [],
					value: (saved_settings[item.key] !== undefined) ? saved_settings[item.key] : item.def
				}
				
				if (item.type == 'multibool' && setting.value === undefined) setting.value = item.options
				this.settings[item.key] = setting
			}
		}

		if (done) done()
	},

	settings: {}
}



function detect_forum_skin() {
	if (document.head.querySelector('link[href*="forum_carbon"]')) forum_skin = 0
	else forum_skin = 1
}


/*-- Quick post/post preview --*/
function submit_quick_post() {
	if (location.pathname == "/forum/ajax/popUp") {
		$('.kontent>.submit>div.gog_btn:first-child').click()
	} else {
		var post_text_e = $('.quick_post textarea')
	
		if (post_text_e.length < 1 || post_text_e.val() == '') return
	
		var post_text = post_text_e.val()
		var reply_to = post_text.match(/\[quote_([0-9]+)\]/)
		var reply_to_pid = (reply_to === null) ? undefined : reply_to[1]
					
		$('.submit-quick-post')[0].disabled = true
					
		// submit the post
		$.ajax({
			type:"POST",
			url:"/forum/ajax",
			timeout:15000,
			data:{
				a:"addPost",
				f:$("#f").val(),
				f_arr:$("#f_arr").val(),
				w:$("#w").val(),
				pid:reply_to_pid,
				text:post_text,
				added_images_ids:"",
				added_images_names:"",
				kap:undefined,
				guest_name:undefined,
				btn:"0"
			}
		})
		.done(function(data, textStatus, jqXHR){
			var response = JSON.parse(data)
			if (response.error) {
				alert("A problem occurred while submitting this Quick Post. Try again using a regular post.")
				console.log(data)
				$('.submit-quick-post')[0].disabled = false
			} else {
				window.location = response.result
			}
		})
		.fail(function(data, textStatus, jqXHR){
			$('.submit-quick-post')[0].disabled = false
			alert("A problem occurred while submitting this Quick Post. Try again using a regular post.")
		})
	}
}
function tag_input_text(input, begin_tag, end_tag) {
	var start = input.selectionStart, end = input.selectionEnd
	input.value = (
		input.value.substring(0, start) 
		+ begin_tag
		+ input.value.substring(start, end)
		+ end_tag
		+ input.value.substring(end)
	)
						
	input.selectionStart = start + begin_tag.length
	input.selectionEnd = end + begin_tag.length
}

function parse_node(node) {
	switch (node.nodeType) {
		case 3: { // text node
			return node.nodeValue
		}
		case 1: { // element
			
			var result = ""
			var after = ""
			
			if (node.tagName == 'BR') return "\n"
			else if (node.tagName == 'DIV') {
				if (node.classList.contains('post_text_c')) {
				} else return ""
			} else if (node.tagName == 'A') {
				result = "[url="+encodeURI(node.getAttribute('href'))+"]"
				after = "[/url]"
			} else if (node.tagName == 'I') {
				result = "[i]"
				after = "[/i]"
			} else if (node.tagName == 'SPAN') {
				if (node.classList.contains('podkreslenie')) {
					result = "[u]"
					after = "[/u]"
				} else if (node.classList.contains('bold')) {
					result = "[b]"
					after = "[/b]"
				} else return ""
			} else return ""
			
			
			var child = node.firstChild
			while (!!child) {
				result += parse_node(child)
				child = child.nextSibling
			}
			
			return result + after
		}
	}
	return ""
}

function parse_post(post) {
	return parse_node(post.find('.post_text_c')[0])
}
function post_keydown_handler(event) {
	if (event.ctrlKey && !event.repeat) {
		switch (event.which) {
			case 66:
			case 98: tag_input_text(this, '[b]', '[/b]'); break
					
			case 85:
			case 117: tag_input_text(this, '[u]', '[/u]'); break
					
			case 73:
			case 105: tag_input_text(this, '[i]', '[/i]'); break
					
			case 76:
			case 108: tag_input_text(this, '\n\n[spoiler]\n\n\n\n', '\n\n\n\n[/spoiler]\n\n'); break
					
			case 81:
			case 113: tag_input_text(this, '[quote]', '[/quote]'); break
					
			case 89:
			case 121: {
				var selected_text = this.value.substring(this.selectionStart, this.selectionEnd)
						
				var url = prompt("Enter the URL for the link", selected_text)
				if (!!url) {
					tag_input_text(this, '[url='+encodeURI(url)+']', '[/url]')
				}
				break
			}
					
			case 13: {
				submit_quick_post();
				break
			}
					
			default: return true
		}
		
		event.stopPropagation();
		show_preview()
				
		return false
	}
}
function post_preview_html(source) {
	var tokenexp = /\[\/?(?:[ibu]|(?:url(?:=[^\n\]]*)?|quote(?:_[0-9]*)?))\]/g
	
	var text_tokens = source.split(tokenexp)
	if (!text_tokens) text_tokens = []
	var tag_tokens = source.match(tokenexp)
	if (!tag_tokens) tag_tokens = []
	
	var text_i = 0, tag_i = 0
	var tag_stack = []
	var preview = ""
	var top_tag = null
	var ignore_first_linebreak = false;
	
	while (true) {
		if (text_i < text_tokens.length) {
		
			var text_token = text_tokens[text_i++]
			.replace(/&/g, '&amp;')
			.replace(/\>/g, '&gt;')
			.replace(/\</g, '&lt;')
			.replace(/\n/g, '<br/>')
			
			preview += ignore_first_linebreak ? text_token.replace(/^<br\/>/, '') : text_token
			
		} else break;
		
		if (tag_i < tag_tokens.length) {
		
			var tag = tag_tokens[tag_i++].match(/^\[(\/?)([^=_\]]*)(?:[=_](.*))?\]/)			
			if (tag[2] === undefined) continue
			
			ignore_first_linebreak = false
			
			if (tag[1] == '/') {
			
				if (!!top_tag) {
					preview += top_tag.closing
					
					if (top_tag.tag == 'quote') ignore_first_linebreak = true
					
					if (tag[2] != top_tag.tag)
					preview += '<span class="syntax-warning">[/'+top_tag.tag+']</span>';
					
					top_tag = tag_stack.pop()
				}
				
			} else {
				tag_stack.push(top_tag)
				switch (tag[2]) {
					case 'i': {
						preview += '<i>'
						top_tag = {
							tag: tag[2],
							closing: '</i>'
						}
					} break;
					case 'b': {
						preview += '<b>'
						top_tag = {
							tag: tag[2],
							closing: '</b>'
						}
					} break;
					case 'u': {
						preview += '<u>'
						top_tag = {
							tag: tag[2],
							closing: '</u>'
						}
					} break;
					case 'quote': {
						preview += '<blockquote>';
						ignore_first_linebreak = true
						top_tag = {
							tag: tag[2],
							closing: '</blockquote>'
						}
					} break;
					case 'url': {
						preview +=  (tag[3] === undefined)? '<a href="">' : ('<a href="'+encodeURI(tag[3])+'">');
						top_tag = {
							tag: tag[2],
							closing: '</a>'
						}
					} break;
					default: top_tag = tag_stack.pop();
				}
			}
		}
	}

	while (!!top_tag) {
		preview += top_tag.closing
		preview += '<span class="syntax-warning">[/'+top_tag.tag+']</span>';
		top_tag = tag_stack.pop()
	}

	return preview
}
function show_preview() {
	$('.BE-preview').html(post_preview_html($('.quick_post textarea, form#f_text>textarea#text').val()))
}

/*-- Feature functions --*/

function feature_navbar_style() {
	function on_update(value) {
		switch (value) {
			case 'new and colourful':
				style.text(to_css([
					'nav.top-nav', [
						'background: linear-gradient(to bottom, #E5E5E5, #DEDEDE) no-repeat scroll 1080px 0px transparent',
					],
					'.top-nav__inner:after', [
						'display: none',
					],
					'.top-nav__logo:hover:before', [
						'background-color: #000',
						'box-shadow: 0px 0px 5px -1px #0C0A32'
					],
					'.top-nav__logo:before', [
						'display: block',
						'z-index: 0',
						'background: transparent url()',
						'height: 42px',
						'margin: 7px 0',
						'border-radius: 3px'
					],
					'.top-nav__logo', [
						'background: rgba(0, 0, 0, 0.45) url();',
					],
					'.top-nav__logo:hover', [
					],
				]))
				break
			case 'new with old logo':
				style.text(to_css([
					'nav.top-nav', [
						'background: linear-gradient(to bottom, #E5E5E5, #DEDEDE) no-repeat scroll 1080px 0px transparent',
					],
					'.top-nav__inner:after', [
						'display: none',
					],
					'.top-nav__home', [
						'margin-left: -45px',
						'padding: 0',
						'background: linear-gradient(to bottom, #E5E5E5, #DEDEDE)',
					],
					'.top-nav__logo', [
						'background: url(https://secure.gog.com/www/default/-img/_nav.38808059.png) no-repeat 0 -54px',
						'width: 98px',
						'height: 22px',
						'margin-top: 19px'
					]
				]))
				break
			case 'old-school':
				style.text(to_css([
					'.top-nav__dd-items', [
						'background-color: #C3C3C3'
					],
					'.is-anonymous .top-nav__login', [
						'background: transparent',
					],
					'.top-nav__login-inner', [
						'height: 47px',
					],
					'.top-nav__login-content', [
						'border-right: none',
					],
					'body > nav.top-nav, body .universe nav.top-nav', [
						'background: linear-gradient(#c3c3c3, #a2a2a2) no-repeat scroll 1080px 0px transparent',
					],
					'.top-nav__inner:after', [
						'display: none',
					],
					'.top-nav__home', [
						'margin-left: -45px',
						'padding: 0',
					],
					'.top-nav__home, .top-nav-extras-search-holder, .top-nav__extras, nav.top-nav, .top-nav__item', [
						'background: transparent',
						'height: 47px',
					],
					'.top-nav__logo', [
						'background: url(https://secure.gog.com/www/default/-img/_nav.38808059.png) no-repeat 0 -54px',
						'width: 98px',
						'height: 22px',
						'margin-top: 15px'
					],
					'.top-nav__item', [
						'line-height: 47px',
						'padding: 0 20px',
						'width: 17%',
						'border-right: none',
						'text-transform: lowercase'
					],
					'.top-nav__item:first-child', [
						'border-left: none',
					],
					'.top-nav__inner:before, .top-nav__container, .top-nav__inner', [
						'background: linear-gradient(#c3c3c3, #a2a2a2)',
						'background-repeat: repeat-x;',
					],
					'.top-nav__item ._dropdown__toggle:hover, .top-nav__item:hover:not(.is-expanded):not(.top-nav__login)', [
						'background: #C3C3C3',
					],
					'.top-nav__item.is-active', [
						'background: #A1A1A1',
					],
					'.top-nav__item.is-expanded', [
						'padding: 0',
					],
					'.top-nav__item.is-expanded>a._dropdown__toggle', [
						'background: #C3C3C3',
						'padding: 0 20px'
					],
					'.top-nav__dd-item', [
						'background: #C3C3C3',
						'border-top: none !important',
					],
					'.top-nav__dd-item:hover', [
						'background: #B3B3B3',
					],
					'.top-nav__cart, .top-nav__social, .top-nav-extras-search-holder', [
						'line-height: 47px',
					]
				]))
				break
			default:
				style.text(to_css([
					'nav.top-nav', [
						'background: linear-gradient(to bottom, #E5E5E5, #DEDEDE) no-repeat scroll 1080px 0px transparent',
					],
					'.top-nav__inner:after', [
						'display: none',
					],
				]))
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('navbar-style', on_update)
}

function feature_navbar_opacity() {
	function on_update(value) {
		style.text(to_css([
			'body>nav.top-nav', [
				'opacity: '+value,
				'transition: opacity 0.5s linear'
			],
			'body>nav.top-nav:hover', [
				'opacity: 1',
			],
		]))
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('navbar-opacity', on_update)
}

function feature_notification_style() {
	function on_update(value) {
		switch (value) {
			case 'red':
				style.text(to_css([
					'.top-nav__item-count', [
						'background: #BF0B0B',
						'padding-bottom: 1px',
					],
				]))
				break;
			case 'green':
				style.text(to_css([
					'.top-nav__item-count', [
						'background: #008500',
						'padding-bottom: 1px',
					],
				]))
				break;
			default:
				style.text('')
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('navbar-notification-style', on_update)
}

function feature_BE_style() {
	function on_update(value) {
		switch (value) {
			default:
				style.text(to_css([
					'.BE-error', [
						'color: #B80000',
						'font-weight: bold',
					],
					'.BE-in-progress', [
						'color: #808000',
						'font-weight: bold',
					],
					'.BE-success', [
						'color: #006000',
						'font-weight: bold',
					],
					'.BE-sync-progress>p', [
						'margin: 1em 0',
					],
					'.BE-sync-progress p>span:nth-child(1)', [
						'display: inline-block',
						'text-align: right',
						'margin-right: 0.5em',
						'min-width: 25%',
					],
					'.BE-popup h2', [
						'font-size: 14px',
						'font-weight: bold',
						'margin: 1.5em 0 0.5em',
					],
					'.BE-popup .BE-changelog + p', [
						'margin: 1em',
						'font-style: italic',
						'-moz-user-select: none',
						'user-select: none',
					],
					'.BE-older-changes', [
						'padding-top: 1px',
					],
					'.BE-changelog li:before', [
						'content: "•"',
						'margin-left: -1em',
						'position: absolute',
					],
					'.BE-changelog li', [
						'margin: 0.3em 0',
						'display: block',
						'padding-left: 1em',
						'line-height: 1.4',
						'position: relative',
					],
					'.BE-changelog', [
						'list-style: disc inside none',
						'font-size: 12px',
						'display: block',
						'margin-right: 2em',
					],
					'.BE-popup a', [
						'cursor: pointer',
						'color: blue',
					],
					'.BE-popup a:hover', [
						'text-decoration: underline'
					],
					'.BE-popup input[type=checkbox]', [
						'width: auto',
						'margin: 2px 0 0 6px',
					],
					'.BE-popup .BE-multibool input[type=checkbox]', [
						'vertical-align: middle',
						'margin: 0 0.5em 0 0',
						'line-height: 1'
					],
					'.BE-popup .BE-multibool label', [
						'float: none',
						'width: auto',
						'text-align: left',
						'line-height: 1',
						'margin-bottom: 0.2em'
					],
					'.BE-popup .BE-multibool', [
						'overflow: hidden',
						'padding: 0 0 0 0.5em',
						'border-left: 1px solid #676767',
						'margin-top: 5px',
					],
					'.BE-popup input[type=range]::-moz-focus-outer', [
						'border: none',
						'border-right: 2px solid #808080',
						'border-left: 2px solid #808080',
					],
					'.BE-popup input[type=range]', [
						'padding: 0 4px',
						'margin-top: 2px',
						'border: none',
					],
					'.BE-popup select', [
						'border: 1px solid #808080',
						'font-family: "Lucida Grande",Arial,Verdana,sans-serif',
					],
					'.BE-popup select, .BE-popup input, .BE-popup .BE-multibool', [
						'font-size: 11px',
						'width: 20em',
						'max-width: 70%',
						'box-sizing: border-box',
					],
					'.BE-popup p small', [
						'display: block',
						'font-style: italic',
						'clear: both',
					],
					'.BE-popup p', [
						'margin: 0.5em 0',
						'overflow: hidden',
						'padding-bottom: 2px',
					],
					'.BE-popup label', [
						'float: left',
						'display: block',
						'width: 40%',
						'clear: both',
						'text-align: right',
						'margin-right: 1em',
						'line-height: 1.8em',
						'cursor: default',
						'-moz-user-select: none'
					],
					'.BE-popup h1', [
						'font-size: 14pt',
						'font-weight: normal',
						'padding: 0.5em 0 .3em 2em',
						'margin: 0 0 0.7em',
						'line-height: normal',
						'border-bottom: 1px solid #676767',
					],
					'.BE-popup', [
						'background: #E1E1E1',
						'width: 850px',
						'position: fixed',
						'top: 10%',
						'height: 85%',
						'right: calc(50% - 425px)',
						'z-index: 600',
						'box-shadow: 1px 1px 10px 0 black',
						'box-sizing: border-box',
						'display: flex',
						'font-size: 11px',
						'font-family: "Lucida Grande",Arial,Verdana,sans-serif',
						'color: #212121',
					],
					'.BE-popup>div', [
						'flex: 1 1 auto',
						'padding: 1em',
						'overflow: auto',
					],
					'.BE-popup>nav>ul>li:hover', [
						'color: inherit',
					],
					'.BE-popup>nav>ul>li+li', [
						'border-top: 1px solid #676767',
					],
					'.BE-popup>nav>ul>li.active', [
						'background: #E1E1E1',
						'color: #4A4A4A'
					],
					'.BE-popup>nav>ul>li:not(.active):hover', [
						'color: #fff',
						'text-shadow: 1px 1px 0px black',
						'background: #606060',
					],
					'.BE-popup>nav>ul>li', [
						'padding: 1em 2em',
						'cursor: pointer',
						'line-height: 1',
					],
					'.BE-popup>nav>ul', [
						'margin: 0',
						'padding: 0',
						'display: block',
					],
					'.BE-popup>nav', [
						'all: unset',
						'background: #4A4A4A',
						'color: #E1E1E1',
						'font-size: 11px',
						'font-family: "Lucida Grande",Arial,Verdana,sans-serif',
						'min-width: 134px',
					]
				]))
				break;
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('BE-style', on_update)
}

function feature_cart_style() {
	function on_update(value) {
		switch (value) {
			case 'green':
				style.text(to_css([
					'.top-nav__count', [
						'background-color: #008500',
						'font-size: 10px',
						'border-radius: 2px',
						'vertical-align: middle',
						'line-height: 1',
						'color: #FFF',
						'padding: 3px 3px 3px 4px',
						'margin-bottom: 4px',
					]
				]))
				break;
			default:
				style.text('')
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('navbar-cart-style', on_update)
}


/*-- sync functions --*/

	function shelf_url(page, order, timestamp) { return "https://www.gog.com/account/ajax?a=gamesShelfMore&p="+page+"&s="+order+"&q=&t="+timestamp }
	function list_url(page, order, timestamp) { return "https://www.gog.com/account/ajax?a=gamesListMore&p="+page+"&s="+order+"&q=&t="+timestamp }
	function wishlist_url(page, order, timestamp) { return "https://www.gog.com/account/ajax?a=wishlistSearch&p="+page+"&s="+order+"&q=&t="+timestamp }


	function get_account_information(job) {
		sync_status_account.attr('class', 'BE-in-progress').text("Starting...")
		GM_xmlhttpRequest({
			method: "GET",
			url:'https://www.gog.com/account/settings',
			context: job,
			onload:function(response) {
				var job = this.context
				var html = $(response.responseText.replace(/src=/g, "alt="))


				job.avatar_url = html.find('.avatar').attr('alt')
				job.gogname = html.find('.nickname').text().trim()
				job.days = html.find('.days span').text()

				job.birthday = 0.001 * Date.parse(
					html.find('#accountEditBdaySpan').text() + " " + html.find('#accountEditBmonthSpan').text() + " " + html.find('#accountEditByearSpan').text()
					+ ' GMT'
				)

				sync_status_account.attr('class', 'BE-success').text("Done")
				this.context.count_down()
			},
			onerror:function(response) {
				sync_status_account.attr('class', 'BE-error').text("Error").addClass('error')
				this.context.errors = true
				this.context.count_down()
			}
		})
	}

	function getgame(id, games) {
		var game = games[id]
		if (game === undefined) {
			game = {id:id}
			games[id] = game
		}
		return game
	}

	function write_fragment(fragment, element) {
		switch (typeof(fragment)) {
			case 'string': {
				element.append(fragment)
				break;
			}
			case 'object': {
				if (fragment instanceof Array) {
					for (var i = 0; i < fragment.length; i += 1) {
						write_fragment(fragment[i], element)
					}
				} else {
					var link = fragment.link
					var text = fragment.text
					if (link && text) element.append(
						$('<a>').attr('href', link).attr('target', '_blank').text(text)
					)
				}
				break;
			}
		}
	}
	function write(paragraphs, container) {
		for (var i = 0; i < paragraphs.length; i += 1) {
			var p = $('<p>').appendTo(container)
			write_fragment(paragraphs[i], p)
		}
	}

	function purge_user_page(job) {
		// status_purging.text("In Progress")
		GM_xmlhttpRequest({
			url:'http://www.gogwiki.com/wiki/Special:GOGUser/'+escape(job.gogname)+'/purge',
			method:'POST',
			// data: $.param(data),
			// headers: { "Content-Type": "application/x-www-form-urlencoded" },
			onload: function(response) {
				// status_purging.text("Done")
			},
			onerror:function() {
				// status_purging.text("Error (not serious)").addClass('error')
			}
		})
	}

	function send_to_gogwiki(job) {
		sync_status_send.text("Sending...")

		var data = {
			version: version,
			branch: branch,
			games: [],
			wishlist: job.wishlist,
			avatar_url: job.avatar_url,
			birthday: job.birthday,
			days: job.days,
			gogname: job.gogname
		}
		for  (var i in job.games) {
			data.games.push(job.games[i])
		}

		GM_xmlhttpRequest({
			url:'http://www.gogwiki.com/wiki/Special:GOGSync',
			method:'POST',
			context: job,
			data: $.param({data: JSON.stringify(data)}),
			headers: { "Content-Type": "application/x-www-form-urlencoded" },
			onload: function(response) {
				sync_status_send.text("Done").attr('class', 'BE-success')
				purge_user_page(this.context)
				try {
					sync_status_output.empty()
					write(JSON.parse(response.responseText), sync_status_output)
					// sync_status_output.text(JSON.parse(response.responseText))
				} catch (exception) {
					sync_status_output.html("<p>Oops - it looks like some kind of server error happened. Your data was sent but we cannot tell whether GOGWiki stored it correctly.</p>")
				}

				this.context.count_down()
			},
			onerror:function(response) {
				sync_status_send.text("Error").attr('class', 'BE-error')
				this.context.count_down()
				response.erros = true
			}
		})
	}

	function collect_wishlist(job) {
		sync_status_wishlist.text("Starting...").attr('class', 'BE-in-progress')
		var position = 0

		function read_page(order, page, job) {
			var r = GM_xmlhttpRequest({
				method: "GET",
				url: wishlist_url(page, order, job.timestamp),
				context: job,
				onload:(function(order, response) {
					var job = this
					var json = $.parseJSON(response.responseText)
					position += json.count
					if (json.count > 0) {
						var html = $(json.html.replace(/src=/g, "data-src="))
						html.each(function() {
							var img = $(this).find('.game-item-img')
							var info = $(this).find('.on-wishlist')
							var title = img.attr('alt')
							var wish = {
								thumb_url: img.attr('data-src'),
								sort_title:title.replace(/ ?[\u00AE\u2122]/g, ''),
								title:title.replace(/ ?[\u00AE\u2122]/g, '').replace(/^(.*), The$/, 'The $1'),
								index:info.attr('data-gameindex')
							}
							job.wishlist.push(wish)
						})

						sync_status_wishlist.text(position + " games counted so far")
						read_page(order, page + 1, job)
					} else {
						sync_status_wishlist.text("Done").attr('class', 'BE-success')
						job.count_down()
					}
				}).bind(job, order),
				onerror:function(response) {
					sync_wishlist_shelf.text("Error").attr('class', 'BE-error')
					this.context.errors = true
					this.context.count_down()
				}
			})
		}

		// read the wishlist page
		GM_xmlhttpRequest({
			method: "GET",
			url:"https://www.gog.com/account/wishlist",
			context: job,
			onload:function(response) {
				var html = $(response.responseText.replace(/src=/g, 'data-src='))

				var order = html.find('#account_wishlist_order').val()
				
				read_page(order, 1, this.context)
			},
			onerror:function(response) {
				sync_wishlist_shelf.text("Error").attr('class', 'BE-error')
				this.context.count_down()
			}
		})
	}


	function collect_list(job) {	
		sync_status_list.text("Starting...").attr('class', 'BE-in-progress')
		var position = 0

		function read_page(order, page, job) {
			GM_xmlhttpRequest({
				method: "GET",
				url: list_url(page, order, job.timestamp),
				context: job,
				onload:function(response) {
					var job = this.context
					var json = $.parseJSON(response.responseText)

					if (json.count > 0) {
						var html = $(json.html.replace(/src=/g, "alt="))
						html.each(function() {
							var e = $(this)
							var game_id = e.attr('id')
							if (game_id === undefined) return
							game_id = game_id.replace('game_li_', '')
							var game = getgame(game_id, job.games)

							game.thumb_url = e.find('img.list_image').attr('alt')
							game.title = e.find('.game-title-link').text().trim().replace(/ ?[\u00AE\u2122]/g, '').replace(/^(.*), The$/, 'The $1')
							game.list_pos = position++
						})

						sync_status_list.text(position + " games counted so far")
						read_page(order, page + 1, job)
					} else {
						sync_status_list.text("Done").attr('class', 'BE-success')
						this.context.count_down()
					}
				},
				onerror:function(response) {
					sync_status_list.text("Error").attr('class', 'BE-error')
					this.context.errors = true
					this.context.count_down()
				}
			})
		}

		GM_xmlhttpRequest({
			method: "GET",
			url:"https://www.gog.com/account/games/list",
			context: job,
			onload:function(response) {
				var job = this.context
				var html = $(response.responseText.replace(/src=/g, 'alt='))

				var order = html.find('input#account_list_order').val()
				read_page(order, 1, job)
			},
			onerror:function(response) {
				sync_status_list.text("Error").attr('class', 'BE-error')
				this.context.errors = true
				this.context.count_down()
			}
		})
	}
			
	function collect_shelf(job) {
		sync_status_shelf.text("Starting...").attr('class', 'BE-in-progress')
		var position = 0

		function read_page(order, page, job) {
			GM_xmlhttpRequest({
				method: "GET",
				url: shelf_url(page, order, job.timestamp),
				context: job,
				onload:function(response) {
					var job = this.context
					var json = $.parseJSON(response.responseText)
					if (json.count > 0) {
						var html = $(json.html.replace(/src=/g, "alt="))

						html.each(function() {
							var e = $(this)
							var game_id = e.attr('data-gameid')
							if (game_id === undefined) return
							var game = getgame(game_id, job.games)
							
							game.index = e.attr('data-gameindex')
							game.box_url = e.find('img.shelf_game_box').attr('alt')
							game.shelf_pos = position++
						})

						sync_status_shelf.text('Shelf: '+position + " games counted so far")
						read_page(order, page + 1, job)
					} else {
						sync_status_shelf.text("Done").attr('class', 'BE-success')
						this.context.count_down()
					}
				},
				onerror:function(response) {
					sync_status_shelf.text("Error").attr('class', 'BE-error')
					this.context.errors = true
					this.context.count_down()
				}
			})
		}

		GM_xmlhttpRequest({
			method: "GET",
			url:"https://www.gog.com/account/games/shelf",
			context: job,
			onload:function(response) {
				var html = $(response.responseText.replace(/src=/g, 'data-src='))

				var order = html.find('input#account_shelf_order').val()
				read_page(order, 1, job)
			},
			onerror:function(response) {
				sync_status_shelf.text("Error").attr('class', 'BE-error')
				this.context.errors = true
				this.context.count_down()
			}
		})
	}

function feature_forum_username_link() {
	function on_update(value) {
		switch (value) {
			case true:
				$('.big_user_info .b_u_name').each(function() {
					var div = $(this)
					var name = div.text()
					div.text("")
					
					$("<a>")
					.text(name)
					.attr("href", "http://www.gogwiki.com/wiki/Special:GOGUser/"+escape(name))
					.attr("target", "_blank")
					.appendTo(div)
				})
				break;
			default:
				$('.big_user_info .b_u_name>a').each(function() {
					var a = $(this)
					var div = a.parent()
					var name = a.text()
					div.empty().text(name)
				})
				break;
		}
	}


	$('<style>')
	.text(to_css([
		'.b_u_name>a', [
			'color: inherit'
		]
	]))
	.appendTo(document.head)

	settings.onchange('forum-username-link', on_update)
}

function feature_forum_group_giveaways() {
	function on_update(value) {

		if (value == 'group below other topics') {
			giveaway_topics.remove().insertAfter($('#t_norm'))
			.find('.list_bar_h')
			.click(function() {
				$(this).siblings('.list_row_h').slideToggle()
			})
			$('.list_bottom_bg').remove().appendTo(giveaway_topics)
		} else {
			giveaway_topics.remove().insertBefore($('#t_norm'))
			.find('.list_bar_h')
			.click(function() {
				$(this).siblings('.list_row_h').slideToggle()
			})
			$('.list_bottom_bg').remove().appendTo($('#t_norm'))
		}

		switch (value) {
			case 'hide':
				style.text(to_css([
					'.BE-giveaway-topics, .BE-giveaway-topic', [
						'display: none'
					]
				]))
				break
			case 'group and collapse':
				style.text(to_css([
					'.BE-giveaway-topics', [
						'display: block'
					],
					'.BE-giveaway-topic', [
						'display: none'
					],
				]))
				giveaway_topics.children('.list_row_h').hide()
				break
			case 'group and expand':
			case 'group below other topics':
				style.text(to_css([
					'.BE-giveaway-topics', [
						'display: block'
					],
					'.BE-giveaway-topic', [
						'display: none'
					],
				]))
				giveaway_topics.children('.list_row_h').show()
				break
			default:
				style.text(to_css([
					'.BE-giveaway-topics', [
						'display: none'
					],
					'.BE-giveaway-topic', [
						'display: block'
					],
				]))
		}
	}

	var setting = settings.get('forum-group-giveaways')
	
	var giveaway_topics = $('<div class="favourite_h BE-giveaway-topics">')

	var list = $('<div class="list_row_h">')

	var giveaway_count = 0

	$('#t_norm').find('div.topic_s>a').each(function() {
		if (/giveaway[^/]*$/.test(this.getAttribute('href'))) {
			var row = $(this).closest('.list_row_odd')
			row.clone().appendTo(list)
			row.addClass('BE-giveaway-topic')
			giveaway_count += 1
		}
	})

	if (giveaway_count > 0) {

		var bar = $('<div class="list_bar_h">')
		.append(
			$('<div class="lista_icon_3">'),
			$('<div class="lista_bar_text">').text("Topics which appear to be giveaways ("+giveaway_count+")")
		)
		.css('cursor', 'pointer')
	
		giveaway_topics
		.append(bar, list)
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-group-giveaways', on_update)
}

function feature_hide_alerts() {
	function on_update(value) {

		var total_element = $('._dropdown__toggle>.top-nav__item-count')
		if (total_element.length > 0) {
			var total = 0
			
			if (value["game updates"]) {
				var c = parseInt($('.top-nav__dd-items a[href$="/account"]>.top-nav__item-count').text())
				if (c > 0) total += c
			}
			if (value["forum replies"]) {
				var c = parseInt($('.top-nav__dd-items a[href$="/myrecentposts"]>.top-nav__item-count').text())
				if (c > 0) total += c
			}
			if (value["new messages"]) {
				var c = parseInt($('.top-nav__dd-items a[href$="/messages"]>.top-nav__item-count').text())
				if (c > 0) total += c
			}

			if (total > 0) {
				total_element.text(total)
				total_element.show()
			} else total_element.hide()
		}

	}

	setTimeout(settings.onchange.bind(settings, 'navbar-show-alerts', on_update), 1)
}

function feature_show_sections() {
	function on_update(value) {
		$('.top-nav__items>.top-nav__dropdown--games>a[href="/games"]').parent().toggle(value.games)
		$('.top-nav__items>.top-nav__item[href="/movies"]').toggle(value.movies)
	}

	settings.onchange('navbar-show-sections', on_update)
}

function feature_favicon() {
	function on_update(value) {
		$('head link[rel=icon]').remove()
		switch (value) {
			case 'old favicon':
				$('<link rel="icon" type="image/vnd.microsoft.icon" href="https://secure.gog.com/favicon.ico" />').appendTo(document.head)
				break
			default:
				$('<link rel="icon" type="image/vnd.microsoft.icon" href="/favicon.ico" />').appendTo(document.head)
		}
	}

	settings.onchange('favicon', on_update)
}

function feature_gamecard_show_descriptions() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.description__text a.description__more', [
					'display: none'
				],
				'.description__text[ng-hide=showAll]', [
					'display: none !important'
				],
				'.description__text[ng-show=showAll]', [
					'display: block !important'
				],
			]))
		} else {
			style.text('')
		}
	}

	var style = $('<style>').appendTo(document.head)
	settings.onchange('gamecard-show-descriptions', on_update)
}

function feature_navbar_library_links() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.BE-navbar-library-links:hover', [
					// 'background: #c3c3c3'
				],
				'.top-nav__dd-item._dropdown__item:hover + .BE-navbar-library-links', [
					// 'background: #B3B3B3'
				],
				'.is-contracted .BE-navbar-library-links', [
					'display: none',
				],
				'.BE-navbar-library-links', [
					'cursor: default',
					'height: 25px !important',
					'text-align: center',
					'border-top: none !important',
					'margin-bottom: 5px'
				],
				'.BE-navbar-library-links a', [
					'width: 16px',
					'height: 12px',
					'display: inline-block',
					'margin: 0px 9px',
					'border: 1px solid #A1A1A1',
					'padding: 5px',
					'background-clip: content-box, padding-box',
					'vertical-align: top',
					'border-radius: 3px',
					'background-image: url("/www/default/-img/acc_sprt.2b08c763.png"), linear-gradient(to bottom, #E6E6E6, #CCC)',
				],
				'.BE-navbar-library-links a:hover', [
					'background-image: url("/www/default/-img/acc_sprt.2b08c763.png"), linear-gradient(to bottom, #666, #939393)',
				],
				'.BE-navbar-library-links a:nth-child(1)', [
					'background-position: -11px -52px, 0 0',
				],
				'.BE-navbar-library-links a:nth-child(2)', [
					'background-position: -11px -64px, 0 0',
				],
				'.BE-navbar-library-links a:nth-child(3)', [
					'background: linear-gradient(to bottom, #E6E6E6, #CCC)',
				],
				'.BE-navbar-library-links a:nth-child(1):hover', [
					'background-position: 5px -52px, 0 0',
				],
				'.BE-navbar-library-links a:nth-child(2):hover', [
					'background-position: 5px -65px, 0 0',
				],
				'.BE-navbar-library-links a:nth-child(3):hover', [
					'background: linear-gradient(to bottom, #666, #939393)',
				],
			]))
		} else {
			style.text(to_css([
				'.BE-navbar-library-links', [
					'display: none'
				],
			]))
		}
	}

	$('<div class="top-nav__dd-item _dropdown__item BE-navbar-library-links">')
	.append($('<a href="https://www.gog.com/account/games/shelf">'))
	.append($('<a href="https://www.gog.com/account/games/list">'))
	.append($('<a href="https://www.gog.com/account/gifts">'))
	.insertAfter($('.top-nav__dropdown--account .top-nav__dd-item._dropdown__item[href$="/account"]'))

	var style = $('<style>').appendTo(document.head)
	settings.onchange('navbar-library-links', on_update)
}

function feature_gamecard_gogwiki_link() {
	function on_update(value) {
		if (value) {
			$('dl.product-details')
			.append(dt)
			.append(dd)
		} else {
			dt.remove()
			dd.remove()
		}
	}

	var dt = $('<dt class="product-details__category">').text("GOGWiki")

	var title = (unsafeWindow?unsafeWindow:window).gogData.gameProductData.title
		.replace(/[\u2122\u00ae\u2018]/g, '')
		.replace("\u2019", "'")
		.replace("\u2013", "-")
		.replace(/^(.*), The/, 'The $1')
			
	var dd = $('<dd class="product-details__data">')
	.append(
		$('<a target="_blank" class="un">')
		.attr('href', "http://www.gogwiki.com/wiki/"+escape(title))
		.text(title)
	)

	var style = $('<style>').appendTo(document.head)
	settings.onchange('gamecard-gogwiki-link', on_update)
}

function feature_avatar_zoom() {
	var zoom_on_click

	function on_update(value) {
		zoom_on_click = value
		if (value) {
			style.text(to_css([
				'.spot_h:first-child+.BE-fullsize-avatar', [
					'margin-top: 13px',
				],
				'.BE-fullsize-avatar', [
					'box-shadow: 1px 1px 3px 0px #000000',
					'z-index: 101',
					'margin-left: 12px',
					'margin-top: 30px',
					'max-height: 420px',
					'cursor: zoom-out',
					'position: absolute'
				],
				'.b_p_avatar_h img', [
					'cursor: zoom-in'
				],
			]))
		} else {
			style.text('')
			$('.BE-fullsize-avatar').click()
		}
	}

	$('.b_p_avatar_h img').click(function() {
		if (!zoom_on_click) return

		var img = $(this)
		var src = img.attr('src')

		if (!img.data('BE-zoomed')) {
		
			img.data('BE-zoomed', true)
			
			// clear all fullsize avatars
			$('.BE-fullsize-avatar').data('BE-thumb', null).remove()
			$('.b_p_avatar_h img').data('BE-zoomed', false)
		
			var post = img.closest('.spot_h')
			
			var fullsize = $('<img alt="" class="BE-fullsize-avatar">')
			.attr('src', src.replace(/_t\.jpg$/, '.jpg'))
			.insertBefore(post)
			.one('click', function() {
				$(this).data('BE-thumb').data('BE-zoomed', false)
				$(this).data('BE-thumb', null)
				$(this).remove()
			})
			
			fullsize.data('BE-thumb', img)
			
		}
	})

	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-avatar-zoom', on_update)
}


function feature_forum_quick_post() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.quick_post .BE-hints p', [
					'margin: 7px 0'
				],
				'.quick_post .BE-hints key', [
					'border: 1px solid',
					'margin: 0 1px',
					'padding: 1px 4px',
					'border-radius: 4px',
					'font-family: serif'
				],
				'.quick_post.BE-focus .BE-hints.BE-focus', [
					'opacity: 1'
				],
				'.quick_post.BE-focus .BE-hints', [
					'opacity: 0'
				],
				'.quick_post .BE-hints.BE-focus', [
					'opacity: 0'
				],
				'.quick_post .BE-hints', [
					'color: ' + ((forum_skin==0)?'#aaa': '#878787'),
					'position: absolute',
					'margin-top: 13px'
				],
				'.quick_post textarea:invalid', [
					'box-shadow: none',
					'height: 30px'
				],
				'.quick_post > textarea:invalid ~ *', [
					'opacity: 0'
				],
				'.quick_post > *', [
					'transition: opacity ease 0.5s'
				],
				'.quick_post', [
					'background: url("'+((forum_skin==0)?"/www/forum_carbon/-img/post_bg.851b4700.gif": "/www/forum_alu/-img/post_bg.b7d2258c.gif")+'") repeat-x scroll 0 -161px transparent',
					'border-radius: 5px 5px 5px 5px',
					'margin: 0 auto',
					'min-height: 90px',
					'overflow: hidden',
					'padding: 2px 12px',
					'width: 926px'
				],
				'.quick_post h1', [
					'color: ' + ((forum_skin==0)?'#aaa': '#878787'),
					'font-family: "Lucida Grande",Arial,Verdana,sans-serif',
					'font-size: 10px',
					'font-weight: normal',
					'margin: 10px 0 0 58px',
					'clear: both'
				],
				'.quick_post textarea:focus, .quick_post textarea:hover', [
					'border: 1px solid '+((forum_skin==0)?'#DBDBDB': '#4C4C4C')
				],
				'.quick_post textarea:focus', [
					'height: 150px'
				],
				'.quick_post textarea', [
					'background: none repeat scroll 0 0 '+((forum_skin==0)?'#676767': '#D1D1D1'),
					'border: 1px solid '+((forum_skin==0)?'#929292': '#929292'),
					'color: ' + ((forum_skin==0)?'#DBDBDB':'#4C4C4C'),
					'height: 150px',
					'transition: height 0.5s ease 0s',
					'margin: 0 0 5px 168px',
					'padding: 2px',
					'width: 751px',
					'font-family: Arial',
					'font-size: 12px'
				],
				'button.BE-button::-moz-focus-inner,'+
				'button.submit-quick-post::-moz-focus-inner', [
					'border: none'
				],
				'button.BE-button:focus,'+
				'button.submit-quick-post:focus', [
					'color: #ffffff'
				],
				'button.BE-button:hover,'+
				'button.submit-quick-post:hover', [
					'background: linear-gradient(#303030, #393939) repeat scroll 0 0 transparent',
					'background: ' + ((forum_skin==0)?"-webkit-linear-gradient(#303030, #393939)": "-webkit-linear-gradient(#959595, #646464)")+' repeat scroll 0 0 transparent',
					'background: ' + ((forum_skin==0)?"linear-gradient(#303030, #393939)": "linear-gradient(#959595, #646464)")+' repeat scroll 0 0 transparent'
				],
				'button.BE-button:active,'+
				'button.submit-quick-post:active', [
					'background: ' + ((forum_skin==0)?"#4c4c4c": "")
				],
				'button.BE-button:disabled,'+
				'button.submit-quick-post:disabled', [
					'background: linear-gradient(#757575, #828282) repeat scroll 0 0 transparent',
					'box-shadow: none',
					'color: #525252'
				],
				'button.BE-button,'+
				'button.submit-quick-post', [
					'background: ' + ((forum_skin==0)?"-webkit-linear-gradient(#393939, #434343)": "-webkit-linear-gradient(#646464, #959595)")+' repeat scroll 0 0 transparent',
					'background: ' + ((forum_skin==0)?"linear-gradient(#393939, #434343)": "linear-gradient(#646464, #959595)")+" repeat scroll 0 0 transparent",
					'border: medium none',
					'border-radius: 12px 12px 12px 12px',
					'color: ' + ((forum_skin==0)?"#BEBEBE": "#F0F0F0"),
					'font-family: arial, sans-serif',
					'font-size: 11px',
					'font-weight: normal',
					'line-height: 25px',
					'cursor: pointer',
					'margin: 10px 0',
					'display: block',
					'padding: 0 15px',
					'vertical-align: middle',
					'transition: color 0.3s ease'
				],
				'.quick_post div.submit', [
					'float: right',
					'clear: none',
					'overflow: visible',
					'padding: 11px 3px 0 0'
				],
				'.big_post_h:hover .BE-quick-reply', [
					'display: inline-block'
				],
				'.BE-quick-reply', [
					'background: ' + ((forum_skin==0)?"-webkit-linear-gradient(#656566, rgba(101, 101, 102, 0))": "-webkit-linear-gradient(#E5E5E5, rgba(229, 229, 229, 0))"),
					'background: ' + ((forum_skin==0)?"linear-gradient(#656566, rgba(101, 101, 102, 0))": "linear-gradient(#E5E5E5, rgba(229, 229, 229, 0))"),
					'border-radius: 3px 3px 3px 3px',
					'color: ' + ((forum_skin==0)?"#E0E0E0": "#606060"),
					'display: none',
					'height: 23px',
					'margin: 8px 0 0',
					'padding-top: 4px',
					'text-align: center',
					'text-shadow: ' + ((forum_skin==0)?"0px 1px 1px #333333": "none"),
					'width: 60px',
					'cursor: pointer'
				],
			]))

			$('.p_button_right_h').each(function() {
				$('<div class="BE-quick-reply">')
				.text('quick reply')
				.appendTo(this)
				.click(function() {
				
					window.scrollTo(0, $('.quick_post').position().top - Math.round(window.innerHeight*0.4))
					
					var textarea = $('.quick_post textarea').focus()
					
					var quoted = $(this).closest('.big_post_h')
					var quote_nr = quoted.find('.post_nr').text()
					
					var val = textarea.val()
					textarea.val(
						((val.length==0)?"":(val+"\n\n"))
						+ '[quote_'+quote_nr+']'
						+ parse_post(quoted)
						+'[/quote]\n'
					)
					
					show_preview()
				})
			})

		} else {
			$('.BE-quick-reply').remove()

			style.text(to_css([
				'.quick_post', [
					'display: none'
				],
			]))
		}
	}

	var last_post = $('.spot_h:last')
	if (last_post.length > 0) {
		var new_post = $('<textarea required>')
		var preview = $('<p class="BE-preview">')
		var post_button = $('<button class="submit-quick-post">').text('submit quick post')

		$('<div class="quick_post">')
		.append(
			$('<h1>').text('Quick Post'),

			'<div class="BE-hints"><p>Shortcut: <key>Ctrl</key> <key>Space</key></p></div>',
			'<div class="BE-hints BE-focus">'
			+'<p><key>Ctrl</key> <key>I</key>: Italic</p>'
			+'<p><key>Ctrl</key> <key>B</key>: Bold</p>'
			+'<p><key>Ctrl</key> <key>U</key>: Underline</p>'
			+'<p><key>Ctrl</key> <key>Y</key>: Hyperlink</p>'
			+'<p><key>Ctrl</key> <key>L</key>: Spoiler</p>'
			+'<p><key>Ctrl</key> <key>Enter</key>: Submit</p>'
			+'</div>',

			new_post,
			
			$('<div class="submit">')
			.append(
				post_button
			),
			
			$('<h1>').text('Preview'),
			preview
		)
		.insertAfter(last_post)

		// hints respond to focus
		new_post
		.focus( function() { $(this).closest('.quick_post').addClass('BE-focus') } )
		.blur( function() { $(this).closest('.quick_post').removeClass('BE-focus') } )

		// handle shortcut keys in quick post
		new_post.keypress(post_keydown_handler)
		
		// refresh the preview on quic post input
		new_post[0].addEventListener('input', show_preview)
		
		post_button.click(submit_quick_post)
	}

	$(document).keydown(function(event) {
		if (event.ctrlKey && !event.repeat && event.which == 32) {
			window.scrollTo(0, $('.quick_post').position().top - Math.round(window.innerHeight*0.4))
			$('.quick_post textarea').focus()
			return false
		}
	})

	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-quick-post', on_update)
}

function feature_post_preview() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.text_1, .text_1_bad', [
					'height: 180px',
				],
				'div.files', [
					'z-index: 1',
				],
				'div.submit', [
					'float: right',
					'clear: none',
					'overflow: visible',
					'padding: 11px 3px 0 0'
				],
				'html', [
					'height: 100%',
					'overflow: auto',
				],
				'div.files', [
					'float: left',
					'width: 460px',
					'overflow: hidden',
					'height: auto'
				],
				'.zawartosc', [
					'height: 100%'
				],
				'.zawartosc .kontent', [
					'height: initial'
				],
			]))

			var message = document.getElementById('text')
			var preview = $('<p class="BE-preview">').insertAfter($('.kontent>.submit'))
		
			var refresh_preview = function() {
				var text = post_preview_html(message.value)
			
				preview.html(text)
			}

			// capture the offset of the preview and make it absolute
			preview.css({
				'position': 'absolute',
				'bottom': '0px',
				'height': 'auto',
				'top': preview.offset().top + 'px'
			})
		
			$(message).keydown(post_keydown_handler)
			message.addEventListener('input', refresh_preview)
		
			$('.btn_h>.btn').click(refresh_preview)

			refresh_preview()
		} else {
			$('.BE-preview').remove()
			style.text('')
		}
	}

	var style = $('<style>').appendTo(document.head)
	
	settings.onchange('forum-post-preview', on_update)
}

function add_preview_styles() {
	var style = $('<style>')
	.text(to_css([
		'.BE-preview a', [
			'background: url("'+((forum_skin==0)?"/www/forum_carbon/-img/post_un.a8689b98.gif":"http://static.gog.com/www/forum_alu/-img/zig_underl.8b625731.gif")+'") repeat-x scroll center bottom transparent',
			'color: ' + ((forum_skin==0)?'#DBDBDB':'#4C4C4C')
		],
		'.quick_post .BE-preview', [
			'margin: 0 0 0 168px',
			'padding: 3px',
			'width: 751px',
			'height: auto',
			'overflow: hidden'
		],
		'.BE-preview', [
			'height: 150px',
			'overflow: auto',
			'margin: 0 0 0 143px',
			'padding: 5px',
			'clear: both',
			'width: 650px',
			'word-wrap: break-word',
			'color: ' + ((forum_skin==0)?'#DBDBDB':'#4C4C4C'),
			'transition: height 0.5s ease 0s',
			'font-family: Arial',
			'font-size: 12px'
		],
		'.BE-preview .syntax-warning', [
			'background: #6F3E3E',
			'border-radius: 5px 5px 5px 5px',
			'color: #DBDBDB',
			'display: inline-block',
			'font-size: 11px',
			'font-weight: normal',
			'font-style: normal',
			'font-family: monospace, sans-serif',
			'margin: 0 5px',
			'padding: 0 2px'
		],
		'.BE-preview blockquote', [
			'border-left: 1px solid #929292',
			'padding: 0 0 0 8px',
			'margin: 0'
		],
	]))
	.appendTo(document.head)
}

function feature_click_own_title() {
	function on_update(value) {
		if (value) {
			$('.edit_h_EN').each(function() {
			  var post = $(this).closest('.big_post_main')
			  if (post) {
				var stat = post.find('.b_u_stat')
				if (stat) {
					var stat_text = stat.text()
					stat.empty().append($('<a href="/forum/mysettings" title="Click here to change your forum title">').text(stat_text))
				}
			  }
			})
		} else {
			$('.b_u_stat>a').each(function() {
				var parent = $(this).parent()
				$(this).remove()
				parent.text($(this).text())
			})
		}
	}

	$('<style>').text('.b_u_stat>a { color: inherit; }').appendTo(document.head)

	settings.onchange('forum-title-settings', on_update)
}

function feature_forum_move_edit_note() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.big_post_main:hover .BE-edit-note .pencil_text', [
					'visibility: visible'
				],
				'.BE-edit-note .pencil_text', [
					'visibility: hidden'
				],
				'.BE-edit-note', [
					'padding-top: 4px',
					'left: 180px',
					'position: absolute'
				],
			]))
			$('.post_attaczments .pencil_text').each(
				function() {
					var element = $(this).parent()

					var edit_note = $('<div class="BE-edit-note">')

					element.closest('.big_post_main').find('.post_header_h .p_button_left_h')
					.after(edit_note)

					element.remove()
					element.appendTo(edit_note)
				}
			)
		} else {
			style.text('')
			$('.BE-edit-note .post_attaczments').each(
				function() {
					var element = $(this)
					var parent = element.parent()
					element.remove()

					parent.closest('.big_post_main').find('.post_text')
					.after(element)

					parent.remove()
				}
			)
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-move-edit-note', on_update)
}

function feature_forum_quotation_style() {
	function on_update(value) {
		switch (value) {
			case 'distinct':
				style.text(to_css([
					'.post_text_c div.quot', [
						'font-style: normal'
					],
					'.post_text_c div.quot:not(.gog_color),'
					+'.post_text_c div.quot:not(.gog_color) a,'
					+'.post_text_c div.quot:not(.gog_color) span.bold,'
					+'.BE-preview blockquote,'
					+'.BE-preview blockquote a', [
						'color: ' + ((forum_skin==0)?'#A8A8A8':'#686868')
					]
				]))
				break;
			case 'clear':
				style.text(to_css([
					'.post_text_c div.quot div.quot', [
						'background: transparent'
					],
					'.post_text_c div.quot', [
						'font-style: normal',
						'padding: 2px 5px 2px 7px',
						'background: ' + ((forum_skin==0)?'rgba(0, 0, 0, 0.1)':'rgba(255, 255, 255, 0.2)')
					],
					'.post_text_c div.quot:not(.gog_color),'
					+'.post_text_c div.quot:not(.gog_color) a,'
					+'.post_text_c div.quot:not(.gog_color) span.bold,'
					+'.BE-preview blockquote,'
					+'.BE-preview blockquote a', [
						'font-size: 12px'
					]
				]))
				break;
			default:
				style.text('')
		}
	}

	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-quotation-style', on_update)
}

function feature_wiki_sync() {

	function collect_data() {
		var data = {
			wishlist: [],
			games: {},
			errors: false,
			countdown: 4,
			timestamp: (new Date()).getTime(),
			count_down: function() {
				this.countdown -= 1
				if (this.countdown == 0) {
					if (this.errors) {
						sync_status_output
						.html("<p>It seems that there were problems collecting information from your account. Try again later - it might just be temporary issue. If you keep having this problem then <a href=\"https://www.gog.com/forum/general/barefoot_essentials_2_gogcom_enhancement\" target=\"blank\">let me know</a>.</p>")
						.show()
					
						sync_status_send.text('Stopped').attr('class', 'BE-error')
						sync_status_restart.show()//.appendTo(gog_sync_element)
					} else {
						send_to_gogwiki(this)
					}
				} else if (this.countdown < 0) {
					sync_status_restart.show()//.appendTo(gog_sync_element)
					sync_status_output.show()
				}
			}
		}

		get_account_information(data)
		collect_wishlist(data)
		collect_shelf(data)
		collect_list(data)
		sync_status_send.text('Waiting...').attr('class', 'BE-in-progress')

		sync_status_start.remove()
		sync_status_restart.hide()
		sync_status_output.hide()
		sync_status_progress.show()
	}

	gog_sync_element = $('<div>')

	sync_status_start = $('<p><a>Click here to share your library and wishlist on GOGWiki</a></p>')
	.click(collect_data)
	.appendTo(gog_sync_element)

	sync_status_progress = $('<div class="BE-sync-progress">')
	.hide()
	.appendTo(gog_sync_element)
	.append(
		'<h2>Sync progress</h2>',
		$('<p><span>Loading your GOG account:</span>').append(sync_status_account = $('<span>')),
		$('<p><span>Reading your game shelf:</span>').append(sync_status_shelf = $('<span>')),
		$('<p><span>Reading your game list:</span>').append(sync_status_list = $('<span>')),
		$('<p><span>Reading your wishlist:</span>').append(sync_status_wishlist = $('<span>')),
		$('<p><span>Sending to GOGWiki:</span>').append(sync_status_send = $('<span>'))
	)

	sync_status_output = $('<div>')
	.hide()
	.appendTo(gog_sync_element)

	sync_status_restart = $('<p><a>Click here to sync with GOGWiki again</a></p>')
	.click(collect_data)
	.hide()
	.appendTo(gog_sync_element)
}


function feature_enhance_bold_text() {
	function on_update(value) {
		if (value) style.text(to_css([
			'.post_text span.bold, .BE-preview b', [
				'font-weight: 800'
			],
		]))
		else style.text('')
	}
	var style = $('<style>').appendTo(document.head)
	
	settings.onchange('forum-bold-text', on_update)
}

function feature_hide_spoilers() {
	if (!settings.get('forum-hide-spoilers')) return

	var style = $('<style>').appendTo(document.head)
	style.text(to_css([
		'.BE-spoiler', [
			'height: 0',
			'display: none'
		],
		'.BE-spoiler.BE-visible', [
			'height: auto',
			'display: block',
			'border: 1px solid '+((forum_skin==0)?"#676767": "#9a9a9a"),
			'padding: 1em',
			'margin: 1em',
			'background: ' + ((forum_skin==0)?"#585858": "#D7D7D7"),
			'border-radius: 0.5em'
		],
		'.BE-spoiler .BE-spoiler .BE-spoiler.BE-visible', [
			'padding: 1em 5px',
			'margin: 1em 1px'
		],
	]))


	function traverse_to_hide_spoilers(parent) {
	    function toggle_spoiler() {
	        var visible = !$(this).data('visible')
	        $(this).data('visible', visible)
	        if (visible) {
	            $(this).text("hide spoiler")
	            .next("div.BE-spoiler").addClass('BE-visible')
	        } else {
	            $(this).text("show spoiler")
	            .next("div.BE-spoiler").removeClass('BE-visible')
	        }
	    }
	    var openings = []
	    for (var n = parent.firstChild; n !== null; n = n.nextSibling) {
	        if (n.nodeType == 3) {
	            
	            if (/^\s*\[spoiler\]\s*$/.test(n.nodeValue)) {
	                openings.push(n)
	            } else if (/^\s*\[\/spoiler\]\s*$/.test(n.nodeValue)) {
	            	var opening = openings.pop()
	            	if (opening === null) continue
	            	
	                var spoiler_button = $('<button class="BE-button">')
	                .text("show spoiler")
	                .click(toggle_spoiler)
	                .insertBefore(opening)
	                var spoiler_content = $('<div class="BE-spoiler">')
	                .insertAfter(spoiler_button)
	                
	                var next = opening.nextSibling
	                for (nn = opening; nn !== null; nn = next) {
	                    next = nn.nextSibling
	                    parent.removeChild(nn)
	                    if (nn !== opening && nn !== n) spoiler_content.append(nn)
	                    if (nn === n) break
	                }
	                
	                n = spoiler_content.get(0)
	            }
	        } else if (n.nodeType == 1) {
	        	traverse_to_hide_spoilers(n)
	        }
	    }
	}

	$('.post_text_c').each(function() {
	    if (0 <= this.textContent.indexOf('[spoiler]')) {
	        traverse_to_hide_spoilers(this)
	    }
	})
	
	// remove all extra linebreaks around spoilers
	$('.BE-button').each(function() {
		for (var prev = this.previousSibling; prev; prev = this.previousSibling) {
		
			if (prev.nodeType == 1) {
				if (prev.nodeName != 'BR') break
			} else if (prev.nodeType == 3) {
				if (!/^[\s]*$/.test(prev.nodeValue)) break
			} else break
			this.parentElement.removeChild(prev)
		}
	})
	$('.BE-spoiler').each(function() {
	
		// strip after the spoiler
		for (var next = this.nextSibling; next; next = this.nextSibling) {
		
			if (next.nodeType == 1) {
				if (next.nodeName != 'BR') break
			} else if (next.nodeType == 3) {
				if (!/^[\s]*$/.test(next.nodeValue)) break
			} else break
			this.parentElement.removeChild(next)
		}
		
		// strip the front
		for (var next = this.firstChild; next; next = this.firstChild) {
			if (next.nodeType == 1) {
				if (next.nodeName != 'BR') break
			} else if (next.nodeType == 3) {
				if (!/^[\s]*$/.test(next.nodeValue)) break
			} else break
			this.removeChild(next)
		}
		// strip the tail
		for (var next = this.lastChild; next; next = this.lastChild) {
			if (next.nodeType == 1) {
				if (next.nodeName != 'BR') break
			} else if (next.nodeType == 3) {
				if (!/^[\s]*$/.test(next.nodeValue)) break
			} else break
			this.removeChild(next)
		}
	})
}


function feature_add_essentials_link() {
	function on_update(value) {
		$('.BE-essentials-menu-item').remove()

		var menu = $('.top-nav__dropdown--account .top-nav__dd-items._dropdown__items')

		var item = $('<span class="top-nav__dd-item _dropdown__item BE-essentials-menu-item">')
		.text('ESSENTIALS')
		.click(popup.show.bind(popup, 'Changelog'))


		switch (value) {
			case 'bottom':
				menu.append(item)
				break;
			default:
				menu.prepend(item)
		}
	}
	var style = $('<style>').appendTo(document.head)
	
	settings.onchange('navbar-essentials-position', on_update)
}

function feature_library_always_show_count() {
	if (settings.get('library-always-show-count')) {
		try {
			var observer = new MutationObserver(function(mutations) {
	
				mutations.forEach(function(mutation) {
				
					var target = mutation.target
				
					if (mutation.attributeName == 'class') switch (target.className) {
						case 'middle_btns': {
							if (!(target.childElementCount > 0)) {
						
								// count games
								var num_games = document.querySelectorAll('#shelfGamesList .shelf_game:not(.empty), #hiddenGamesList .shelf_game:not(.empty)').length
		
								// display game count
								document.querySelector('#tagButtons').innerHTML = '<span class="shelf_btn css3pie new tag-btn">NEW &amp; UPDATED <span class="count">0</span></span><span class="shelf_btn css3pie all active tag-btn">ALL <span class="count">'+num_games+'</span></span>'
							}
						} break;
						case 'list_header': {
							if (!(target.childElementCount > 0)) {
						
								// count games
								var num_games = document.querySelectorAll('#gamesList .game-item, #hiddenGamesList .game-item').length
						
								// display game count
								document.querySelector('#tagButtons').innerHTML = 'My collection <span class="list_btn css3pie new tag-btn">NEW &amp; UPDATED <span class="count">0</span></span><span class="list_btn css3pie all active tag-btn">ALL <span class="count">'+num_games+'</span></span>'
							}
						} break;
					}
				})
		
			})
			observer.observe(document.querySelector('#tagButtons'), {attributes: true})
		} catch (exception) {
			// no action necessary
		}
	}
}

function feature_navbar_position() {
	function on_update(value) {
		switch (value) {
			case 'absolute':
				style.text(to_css([
					'.top-nav', [
						'position: absolute',
					],
				]))
				break;
			default:
				style.text('')
		}
	}
	var style = $('<style>').appendTo(document.head)
	
	settings.onchange('navbar-position', on_update)
}

function feature_catalogue_hide_owned() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.product-row.is-owned', [
					'display: none'
				]
			]))
		} else {
			style.text('')
		}
	}
	var style = $('<style>').appendTo(document.head)

	settings.onchange('catalogue-hide-owned', on_update)
}

function feature_catalogue_add_hide_owned_toggle() {
	function on_update(value) {
		toggle.toggle(value)
	}
	function on_show_or_hide(value) {
		state.text(value ? 'HIDDEN' : 'SHOWN')
	}
	var toggle = $('<div class="header__switch">')
	.text('OWNED: ')

	var state = $('<span class="header__dropdown module-header-dd _dropdown is-contracted">')
	.text('SHOW')
	.click(function() {
		settings.set('catalogue-hide-owned', !settings.get('catalogue-hide-owned'))
	})
	.appendTo(toggle)

	toggle.appendTo($('.header__switches'))

	settings.onchange('catalogue-show-hide-owned-toggle', on_update)
	settings.onchange('catalogue-hide-owned', on_show_or_hide)
}

function feature_forum_show_hover_elements() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'div.on_off_stat, div.p_button_left_h', [
					'display: block'
				],
				'.BE-edit-note .pencil_text', [
					'visibility: visible'
				]
			]))
		} else {
			style.text('')
		}
	}
	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-show-hover-elements', on_update)
}

function feature_collapsible_footer() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.main-footer__dark-footer', [
					'position: absolute',
					'left: 0',
					'right: 0',
					'transition: top 0.5s linear',
					'top: 5px',
				],
				'footer:hover .main-footer__dark-footer', [
					'top: 312px',
				],
				'.main-footer', [
					'position: absolute',
					'height: 98px',
					'left: 0',
					'right: 0',
					'transition: height 0.5s linear, margin-top 0.5s linear',
				],
				'.main-footer:hover', [
					'height: 405px',
					'margin-top: -307px',
				]
			]))
		} else {
			style.text('')
		}
	}
	var style = $('<style>').appendTo(document.head)

	settings.onchange('forum-collapsible-footer', on_update)
}

function feature_reduce_footer_spacer() {
	function on_update(value) {
		if (value) {
			style.text(to_css([
				'.footer-spacer', [
					'height: 150px',
				]
			]))
		} else {
			style.text('')
		}
	}
	var style = $('<style>').appendTo(document.head)

	settings.onchange('bugfix-collapsible-footer', on_update)
}


try {
	settings.initialise(config, function() {

		feature_favicon()
		feature_BE_style()
		feature_wiki_sync()

		// navbar
		feature_navbar_position()
		feature_navbar_style()
		feature_navbar_opacity()
		feature_notification_style()
		feature_cart_style()
		feature_hide_alerts()
		feature_show_sections()
		feature_navbar_library_links()
		feature_add_essentials_link()

		// forum
		if (/^\/forum/.test(window.location.pathname)) {
	
			detect_forum_skin()
			add_preview_styles()

			// forum section
			if (/^\/forum\/[^/]*(?:\/(?:page[0-9]+)?)?$/.test(window.location.pathname)) {
				feature_forum_group_giveaways()
			}

			// forum popup
			else if (location.pathname == "/forum/ajax/popUp") {
				feature_post_preview()
			}

			// forum topic
			else if (/^\/forum\/[^/]*\/[^/]*(?:\/(?:page[0-9]+|post[0-9]+)?)?$/.test(window.location.pathname)) {
				feature_forum_username_link()
				feature_avatar_zoom()
				feature_forum_quick_post()
				feature_click_own_title()
				feature_forum_move_edit_note()
				feature_forum_quotation_style()
				feature_enhance_bold_text()
				feature_hide_spoilers()
				feature_forum_show_hover_elements()
				feature_collapsible_footer()
				feature_reduce_footer_spacer()
			}
		}

		// moviecard
		else if (/^\/movie\/[^/]*$/.test(window.location.pathname)) {
			feature_gamecard_show_descriptions()
		}
		// gamecard
		else if (/^\/game\/[^/]*$/.test(window.location.pathname)) {
			feature_gamecard_show_descriptions()
			feature_gamecard_gogwiki_link()
		}
		// library
		else if ((new RegExp("^/account(?:/(?:games|movies)(?:/(?:shelf|list))?)?$")).test(window.location.pathname)) {
			feature_library_always_show_count()
		}
		// catalogue
		else if ((new RegExp('^/(?:games|movies)(?:/.*)?$')).test(window.location.pathname)) {
			feature_catalogue_hide_owned()
			feature_catalogue_add_hide_owned_toggle()
		}


	})

	// check version, and show changelog if new
	var last_BE_version = GM_getValue('last_BE_version')
	if (last_BE_version === undefined) last_BE_version = default_prev_version
	else if (cmpVersion(last_BE_version, version) < 0) {
		popup.show('Changelog')
	}
	GM_setValue('last_BE_version', version)

} catch (exception) { console.error(exception)}