Uniapp + Vue3 + TS 搭建基礎(chǔ)項目

基礎(chǔ)介紹

個人使用背景是因為公司業(yè)務(wù)覆蓋多個小程序端,主要包括:支付寶小程序回论、微信小程序散罕、百度小程序、快應(yīng)用傀蓉。項目基于npm包管理欧漱,vscode相關(guān)插件提供更好的開發(fā)體驗。

  • 主要用到相關(guān)技術(shù):uni-app相關(guān) / vue3 / typescript / axios / sass / eslint / prettier

下面直接開始從頭開始搭建一個可用的初始項目~ 如果"需求緊急"可以參考底部的GitHub源碼葬燎。如果遇到問題可參照對應(yīng)步驟误甚。

使用npm安裝uni-app

npm install -g @vue/cli
vue create -p dcloudio/uni-preset-vue#vue3 uni-app-vue3-demo
// 等待片刻 ...
> 請選擇 uni-app 模板: 默認(rèn)模板(TypeScript)

cd uni-app-vue3-demo

npm install
// 安裝較慢時可使用
npm install --registry=https://registry.npm.taobao.org

// 卸載不需要的包
npm uninstall vue-class-component vue-property-decorator
// 更新包
npm install vue@3.2.20 vuex@4.0.2 typescript@4.4.4

復(fù)制代碼

配置 eslint + prettier 自動格式化代碼

  • 安裝相關(guān)npm包
npm install eslint prettier --save-dev
npm install eslint-config-prettier eslint-plugin-prettier eslint-plugin-vue --save-dev
npm install @typescript-eslint/eslint-plugin @typescript-eslint/parser --save-dev

復(fù)制代碼
module.exports = {
  parser: 'vue-eslint-parser',
  extends: ['plugin:vue/recommended', 'plugin:prettier/recommended'],
  parserOptions: {
    parser: '@typescript-eslint/parser',
    ecmaVersion: 2018,
    sourceType: 'module',
  },
  plugins: ['vue', '@typescript-eslint'],
}

復(fù)制代碼
  • prettier配置 .prettierrc.js
module.exports = {
  printWidth: 120,
  tabWidth: 2,
  tabs: false,
  semi: false,
  singleQuote: true,
  quoteProps: 'as-needed',
  bracketSpacing: true,
  jsxBracketSameLine: false,
  arrowParens: 'always',
  endOfLine: 'auto',
}
復(fù)制代碼
  • vscode 配置 .vscode/settings.json
{
  "vetur.validation.template": false,
  "editor.formatOnSave": true,
  "editor.codeActionsOnSave": {
    "source.fixAll.eslint": true
  },
  "elint.validate": ["typescript", "vue", "html", "json"],
  "editor.defaultFormatter": "esbenp.prettier-vscode",
  "[javascript]": {
    "editor.defaultFormatter": "esbenp.prettier-vscode"
  },
  "[vue]": {
    "editor.defaultFormatter": "rvest.vs-code-prettier-eslint"
  },
  "[html]": {
    "editor.defaultFormatter": "esbenp.prettier-vscode"
  },
  "[json]": {
    "editor.defaultFormatter": "esbenp.prettier-vscode"
  },
  "json.format.enable": false
}
復(fù)制代碼
  • 測試代碼自動格式化 (重啟vscode 打開 src pages/index.vue 輸入幾個空格 然后在保存看是否會自動格式化)

Typescript 配置允許默認(rèn) any 類型 tsconfig.json

{
  compilerOptions: {
    "noImplicitAny": false,
  }
}
復(fù)制代碼

配置 SASS

  • 安裝 npm 相關(guān)包
 npm install node-sass@4.0.0 sass-loader@7.3.1 --save-dev 
 // 如遇到安裝問題可手動在package.json devDependencies 手動中添加
 // 刪除 node_modules 然后 重新安裝npm install
復(fù)制代碼
  • 在 vue 中使用
<style lang="scss">
</style>
復(fù)制代碼

配置全局公共變量 并 自動導(dǎo)入

  • 新建全局樣式變量 style/var.scss
$test-size: 400rpx;
復(fù)制代碼
  • 在全局自動導(dǎo)入 vue.config.js
process.env.UNI_USING_VUE3 = true
process.env.UNI_USING_VUE3_OPTIONS_API = true
module.exports = {
  css: {
    loaderOptions: {
      sass: {
        prependData: `@import "@/styles/var.scss";`, //引入全局變量
      },
    },
  },
  configureWebpack: {
    resolve: {
      extensions: ['.js', '.ts', '.vue', '.json'],
    },
  },
}

復(fù)制代碼
  • vscode 相關(guān)插件
image.png

安裝 axios api層 適配小程序

import axios, { AxiosError } from 'axios'
import qs from 'qs'
let TOKEN_KEY = 'x-token' // TODO 根據(jù)自己后端配置
let API_SOURCE = ''
// #ifdef  MP-WEIXIN
API_SOURCE = 'weixin' // 微信小程序
// #endif
// #ifdef  MP-BAIDU
API_SOURCE = 'bdapplets' // 百度小程序
// #endif

// 設(shè)置表單類型
axios.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded'

let baseURL = process.env.VUE_APP_API_HOST

const fetch = axios.create({
  withCredentials: true,
  baseURL,
  timeout: 5000,
})
let jumpLoginCount = 0

// request攔截器,在請求之前做一些處理
fetch.interceptors.request.use(
  (config) => {
    Object.assign(config.headers, {
      Authorization: uni.getStorageSync(TOKEN_KEY),
    })
    config.data = qs.stringify(config.data)

    return config
  },
  (error) => {
    return Promise.reject(error)
  }
)

// 配置成功后的攔截器
fetch.interceptors.response.use(
  (res) => {
    const params = qs.parse(res.config.data)

    // 設(shè)置 token
    if (res.headers.Authorization) {
      uni.setStorageSync(TOKEN_KEY, res.headers.Authorization)
    }
    // TODO 根據(jù)后端成功狀態(tài)配置
    if (['20001'].includes(`${res.data.code}`)) {
      // 是否返回根數(shù)據(jù)
      if (params.rootResult) return res
      else return res.data
    } else {
      return Promise.reject(res.data)
    }
  },
  (error: AxiosError) => {
    const params = qs.parse(error.config.data)
    // 未登錄 跳轉(zhuǎn)登錄頁
    if (error.response!.status == 401) {
      if (jumpLoginCount == 0) {
        uni.navigateTo({
          url: '/pages/login/index', // TODO 配置成自己的登錄頁路徑
        })
        ++jumpLoginCount
        setTimeout(() => (jumpLoginCount = 0), 2000)
      }
      return Promise.reject(error)
    }

    return Promise.reject(error)
  }
)

// 適配 小程序
fetch.defaults.adapter = function (config: any) {
  return new Promise((resolve, reject) => {
    var settle = require('axios/lib/core/settle')
    var buildURL = require('axios/lib/helpers/buildURL')

    uni.request({
      method: config.method.toUpperCase(),
      url: config.baseURL + buildURL(config.url, config.params, config.paramsSerializer),
      header: config.headers,
      data: config.data,
      dataType: config.dataType,
      responseType: config.responseType,
      sslVerify: config.sslVerify,
      complete: function complete(response: any) {
        response = {
          data: response.data,
          status: response.statusCode,
          errMsg: response.errMsg,
          headers: response.header, // 注意此處 單復(fù)數(shù)
          config: config,
        }
        settle(resolve, reject, response)
      },
    })
  })
}

export { fetch, API_SOURCE }

復(fù)制代碼

總結(jié)

通過上述步驟后,就能愉快的進(jìn)行踩坑之旅了~ 如有需要可訪問源碼
uni-app-vue3-demo

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末谱净,一起剝皮案震驚了整個濱河市窑邦,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌岳遥,老刑警劉巖奕翔,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異浩蓉,居然都是意外死亡派继,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進(jìn)店門捻艳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來驾窟,“玉大人,你說我怎么就攤上這事认轨∩鹇纾” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵嘁字,是天一觀的道長恩急。 經(jīng)常有香客問我,道長纪蜒,這世上最難降的妖魔是什么衷恭? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮纯续,結(jié)果婚禮上随珠,老公的妹妹穿的比我還像新娘灭袁。我一直安慰自己,他們只是感情好窗看,可當(dāng)我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布茸歧。 她就那樣靜靜地躺著,像睡著了一般显沈。 火紅的嫁衣襯著肌膚如雪软瞎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天构罗,我揣著相機(jī)與錄音铜涉,去河邊找鬼。 笑死遂唧,一個胖子當(dāng)著我的面吹牛芙代,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播盖彭,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼纹烹,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了召边?” 一聲冷哼從身側(cè)響起铺呵,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎隧熙,沒想到半個月后片挂,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡贞盯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年音念,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片躏敢。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡闷愤,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出件余,到底是詐尸還是另有隱情讥脐,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布啼器,位于F島的核電站旬渠,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏端壳。R本人自食惡果不足惜坟漱,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望更哄。 院中可真熱鬧芋齿,春花似錦、人聲如沸成翩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽麻敌。三九已至栅炒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間术羔,已是汗流浹背赢赊。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留级历,地道東北人释移。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像寥殖,于是被迫代替她去往敵國和親玩讳。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容