最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

dom - Check if active element has particular class via javascript - Stack Overflow

programmeradmin2浏览0评论

I'm writing a single-field form and want the Enter key to advance the form to the next input field. Since there's another form on the page, I only want the Enter key to advance the form when one of the inputs in that form is the activeElement.

I've achieved this with what seems like an extremely verbose if() statement here:

document.addEventListener( 'keydown', function( ev ) {
        var keyCode = ev.keyCode || ev.which;
        var ids = [ 'this', 'that', 'there', 'thing', 'other' ];
        if ( document.getElementById( ids[0] ) === document.activeElement || document.getElementById( ids[1] ) === document.activeElement || document.getElementById( ids[2] ) === document.activeElement || document.getElementById( ids[3] ) === document.activeElement || document.getElementById( ids[4] ) === document.activeElement) {
            if( keyCode === 13 ) {
                ev.preventDefault();
                self._nextQuestion();
            }
        }
    } );

Each of the inputs are the same class: .questions. I've tried something like:

document.addEventListener( 'keydown', function( ev ) {
    var keyCode = ev.keyCode || ev.which;
    var ids = [ 'this', 'that', 'there', 'thing', 'other' ];
    if ( document.querySelector('.questions') === document.activeElement) {
        if( keyCode === 13 ) {
            ev.preventDefault();
            self._nextQuestion();
        }
    }
} );

But of course, this only accesses the first instance of .questions on the page. I don't want to iterate over nodes, as it doesn't seem much better than what I have already.

I'm a novice, and I'm looking for more concise logic.

I'm writing a single-field form and want the Enter key to advance the form to the next input field. Since there's another form on the page, I only want the Enter key to advance the form when one of the inputs in that form is the activeElement.

I've achieved this with what seems like an extremely verbose if() statement here:

document.addEventListener( 'keydown', function( ev ) {
        var keyCode = ev.keyCode || ev.which;
        var ids = [ 'this', 'that', 'there', 'thing', 'other' ];
        if ( document.getElementById( ids[0] ) === document.activeElement || document.getElementById( ids[1] ) === document.activeElement || document.getElementById( ids[2] ) === document.activeElement || document.getElementById( ids[3] ) === document.activeElement || document.getElementById( ids[4] ) === document.activeElement) {
            if( keyCode === 13 ) {
                ev.preventDefault();
                self._nextQuestion();
            }
        }
    } );

Each of the inputs are the same class: .questions. I've tried something like:

document.addEventListener( 'keydown', function( ev ) {
    var keyCode = ev.keyCode || ev.which;
    var ids = [ 'this', 'that', 'there', 'thing', 'other' ];
    if ( document.querySelector('.questions') === document.activeElement) {
        if( keyCode === 13 ) {
            ev.preventDefault();
            self._nextQuestion();
        }
    }
} );

But of course, this only accesses the first instance of .questions on the page. I don't want to iterate over nodes, as it doesn't seem much better than what I have already.

I'm a novice, and I'm looking for more concise logic.

Share Improve this question edited Nov 18, 2016 at 18:31 CollinD 7,5732 gold badges24 silver badges47 bronze badges asked Nov 18, 2016 at 17:57 J. Adam ConnorJ. Adam Connor 1,7344 gold badges21 silver badges38 bronze badges 4
  • 2 Could you check document.activeElement.className, maybe using .indexOf? – qxz Commented Nov 18, 2016 at 18:00
  • And does the forms have id's or anything else to distinguish them? If so, just listen on the form rather than the document – adeneo Commented Nov 18, 2016 at 18:00
  • Yes it has an id and a class, but when the cursor is in the input it is the only activeElement, not the form. – J. Adam Connor Commented Nov 18, 2016 at 18:05
  • 1 My point being, the event bubbles, first to the form, then to the document, if you only want to capture key events on one form, use a selector to target that form, – adeneo Commented Nov 18, 2016 at 18:07
Add a ment  | 

4 Answers 4

Reset to default 7

Just check if the activeElement has the questions class.

var pattern = /(?:^|\s)questions(?:\s|$)/
if (document.activeElement.className.match(pattern)) {
  ...
}

squint provided an improved regex that will account for more funky situations in the classname.

Try this:

if ((" "+document.activeElement.className+" ").indexOf(" questions ") > -1) {
    ...
}

First of all, you can utilise classList api to check element for given className presence, e.g. document.activeElement && document.activeElement.classList.contains('question').

All the proposed approaches are really helpful and should fix the problem. On the other hand, you might want to keep you application's state in one place so you can manage it easily. By doing so, you make your application more predictable and easier to debug.

Here's an approach you might want to take:

HTML Code:

<input class="question" value="1"/>
<input class="question" value="2"/>
<input class="question" value="3"/>
<input class="question" value="4"/>
<input class="question" value="5"/>

JavaScript Code:

var ponent = {
  activeQuestion: 0,
  questionSelector: '.question',
  get activeQuestionElement () {
    return ponent.allQuestionElements[ponent.activeQuestion];
  },
  get allQuestionElements () {
    return document.querySelectorAll(ponent.questionSelector);
  },
  next: function () {
    var activeElement = document.activeElement;

    if (activeElement && activeElement === ponent.activeQuestionElement && ponent.allQuestionElements[ponent.activeQuestion+1]) {
      ponent.activeQuestion++;
      ponent.highlightQuestion();
    }
  },
  highlightQuestion: function () {
    ponent.activeQuestionElement.focus();
  },
  install: function () {
    document.addEventListener('keyup', function (event) {
      if (event.which === 13) {
        ponent.next.call(ponent);
      }
    });
    document.addEventListener('click', function (event) {
      var className = ponent.questionSelector.slice(1);

      if (event.target.classList.contains(className)) {
        ponent.activeQuestion = [].slice.call(document.querySelectorAll(ponent.questionSelector)).indexOf(event.target);
        ponent.highlightQuestion();
      }
    })
  }
};

ponent.install();

As you can see above, ponent is a single object instance that holds useful information like activeQuestion index, question selector, some puted properties that return DOM elements. The install method binds event listeners which manage the state when events occur.

Here's a demo:

http://jsfiddle/maciejsmolinski/xzyvye8z/embedded/result/

When you click enter when any of the fields is focused, it is going to move focus to the next one. In addition to that, clicking on any of the fields changes active question as well. You can easily alter the behaviour by modifying the code I posted above. It is just a basic skeleton you can use to build your own functionality on top of.

This is a very basic ponent. You can build anything on top of it. You can introduce ES6 syntax (classes, arrow functions, const instead of var) but I intentionally left that part untouched to make it easier to understand.

Hope it helps!

You guys pointed me in the right direction with document.activeElement.className, so I upvoted all who mentioned it, but the most concise solution that works here seems to be this:

document.addEventListener( 'keydown', function( ev ) {
        var keyCode = ev.keyCode || ev.which;
        // enter
        if ( document.activeElement.className === 'questions') {
            if( keyCode === 13 ) {
                ev.preventDefault();
                self._nextQuestion();
            }
        }
    } );

I'll give you guys some time to critique this before I mark it as the answer...

发布评论

评论列表(0)

  1. 暂无评论