Canny Google Search

Adds Google Custom Search Box into Canny pages.

// ==UserScript==
// @name        Canny Google Search
// @name:ja     Canny Google検索
// @description Adds Google Custom Search Box into Canny pages.
// @description:ja CannyのページへGoogleカスタム検索窓を追加します。
// @namespace   https://greasyfork.org/users/137
// @version     1.1.0
// @match       https://*.canny.io/*
// @exclude     https://canny.io/*
// @require     https://greasyfork.org/scripts/17896/code/start-script.js?version=112958
// @require     https://greasyfork.org/scripts/19616/code/utilities.js?version=230651
// @license     MPL-2.0
// @compatible  Edge
// @compatible  Firefox Firefoxを推奨 / Firefox is recommended
// @compatible  Opera
// @compatible  Chrome
// @grant       dummy
// @noframes
// @run-at      document-start
// @icon        
// @author      100の人
// @homepageURL https://greasyfork.org/scripts/392585
// ==/UserScript==

'use strict';

startScript(
	function () {
		document.head.insertAdjacentHTML('beforeend', `<style>
			[action="https://www.google.com/search"] {
				margin-left: auto;
				display: flex;
				align-items: center;
			}
		</style>`);

		const secondaryNav = document.getElementsByClassName('secondaryNav')[0];
		document.getElementsByClassName('secondaryNav')[0].insertAdjacentHTML('beforeend', h`
			<form action="https://www.google.com/search">
				<input type="search" name="q" size="31" />
				<input type="hidden" name="as_sitesearch" value="${location.origin}" />
				<input type="submit" value="Search" />
				<img alt="Google™ Custom Search"
					src="https://cse.google.com/cse/images/google_custom_search_smwide.gif" />
			</form>
		`);
		const form = secondaryNav.lastElementChild;

		// CSPの回避
		form.addEventListener('submit', function (event) {
			event.preventDefault();
			location.assign(form.action + '?' + new URLSearchParams(new FormData(form)));
		});

		new MutationObserver(function (mutations, observer) {
			if (!mutations.some(mutation => mutation.removedNodes[0] === form)) {
				return;
			}
			observer.disconnect();
			secondaryNav.append(form);
		}).observe(secondaryNav, { childList: true });

		new MutationObserver(function () {
			document.getElementsByClassName('secondaryNav')[0].append(form);
		}).observe(document.getElementsByClassName('publicContainer')[0], { childList: true });
	},
	parent => parent.classList.contains('secondaryNav'),
	target => target.classList.contains('boards'),
	() => document.getElementsByClassName('boards')[0]
);