preface

This is to build the general appearance framework (basically modeled after wechat), and it is not unchanged and may be changed later. There are many resources available with off-the-shelf open source products that can be leveraged directly for your own projects.

  • Element – UI is a Vue 2.0-based desktop component library for developers, designers, and product managers. The front-end component library most commonly used by VUE projects has a rich set of components to work with.

  • Iconfont Ali Mom MUX dedicated to create vector icon management, communication platform. Designers upload ICONS to the Iconfont platform, and users can customize and download ICONS in various formats. The platform can also convert ICONS into fonts, which is convenient for front-end engineers to adjust and call freely.

  • Chinese color Chinese traditional color matching website.

1. The integrated element – the UI

Add element-UI to dependencies in package.json

 "dependencies": {
    "core-js": "^ 3.6.5." "."vue": "^ 2.6.11." "."vue-router": "^ 3.2.0"."vuex": "^ 3.4.0"."element-ui": "^ 2.13.2"
  }
Copy the code

Run the yarn command in the root directory of the console project. Because it is a desktop software, it does not consider on-demand reference, directly all reference can be. Add to main.js:

/ / into the element
import ElementUI from 'element-ui'
import 'element-ui/lib/theme-chalk/index.css'

import Vue from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'

Vue.config.productionTip = false

/ / into the element
Vue.use(ElementUI)
Copy the code

2. Integrate the Remote module

Add to main.js:

// Get the remote call
const remote = require('electron').remote
Vue.prototype.$remote = remote
Copy the code

3. Window shadows

Desktop Windows lack shadows, but custom window shadows require a little trickery after electron removes the built-in window frame.

First set window background transparency in background.js:

win = new BrowserWindow({
  transparent: true // Window transparent
})
Copy the code

Then in app.vue, make # App absolutely laid out and the background transparent, the child div absolutely laid out and 4px away from the external div and set the shadow.

Note: If the full screen does not appear window shadows, otherwise there will be gaps. At this point, it is necessary to determine whether to enable shadow by determining whether to enable full screen. This will come back later when we talk about the top toolbar buttons (minimize, maximize, close).

<template>
  <! -- Disable mouse dragging pictures -->
  <div id="app" ondragstart="return false;">
    <div class="outer-radius" :class="this.$store.getters.getFullScreenChange ? 'no-app-shadow' : 'app-shadow'">
      <router-view />
    </div>
  </div>
</template>

<script>
  export default {}
</script>

<style>
  @font-face {
    font-family: Apr;
    src: url('./assets/font/Alibaba-PuHuiTi-Regular.otf');
  }

  #app {
    font-family: Apr, 'Helvetica Neue', Helvetica, 'PingFang SC'.'Hiragino Sans GB'.'Microsoft YaHei'.Microsoft Yahei, Arial, sans-serif;
    -webkit-font-smoothing: antialiased;
    -moz-osx-font-smoothing: grayscale;
    position: absolute;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
    background-color: transparent;
  }

  .outer-radius {
    border-radius: 2px;
  }

  .no-app-shadow {
    position: absolute;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
  }

  .app-shadow {
    position: absolute;
    top: 4px;
    bottom: 4px;
    left: 4px;
    right: 4px;
    box-shadow: 0 2px 4px rgba(0.0.0.0.12), 0 0 6px rgba(0.0.0.0.04);
  }

  input,
  select,
  option,
  textarea {
    outline: none;
  }

  /* Displays a line with an ellipsis */
  .text-elip {
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
  }

  /* El-scrollbar global setting 100% height */
  .el-scrollbar {
    height: 100%;
  }

  /* El-scrollBar removes the X-axis slider */
  .el-scrollbar__wrap {
    overflow-x: hidden;
  }

  /* Fix elemental-UI dropdown */
  .el-select-dropdown .el-scrollbar {
    padding-bottom: 17px;
  }

  /* Transition animation - fade in */
  .fade-enter-active..fade-leave-active {
    transition: opacity 0.2 s;
  }
  .fade-enter..fade-leave-to /*.fade-leave-active below version 2.1.8 */ {
    opacity: 0;
  }
