<div id="c100x50">
  <div id="area">
    <div class="bg"></div>
    <div class="house" v-bind:style="houseStyle"></div>
    <div class="trees" v-bind:style="treesStyle"></div>
    <div class="bushes" v-bind:style="bushesStyle"></div>
    <div class="stone_flower" v-bind:style="stone_flowerStyle"></div>
    <div class="fence" v-bind:style="fenceStyle"></div>
    <div class="event-mask"
         v-on:mousemove="parallaxMove"
         v-on:mouseout="originalPos"></div>
  </div>
</div>
:root {
  --bg: url("");
  --house: url("");
  --trees: url("");
  --bushes: url("");
  --stone_flower: url("");
  --fence: url("");
}

html, body {
  padding: 0;
  margin: 0;
}

#c100x50 {
  background-color: #303030;
  min-height: 100vh;
  margin: 0;
}

#area {
  width: 100px;
  height: 50px;
  border: 3px solid black;
  overflow: hidden;
  position: absolute;
  top: 50%;
  left: 50%;
  box-shadow: 0 8px 20px #000000;
  transform: translate(-50%, -50%);
}

.bg {
  background-image: var(--bg);
  width: 100px;
  height: 50px;
  position: absolute;
  top: 0;
  left: 0;
}

.house {
  background-image: var(--house);
  width: 110px;
  height: 60px;
}

.trees {
  background-image: var(--trees);
  width: 120px;
  height: 70px;
}

.bushes {
  background-image: var(--bushes);
  width: 130px;
  height: 80px;
}

.stone_flower {
  background-image: var(--stone_flower);
  width: 140px;
  height: 90px;
}

.fence {
  background-image: var(--fence);
  width: 150px;
  height: 100px;
}

.house, .trees, .bushes, .stone_flower, .fence {
  position: absolute;
  top: 50%;
  left: 50%;
  transition: top 400ms ease-out, left 400ms ease-out;
}

.event-mask {
  width: 100%;
  height: 100%;
  position: absolute;
  top: 0;
  left: 0;
}
const els = {
  house:        { w: 110, h: 60,  o: -5  }, 
  trees:        { w: 120, h: 70,  o: -10 }, 
  bushes:       { w: 130, h: 80,  o: -15 }, 
  stone_flower: { w: 140, h: 90,  o: -20 }, 
  fence:        { w: 150, h: 100, o: -25 }
};

new Vue({
  el: '#c100x50',
  data: {
    houseStyle:        { top: '-5px',  left: '-5px'  },
    treesStyle:        { top: '-10px', left: '-10px' },
    bushesStyle:       { top: '-15px', left: '-15px' },
    stone_flowerStyle: { top: '-20px', left: '-20px' },
    fenceStyle:        { top: '-25px', left: '-25px' }
  },
  methods: {
    calc: function(el, w2, h2, x, y) {
      const w1 = el.w;
      const h1 = el.h;
      const w  = w1 - w2;
      const h  = h1 - h2;
      
      return {
        top:  (w * y * -1) + 'px',
        left: (w * x * -1) + 'px'
      };
    },
    originalPos: function(e) {
      for (let i in els) {
        this[i + 'Style'] = {
          top:  els[i].o + 'px',
          left: els[i].o + 'px'
        };
      }
    },
    parallaxMove: function(e) {
      const el = e.target;
      const x  = e.offsetX + 1;
      const y  = e.offsetY + 1;
      const w  = el.clientWidth;
      const h  = el.clientHeight;
      const px = x / w;
      const py = y / h;

      for (let i in els) {
        this[i + 'Style'] =  this.calc(els[i], w, h, px, py);
      }
    }
  }
});
View Compiled

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. https://cdnjs.cloudflare.com/ajax/libs/vue/2.6.10/vue.min.js