TDK Arama Yardımcısı

TDK'nın sözlük sitesinde URL'den arama yapmaya olanak sağlar (URL örneği: http://sozluk.gov.tr/?q=KAİDE), tarayıcı geçmişinde aranan kelimeler için kayıt oluşturur ve site anasayfasında eski aramaları gösterir.

// ==UserScript==
// @name         TDK Arama Yardımcısı
// @namespace    https://github.com/nhtctn
// @version      1.6
// @description  TDK'nın sözlük sitesinde URL'den arama yapmaya olanak sağlar (URL örneği: http://sozluk.gov.tr/?q=KAİDE), tarayıcı geçmişinde aranan kelimeler için kayıt oluşturur ve site anasayfasında eski aramaları gösterir.
// @author       nht.ctn & Magnum357
// @include      *://sozluk.gov.tr/*
// @icon         

// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_info
// @run-at       document-idle

// @require	 https://ajax.googleapis.com/ajax/libs/jquery/3.3.1/jquery.min.js
/* global $ */
/*jshint esversion: 6 */
// ==/UserScript==
(function() {
	'use strict';

	/*
	 ** [ ESSENTIALS ]
	 */

	var urlParams = new URLSearchParams(window.location.search);
	var word = urlParams.get('q');
	var dict = urlParams.get('dic');
	var inputEl = document.querySelector('#tdk-srch-input');
	var formEl = document.querySelector('#tdk-srch-form');
	var buttonEl = document.querySelector('#tdk-search-btn');
	var suggestionPanelEl = document.querySelector('.autocmp');
	if (GM_getValue("pastArray") == null) GM_setValue("pastArray", []);
	var enter = `
`;

	// Storage update
	storageUpt();

	// Show the result by a query when the page opened
	if (word != undefined) {
		showResult(word, dict);
		setWord(word, dict);
	}

	// Select the first suggestion on the enter press
	inputEl.addEventListener('keypress', goToFirstSuggestion);

	// Refresh query
	formEl.onsubmit = realTimeSearch;

	function showResult(s_word, s_dict) {
		// If specified, select dictionary to search
		if (s_dict) {
			var dictList = $('#dictList1, #dictList2').find('input[type="checkbox"]').map(function(i, item) {
				return $(item).attr("id");
			}).get();
			var checks = s_dict.split(" ");
			for (let x = 0; x < dictList.length; x++) {
				var d = dictList[x];
				if (checks.includes(d)) {
					$('#' + d).prop('checked', true);
				} else {
					$('#' + d).prop('checked', false);
				}
			}
		}

		inputEl.value = s_word;
		buttonEl.click();
	}

	function setWord(s_word, s_dict) {
		// Browser History
		if (history.pushState) {
			// Refresh the query and write it history
			var newurl = window.location.protocol + '//' + window.location.host + window.location.pathname + '?q=' + s_word + (s_dict ? '&dic=' + s_dict.replace(/ /g, "+") : '');
			window.history.pushState({}, '', newurl);
		}

		// Script History (maximum 1000 entry)
		var pastArray = GM_getValue("pastArray");
		if (pastArray.length == 0 || s_word != pastArray[pastArray.length - 1].word || s_dict != pastArray[pastArray.length - 1].dict) {
			let item = {
				word: s_word,
				time: new Date().getTime(),
				dict: s_dict
			};
			pastArray[pastArray.length] = item;
		} else {
			pastArray[pastArray.length - 1].time = new Date().getTime(); // If the word searched for second time, just update the time.
		}
		var cleaner = (pastArray.length > 1000) ? 1 : 0; // If the word number goes beyond 1000, it will delete one word for each new word. The number will remain 1000.
		GM_setValue("pastArray", pastArray.splice(cleaner, 1000));
	}

	function realTimeSearch() {
		var realWord = inputEl.value;
		var realDict = $('#dictList1, #dictList2').find('input[type="checkbox"]:checked').map(function(i, item) {
			return $(item).attr("id");
		}).get().toString().replace(/,/g, " ");
		if (realWord.length > 0 && realDict.length > 0) {
			realDict = (realDict == "gts") ? null : realDict;
			setWord(realWord, realDict);
			addPastToPage();
		}
	}

	function goToFirstSuggestion(e) {
		if (e.keyCode == 13) {
			var wordEl = suggestionPanelEl.querySelector('.selected');

			if (wordEl != undefined) {
				inputEl.value = wordEl.innerText;
			}
		}
	}

	/*
	 ** [ HIDING SIGN LANGUAGE ]
	 */

	var signLangContHTML = `
<div id="signs_648" class="card">
    <div class="card-header" id="headingThree">
        <h5 class="mb-0">
            <button class="btn collapsed" btn-link="" collapsed="" type="button" data-toggle="collapse" data-target="#isaret-gts0" aria-expanded="false" aria-controls="collapseThree">
                <strong style="padding:5px;font-size:15px">İşaret Dili</strong>
            </button>
        </h5>
    </div>
    <div id="isaret-gts0" class="collapse" aria-labelledby="headingThree" data-parent="#accordionExample-gts0" aria-expanded="false" style="height: 0px;">
        <div id="isaret-gts0" class="card-body" birlesik-gts0="">
        </div>
    </div>
</div>
`;

	var accordionHTML = `<div class="accordion" id="accordionExample-gts0"> </div>`;

	waitForKeyElements('div#isaretBulunan', hideSignLang, false);

	function hideSignLang() {
		var accordion = document.querySelector('div.accordion');
		if (accordion == null) {
			document.querySelector('div#maddeleri-sar').insertAdjacentHTML("beforeend", accordionHTML);
		}
		document.querySelector('div.accordion').insertAdjacentHTML("beforeend", signLangContHTML);
		var signs = document.querySelector('div#isaretSoz');
		var signContainer = document.querySelector('div#isaret-gts0');
		signs.parentElement.style.display = "none";
		signContainer.insertAdjacentElement("afterbegin", signs);
	}

	/*
	 ** [ PAST MODULE ]
	 */

	var pastContainerHTML = `
<div class="contain">
    <div id="past" style="padding: 0 5px;">
        <span class="thumbnail text-center" style="height: auto;">
            <h4 class="text-danger" style="color:red; font-weight:bolder; margin: 15px 0;">Geçmiş</h4>
            <ul class="pastUl">
            </ul>
            <div style="margin: 15px; text-align: left;">
                <button id="deletePast" class="btn" type="button"> <strong style="padding:5px;font-size:15px">Geçmişi Sil</strong> </button>
            </div>
        </span>
    </div>
</div>
`;

	// Sonradan eklenebilir:
	// <button class="btn" type="button"> <strong style="padding:5px;font-size:15px">Geçmiş Detayları</strong> </button>

	GM_addStyle(`
.pastUl {margin: 15px; list-style: none; overflow: auto;}
.wordElements, .wordElements:hover {color:#cd853f; font-size:18px; font-weight: 700; margin: 0 10px 5px 10px; display: inline-block;}
`);

	addPastToPage();
	document.querySelector('button#deletePast').onclick = function() {
		deleteWord("all");
	};
	waitForKeyElements('div#bulunmayan-gts.hata', function() {
		deleteWord("last");
	}, false);

	function addPastToPage() {
		// Delete old
		var oldSignHTML = document.querySelector('div#signs_648');
		if (oldSignHTML) oldSignHTML.remove();

		// Container
		var pastContainer = document.querySelector('div#past > span > ul');
		if (pastContainer == null) {
			document.querySelector('div#kelime').parentElement.parentElement.parentElement.insertAdjacentHTML("afterbegin", pastContainerHTML);
		}

		// Words
		getPast();
	}

	function getPast() {
		var wordElements = '';
		var counter = 0;
		var pastArray = GM_getValue("pastArray");
		for (var x = pastArray.length - 1; x > -1 && counter < 40; x--) {
			if (wordElements.indexOf('?q=' + pastArray[x].word + '"') <= 0) {
				wordElements += '<a href="http://sozluk.gov.tr/?q=' + pastArray[x].word + (pastArray[x].dict != null ? '&dic=' + pastArray[x].dict.replace(/ /g, "+") : '') + '" class="wordElements" title="' + convertDate(pastArray[x].time) + '">' + pastArray[x].word + '</a>';
				counter++;
			}
		}
		document.querySelector('div#past > span > ul').innerHTML = wordElements;
	}

	function convertDate(milliseconds) {
		var months = ["Ocak", "Şubat", "Mart", "Nisan", "Mayıs", "Haziran", "Temmuz", "Ağustos", "Eylül", "Ekim", "Kasım", "Aralık"];
		var d = new Date(milliseconds);
		var newDate = d.getDate() + ' ' + months[d.getMonth()] + ' ' + d.getFullYear() + ' (' + d.getHours().toString().padStart(2, "0") + ':' + d.getMinutes().toString().padStart(2, "0") + ')';
		return newDate;
	}

	function deleteWord(type) {
		if (type == "all") {
			var c = confirm("Arama geçmişinizi silmek istediğinize emin misiniz?");
			if (c) {
				GM_setValue("pastArray", []);
				document.querySelector('div#past > span > ul').innerHTML = '';
			}
		} else if (type == "last") {
			var pastArray = GM_getValue("pastArray");
			pastArray.splice(pastArray.length - 1, 1);
			GM_setValue("pastArray", pastArray);
			document.querySelector('div#past > span > ul > a').remove();
		} else if (isNumber(type)) {
			// En mükemmel kod henüz yazılmamış olandır.
		}
	}

	function isNumber(val) {
		return (val >= 0 || val < 0);
	}

	function storageUpt() {
		try {
			// İlk verisonlu sürüme (1.6) geçiş. Kelime ve zaman kütüphanelerini birleştirilip tek obje yap. ========
			if (GM_getValue("scriptVer") == null && GM_getValue("pastArray_word") != null) {
				var newPast = [];
				var oldPast_word = GM_getValue("pastArray_word");
				var oldPast_time = GM_getValue("pastArray_time");
				for (let x = 0; x < oldPast_word.length; x++) {
					newPast[x] = {
						word: oldPast_word[x],
						time: oldPast_time[x],
						dict: null
					};
				}
				GM_setValue("pastArray", newPast);
				GM_setValue("pastArray_word", null);
				GM_setValue("pastArray_time", null);
				GM_setValue("scriptVer", 1.6);
				alert("TDK Arama Yardımcısı Betiği:" + enter + enter + "Güncelleme işlemi tamamlandı. Bu yeni sürümde URL üzerinden TDK'nın diğer sözlüklerini de aramaya dahil edebilirsiniz. Derleme Sözlüğü, Atasözleri ve Deyimler Sözlüğü vs. Daha fazla ayrıntı için Greasyfork'taki eklenti görsellerine bakın.");
			}
			else if (GM_info.script.version > GM_getValue("scriptVer")) {
				GM_setValue("scriptVer", GM_info.script.version);
			}
		} catch (err) {
			alert("TDK Arama Yardımcısı Betiği:" + enter + enter + "Güncelleme işlemi sırasında bir sorun oluştu." + enter + "Sürüm " + GM_getValue("scriptVer") + " => Sürüm " + GM_info.script.version + enter + enter + "Lütfen aşağıdaki hata mesajı ile birlikte uygulama yazarına haber verin." + enter + enter + err.message);
		}
	}

	function waitForKeyElements(
		selectorTxt,
		/* Required: The jQuery selector string that
		                        specifies the desired element(s).
		                    */
		actionFunction,
		/* Required: The code to run when elements are
		                           found. It is passed a jNode to the matched
		                           element.
		                       */
		bWaitOnce,
		/* Optional: If false, will continue to scan for
		                      new elements even after the first match is
		                      found.
		                  */
		iframeSelector
		/* Optional: If set, identifies the iframe to
		                          search.
		                      */
	) {
		var targetNodes, btargetsFound;

		if (typeof iframeSelector == "undefined")
			targetNodes = $(selectorTxt);
		else
			targetNodes = $(iframeSelector).contents()
			.find(selectorTxt);

		if (targetNodes && targetNodes.length > 0) {
			btargetsFound = true;
			/*--- Found target node(s).  Go through each and act if they
			    are new.
			*/
			targetNodes.each(function() {
				var jThis = $(this);
				var alreadyFound = jThis.data('alreadyFound') || false;

				if (!alreadyFound) {
					//--- Call the payload function.
					var cancelFound = actionFunction(jThis);
					if (cancelFound)
						btargetsFound = false;
					else
						jThis.data('alreadyFound', true);
				}
			});
		} else {
			btargetsFound = false;
		}

		//--- Get the timer-control variable for this selector.
		var controlObj = waitForKeyElements.controlObj || {};
		var controlKey = selectorTxt.replace(/[^\w]/g, "_");
		var timeControl = controlObj[controlKey];

		//--- Now set or clear the timer as appropriate.
		if (btargetsFound && bWaitOnce && timeControl) {
			//--- The only condition where we need to clear the timer.
			clearInterval(timeControl);
			delete controlObj[controlKey];
		} else {
			//--- Set a timer, if needed.
			if (!timeControl) {
				timeControl = setInterval(function() {
						waitForKeyElements(selectorTxt,
							actionFunction,
							bWaitOnce,
							iframeSelector
						);
					},
					300
				);
				controlObj[controlKey] = timeControl;
			}
		}
		waitForKeyElements.controlObj = controlObj;
	}
})();