</style>
Copy the code

4. Side menu bar

Select the appropriate icon in the Icon component of the Element-UI and register the icon click event to pass the click number.

<template>
  <div class="side-bar">
    <div class="user-icon">
      <img src=".. /assets/img/icon.svg" />
    </div>
    <div title="Chat" class="side-btn my-chat" :class="tabIndex === 0 ? 'tab-this' : 'tab-other'" @click="_btnTab(0)">
      <i class="el-icon-chat-dot-round"></i>
    </div>
    <div title="Address book" class="side-btn my-address" :class="tabIndex === 1 ? 'tab-this' : 'tab-other'" @click="_btnTab(1)">
      <i class="el-icon-user"></i>
    </div>
    <div title="Folder" class="side-btn my-file" :class="tabIndex === 2 ? 'tab-this' : 'tab-other'" @click="_btnTab(2)">
      <i class="el-icon-document"></i>
    </div>
    <div title="Settings" class="side-btn my-setting" :class="tabIndex === 3 ? 'tab-this' : 'tab-other'" @click="_btnTab(3)">
      <i class="el-icon-setting"></i>
    </div>
  </div>
</template>

<script>
  export default {
    data() {
      return {
        tabIndex: 0}},methods: {
      _btnTab: function(index) {
        if (this.tabIndex === index) {
          return
        }
        this.tabIndex = index
        // Register click events
        this.$emit('tabClick', index)
      }
    }
  }
</script>

<style scoped>
  .side-bar {
    position: absolute;
    top: 0;
    left: 0;
    bottom: 0;
    width: 60px;
    background-color: #1f2623;
    border-top-left-radius: 2px;
    border-bottom-left-radius: 2px;
    -webkit-app-region: drag;
    -webkit-user-select: none;
  }

  .user-icon {
    position: absolute;
    top: 20px;
    left: 10px;
    right: 10px;
    height: 40px;
    -webkit-app-region: no-drag;
  }

  .user-icon img {
    height: 100%;
    width: 100%;
  }

  .side-btn {
    height: 30px;
    width: 30px;
    text-align: center;
    line-height: 30px;
    font-size: 28px;
    -webkit-app-region: no-drag;
    cursor: pointer;
  }

  .tab-other {
    color: #dcdfe6;
  }

  .tab-this {
    color: aquamarine;
  }

  .tab-other:hover {
    color: #f2f6fc;
  }

  .my-chat {
    position: absolute;
    top: 100px;
    left: 15px;
  }

  .my-address {
    position: absolute;
    top: 160px;
    left: 15px;
  }

  .my-file {
    position: absolute;
    top: 220px;
    left: 15px;
  }

  .my-setting {
    position: absolute;
    bottom: 20px;
    left: 15px;
  }
</style>
Copy the code

5. Top function bar

The top is only responsible for window dragging and maximizing, minimizing, closing, top and other functions. At this time, the electron remote module is used. Since the remote module is already registered in the main.js entry function, it can be referenced in vue directly by $remote.

5.1 Questions here

  • The electron window maximization is disabled

Windows set frame: True, and then this configuration, maximized invalid, gray.

  • Win.ismaximized () always returns false

Windows sets frame: false, and then with this configuration, win.ismaximized () will always return false.

The program enters the non-full-screen state by default. You can customize the full-screen state in the status management.

  • Electron9.x Quit the program app.quit() is invalid

The cause is not found at the moment, only app.exit() can be used instead.

The difference between app.quit() and app.exit() is that if all Windows are closed, the quit event is triggered directly. Otherwise, the Electron will fire before-quit first, then start closing all Windows, and then fire the will-quit event. Note that in this case the window-all-closed event will not be fired. So you can use app.quit() in window-all-closed without worrying about infinite recursion. App.exit () shuts down the process and forces it to exit.

Note: the “X” in the upper right corner should be the tray left in the window, not the exit program. Only in the tray can you really exit the program. The tray function will be explained in a later article, where X is the temporary exit program.

