4

I have a simple web kiosk that displays a user action prompt after 30 seconds of the mouse not moving with jQuery.

var i = null;
  $("#body-wrap").mousemove(function() {
    clearTimeout(i);
    $("#overlay").fadeOut();
    i = setTimeout(function() {
      $("#overlay").fadeIn();
    }, 30000);
  })

The kiosk is a video-selection stand. The video is played full screen and I'm running into an issue where the action is displayed beneath the video frame because the mouse hasn't moved.

Is there a way to check for browser activity with JavaScript other than mouse or key events?

Brian
  • 4,274
  • 2
  • 27
  • 55
  • I doubt it. See http://stackoverflow.com/questions/667555/detecting-idle-time-in-javascript-elegantly – Klompenrunner Sep 18 '16 at 03:26
  • How is "browser activity" determined? Do you want to prompt user every 30 seconds? – guest271314 Sep 18 '16 at 03:39
  • @guest271314 It's a pseudo-screensaver. If the mouse hasn't moved, it's there so people know how to interact. But, if a video is longer than 30 seconds (they are) it kicks them back to that prompt. So, there is no user activity, but the browser window is definitely active. – Brian Sep 18 '16 at 19:09
  • Should the prompt occur every 30 seconds until end of video? – guest271314 Sep 18 '16 at 19:19
  • @BrianBennett Is the issue with approach at Question is that during full screen the prompt is displayed outside of the full screen viewport? – guest271314 Sep 18 '16 at 20:54
  • @guest271314 Yeah, each video is a hidden `div` element and the correct video is pulled up by user action. I have a demo of the site set up here: http://dev.ohheybrian.com/church – Brian Sep 19 '16 at 12:39

2 Answers2

2

The video is played full screen and I'm running into an issue where the action is displayed beneath the video frame because the mouse hasn't moved.

If the issue is message to user being rendered beneath <video> element when user has requested fullscreen display, you can use <track> element with src set to .vtt file, or .addTextTrack() method of HTMLMediaElement, VTTCue(), .addCue() to add a cue message to be displayed at <video> element at specific time.

Use loadedmetadata and ended event of HTMLMediaElement to set, reset TextTrackCue. Also set, reset cue to 30 seconds from current user activity at mousemove event handler.

You can view the linked plnkr at a new window by clicking "Launch the preview in a separate window" icon, then click <button> element having text "toggle fullscreen".

<!DOCTYPE html>
<html>
<head>
 <style>
 ::cue {
   background: transparent;
 }

 ::cue(.prompt) {
   color: #d81159;
   background: transparent;
   text-shadow: 2px 2px 2px #ffbc42;
   font-family: Menlo, Sans, Monospace;
   font-size: 36px;
   display: block;
   width: 100px;
 }

 ::cue(b) {
   display: block;
   width: 100px;
 }
 </style>
</head>
<body>
<button>
  toggle fullscreen
</button>
<br>
<video 
  width="320px"
  height="280px"
  autoplay="autoplay" 
  controls="controls"
  src="http://nickdesaulniers.github.io/netfix/demo/frag_bunny.mp4"
  poster="">     
</video>
<script>
  var video = document.querySelector("video");
  var button = document.querySelector("button");
  video.requestFullscreen = video.requestFullscreen 
                            || video.mozRequestFullscreen 
                            || video.webkitRequestFullscreen;
  video.exitFullscreen = video.exitFullscreen 
                         || video.mozExitFullscreen 
                         || video.webkitExitFullscreen;
  var track = video.addTextTrack("captions", "prompt", "en");
  track.mode = "showing";
  var overlay = "<c.prompt><b>No browser activity<b></c>";
  var isFullscreen = false;
  var currentCue;
  var duration;

  video.onloadedmetadata = function(e) {
    duration = this.duration;
    currentCue = createCue(30, Math.round(duration));
    track.addCue(currentCue);
  }

  video.onended = video.onpause = function() {
    removeCues(track);
    console.log(track.cues);
    currentCue = createCue(30
                   , Math.round(duration || video.duration));
    track.addCue(currentCue);
  }

  button.onclick = function() {
    if (isFullscreen) {
      video.exitFullscreen;
      video.setAttribute("width", "320px");
      video.setAttribute("height", "280px");
      isFullscreen = false;
    } else {
      video.removeAttribute("width");
      video.removeAttribute("height");
      video.requestFullscreen();
      isFullscreen = true;
    }
  }
  
  window.onmousemove = function() {
    if (track.cues.length && currentCue && duration) {
      removeCues(track);
      currentCue = createCue(
        Math.round(video.currentTime) + 30
        , Math.round(duration || video.duration)
      );
      track.addCue(currentCue);
    }
  }

  function createCue(begin, end) {
    var cue = new VTTCue(begin, end, overlay);
    cue.line = 5;
    cue.size = 100;
    return cue
  }

  function removeCues(track) {
    for (var i = 0; i < track.cues.length; i++) {
      track.removeCue(track.cues[i]);
    }
  }
</script>
</body>
</html>
guest271314
  • 1
  • 15
  • 104
  • 177
0

Depending on how you're playing the video, toggle a global flag, videoPlaying, on video play/pause. Then;

$("#body-wrap").mousemove(function() {
    clearTimeout(i);
    $("#overlay").fadeOut();
    i = setTimeout(function() {
      if(!videoPlaying)
        $("#overlay").fadeIn();
    }, 30000);
  })
Jake Haller-Roby
  • 6,335
  • 1
  • 18
  • 31