nicosearchで自動補完

    @@ -18,8 +18,8 @@ // IE9以降と最新のFirefox, Chromeで動作すると思います。Operaは知りません。 // 【直近の変更】 -// ・ニコニコ大百科へのリンクがクリックできなかったのを修正しました。 -// ・長すぎる候補が出たとき、文字を丸めるようにしました。 +// ・AND検索に対応しました。 +//  適当な実装なので、全角スペースで区切っていても候補を選択すると半角スペース区切りになります。 if (!Element.prototype.matchesSelector) { @@ -53,6 +53,7 @@ var REQUEST_METHOD = "post"; var REQUEST_URL = "http://nicoapi.com/ns/suggest.xml"; var RESPONSE_LENGTH = 10; +var QUERY_DELIMITER = /[\s ]/; var cache = {}; // 検索ワードと結果のペアを保持 var walker = null; // カーソルキーで動かした場合のTreeWalkerを保持 @@ -127,7 +128,6 @@ || document.getElementById("search-box") // ニコニコ大百科 || document.getElementById("jsSearchText") // SP版ニコニコ動画 || document.querySelector("#globalSearch .searchString") // SP版ニコニコチャンネル - ; return target; })(); @@ -172,6 +172,9 @@ }, false); searchBox.addEventListener("keydown", function(evt){ + var valueArr = searchBox.value.split(QUERY_DELIMITER); + valueArr.pop(); + var suggestbox = document.getElementById("nicosearch-suggestbox"); if (suggestbox) { if (!walker) walker = document.createTreeWalker( @@ -194,7 +197,8 @@ walker.currentNode.nextElementSibling.className = ""; walker.currentNode.className = "selected"; } - searchBox.value = walker.currentNode.textContent; + valueArr.push(walker.currentNode.textContent); + searchBox.value = valueArr.join(" "); break; case 40: // ↓ if (!suggestbox.querySelector(".selected")) { @@ -205,7 +209,8 @@ walker.currentNode.previousElementSibling.className = ""; walker.currentNode.className = "selected"; } - searchBox.value = walker.currentNode.textContent; + valueArr.push(walker.currentNode.textContent); + searchBox.value = valueArr.join(" "); break; case 13: // Enter if (suggestbox.querySelector(".selected")) { @@ -229,7 +234,8 @@ return false; } - var word = evt.target.value; + // 空白で区切った配列の最後を代入(AND検索用) + var word = evt.target.value.split(QUERY_DELIMITER).pop(); // 文字列が空の場合はサジェストを消す if (word.length === 0) { removeSuggestWindow();
  • /*
     * @title nicosearchで自動補完
     * @description niconicoにおいて、検索ワードのサジェストを表示します
     * @include http://www.nicovideo.jp/*
     * @include http://seiga.nicovideo.jp/*
     * @include http://live.nicovideo.jp/*
     * @include http://ch.nicovideo.jp/*
     * @include http://com.nicovideo.jp/*
     * @include http://dic.nicovideo.jp/*
     * @include http://sp.nicovideo.jp/*
     * @include http://sp.live.nicovideo.jp/*
     * @include http://sp.ch.nicovideo.jp/*
     * @license NYSL
     */
    
    
    // 個人的にニコニコ静画、ニコニコ生放送などの"ニコニコ動画以外"を全く使わないので、動作検証は不十分です。
    // IE9以降と最新のFirefox, Chromeで動作すると思います。Operaは知りません。
    
    // 【直近の変更】
    // ・AND検索に対応しました。
    //  適当な実装なので、全角スペースで区切っていても候補を選択すると半角スペース区切りになります。
    
    
    if (!Element.prototype.matchesSelector) {
    	Object.defineProperty(Element.prototype, "matchesSelector", {
    		enumerable: false,
    		configurable: true,
    		value: function(property){
    			if (this.webkitMatchesSelector) {
    				return this.webkitMatchesSelector(property);
    			}
    			else if (this.mozMatchesSelector) {
    				return this.mozMatchesSelector(property);
    			}
    			else if (this.msMatchesSelector) {
    				return this.msMatchesSelector(property);
    			}
    			else if (this.oMatchesSelector) {
    				return this.oMatchesSelector(property);
    			}
    			else
    				return null;
    		}
    	});
    }
    
    
    // ref: http://nicoapi.com/docs/nicosearchapi.html
    // ref: https://chrome.google.com/webstore/detail/pafehjbkojbbenbdklnlajbmlojnecjl
    (function main(){
    
    var REQUEST_METHOD  = "post";
    var REQUEST_URL     = "http://nicoapi.com/ns/suggest.xml";
    var RESPONSE_LENGTH = 10;
    var QUERY_DELIMITER = /[\s ]/;
    
    var cache  = {};       // 検索ワードと結果のペアを保持
    var walker = null;     // カーソルキーで動かした場合のTreeWalkerを保持
    var tID    = null;     // sendRequestのtimeout用
    
    (function createStyle(){
    	if (document.querySelector("#nicosearch-style")) {
    		return false;
    	}
    	var style = document.createElement("style"), i = 0;
    	style.id = "nicosearch-style";
    	document.getElementsByTagName("head")[0].appendChild(style);
    	([
    		"#nicosearch-suggestbox {\
    			position: absolute;\
    			border:  1px solid #778899;\
    			background-color: #fff;\
    			z-index: 100000;\
    			-webkit-user-select: none;\
    			 -khtml-user-select: none;\
    			   -moz-user-select: none;\
    			    -ms-user-select: none;\
    			     -o-user-select: none;\
    			        user-select: none;\
    		}",
    		"#nicosearch-suggestbox * {\
    			text-align: left;\
    		}",
    		"#nicosearch-suggestbox > div {\
    			position: relative;\
    			line-height: normal;\
    			overflow:    hidden;\
    		}",
    		"#nicosearch-suggestbox > div.selected\
    		,#nicosearch-suggestbox > div:hover {\
    			background-color: #afeeee;\
    		}",
    		"#nicosearch-suggestbox a {\
    			display: block;\
    		}",
    		".nicosearch-dic-icon {\
    			padding-top:    1px;\
    			padding-bottom: 1px;\
    		}",
    		".nicosearch-name {\
    			position:      absolute;\
    			top:           0;\
    			left:          18px;\
    			width:         calc(100% - 18px);\
    			height:        18px;\
    			font-size:     14px;\
    			text-overflow: ellipsis;\
    			word-wrap:     normal;\
    			white-space:   nowrap;\
    			overflow:      hidden;\
    			cursor:        pointer;\
    		}"
    	]).forEach(function(val){
    		style.sheet.insertRule(val, i++);
    	});
    })();
    
    var searchBox = (function getSearchBox(){
    	var target = document.getElementById("searchWord")                    // トップページ
    	|| document.getElementById("bar_search")                              // ニコニコ動画
    	|| document.getElementById("search_united")                           // 検索結果
    	|| document.querySelector(".searchText input")                        // 動画ページ
    	|| document.getElementById("bar_search")                              // ニコニコ静画
    	|| document.querySelector(".search_program .search_word")             // ニコニコ生放送
    	|| document.getElementById("searchtxt")                               // ニコニコチャンネル
    	|| document.getElementById("search_text")                             // ニコニココミュニティ
    	|| document.getElementById("search-box")                              // ニコニコ大百科
    	|| document.getElementById("jsSearchText")                            // SP版ニコニコ動画
    	|| document.querySelector("#globalSearch .searchString")              // SP版ニコニコチャンネル
    	;
    	return target;
    })();
    if (searchBox === null) {
    	window.alert("入力欄が見つかりません。");
    	return false;
    }
    
    var XmlHttp = function(word){
    	var xhr = new XMLHttpRequest();
    	xhr.open(REQUEST_METHOD, REQUEST_URL, true);
    	xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8");
    	xhr.addEventListener("readystatechange", function(evt){
    		if (xhr.readyState === 4) {
    			switch(xhr.status) {
    				case 200:
    					cache[word.toLowerCase()] = xhr.responseXML; // キャッシュに追加
    					createSuggestWindow(xhr.responseXML);
    					break;
    			}
    		}
    	}, false);
    	return xhr;
    };
    // どうやらIE9はXMLHttpRequestをクロスドメインには使えないみたいで…
    var Xdr = function(word, query){
    	var xdr = new XDomainRequest();
    	xdr.open("GET", REQUEST_URL + "?" + query);
    	xdr.onload = function(){
    		var xml = DOMParser().parseFromString(xdr.responseText, "application/xml");
    		cache[word.toLowerCase()] = xml; // キャッシュに追加
    		createSuggestWindow(xml);
    	};
    	return xdr;
    };
    
    
    document.addEventListener("click", function(evt){
    	if (!evt.target.matchesSelector("#nicosearch-suggestbox") || evt.target !== searchBox) {
    		removeSuggestWindow();
    	}
    }, false);
    
    searchBox.addEventListener("keydown", function(evt){
    	var valueArr = searchBox.value.split(QUERY_DELIMITER);
    	valueArr.pop();
    	
    	var suggestbox = document.getElementById("nicosearch-suggestbox");
    	if (suggestbox) {
    		if (!walker) walker = document.createTreeWalker(
    			suggestbox,
    			NodeFilter.SHOW_ELEMENT,
    			function(node){
    				return node.parentElement === suggestbox
    				? NodeFilter.FILTER_ACCEPT
    				: NodeFilter.FILTER_SKIP;
    			},
    			false
    		);
    		switch(evt.keyCode) {
    			case 38: // ↑
    				if (!suggestbox.querySelector(".selected")) {
    					walker.lastChild();
    					walker.currentNode.className = "selected";
    				}
    				else if (walker.previousNode()) {
    					walker.currentNode.nextElementSibling.className = "";
    					walker.currentNode.className = "selected";
    				}
    				valueArr.push(walker.currentNode.textContent);
    				searchBox.value = valueArr.join(" ");
    				break;
    			case 40: // ↓
    				if (!suggestbox.querySelector(".selected")) {
    					walker.firstChild();
    					walker.currentNode.className = "selected";
    				}
    				else if (walker.nextNode()) {
    					walker.currentNode.previousElementSibling.className = "";
    					walker.currentNode.className = "selected";
    				}
    				valueArr.push(walker.currentNode.textContent);
    				searchBox.value = valueArr.join(" ");
    				break;
    			case 13: // Enter
    				if (suggestbox.querySelector(".selected")) {
    					walker.currentNode.getElementsByTagName("p")[0].click();
    				}
    				break;
    			default:
    				if (suggestbox.querySelector(".selected")) {
    					suggestbox.getElementsByClassName("selected")[0].className = "";
    				}
    		}
    	}
    }, false);
    
    searchBox.addEventListener("keyup", function(evt){
    	window.clearTimeout(tID);
    	
    	// 選択状態の場合はなにもしない
    	var suggestbox = document.getElementById("nicosearch-suggestbox");
    	if (suggestbox && suggestbox.querySelector(".selected")) {
    		return false;
    	}
    	
    	// 空白で区切った配列の最後を代入(AND検索用)
    	var word = evt.target.value.split(QUERY_DELIMITER).pop();
    	// 文字列が空の場合はサジェストを消す
    	if (word.length === 0) {
    		removeSuggestWindow();
    		return false;
    	}
    	// キャッシュをチェック
    	// 既に結果をもらっている場合は二度送らない
    	if (cache[word.toLowerCase()]) {
    		createSuggestWindow(cache[word.toLowerCase()]);
    	}
    	else {
    		tID = window.setTimeout(function(){
    			sendRequest(word);
    		}, 500);
    	}
    }, false);
    
    
    function sendRequest(word){
    	var xhr;
    	var query = "i=" + encodeURI(word) + "&l=" + RESPONSE_LENGTH;
    	try {
    		xhr = new XmlHttp(word);
    	} catch(e) {
    		xhr = new Xdr(word, query); // IE9用
    	}
    	xhr.send(query);
    }
    
    function removeSuggestWindow(){
    	var suggestbox = document.getElementById("nicosearch-suggestbox");
    	walker = null;
    	if (suggestbox) {
    		suggestbox.parentElement.removeChild(suggestbox);
    	}
    }
    
    function createSuggestWindow(xml){
    	if (xml.documentElement.getAttribute("status") !== "ok") return false;
    	
    	walker = null;
    	var wrapper = document.getElementById("nicosearch-suggestbox");
    	if (!wrapper) {
    		wrapper = (function createWrapper(){
    			var wrapper = document.createElement("div");
    			var boxRect = searchBox.getBoundingClientRect();
    			wrapper.id = "nicosearch-suggestbox";
    			wrapper.style.top      = boxRect.top + boxRect.height + "px";
    			wrapper.style.left     = boxRect.left + "px";
    			wrapper.style.minWidth = searchBox.offsetWidth  + "px";
    			return wrapper;
    		})();
    	}
    	else {
    		while (wrapper.hasChildNodes()) {
    			wrapper.removeChild(wrapper.firstChild);
    		}
    	}
    	
    	var img = (function createImage(){
    		var img = document.createElement("img");
    		img.className = "nicosearch-dic-icon";
    		img.src = "";
    		return img;
    	})();
    	
    	Array.prototype.forEach.call(xml.getElementsByTagName("item"), function(val){
    		var div = document.createElement("div");
    		if (document.ontouchstart) {
    			div.style.padding = "4px 0"; // SP版の場合ちょっと縦に幅を持たせる
    		}
    		var a = (function createAnchor(){
    			var a = document.createElement("a");
    			a.href   = "http://dic.nicovideo.jp/a/" + encodeURIComponent(val.textContent);
    			a.target = "_blank";
    			return a;
    		})();
    		var p = (function createParagraph(){
    			var p = document.createElement("p");
    			p.className = "nicosearch-name";
    			return p;
    		})();
    		p.addEventListener("click", function(evt){
    			searchBox.value = val.textContent;
    			removeSuggestWindow();
    		}, false);
    		a.appendChild(img.cloneNode(false));
    		div.appendChild(a);
    		p.appendChild(document.createTextNode(val.textContent));
    		div.appendChild(p);
    		wrapper.appendChild(div);
    	});
    	document.body.appendChild(wrapper);
    }
    
    })();
  • Permalink
    このページへの個別リンクです。
    RAW
    書かれたコードへの直接のリンクです。
    Packed
    文字列が圧縮された書かれたコードへのリンクです。
    Userscript
    Greasemonkey 等で利用する場合の .user.js へのリンクです。
    Loader
    @require やソースコードが長い場合に多段ロードする Loader コミのコードへのリンクです。
    Metadata
    コード中にコメントで @xxx と書かれたメタデータの JSON です。