5.2 Listening to and Exiting the full screen

  1. Add the computed properties of the full-screen event state to the status management store:
import Vue from 'vue'
import Vuex from 'vuex'

Vue.use(Vuex)

export default new Vuex.Store({
  state: {
    isFullScreen: false
  },
  mutations: {
    // Change the full screen state
    switchFullScreen(state, payload) {
      state.isFullScreen = payload
    }
  },
  getters: {
    // Get the full-screen status calculation property
    getFullScreenChange(state) {
      return state.isFullScreen
    }
  },
  actions: {},
  modules: {}})Copy the code
  1. Where need to use $store. Getters. GetFullScreenChange state to represent the full screen, such as full screen the cancellation window frame shadow:
<div class="outer-radius" :class="this.$store.getters.getFullScreenChange ? 'no-app-shadow' : 'app-shadow'">
  <router-view />
</div>
Copy the code

5.3 Top function bar topbar.vue code

<template>
  <div class="top-bar">
    <div class="left"></div>
    <div class="right">
      <div class="top-icon close" title="Closed" @click="btnClick('close')">
        <i class="el-icon-close"></i>
      </div>
      <div class="top-icon max" title=maximization @click="btnClick('max')">
        <i v-if=! "" isFull" class="el-icon-full-screen"></i>
        <i v-if="isFull" class="el-icon-copy-document"></i>
      </div>
      <div class="top-icon min" title=Minimize @click="btnClick('min')">
        <i class="el-icon-minus"></i>
      </div>
      <div class="top-icon top" :class="isAlwaysOnTop ? 'always-top' : ''" title="Top" @click="btnClick('top')">
        <i class="el-icon-paperclip"></i>
      </div>
    </div>
  </div>
</template>

<script>
export default {
  data() {
    return {
      isFull: false.isAlwaysOnTop: false}},methods: {
    btnClick: function(key) {
      let vm = this
      switch (key) {
        case 'close':
          vm.$remote.app.exit()
          break
        case 'max':
          if (vm.isFull) {
            vm.$remote.getCurrentWindow().unmaximize()
            vm.isFull = false
          } else {
            vm.$remote.getCurrentWindow().maximize()
            vm.isFull = true
          }
          vm.$store.commit('switchFullScreen', vm.isFull)
          break
        case 'min':
          vm.$remote.getCurrentWindow().minimize()
          break
        case 'top': vm.$remote.getCurrentWindow().setAlwaysOnTop(! vm.isAlwaysOnTop) vm.isAlwaysOnTop = ! vm.isAlwaysOnTopbreak}}}}</script>

<style scoped>
.top-bar {
  position: absolute;
  top: 0;
  left: 60px;
  right: 0;
  height: 26px;
  border-top-right-radius: 4px;
  -webkit-app-region: drag;
  -webkit-user-select: none;
}

.top-bar .left {
  position: absolute;
  top: 0;
  left: 0;
  height: 100%;
  width: 250px;
  background-color: #c3d7df;
}

.top-bar .right {
  position: absolute;
  top: 0;
  left: 250px;
  height: 100%;
  right: 0;
  background-color: rgba(198.223.200);
  border-top-right-radius: 2px;
}

.top-icon {
  position: absolute;
  -webkit-app-region: no-drag;
  top: 0;
  height: 26px;
  width: 30px;
  line-height: 26px;
  font-size: 14px;
  text-align: center;
  cursor: pointer;
  color: # 909399;
}

.top-icon:hover {
  background-color: #f2f6fc;
}

.close {
  right: 0;
  border-top-right-radius: 2px;
}

.close:hover {
  background-color: #f56c6c;
  color: #ffffff;
}

.max {
  right: 30px;
}

.min {
  right: 60px;
}

.top {
  right: 90px;
}

.always-top {
  color: # 000000;
}
</style>

Copy the code

6. Summary

The general UI framework has been completed, all that remains is the need to supplement the various secondary interfaces.

To be continued…