Vue實現移動端頁面切換效果

找了好多博客實現效果都……emmm……html

 

應用Vue自帶的過渡 《 進入/離開 & 列表過渡 》和 嵌套路由 和 fixed定位實現vue

其實仍是挺簡單的。vue-router

在子頁面把整個頁面作絕對定位,覆蓋整個屏幕,子父頁面將 router-view 用 transition 套起來,並加上過渡動畫就能夠啦。app

代碼:ide

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
  <title>Document</title>
  <style>
    * { padding: 0; margin: 0; }
    html, body, #app { width: 100%; height: 100%; }
    .one { height: 100%; background-color: yellow; }
    .two { background-color: tomato; position: fixed; top: 0; bottom: 0; left: 0; right: 0; }
    .three { background-color: #ffe69f; position: fixed; top: 0; bottom: 0; left: 0; right: 0; }
    .v-enter-active, .v-leave-active { transition: all 0.3s; }
    .v-enter, .v-leave-to { transform: translateX(100%); }
  </style>
</head>
<body>
  <div id="app">
    <div class="one">
      <p>
        <router-link to="/foo">下一層</router-link>
      </p>
      <h1>第一層</h1>
    </div>
    <transition>
      <router-view></router-view>
    </transition>
  </div>

  <script src="https://unpkg.com/vue/dist/vue.js"></script>
  <script src="https://unpkg.com/vue-router/dist/vue-router.js"></script>
  <script>
    const Foo = {
      template: `
        <div class="whole-page two">
          <router-link to="/foo/bar">下一層</router-link>
          <router-link to="/">返回</router-link>
          <h1>第二層</h1>
          <transition>
            <router-view></router-view>
          </transition>
        </div>
      `
    }
    const Bar = {
      template: `
        <div class="whole-page three">
          <router-link to="/foo">返回</router-link>
          <h1>第三層</h1>
          <transition>
            <router-view></router-view>
          </transition>
        </div>
      `
    }
    const routes = [ 
      { path: '/foo', component: Foo, children: [ { path: 'bar', component: Bar } ] }
    ]
    const router = new VueRouter({ routes })
    const app = new Vue({ router }).$mount('#app')
  </script>
</body>
</html>

 

效果:動畫

 

有一個問題須要注意一下,ui

咱們知道,在應用transform屬性的時候,fixed定位會變成absolute。spa

這裏,頁面轉換的時候,就變成了相對translation定位。因此若是子頁面中有絕對定位的話,移動的過程當中頁面會變形。scala

簡單舉個栗子,code

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
  <title>Document</title>
  <style>
* { padding: 0; margin: 0; }
html, body, #app { width: 100%; height: 100%; }
#app { padding-top: 50px; }
.one { height: 100%; background-color: yellow;}
.two { background-color: tomato; position: fixed; top: 100px; bottom: 0; left: 0; right: 0; }.v-enter-active, .v-leave-active { transition: all 0.3s; }
.v-enter, .v-leave-to { transform: translateX(100%); }
header { height: 50px; background-color: #000; width: 100%; position: fixed; top: 0; color: #fff; line-height: 50px; text-align: center; }
.two header { top: 50px; background-color: #666; }
  </style>
</head>
<body>
  <div id="app">
    <header>我是一個標題</header>
    <div class="one">
      <p>
        <router-link to="/foo">下一層</router-link>
      </p>
      <h1>第一層</h1>
      <transition>
        <router-view></router-view>
      </transition>
    </div>
  </div>

  <script src="https://unpkg.com/vue/dist/vue.js"></script>
  <script src="https://unpkg.com/vue-router/dist/vue-router.js"></script>
  <script>
    const Foo = {
      template: `
        <div class="whole-page two">
          <router-link to="/">返回</router-link>
          <header>我也是一個標題</header>
          <h1>第二層</h1>
          <transition>
            <router-view></router-view>
          </transition>
        </div>
      `
    }
    const routes = [ 
      { path: '/foo', component: Foo }
    ]
    const router = new VueRouter({ routes })
    const app = new Vue({ router }).$mount('#app')
  </script>
</body>
</html>

看下效果:

 

OKOK,反正就是這種bug嘛。

解決辦法就是,就是,儘可能讓頁面fixed定位都是0 0 0 0,而後偏移用padding實現。

大概吧……反正我是這麼解決的……

好比上面那個能夠把CSS改爲這樣解決問題。

* { padding: 0; margin: 0; }
html, body, #app { width: 100%; height: 100%; }
#app { padding-top: 50px; }
.one { height: 100%; background-color: yellow;}
.two { background-color: tomato; position: fixed; top: 0; padding-top: 100px; bottom: 0; left: 0; right: 0; }.v-enter-active, .v-leave-active { transition: all 0.3s; }
.v-enter, .v-leave-to { transform: translateX(100%); }
header { height: 50px; background-color: #000; width: 100%; position: fixed; top: 0; color: #fff; line-height: 50px; text-align: center; z-index: 100; }
.two header { top: 50px; background-color: #666; }

 

嗯嗯 還有一個問題,還有個滑動穿透的問題,(真開心! 這麼多問題!

我再舉個栗子,

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0, user-scalable=no">
  <title>Document</title>
  <style>
* { padding: 0; margin: 0; }
html, body, #app { width: 100%; height: 100%; }
.one { min-height: 100%; background-color: yellow;}
.two { background-color: tomato; position: fixed; top: 0; bottom: 0; left: 0; right: 0; }
.three { background-color: #ffe69f; position: fixed; top: 50px; bottom: 0; left: 0; right: 0; }
.v-enter-active, .v-leave-active { transition: all 0.3s; }
.v-enter, .v-leave-to { transform: translateX(100%); }
  </style>
</head>
<body>
  <div id="app">
    <div class="one">
      <p>
        <router-link to="/foo">下一層</router-link>
      </p>
      <h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1>
      <h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1>
      <h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1><h1>第一層</h1>
      <transition>
        <router-view></router-view>
      </transition>
    </div>
  </div>

  <script src="https://unpkg.com/vue/dist/vue.js"></script>
  <script src="https://unpkg.com/vue-router/dist/vue-router.js"></script>
  <script>
    const Foo = {
      template: `
        <div class="whole-page two">
          <router-link to="/">返回</router-link>
          <h1>第二層</h1>
          <transition>
            <router-view></router-view>
          </transition>
        </div>
      `
    }
    const routes = [ 
      { path: '/foo', component: Foo }
    ]
    const router = new VueRouter({ routes })
    const app = new Vue({ router }).$mount('#app')
  </script>
</body>
</html>

看效果,第二頁的高度明明就是視窗的高度,可是它有一個滾動條,實際上那是第一個頁面的滾動條。

網上找了好多方法,一一試了,所有不生效。(固然頗有多是個人方法不對。

 

最後沒辦法只有找最笨的方法啦,就是經過  v-if  把父頁面不顯示就行了。

固然不能直接不顯示,由於動畫還沒結束父元素就空白了呀!setTimeout 就行了……

具體代碼就不寫了,這個應該很容易理解。

 

還有什麼問題,等我想起來在補充。或者還有什麼沒注意到的問題,歡迎路過的大佬們提出呀,反正我也解決不了。

相關文章
相關標籤/搜索