History

  1. 2013/05/28 03:43:42 - 2013-05-28
  2. 2013/05/25 20:26:56 - 2013-05-25
  3. 2013/05/25 20:21:07 - 2013-05-25
  4. 2013/05/25 20:14:54 - 2013-05-25
  5. 2013/05/25 01:46:27 - 2013-05-25
  6. 2013/05/21 19:07:56 - 2013-05-21
  7. 2013/05/18 02:03:22 - 2013-05-18
  8. 2013/05/15 21:49:02 - 2013-05-15
  9. 2013/05/15 21:21:32 - 2013-05-15
  10. 2013/05/15 00:33:54 - 2013-05-15
  11. 2013/05/15 00:16:22 - 2013-05-15
  12. 2013/05/15 00:15:10 - 2013-05-15
  13. 2013/05/13 18:36:18 - 2013-05-13
  14. 2013/05/13 15:54:32 - 2013-05-13
  15. 2013/05/13 15:23:01 - 2013-05-13
  16. 2013/05/13 15:08:07 - 2013-05-13
  17. 2013/05/13 08:57:59 - 2013-05-13
  18. 2013/05/13 08:44:37 - 2013-05-13
  19. 2013/05/13 08:20:17 - 2013-05-13
  20. 2013/05/13 08:17:31 - 2013-05-13