VBtn.mjs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. import { mergeProps as _mergeProps, resolveDirective as _resolveDirective, createVNode as _createVNode } from "vue";
  2. // Styles
  3. import "./VBtn.css";
  4. // Components
  5. import { VBtnToggleSymbol } from "../VBtnToggle/VBtnToggle.mjs";
  6. import { VDefaultsProvider } from "../VDefaultsProvider/index.mjs";
  7. import { VIcon } from "../VIcon/index.mjs";
  8. import { VProgressCircular } from "../VProgressCircular/index.mjs"; // Composables
  9. import { makeBorderProps, useBorder } from "../../composables/border.mjs";
  10. import { makeComponentProps } from "../../composables/component.mjs";
  11. import { makeDensityProps, useDensity } from "../../composables/density.mjs";
  12. import { makeDimensionProps, useDimension } from "../../composables/dimensions.mjs";
  13. import { makeElevationProps, useElevation } from "../../composables/elevation.mjs";
  14. import { makeGroupItemProps, useGroupItem } from "../../composables/group.mjs";
  15. import { IconValue } from "../../composables/icons.mjs";
  16. import { makeLoaderProps, useLoader } from "../../composables/loader.mjs";
  17. import { makeLocationProps, useLocation } from "../../composables/location.mjs";
  18. import { makePositionProps, usePosition } from "../../composables/position.mjs";
  19. import { makeRoundedProps, useRounded } from "../../composables/rounded.mjs";
  20. import { makeRouterProps, useLink } from "../../composables/router.mjs";
  21. import { useSelectLink } from "../../composables/selectLink.mjs";
  22. import { makeSizeProps, useSize } from "../../composables/size.mjs";
  23. import { makeTagProps } from "../../composables/tag.mjs";
  24. import { makeThemeProps, provideTheme } from "../../composables/theme.mjs";
  25. import { genOverlays, makeVariantProps, useVariant } from "../../composables/variant.mjs"; // Directives
  26. import { Ripple } from "../../directives/ripple/index.mjs"; // Utilities
  27. import { computed, withDirectives } from 'vue';
  28. import { genericComponent, propsFactory, useRender } from "../../util/index.mjs"; // Types
  29. export const makeVBtnProps = propsFactory({
  30. active: {
  31. type: Boolean,
  32. default: undefined
  33. },
  34. activeColor: String,
  35. baseColor: String,
  36. symbol: {
  37. type: null,
  38. default: VBtnToggleSymbol
  39. },
  40. flat: Boolean,
  41. icon: [Boolean, String, Function, Object],
  42. prependIcon: IconValue,
  43. appendIcon: IconValue,
  44. block: Boolean,
  45. readonly: Boolean,
  46. slim: Boolean,
  47. stacked: Boolean,
  48. ripple: {
  49. type: [Boolean, Object],
  50. default: true
  51. },
  52. text: String,
  53. ...makeBorderProps(),
  54. ...makeComponentProps(),
  55. ...makeDensityProps(),
  56. ...makeDimensionProps(),
  57. ...makeElevationProps(),
  58. ...makeGroupItemProps(),
  59. ...makeLoaderProps(),
  60. ...makeLocationProps(),
  61. ...makePositionProps(),
  62. ...makeRoundedProps(),
  63. ...makeRouterProps(),
  64. ...makeSizeProps(),
  65. ...makeTagProps({
  66. tag: 'button'
  67. }),
  68. ...makeThemeProps(),
  69. ...makeVariantProps({
  70. variant: 'elevated'
  71. })
  72. }, 'VBtn');
  73. export const VBtn = genericComponent()({
  74. name: 'VBtn',
  75. props: makeVBtnProps(),
  76. emits: {
  77. 'group:selected': val => true
  78. },
  79. setup(props, _ref) {
  80. let {
  81. attrs,
  82. slots
  83. } = _ref;
  84. const {
  85. themeClasses
  86. } = provideTheme(props);
  87. const {
  88. borderClasses
  89. } = useBorder(props);
  90. const {
  91. densityClasses
  92. } = useDensity(props);
  93. const {
  94. dimensionStyles
  95. } = useDimension(props);
  96. const {
  97. elevationClasses
  98. } = useElevation(props);
  99. const {
  100. loaderClasses
  101. } = useLoader(props);
  102. const {
  103. locationStyles
  104. } = useLocation(props);
  105. const {
  106. positionClasses
  107. } = usePosition(props);
  108. const {
  109. roundedClasses
  110. } = useRounded(props);
  111. const {
  112. sizeClasses,
  113. sizeStyles
  114. } = useSize(props);
  115. const group = useGroupItem(props, props.symbol, false);
  116. const link = useLink(props, attrs);
  117. const isActive = computed(() => {
  118. if (props.active !== undefined) {
  119. return props.active;
  120. }
  121. if (link.isLink.value) {
  122. return link.isActive?.value;
  123. }
  124. return group?.isSelected.value;
  125. });
  126. const color = computed(() => isActive.value ? props.activeColor ?? props.color : props.color);
  127. const variantProps = computed(() => {
  128. const showColor = group?.isSelected.value && (!link.isLink.value || link.isActive?.value) || !group || link.isActive?.value;
  129. return {
  130. color: showColor ? color.value ?? props.baseColor : props.baseColor,
  131. variant: props.variant
  132. };
  133. });
  134. const {
  135. colorClasses,
  136. colorStyles,
  137. variantClasses
  138. } = useVariant(variantProps);
  139. const isDisabled = computed(() => group?.disabled.value || props.disabled);
  140. const isElevated = computed(() => {
  141. return props.variant === 'elevated' && !(props.disabled || props.flat || props.border);
  142. });
  143. const valueAttr = computed(() => {
  144. if (props.value === undefined || typeof props.value === 'symbol') return undefined;
  145. return Object(props.value) === props.value ? JSON.stringify(props.value, null, 0) : props.value;
  146. });
  147. function onClick(e) {
  148. if (isDisabled.value || link.isLink.value && (e.metaKey || e.ctrlKey || e.shiftKey || e.button !== 0 || attrs.target === '_blank')) return;
  149. link.navigate?.(e);
  150. group?.toggle();
  151. }
  152. useSelectLink(link, group?.select);
  153. useRender(() => {
  154. const Tag = link.isLink.value ? 'a' : props.tag;
  155. const hasPrepend = !!(props.prependIcon || slots.prepend);
  156. const hasAppend = !!(props.appendIcon || slots.append);
  157. const hasIcon = !!(props.icon && props.icon !== true);
  158. return withDirectives(_createVNode(Tag, _mergeProps({
  159. "type": Tag === 'a' ? undefined : 'button',
  160. "class": ['v-btn', group?.selectedClass.value, {
  161. 'v-btn--active': isActive.value,
  162. 'v-btn--block': props.block,
  163. 'v-btn--disabled': isDisabled.value,
  164. 'v-btn--elevated': isElevated.value,
  165. 'v-btn--flat': props.flat,
  166. 'v-btn--icon': !!props.icon,
  167. 'v-btn--loading': props.loading,
  168. 'v-btn--readonly': props.readonly,
  169. 'v-btn--slim': props.slim,
  170. 'v-btn--stacked': props.stacked
  171. }, themeClasses.value, borderClasses.value, colorClasses.value, densityClasses.value, elevationClasses.value, loaderClasses.value, positionClasses.value, roundedClasses.value, sizeClasses.value, variantClasses.value, props.class],
  172. "style": [colorStyles.value, dimensionStyles.value, locationStyles.value, sizeStyles.value, props.style],
  173. "aria-busy": props.loading ? true : undefined,
  174. "disabled": isDisabled.value || undefined,
  175. "tabindex": props.loading || props.readonly ? -1 : undefined,
  176. "onClick": onClick,
  177. "value": valueAttr.value
  178. }, link.linkProps), {
  179. default: () => [genOverlays(true, 'v-btn'), !props.icon && hasPrepend && _createVNode("span", {
  180. "key": "prepend",
  181. "class": "v-btn__prepend"
  182. }, [!slots.prepend ? _createVNode(VIcon, {
  183. "key": "prepend-icon",
  184. "icon": props.prependIcon
  185. }, null) : _createVNode(VDefaultsProvider, {
  186. "key": "prepend-defaults",
  187. "disabled": !props.prependIcon,
  188. "defaults": {
  189. VIcon: {
  190. icon: props.prependIcon
  191. }
  192. }
  193. }, slots.prepend)]), _createVNode("span", {
  194. "class": "v-btn__content",
  195. "data-no-activator": ""
  196. }, [!slots.default && hasIcon ? _createVNode(VIcon, {
  197. "key": "content-icon",
  198. "icon": props.icon
  199. }, null) : _createVNode(VDefaultsProvider, {
  200. "key": "content-defaults",
  201. "disabled": !hasIcon,
  202. "defaults": {
  203. VIcon: {
  204. icon: props.icon
  205. }
  206. }
  207. }, {
  208. default: () => [slots.default?.() ?? props.text]
  209. })]), !props.icon && hasAppend && _createVNode("span", {
  210. "key": "append",
  211. "class": "v-btn__append"
  212. }, [!slots.append ? _createVNode(VIcon, {
  213. "key": "append-icon",
  214. "icon": props.appendIcon
  215. }, null) : _createVNode(VDefaultsProvider, {
  216. "key": "append-defaults",
  217. "disabled": !props.appendIcon,
  218. "defaults": {
  219. VIcon: {
  220. icon: props.appendIcon
  221. }
  222. }
  223. }, slots.append)]), !!props.loading && _createVNode("span", {
  224. "key": "loader",
  225. "class": "v-btn__loader"
  226. }, [slots.loader?.() ?? _createVNode(VProgressCircular, {
  227. "color": typeof props.loading === 'boolean' ? undefined : props.loading,
  228. "indeterminate": true,
  229. "width": "2"
  230. }, null)])]
  231. }), [[Ripple, !isDisabled.value && props.ripple, '', {
  232. center: !!props.icon
  233. }]]);
  234. });
  235. return {
  236. group
  237. };
  238. }
  239. });
  240. //# sourceMappingURL=VBtn.mjs.map