ホームページ > ウェブフロントエンド > Vue.js > Vue3+ViteでのJSXの使い方を詳しく解説

Vue3+ViteでのJSXの使い方を詳しく解説

青灯夜游
リリース: 2022-12-09 20:27:28
転載
3106 人が閲覧しました

Vue Vite で JSX を使用するにはどうすればよいですか?以下の記事ではVue3 ViteでのJSXの使い方を紹介していますので、ご参考になれば幸いです。

Vue3+ViteでのJSXの使い方を詳しく解説

[関連する推奨事項: vuejs ビデオ チュートリアルWeb フロントエンド開発]

JSX の概要

JSX (JavaScript および XML) は、React で初めて導入された JavaScript 内 HTML 構文拡張機能です。 JSX は、UI がどのように対話するかという本質において、UI がどのように動作するかを記述する優れた方法です。 JSX は JavaScript 構文の拡張であるため、HTML に似たコードは JS と共存できます。例:

const button = <MyButton color="blue" shadowSize={2}>
  Click Me
</MyButton>
ログイン後にコピー

ボタン定数は JSX 式と呼ばれます。これを使用して、アプリケーションで <MyButton> タグをレンダリングできます。ブラウザは JSX を直接読み取って解析することはできません。 (Babel や Parcel などのツールを使用して) コンパイルされた後の JSX 式は次のようになります。

