DOMおれおれAdvent Calendar 2015 – 24日目
最終日です。
jQueryでやってる事をjQueryなしでやるとしたらどうするのってやつです。
$(selector) → document.querySelectorAll(selector)
これが一番基本ですね。
// jQuery
var $el = $('.foo');
console.log($el.length);
console.log($el[0]);
// DOM
var els = document.querySelectorAll('.foo');
console.log(els.length);
console.log(els[0]);
// DOM(一件のみ)
var el = document.querySelector('.foo');
console.log(el);
find() → querySelectorAll()
// jQuery
var $el = $('.foo');
console.log($el.find('.bar'));
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].querySelectorAll('.bar'));
each() → Array.prototype.forEach()
document.querySelectorAll() の結果には each() のように「全てに適用」する仕組みがありません。自前で for 文を書くか、配列の forEach() を利用するのが便利です。
// jQuery
var $el = $('.foo');
$el.each(function(index, el) {
    console.log(index, el);
});
// DOM
var els = document.querySelectorAll('.foo');
Array.prototype.forEach.call(els, function(el, index) {
    console.log(index, el);
});
コールバック関数に与えられる引数の順序が異なる点に気を付けてください。
on() → addEventListener()
// jQuery
var $el = $('.foo');
$el.on('click', function(event) {
    console.log(event.type);
});
// DOM
var els = document.querySelectorAll('.foo');
els[0].addEventListener('click', function(event) {
    console.log(event.type);
});
document.querySelectorAll() の結果は複数件になりますが、 addEventListener() は一件ずつしか適用できません。複数件を対象にイベント監視する場合は別項で紹介した Array.prototype.forEach() を使います。
以降も同じです。
デリゲートは event.target.matches(selector)
デリゲートっていう表現してたよね? $area.on('click', '.inner', listener) みたいなやつ。
// jQuery
var $el = $('.foo');
$el.on('click', '.bar', function(event) {
    console.log(event.type);
});
// DOM
var els = document.querySelectorAll('.foo');
els[0].addEventListener('click', function(event) {
    if (event.target.matches('.bar')) {
        console.log(event.type);
    }
});
off() → removeEventListener()
var callback = function(event) {
    console.log(event.type);
};
// jQuery
var $el = $('.foo');
$el.off('click', callback);
// DOM
var els = document.querySelectorAll('.foo');
els[0].removeEventListener('click', callback);
解除するコールバック関数のインスタンスが必要なのがちょっと面倒なところ。WeakMapなんか使えると管理が楽そうなんだけどなー。
trigger() → dispatchEvent()
割と面倒くさい。
// jQuery
var $el = $('.foo');
$el.trigger('mogu');
// DOM
var els = document.querySelectorAll('.foo');
var myEvent = document.createEvent('CustomEvent');
myEvent.initCustomEvent('mogu', true, true, null);
els[0].dispatchEvent(myEvent);
クラス操作 → classList
addClass() → classList.add()
// jQuery
var $el = $('.foo');
$el.addClass('is-active');
// DOM
var els = document.querySelectorAll('.foo');
els[0].classList.add('is-active');
removeClass() → classList.remove()
// jQuery
var $el = $('.foo');
$el.removeClass('is-active');
// DOM
var els = document.querySelectorAll('.foo');
els[0].classList.remove('is-active');
toggleClass() → classList.toggle()
// jQuery
var $el = $('.foo');
$el.toggleClass('is-active');
// DOM
var els = document.querySelectorAll('.foo');
els[0].classList.toggle('is-active');
hasClass() → classList.has()
// jQuery
var $el = $('.foo');
$el.hasClass('is-active');
// DOM
var els = document.querySelectorAll('.foo');
els[0].classList.contains('is-active');
css() → style 、 getComputedStyle()
数値 200 は設定してもCSS的に不正なので、単位付き "200px" で指定する必要があります。取得時も単位付きで得られます。
// jQuery
var $el = $('.foo');
$el.css({ color:'red, width:200 });
console.log($el.css('color'));
// DOM
var els = document.querySelectorAll('.foo');
els[0].style.color = 'red';
els[0].style.width = '200px';
console.log(getComputedStyle(els[0]).color);
attr() → setAttribute() 、 getAttribute()
// jQuery
var $el = $('.foo');
$el.attr('href', '#top');
console.log($el.attr('href'));
// DOM
var els = document.querySelectorAll('.foo');
els[0].setAttribute('href', '#top');
console.log(els[0].getAttribute('href'));
prop() → プロパティ
// jQuery
var $el = $('.foo');
$el.prop('href', '#top');
console.log($el.prop('href'));
// DOM
var els = document.querySelectorAll('.foo');
els[0].href = '#top';
console.log(els[0].href);
val() → value
// jQuery
var $el = $('.foo');
$el.val('I like it!');
console.log($el.val());
// DOM
var els = document.querySelectorAll('.foo');
els[0].value = 'I like it!';
console.log(els[0].value);
text() → textContent
// jQuery
var $el = $('.foo');
$el.text('<b>BOLD</b>');
console.log($el.text());
// DOM
var els = document.querySelectorAll('.foo');
els[0].textContent = '<b>BOLD</b>';
console.log(els[0].textContent);
html() → innerHTML
// jQuery
var $el = $('.foo');
$el.html('<b>BOLD</b>');
console.log($el.html());
// DOM
var els = document.querySelectorAll('.foo');
els[0].innerHTML = '<b>BOLD</b>';
console.log(els[0].innerHTML);
$(html) → innerHTML を使って自作
// jQuery
var $el = $('<div>');
console.log($el);
// DOM
var wrapper = document.createElement('div');
wrapper.innerHTML = '<div></div>';
var el = wrapper.firstChild;
console.log(el);
is() → matches()
// jQuery
var $el = $('.foo');
console.log($el.is('.foo'));
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].matches('.foo'));
parent() → parentNode
// jQuery
var $el = $('.foo');
console.log($el.parent());
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].parentNode);
closest() → 自作
// jQuery
var $el = $('.foo');
console.log($el.closest('.wrapper'));
// DOM
var els = document.querySelectorAll('.foo');
for (var closest=els[0]; closest; closest=closest.parentElement) {
    if (closest.matches('.wrapper')) {
        break;
    }
}
console.log(closest);
children() → children
// jQuery
var $el = $('.foo');
var $children = $el.children();
for (var i=0, l=$children.length; i<l; i++) {
    console.log($children[i]);
}
// DOM
var els = document.querySelectorAll('.foo');
var children = els[0].children;
for (var i=0, l=children.length; i<l; i++) {
    console.log(children[i]);
}
その他 firstElementChild 、 lastElementChild というのもあります。
prev() → previousElementSibling
// jQuery
var $el = $('.foo');
console.log($el.prev());
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].previousElementSibling);
next() → nextElementSibling
// jQuery
var $el = $('.foo');
console.log($el.next());
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].nextElementSibling);
filter() → Array.prototype.filter()
// jQuery
var $el = $('.foo');
console.log($el.filter('.bar'));
// DOM
var els = document.querySelectorAll('.foo');
var filtered = Array.prototype.filter.call(els, function(el) {
    return el.matches('.bar');
});
console.log(filtered);
append(el) → appendChild(el)
// jQuery
var $el = $('.foo');
var $child = $('.child');
$el.append($child);
// DOM
var els = document.querySelectorAll('.foo');
var children = document.querySelectorAll('.child');
els[0].appendChild(children[0]);
append(html) → innerHTML と appendChild(el) で自作
別項で紹介したやつの組み合わせです。
// jQuery
var $el = $('.foo');
$el.append('<b>BOLD</b>');
// DOM
var els = document.querySelectorAll('.foo');
var wrapper = document.createElement('div');
wrapper.innerHTML = '<b>BOLD</b>';
var child = wrapper.firstChild;
els[0].appendChild(child);
remove() → removeChild()
// jQuery
var $el = $('.foo');
$el.remove();
// DOM
var els = document.querySelectorAll('.foo');
els[0].parentNode.removeChild(els[0]);
width() → clientWidth
// jQuery
var $el = $('.foo');
$el.width(200);
console.log($el.width());
// DOM
var els = document.querySelectorAll('.foo');
els[0].style.width = '200px';
console.log(els[0].clientWidth);
outerWidth() → getBoundingClientRect()
// jQuery
var $el = $('.foo');
console.log($el.outerWidth());
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].getBoundingClientRect().width);
position() → offsetLeft 、 offsetTop
// jQuery
var $el = $('.foo');
var position = $el.position();
console.log(position.left, position.top);
// DOM
var els = document.querySelectorAll('.foo');
console.log(els[0].offsetLeft, els[0].offsetTop);
offset() → getBoundingClientRect()
// jQuery
var $el = $('.foo');
var offset = $el.offset();
console.log(offset.top);
// DOM
var els = document.querySelectorAll('.foo');
var offset = els[0].getBoundingClientRect();
var scrollTop = document.documentElement.scrollTop || document.body.scrollTop;
var top = scrollTop + offset.top;
console.log(top);
その他
DOMじゃないけど。
$.ajax() → XMLHttpRequest で頑張る
簡単な例だけ。
// jQuery
$.ajax({
    url: '/api/sugoi',
    data: { foo:123 }
}).done(function(data) {
    console.log(data);
});
// DOMじゃない
var xhr = new XMLHttpRequest();
xhr.addEventListener('load', function(event) {
    var data = JSON.parse(this.response);
  console.log(data);
});
xhr.open('GET', '/api/sugoi');
xhr.send(JSON.stringify({ foo:123 }));
おしまい
こんな感じの実装を、昨日分の記事の「基本戦略」と組み合わせると「おれおれjQuery」ができあがります。
というわけで、よければgQuery使ってみてください。
