Skip to content

Composables - کامپوزبل ها

نکته

این بخش به دانش پایه‌ای در مورد Composition API نیاز دارد. اگر تاکنون فقط با Options API آشنا شده‌اید، می‌توانید اولویت API را به Composition API تغییر دهید (با استفاده از تاگل در بالای نوار کناری سمت چپ) و فصل‌های مبانی Reactivity و هوک‌های چرخه حیات را مجدداً مطالعه کنید.

یک "Composable" چیست؟

در برنامه‌های Vue، یک "Composable" تابعی است که از Composition API استفاده می‌کند تا منطق stateful را کپسوله‌سازی و قابل استفاده مجدد کند. (کلمه stateful به نگه داشتن وضعیت قبلی سیستم و تاثیر گذار بودن آن در پاسخ اشاره دارد در صورتی که کلمه stateless اشاره به این دارد که وضعیت سیستم در پاسخ گویی تاثیری ندارد و در جایی نگه داشته نمی‌شود)

هنگام ساخت برنامه‌های فرانت‌اند، اغلب به استفاده مجدد از کد برای کارهای رایج نیاز داریم. به عنوان مثال، ممکن است بخواهیم تاریخ‌ها را در بسیاری از منطقه‌ها فرمت کنیم، بنابراین یک تابع قابل استفاده مجدد برای این کار قرار می‌دهیم. این تابع فرمت‌کننده منطق را بصورت stateless کپسوله می‌کند: ورودی‌هایی را دریافت می‌کند و بلافاصله خروجی مورد انتظار را برمی‌گرداند. کتابخانه‌های زیادی برای استفاده مجدد از منطق بصورت stateless وجود دارند - به عنوان مثال lodash و date-fns که ممکن است شما از آن‌ها شنیده باشید.

در مقابل، منطق stateful شامل مدیریت حالت‌هایی است که با گذر زمان تغییر می‌کنند. یک مثال ساده ردیابی موقعیت فعلی ماوس در یک صفحه است. در سناریوهای واقعی‌تر، می‌تواند منطق پیچیده‌تری مانند حرکات لمسی یا وضعیت اتصال به یک پایگاه داده باشد.

مثال ردیاب ماوس

اگر بخواهیم عملکرد ردیابی ماوس را با استفاده از Composition API مستقیماً در داخل یک کامپوننت پیاده‌سازی کنیم، به شکل زیر خواهد بود:

vue
<script setup>
import { ref, onMounted, onUnmounted } from 'vue'

const x = ref(0)
const y = ref(0)

function update(event) {
  x.value = event.pageX
  y.value = event.pageY
}

onMounted(() => window.addEventListener('mousemove', update))
onUnmounted(() => window.removeEventListener('mousemove', update))
</script>

<template>Mouse position is at: {{ x }}, {{ y }}</template>

اما اگر بخواهیم از یک منطق در چندین کامپوننت دوباره استفاده کنیم، چه؟ ما می‌توانیم منطق را به عنوان یک تابع composable در یک فایل خارجی export کنیم:

js
// mouse.js
import { ref, onMounted, onUnmounted } from 'vue'

// شروع می‌شود "use" با composable طبق قرارداد، نام توابع 
export function useMouse() {
  // مدیریت می‌شود composable کپسوله شده و توسط state
  const x = ref(0)
  const y = ref(0)

  // مدیریت شده خود را در طول زمان به روز کند state می‌تواند composable یک
  function update(event) {
    x.value = event.pageX
    y.value = event.pageY
  }

  // همچنین می‌تواند به چرخه عمر کامپوننت مالک خود متصل شود composable یک
  // تا عوارض جانبی را راه‌اندازی کند و یا از بین ببرد
  onMounted(() => window.addEventListener('mousemove', update))
  onUnmounted(() => window.removeEventListener('mousemove', update))

  // مدیریت شده را در اختیار می‌گذارد state مقدار بازگشت داده شده از    
  return { x, y }
}

و به این صورت می‌توان از آن در کامپوننت‌ها استفاده کرد:

vue
<script setup>
import { useMouse } from './mouse.js'

const { x, y } = useMouse()
</script>

<template>Mouse position is at: {{ x }}, {{ y }}</template>
Mouse position is at: 0, 0

آن را در Playground امتحان کنید

همانطور که می‌بینید، منطق اصلی یکسان باقی مانده است - تنها کاری که باید انجام می‌دادیم این بود که آن را در یک تابع خارجی جابجا کنیم و stateهایی که باید نشان داده شوند را برگردانیم. درست مثل داخل یک کامپوننت، می‌توانید از تمام API‌های Composition در composables استفاده کنید. همان useMouse()‎ می‌تواند حالا در هر کامپوننتی استفاده شود.

قسمت جالب composables این است که می‌توانید آن‌ها را داخل یکدیگر قرار دهید: یک تابع ترکیب‌پذیر می‌تواند یک یا چند تابع ترکیب‌پذیر دیگر را صدا بزند. با داشتن این امکان می‌توانیم منطق پیچیده را با استفاده از واحدهای کوچک و مجزا ترکیب کنیم، شبیه به چگونگی ترکیب یک برنامه کامل با استفاده از کامپوننت‌ها. در واقع، به همین دلیلی بود که تصمیم گرفتیم مجموعه API‌هایی که این الگو را ممکن می‌سازند را Composition API بنامیم.

به عنوان مثال، می‌توانیم منطق افزودن و حذف یک شنونده رویداد DOM را در composable خودش export کنیم:

js
// event.js
import { onMounted, onUnmounted } from 'vue'

export function useEventListener(target, event, callback) {
  // if you want, you can also make this
  // support selector strings as target
  onMounted(() => target.addEventListener(event, callback))
  onUnmounted(() => target.removeEventListener(event, callback))
}

و حالا composable ما یعنی useMouse()‎ می‌تواند ساده‌تر شود:

js
// mouse.js
import { ref } from 'vue'
import { useEventListener } from './event'

export function useMouse() {
  const x = ref(0)
  const y = ref(0)

  useEventListener(window, 'mousemove', (event) => {
    x.value = event.pageX
    y.value = event.pageY
  })

  return { x, y }
}

نکته

هر نمونه ساخته شده از کامپوننت که useMouse()‎ را صدا می‌زند، کپی‌های خود را از حالت‌های x و y ایجاد می‌کند تا با یکدیگر تداخل پیدا نکنند. اگر می‌خواهید حالت مشترک بین کامپوننت‌ها را مدیریت کنید، فصل State Management را بخوانید.

مثال Async State

useMouse()‎ هیچ آرگومانی را نمی‌پذیرد، پس بیاید یک مثال دیگر که از یک آرگومان استفاده می‌کند را بررسی کنیم. هنگام دریافت داده‌های Async اغلب باید حالت‌های مختلف را مدیریت کنیم: loading و success و error:

vue
<script setup>
import { ref } from 'vue'

const data = ref(null)
const error = ref(null)

fetch('...')
  .then((res) => res.json())
  .then((json) => (data.value = json))
  .catch((err) => (error.value = err))
</script>

<template>
  <div v-if="error">Oops! Error encountered: {{ error.message }}</div>
  <div v-else-if="data">
    Data loaded:
    <pre>{{ data }}</pre>
  </div>
  <div v-else>Loading...</div>
</template>

تکرار این الگو در هر کامپوننتی که نیاز به دریافت داده دارد، خسته‌کننده خواهد بود. بیایید آن را در یک composable بنویسیم:

js
// fetch.js
import { ref } from 'vue'

export function useFetch(url) {
  const data = ref(null)
  const error = ref(null)

  fetch(url)
    .then((res) => res.json())
    .then((json) => (data.value = json))
    .catch((err) => (error.value = err))

  return { data, error }
}

حالا فقط نیاز است در کامپوننت این کار را انجام دهیم:

vue
<script setup>
import { useFetch } from './fetch.js'

const { data, error } = useFetch('...')
</script>

قبول کردن state های reactive

تابع useFetch()‎ یک آدرس رشته‌ای به عنوان ورودی می‌گیرد، سپس داده را دریافت می‌کند و کار آن تمام می‌شود. ولی اگر بخواهیم هربار که URL عوض شد دریافت دوباره انجام شود چه؟ برای رسیدن به این هدف، باید reactive را به داخل تابع composable پاس دهیم و بگذاریم ناظرهایی را برای انجام کارهایی با استفاده از state پاس داده شده ایجاد کند.

برای مثال تابع useFetch()‎ باید بتواند یک ref را قبول کند:

js
const url = ref('/initial-url')

const { data, error } = useFetch(url)

// این باید باعث درخواست مجدد شود
url.value = '/new-url'

یا یک تابع دریافت کننده (getter) را قبول کند:

js
// تغییر کرد، دوباره درخواست بزند props.id وقتی
const { data, error } = useFetch(() => `/posts/${props.id}`)

می‌توانیم مثال پیاده‌سازی شده خود را با API های toValue()‎ و watchEffect()‎ بازنویسی کنیم:

js
// fetch.js
import { ref, watchEffect, toValue } from 'vue'

export function useFetch(url) {
  const data = ref(null)
  const error = ref(null)

  const fetchData = () => {
    // reset state before fetching..
    data.value = null
    error.value = null

    fetch(toValue(url))
      .then((res) => res.json())
      .then((json) => (data.value = json))
      .catch((err) => (error.value = err))
  }

  watchEffect(() => {
    fetchData()
  })

  return { data, error }
}

تابع toValue()‎ یک API است که در ورژن ۳.۳ اضافه شده که برای عادی سازی ref ها یا getters طراحی شده: اگر آرگومان یک ref باشد، مقدار ref برگردانده می‌شود، اگر آرگومان یک تابع باشد، تابع را صدا می‌زند و مقدار بازگشتی آن را برمی‌گرداند؛ در غیر این صورت، آرگومان را همانطور که هست برمیگرداند. به طور مشابه unref()‎ اما با عملکرد مخصوص برای تابع‌ها.

توجه داشته باشید که toValue()‎ درون تابع بازگشتی watchEffect فراخوانده می‌شود. این تضمین می‌کند که وابستگی‌های reactive که در طول اجرای toValue()‎ به آن دسترسی پیدا می‌کند توسط ناظر دنبال می‌شوند.

این نسخه از useEffect()‎ رشته‌های URL ایستا، refs، و تابع‌های دریافت کننده را قبول می‌کند. این ویژگی آن را انعطاف پذیرتر می‌کند؛ ناظر بلافاصله اجرا می‌شود و وابستگی‌هایی که در طول toValue()‎ به آنها دسترسی پیدا کرده را ردیابی می‌کند، اگر هیچ وابستگی دنبال نشود، مثلا آدرس همان موقع هم یک رشته باشد، تاثیر (effect) فقط یکبار اجرا می‌شود؛ در غیر این صورت هربار که وابستگی دنبال شده تغییر کند دوباره اجرا می‌شود.

اینجا نسخه به روز رسانی شده از()useFetch، با تاخیر مصنوعی و خطای تصادفی برای اهداف نمایشی موجود است.

قرارداد‌ها و بهترین شیوه‌ها

نام‌گذاری

این یک قرارداد است که تابع‌های composable به صورت camelCase و با کلمه use شروع شوند.

آرگومان‌های ورودی

composable می‌تواند refها یا getters را به عنوان آرگومان‌های ورودی قبول کند، حتی اگر برای reactivity به آنها نیازی نداشته باشد، اگر در حال نوشتن یک composable هستید که ممکن است توسط توسعه دهندگان دیگر استفاده شود، ایده خوبی است که حالت‌هایی که ممکن است به جای مقادیر خام، آرگومان های ورودی refs یا getters باشند را هندل کنید. تابع ()toValue برای این هدف مفید خواهد بود:

js
import { toValue } from 'vue'

function useFeature(maybeRefOrGetter) {
  // باشد getter یا ref یک maybeRefOrGetter اگر
  // مقدار نرمال شده آن برگردانده خواهد شد
  // در غیر این صورت همانطور که هست برگردانده می‌شود
  const value = toValue(maybeRefOrGetter)
}

اگر زمانی که ورودی composable شما یک ref یا getter هست اثری reactive ایجاد می‌کند، مطمئن شوید که ref یا getter را به طور واضح با ()watch می‌بینید. همچنین ()toValue را درون ()watchEffect صدا بزنید تا به درستی ردیابی شود.

پیاده‌سازی useFetch()‎ که قبلاً مورد بحث قرار گرفت، یک مثال واضح از یک composable ارائه می‌کند که ref یا getter و یا مقادیر ساده را به عنوان آرگومان ورودی قبول می‌کند.

مقادیر بازگشتی

احتمالاً توجه کرده اید که ما به جای ()reactive در composableها به طور انحصاری از ()ref استفاده کرده ایم. قرارداد پیشنهادی این است که composableها همیشه یک آبجکت ساده و غیر reactive حاوی چندین ref را برگردانند. این عملکرد اجازه می‌دهد تا در کامپوننت‌های سازنده با حفظ reactivity تجزیه شود:

js
// x and y are refs
const { x, y } = useMouse()

برگرداندن یک آبجکت reactive از یک composable باعث می‌شود که ساختار ارتباط واکنش‌پذیری را با فضای داخل composable را از دست بدهد در حالی که ref می‌تواند ارتباط را حفظ کند.

اگر ترجیح می‌دهید از این حالت بازگشتی تابع‌های composable به عنوان ویژگی های آبجکت استفاده کنید، می‌توانید آبجکت برگشتی را با ()reactive پیش ببرید تا ref ها قابل دسترسی شوند، برای مثال:

js
const mouse = reactive(useMouse())
// mouse.x is linked to original ref
console.log(mouse.x)
template
Mouse position is at: {{ mouse.x }}, {{ mouse.y }}

تاثیرات (effect) جانبی

افزودن تاثیرات جانبی، مانند رویدادهای گوش دهنده DOM یا درخواست داده ایرادی ندارد اما باید قوانین زیر را مد نظر داشت:

  • اگر روی برنامه‌ای کار می‌کنید که از رندر سمت سرور (SSR) استفاده می‌کند، مطمئن شوید که اثرهای جانبی خاص DOM در چرخه هوک‌های نصب شده انجام می‌شوند. مثلا onUnmounted()‎. ین هوک‌ها فقط در مرورگر فراخوانی می‌شوند، پس می‌توان مطمئن باشید که کدهای آن به DOM دسترسی دارند.

  • به یاد داشته باشید که اثرهای جانبی را در onUnmounted()‎ حذف کنید، برای مثال اگر یک composable، یک شنونده رویداد DOM را راه اندازی کند، باید آن شنونده را در onUnmounted()‎ حذف کند. همانطور که در مثال useMouse()‎ دیدیم؛ یک ایده خوب این است که از composable ایی استفاده کنیم که به طور خودکار این کار را انجام دهد. مثلا useEventListener()‎

محدودیت در استفاده

composableها فقط باید در <script setup> یا هوک setup()‎ به صورت synchronously (همزمان) صدا زده شوند. در بعضی موارد می‌توان در چرخه هوک‌هایی مانند onMounted()‎ نیز فراخوانی شوند.

این محدودیت‌ها مهم هستند. زیرا با این مضمون Vue می‌تواند تشخیص دهد کدام یک از کامپوننت‌های فعلی، فعال هستند. دسترسی به یک نمونه از کامپوننت های فعال، به دلایل زیر ضروری است:

  1. هوک‌های چرخه حیات را می‌توان در آن ثبت کرد.

  2. ویژگی‌های computed و ناظرهای نصب شده را می‌توان به آن پیوند کرد. به طوری که بعدا آنها را هنگام unmounted شدن یک نمونه، دور انداخت تا از نشت حافظه جلوگیری شود.

نکته

تنها جایی که می‌توانید composableها را بعد از کلمه await صدا بزنید <script setup> است. کامپایلر بعد از اجرای موارد async، محتوای موارد فعال را بازیابی می‌کند.

استفاده از composableها برای سازماندهی کد

composableها نه فقط برای استفاده دوباره، بلکه برای سازماندهی کردن کد نیز می‌توانند استفاده شوند. به نسبت افزایش پیچیدگی کامپوننت‌ها، ممکن است با کامپوننت‌هایی مواجه شوید که برای پیمایش و استدلال بسیار بزرگ هستند، composable کردن API به شما انعطاف پذیری کاملی را می‌دهد تا کد کامپوننت خود را بر اساس منطق های مرتبط سازماندهی کنید:

vue
<script setup>
import { useFeatureA } from './featureA.js'
import { useFeatureB } from './featureB.js'
import { useFeatureC } from './featureC.js'

const { foo, bar } = useFeatureA()
const { baz } = useFeatureB(foo)
const { qux } = useFeatureC(baz)
</script>

تا حدودی می‌توان این composableهای نوشته شده را به عنوان سرویس‌هایی با کامپوننت‌های مشخص‌شده در نظر گرفت که می‌توانند با یکدیگر تعامل کنند.

استفاده از composable در Options API

اگر از Options API استفاده می‌کنید، composableها باید درون setup()‎ صدا زده شوند و پیوند برگردانده شده نیز باید از setup()‎ برگردانده شوند تا توسط this و تمپلیت قابل دسترسی باشند:

js
import { useMouse } from './mouse.js'
import { useFetch } from './fetch.js'

export default {
  setup() {
    const { x, y } = useMouse()
    const { data, error } = useFetch('...')
    return { x, y, data, error }
  },
  mounted() {
    // مشاهده کرد "this" را می‌توان در setup() ویژگی‌های در معرض نمایش گذاشته شده توسط
    console.log(this.x)
  }
  // ...other options
}

مقایسه با سایر تکنیک‌ها

در مقابل Mixins

کاربرانی که از نسخه ۲ می‌آیند ممکن است با آپشن mixins آشنا باشند که همچنین به ما امکان می‌دهد که منطق کامپوننت را در واحد‌هایی با قابلیت استفاده مجدد بنویسیم. سه اشکال اصلی برای mixins وجود دارد:

  1. منبع نامشخص برای پراپرتی‌ها: هنگام استفاده از بسیاری از mixins‌ها، مشخص نمی‌شود که کدام نوع پراپرتی توسط کدام mixin تزریق می‌شود و ردیابی پیاده‌سازی و درک رفتار کامپوننت را دشوار می‌کند. همچنین به همین دلیل است که ما استفاده از "refs + destructure pattern" را برای composableها توصیه می‌کنیم: این موضوع منبع پراپرتی را در کامپوننت‌های استفاده کننده مشخص می‌کند.

  2. تلاقی فضای نام‌ها: چندین mixin از نویسندگان مختلف به طور بالقوه می‌توانند کلیدهای پراپرتی یکسانی را ثبت کنند و باعث برخورد فضای نام شوند، اما هنگام استفاده از composableها، اگر کلیدهای متناقض از composableهای مختلف وجود داشته باشد، می‌توان نام متغیرهای تخریب شده را عوض کرد.

  3. ارتباط بی قید و شرط ترکیب‌های مقابل: چندین mixins که نیاز به تعامل با یکدیگر دارند، باید به کلیدهای پراپرتی مشترک تکیه کنند و آنها را بدون شرط جفت می‌کند. با استفاده از composableها می‌توان مقادیر بازگشتی یک composable را به عنوان آرگومان به دیگری ارسال کرد. درست مانند توابع عادی

به دلایل بالا، استفاده از mixins ها را از این به‌بعد در Vue ۳ توصیه نمی‌کنیم؛ این ویژگی فقط به دلایل آشنایی و مهاجرت نگه‌داری می‌شود.

در مقابل کامپوننت‌های بدون رندر

در فصل اسلات‌ها، الگوی کامپوننت بدون رندر را بر اساس اسلات‌های دارای اسکوپ مورد بحث قرار دادیم. حتی همان نسخه نمایشی ردیابی ماوس را با استفاده از اجزای رندر اجرا کردیم.

مزیت اصلی Composableها نسبت به کامپوننت‌های بدون رندر این است که Composableها هزینه‌های اضافی را متحمل نمی‌شوند. هنگامی که در همه یک برنامه کاربردی استفاده می‌شود، تعداد نمونه های اضافی ایجاد شده توسط الگوی کامپوننت بدون رندر می‌تواند به یک سربار عملکرد قابل توجه تبدیل شود.

توصیه این است که هنگام استفاده مجدد از منطق خالص، از Composables استفاده کنید، و هنگام استفاده مجدد از منطق و طرح بصری، از کامپوننت‌ها.

در مقابل هوک‌های ری‌اکت

اگر تجربه‌ای با React دارید، ممکن است متوجه شوید که این شبیه به کاستوم هوک‌های React به نظر می‌رسد. Composition API تا حدی الهام گرفته از React hooks بوده است، و Vue composables در واقع شبیه به React hooks از نظر توانایی‌های ترکیب منطق هستند. با این حال، Vue composables بر پایه‌ی سیستم واکنش‌پذیری دقیق‌تر ذره‌ای Vue است، که بنیاداً متفاوت از مدل اجرایی React hooks است. این موضوع با جزئیات بیشتری در Composition API FAQ بحث شده است.

اگر تجربه کار با ری‌اکت داشته باشید، ممکن است متوجه شوید که این بسیار شبیه به هوک‌های سفارشی ری‌اکت است. Composition API تا حدی از هوک‌های ری‌اکت الهام گرفته شده است و Composable های Vue در واقع از نظر قابلیت‌های ترکیب منطقی شبیه به هوک های ری‌اکت هستند. با این حال، Composable های Vue مبتنی بر سیستم reactivity دقیق‌تر ذره‌ای Vue هستند که اساساً با نوع اجرای هوک‌های ری‌اکت تفاوت دارد. که با جزئیات بیشتر در سوالات متداول از Composition API مورد بحث قرار گرفته است.

مطالعه بیشتر

  • reactivity در عمق: برای درک نحوه عملکرد سیستم reactivity در Vue در سطح پایین‌تر
  • State Management: برای الگوهای مدیریت state که توسط چندین کامپوننت مشترک هستند
  • تست Composables: نکاتی در مورد تست واحدهای Composables
  • VueUse: مجموعه ای در حال رشد از Vue composables. کد منبع نیز یک منبع یادگیری عالی است.
Composables - کامپوزبل ها has loaded