useOsTheme.js 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. import {ref, onBeforeMount, onBeforeUnmount, getCurrentInstance} from 'vue'
  2. let usedCount = 0, darks, lights, managable = true
  3. const osTheme = ref(null)
  4. const supportMatchMedia = typeof window !== 'undefined' && window.matchMedia !== undefined
  5. //获取系统主题
  6. export const useOsTheme = () => {
  7. if (process.env.NODE_ENV !== 'test' && !supportMatchMedia) {
  8. return osTheme.value
  9. }
  10. if (process.env.NODE_ENV === 'test' && window.matchMedia === undefined) {
  11. return osTheme.value
  12. }
  13. if (usedCount === 0) init()
  14. if (managable && (managable = hasInstance())) {
  15. onBeforeMount(() => {
  16. usedCount += 1
  17. })
  18. onBeforeUnmount(() => {
  19. usedCount -= 1
  20. if (usedCount === 0) clean()
  21. })
  22. }
  23. return osTheme.value
  24. }
  25. const handleDarkMqlChange = (e) => {
  26. if (e.matches) {
  27. osTheme.value = 'dark'
  28. }
  29. }
  30. const handleLightMqlChange = (e) => {
  31. if (e.matches) {
  32. osTheme.value = 'light'
  33. }
  34. }
  35. const init = () => {
  36. darks = window.matchMedia('(prefers-color-scheme: dark)')
  37. lights = window.matchMedia('(prefers-color-scheme: light)')
  38. if (darks.matches) {
  39. osTheme.value = 'dark'
  40. } else if (lights.matches) {
  41. osTheme.value = 'light'
  42. } else {
  43. osTheme.value = null
  44. }
  45. if (darks.addEventListener) {
  46. darks.addEventListener('change', handleDarkMqlChange)
  47. lights.addEventListener('change', handleLightMqlChange)
  48. } else if (darks.addListener) {
  49. darks.addListener(handleDarkMqlChange)
  50. lights.addListener(handleLightMqlChange)
  51. }
  52. }
  53. const clean = () => {
  54. if ('removeEventListener' in darks) {
  55. darks.removeEventListener('change', handleDarkMqlChange)
  56. lights?.removeEventListener('change', handleLightMqlChange)
  57. } else if ('removeListener' in darks) {
  58. darks?.removeListener(handleDarkMqlChange)
  59. lights?.removeListener(handleLightMqlChange)
  60. }
  61. darks = undefined
  62. lights = undefined
  63. }
  64. export const hasInstance = () => {
  65. return getCurrentInstance() !== null
  66. }