React.createElement(
  MyButton,
  {color: &#39;blue&#39;, shadowSize: 2},
  &#39;Click Me&#39;
)
ログイン後にコピー

実際、JSX は React.createElement(component, props, ...children) 関数の糖衣構文にすぎません。 React.createElement() を使用して自分で UI を記述することで、コンパイル手順をスキップできます。ただし、そうすると JSX の宣言的な利点が失われ、コードが読みにくくなります。コンパイルは開発プロセスにおける追加のステップですが、React コミュニティの多くの開発者は、JSX の読みやすさにはそれだけの価値があると信じています。さらに、一般的なツールでは、JSX から JavaScript へのコンパイルがセットアップ プロセスの一部になっています。希望しない限り、自分でコンパイルを設定する必要はありません。特定の JSX が JavaScript に変換されるものをテストしたい場合は、Online Babel コンパイラ を使用してみてください。

React では JSX を使用する必要はありません。ビルド環境で JSX コンパイルを構成したくない場合は、React で JSX を使用しない方が便利です。たとえば、JSX で書かれたコード:

class Hello extends React.Component {
  render() {
    return <div>Hello {this.props.toWhat}</div>;
  }
}

const root = ReactDOM.createRoot(document.getElementById(&#39;root&#39;));
root.render(<Hello toWhat="World" />);
ログイン後にコピー

は、JSX を使用しないコードとして書くことができます:

class Hello extends React.Component {
  render() {
    return React.createElement(&#39;div&#39;, null, `Hello ${this.props.toWhat}`);
  }
}

const root = ReactDOM.createRoot(document.getElementById(&#39;root&#39;));
root.render(React.createElement(Hello, {toWhat: &#39;World&#39;}, null));
ログイン後にコピー

Using JSX in Vue3

Vue では、次の単一のコードを使用します。 -file コンポーネントは、テンプレート、関連スクリプト、CSS を .vue で終わる単一のファイルに統合します。これらのファイルは、最終的に JS パッケージ化またはビルド ツール (Webpack、Vite など) によって処理されます。

<template> 要素には、すべてのマークアップ構造とコンポーネント表示ロジックが含まれています。テンプレートには、有効な HTML だけでなく、Vue 固有の構文も含めることができます。たとえば、<template> タグの lang 属性を設定すると、<template lang="pug"># と設定して Pug テンプレートを使用できます。 ## 標準の HTML を置き換えます。

.vue ファイルの

<script> タグには、コンポーネント内のすべての非表示ロジックが含まれており、デフォルトで JS オブジェクトをエクスポートする必要があります。このオブジェクトは、コンポーネントがローカルに登録され、プロパティが定義され、ローカル状態が処理され、メソッドが定義される場所です。ビルドフェーズでは、テンプレートを含むこのオブジェクトが処理され、render() 関数を使用して Vue コンポーネントに変換されます。

コンポーネントの CSS スタイルは、

<style> タグに記述されます。scoped 属性が追加されると、Vue はスタイルの範囲を次のように制限します。単一ファイルコンポーネントのコンテンツ。これは CSS-in-JS に似たソリューションですが、純粋な CSS を記述できます。 CLI 経由でプロジェクトを作成するときに CSS プリプロセッサを選択した場合は、lang 属性を <style> タグに追加して、Webpack がビルド時にコンテンツを処理できるようにすることができます。

jsx は React によって初めて導入されましたが、JSX 構文は実際にはランタイム セマンティクスを定義しておらず、さまざまな出力形式にコンパイルできます。以前に JSX 構文を使用したことがある場合は、

Vue の JSX コンパイル方法は React の JSX コンパイル方法とは異なるため、Vue アプリケーションでは React の JSX コンパイルを使用できないことに注意してください。 React JSX 構文との顕著な違いは次のとおりです。

  • 可以使用 HTML attributes 比如 classfor 作为 props - 不需要使用 classNamehtmlFor
  • 传递子元素给组件 (比如 slots) 的方式不同

Vue 的类型定义也提供了 TSX 语法的类型推导支持。当使用 TSX 语法时,确保在 tsconfig.json 中配置了 "jsx": "preserve",这样的 TypeScript 就能保证 Vue JSX 语法编译过程中的完整性。

安装插件(@vitejs/plugin-vue-jsx)

vite 官方提供了官方的插件来支持在 vue3 中使用 jsx/tsx,直接安装就行。

npm i @vitejs/plugin-vue-jsx -D
ログイン後にコピー

安装完之后在 vite.config.js 文件中的 plugins 字段中添加 jsx 支持:

import vueJsx from "@vitejs/plugin-vue-jsx";

export default defineConfig({
  plugins: [
    vueJsx(),
  ]
})
ログイン後にコピー

这样就可以在项目中使用 jsx/tsx 了。

新建 jsx 文件

在项目中新建 jsx 或 tsx 后缀的文件,语法和 js 文件类似,但是和 .vue 文件中的 <script> 标签一样需要默认导出一个 JS 对象。该对象是在本地注册组件、定义属性、处理本地状态、定义方法等的地方。

import HelloWorld from &#39;./HelloWorld.vue&#39;
export default {
  setup() {
    return () => <HelloWorld msg="11" />;
  },
};
ログイン後にコピー

语法

1、插值。与 vue 模板语法中的插值一样,但是双大括号 {{}} 变为了单大括号 {}。大括号内支持任何有效的 JavaScript 表达式,比如:2 + 2,user.firstName,formatName(user) 等。

// 模板语法
<span>{{ a + b }}</span>

// jsx/tsx
<span>{ a + b }</span>
ログイン後にコピー

2、class 类名绑定。有两种方式,使用模板字符串或者使用数组。

// 模板字符串
<div className={ `header ${ isBg ? &#39;headerBg&#39; : &#39;&#39; }` }>header</div>
// 数组
<div class={ [ &#39;header&#39;, isBg && &#39;headerBg&#39; ] } >header</div>
ログイン後にコピー

3、style 样式绑定。需要使用双大括号。

const color = &#39;red&#39;
const element = <sapn style={{ color, fontSize: &#39;16px&#39; }}>style</sapn>
ログイン後にコピー

4、条件渲染。由于 jsx 本身具有 js 语法,所以不再需要使用 v-if 指令,使用 if/else 和三元表达式都可以实现。但是支持 v-show 指令。

const element = (name) => {
  if (name) {
    return <h1>Hello, { name }</h1>
  } else {
    return <h1>Hello, Stranger</h1>
  }
}

const element = icon ? <span class="icon"></span> : null;
// 以上代码等效于:
const element = icon && <span class="icon"></span>;
ログイン後にコピー

5、列表渲染。同样,由于 jsx 本身具有 js 语法,所以不再需要使用 v-for 指令,使用 JS 数组的 map 方法即可。

const listData = [
   {name: &#39;Tom&#39;, age: 18},
   {name: &#39;Jim&#39;, age: 20},
   {name: &#39;Lucy&#39;, age: 16}
]
return () => (
   <div>
     <div class={&#39;box&#39;}>
       <span>姓名</span>
       <span>年龄</span>
     </div>
   {
   prop.listData.map(item => <div class={&#39;box&#39;}>
       <span>{item.name}</span>
       <span>{item.age}</span>
     </div>
     })
   </div>
)
ログイン後にコピー

6、标签属性绑定。也是使用大括号包裹,不能使用 v-bind 指令。而 vue 组件中通过 <div v-bind="properties"></div> 批量绑定标签属性,在 JSX 中需要使用 <div {...properties}></div>

const href = &#39;https://cn.vuejs.org/&#39;
const element = <a href={href}>Vue3</a>
ログイン後にコピー

7、事件绑定。使用的也是 单大括号 {},不过事件绑定不是以 @为前缀了,而是改成了 on,与原生相同。例如:click 事件是 onClick 或 onclick。

const confirm = () => {
  // 确认提交
}
<button onClick={confirm}>确定</button>
ログイン後にコピー

如果要带参数,需要使用箭头函数进行包裹:

const confirm = (name) => {
  // 确认提交
}
<button onClick={() => confirm(&#39;Are you sure&#39;)}>确定</button>
ログイン後にコピー

8、事件修饰符。需要使用 withModifiers 方法,接收两个参数,第一个参数是绑定的事件,第二个参数是需要使用的事件修饰符。

import { withModifiers, defineComponent, ref } from &#39;vue&#39;

const App = defineComponent({
  setup() {
    const count = ref(0);

    const inc = () => {
      count.value++;
    };

    return () => (
      <div onClick={ withModifiers(inc, [&#39;self&#39;]) }>{ count.value }</div>
    );
  },
})
export default App
ログイン後にコピー

注意:Vue 模板中 ref 变量是可以直接解构的,但是在 jsx 中不行,需要添加 .value,比如上面的 { count.value }。

9、v-model 双向绑定。需要使用单大括号 {}。如果绑定属性则需要一个数组,第一个元素为绑定的值,第二个元素为绑定的属性。

// 绑定值
<input v-model="show" /> // vue
<input v-model={show.value} /> // jsx

// 绑定属性
<input v-model:prop="show" /> // vue
<input v-model={[show.value,&#39;prop&#39;]} /> // jsx

// 修饰符写法
<input v-model:prop.trim="show" /> // vue
<input v-model={[show.value,&#39;prop&#39;,[&#39;trim&#39;]]} /> // jsx
ログイン後にコピー

10、slot 插槽。jsx/tsx 中无法使用 slot 标签,定义插槽方式一:通过 setup 函数的第一个参数 ctx 上下文对象的 slots 的属性,setup 函数默认接收两个参数:

  • props - 组件传入的参数对象。
  • ctx - 上下文对象,上下文对象暴露了其他一些在 setup 中可能会用到的值,包括:
    • attrs - 透传的 Attributes(非响应式的对象,等价于 $attrs)。
    • slots - 插槽(非响应式的对象,等价于 $slots)。
    • emit - 触发事件的函数(等价于 $emit)。
    • expose - 暴露公共属性的函数。

如果解构了 props 对象,解构出的变量将会丢失响应性,因此推荐通过 props.xxx 的形式来使用其中的 props。如果确实需要解构 props 对象,或者需要将某个 prop 传到一个外部函数中并保持响应性,可以使用 toRefs()toRef() 这两个工具函数:

import { toRefs, toRef } from &#39;vue&#39;

export default {
  setup(props) {
    // 将 `props` 转为一个其中全是 ref 的对象,然后解构
    const { title } = toRefs(props)
    // `title` 是一个追踪着 `props.title` 的 ref
    console.log(title.value)

    // 或者,将 `props` 的单个属性转为一个 ref
    const title = toRef(props, &#39;title&#39;)
  }
}
ログイン後にコピー

ctx 上下文对象是非响应式的,可以安全地解构:

export default {
  setup(props, { attrs, slots, emit, expose }) {
    ...
  }
}
ログイン後にコピー

attrs 和 slots 都是响应式(有状态)的对象,它们总是会随着组件自身的更新而更新。这意味着你应当避免解构它们,并始终通过 attrs.x 或 slots.x 的形式使用其中的属性。此外,和 props 不同,attrs 和 slots 的属性都不是响应式的。如果想要基于 attrs 或 slots 的改变来执行副作用,那么应该在 onBeforeUpdate 生命周期钩子中编写相关逻辑。

expose 函数用于显式地限制该组件暴露出的属性,当父组件通过模板引用访问该组件的实例时,将仅能访问 expose 函数暴露出的内容:

export default {
  setup(props, { expose }) {
    // 让组件实例处于 “关闭状态”
    // 即不向父组件暴露任何东西
    expose()

    const publicCount = ref(0)
    const privateCount = ref(0)
    // 有选择地暴露局部状态
    expose({ count: publicCount })
  }
}
ログイン後にコピー

通过 ctx 上下文对象的 slots 的属性可以获取插槽对象后,就可以定义插槽了。

import { defineComponent } from &#39;vue&#39;

export default defineComponent({
  setup(props, { slots }) { // 逻辑
    return () => {
      return <p>
     <button>{ slots.test?.() }</button>
     <button>{ slots.default?.() }</button>
     </p>
    }
  },
})

// 在引用的组件中
<template #test>slot-test</template>
<template #>slot-default</template>
ログイン後にコピー

定义插槽方式二:使用 renderSlot 函数。

import { renderSlot } from &#39;vue&#39;

<button>
  { renderSlot(slots, &#39;default&#39;) }
</button>
ログイン後にコピー

而如果在 jsx 中使用插槽,可以直接通过标签属性 slot,或通过 v-slots 指令。

import HelloWorld from &#39;./HelloWorld&#39;  
export default defineComponent({  
    setup() {  
        return () => (  
            <div class={&#39;box&#39;}>  
                <HelloWorld v-slots={{  
                    title: () => {  
                        return <p>我是title插槽</p>  
                    },  
                    default: () => {  
                        return <p>我是default插槽</p>  
                    }  
                }} />  
            </div>  
        )  
    }  
})
ログイン後にコピー

11、CSS Modules。引入局部样式,相当于 vue 组件中 <style> 标签的 scoped 属性。

import styles from &#39;./index.module.scss&#39;

<div class={styles.wrap}></div>
ログイン後にコピー

GitHub 源码

(学习视频分享:vuejs入门教程编程基础视频

以上がVue3+ViteでのJSXの使い方を詳しく解説の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:juejin.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート