mirror of
https://github.com/nocodb/nocodb.git
synced 2026-05-01 09:57:21 +00:00
667 lines
20 KiB
Vue
667 lines
20 KiB
Vue
<script lang="ts" setup>
|
|
import { Checkbox, CheckboxGroup, Radio, RadioGroup } from 'ant-design-vue'
|
|
import type { Select as AntSelect } from 'ant-design-vue'
|
|
import { CURRENT_USER_TOKEN, type UserFieldRecordType } from 'nocodb-sdk'
|
|
import { getOptions, getSelectedUsers } from './utils'
|
|
import MdiCloseCircle from '~icons/mdi/close-circle'
|
|
|
|
interface Props {
|
|
modelValue?: UserFieldRecordType[] | UserFieldRecordType | string | null
|
|
rowIndex?: number
|
|
location?: 'cell' | 'filter'
|
|
forceMulti?: boolean
|
|
options?: UserFieldRecordType[]
|
|
}
|
|
|
|
const { modelValue, forceMulti, options: userOptions, location } = defineProps<Props>()
|
|
|
|
const emit = defineEmits(['update:modelValue'])
|
|
|
|
const { isMobileMode } = useGlobal()
|
|
|
|
const { t } = useI18n()
|
|
|
|
const { getColor } = useTheme()
|
|
|
|
const meta = inject(MetaInj)!
|
|
|
|
const isInFilter = inject(IsInFilterInj, ref(false))
|
|
|
|
const column = inject(ColumnInj)!
|
|
|
|
const readOnly = inject(ReadonlyInj)!
|
|
|
|
const isEditable = inject(EditModeInj, ref(false))
|
|
|
|
const activeCell = inject(ActiveCellInj, ref(false))
|
|
|
|
const basesStore = useBases()
|
|
|
|
const { basesUser } = storeToRefs(basesStore)
|
|
|
|
const baseUsers = computed(() => (meta.value.base_id ? basesUser.value.get(meta.value.base_id) || [] : []))
|
|
|
|
const idUserMap = computed(() => {
|
|
return baseUsers.value.reduce((acc, user) => {
|
|
acc[user.id] = user
|
|
acc[user.email] = user
|
|
return acc
|
|
}, {} as Record<string, any>)
|
|
})
|
|
|
|
// use both ActiveCellInj or EditModeInj to determine the active state
|
|
// since active will be false in case of form view
|
|
const active = computed(() => activeCell.value || isEditable.value)
|
|
|
|
const isForm = inject(IsFormInj, ref(false))
|
|
|
|
const isEditColumn = inject(EditColumnInj, ref(false))
|
|
|
|
const isMultiple = computed(() => forceMulti || (column.value.meta as { is_multi: boolean; notify: boolean })?.is_multi)
|
|
|
|
const rowHeight = inject(RowHeightInj, ref(undefined))
|
|
|
|
const isSurveyForm = inject(IsSurveyFormInj, ref(false))
|
|
|
|
const aselect = ref<typeof AntSelect>()
|
|
|
|
const isOpen = ref(false)
|
|
|
|
const isFocusing = ref(false)
|
|
|
|
const isKanban = inject(IsKanbanInj, ref(false))
|
|
|
|
const canvasSelectCell = inject(CanvasSelectCellInj, null)
|
|
|
|
const searchVal = ref<string | null>()
|
|
|
|
const { isUIAllowed } = useRoles()
|
|
|
|
const { showEEFeatures, showUpgradeToUseCurrentUserFilter } = useEeConfig()
|
|
|
|
const isFormListView = computed(() => !isEditColumn.value && isForm.value && parseProp(column.value.meta)?.isList)
|
|
|
|
const options = computed(() => {
|
|
const currentUserField: any[] = []
|
|
if (isEeUI && isInFilter.value && showEEFeatures.value) {
|
|
currentUserField.push({
|
|
id: CURRENT_USER_TOKEN,
|
|
display_name: t('title.currentUser'),
|
|
email: CURRENT_USER_TOKEN,
|
|
})
|
|
}
|
|
return [...currentUserField, ...(userOptions ?? getOptions(column.value, false, isForm.value, baseUsers.value))]
|
|
})
|
|
|
|
const nonDeletedOptions = computed(() => {
|
|
return options.value.filter((op) => !op.deleted)
|
|
})
|
|
|
|
const filteredOptions = computed(() => {
|
|
return nonDeletedOptions.value.filter((op) => {
|
|
return searchVal.value ? searchCompare([op.display_name, op.email], searchVal.value) : true
|
|
})
|
|
})
|
|
|
|
const optionsMap = computed(() => {
|
|
return options.value.reduce((acc, op) => {
|
|
if (op.id) {
|
|
acc[op.id] = op
|
|
}
|
|
if (op.email) {
|
|
acc[op.email.trim()] = op
|
|
}
|
|
return acc
|
|
}, {} as Record<string, (typeof options.value)[number]>)
|
|
})
|
|
|
|
const hasEditRoles = computed(() => isUIAllowed('dataEdit'))
|
|
|
|
const editAllowed = computed(() => (hasEditRoles.value || isForm.value) && active.value)
|
|
|
|
const vModel = computed({
|
|
get: () => {
|
|
return getSelectedUsers(optionsMap.value, modelValue)
|
|
},
|
|
set: (val) => {
|
|
// @ts-expect-error antd select returns string[] instead of { label: string, value: string }[]
|
|
if (isEeUI && val.includes(CURRENT_USER_TOKEN) && showUpgradeToUseCurrentUserFilter()) return
|
|
|
|
// Clear search query after selection is made
|
|
searchVal.value = ''
|
|
|
|
const value: string[] = []
|
|
if (val && val.length) {
|
|
val.forEach((item) => {
|
|
// @ts-expect-error antd select returns string[] instead of { label: string, value: string }[]
|
|
const user = options.value.find((u) => u.id === item)
|
|
if (user) {
|
|
value.push(user.id)
|
|
}
|
|
})
|
|
}
|
|
if (isMultiple.value) {
|
|
emit('update:modelValue', val?.length ? value.join(',') : null)
|
|
} else {
|
|
emit('update:modelValue', val?.length ? value[value.length - 1] : null)
|
|
isOpen.value = false
|
|
}
|
|
},
|
|
})
|
|
|
|
const vModelListLayout = computed(() => {
|
|
if (isMultiple.value) {
|
|
return (vModel.value || []).map((item) => item.value)
|
|
} else {
|
|
return (vModel.value || [])?.[0]?.value || ''
|
|
}
|
|
})
|
|
|
|
watch(isOpen, (n, _o) => {
|
|
// If it is form list view then we don't need to do anything here
|
|
if (isFormListView.value) return
|
|
|
|
if (!n) searchVal.value = ''
|
|
|
|
if (editAllowed.value) {
|
|
if (!n) {
|
|
aselect.value?.$el?.querySelector('input')?.blur()
|
|
} else {
|
|
aselect.value?.$el?.querySelector('input')?.focus()
|
|
}
|
|
}
|
|
})
|
|
|
|
// set isOpen to false when active cell is changed
|
|
watch(active, (n, _o) => {
|
|
if (!n) isOpen.value = false
|
|
})
|
|
|
|
useSelectedCellKeydownListener(
|
|
activeCell,
|
|
(e) => {
|
|
switch (e.key) {
|
|
case 'Escape':
|
|
if (canvasSelectCell) {
|
|
canvasSelectCell.trigger()
|
|
return
|
|
}
|
|
isOpen.value = false
|
|
break
|
|
case 'Enter':
|
|
if (editAllowed.value && active.value && !isOpen.value) {
|
|
e.stopPropagation()
|
|
toggleMenu()
|
|
isOpen.value = true
|
|
}
|
|
break
|
|
// skip space bar key press since it's used for expand row
|
|
case ' ':
|
|
break
|
|
case 'ArrowUp':
|
|
case 'ArrowDown':
|
|
case 'ArrowRight':
|
|
case 'ArrowLeft':
|
|
case 'Delete':
|
|
// skip
|
|
break
|
|
default:
|
|
if (!editAllowed.value) {
|
|
e.preventDefault()
|
|
break
|
|
}
|
|
// toggle only if char key pressed
|
|
if (!(e.metaKey || e.ctrlKey || e.altKey) && e.key?.length === 1 && !isDrawerOrModalExist()) {
|
|
e.stopPropagation()
|
|
isOpen.value = true
|
|
}
|
|
break
|
|
}
|
|
},
|
|
{
|
|
immediate: true,
|
|
isGridCell: true,
|
|
},
|
|
)
|
|
|
|
// close dropdown list on escape
|
|
useSelectedCellKeydownListener(
|
|
isOpen,
|
|
(e) => {
|
|
if (e.key === 'Escape') {
|
|
isOpen.value = false
|
|
canvasSelectCell?.trigger()
|
|
}
|
|
},
|
|
{
|
|
isGridCell: false,
|
|
immediate: true,
|
|
},
|
|
)
|
|
|
|
const search = () => {
|
|
searchVal.value = aselect.value?.$el?.querySelector('.ant-select-selection-search-input')?.value
|
|
}
|
|
|
|
const onTagClick = (e: Event, onClose: Function) => {
|
|
// check clicked element is remove icon
|
|
if (
|
|
(e.target as HTMLElement)?.classList.contains('ant-tag-close-icon') ||
|
|
(e.target as HTMLElement)?.closest('.ant-tag-close-icon')
|
|
) {
|
|
e.stopPropagation()
|
|
onClose()
|
|
}
|
|
}
|
|
|
|
function toggleMenu() {
|
|
if (isFocusing.value) return
|
|
|
|
isOpen.value = editAllowed.value && !isOpen.value
|
|
}
|
|
|
|
const handleClose = (e: MouseEvent) => {
|
|
// close dropdown if clicked outside of dropdown
|
|
if (
|
|
isOpen.value &&
|
|
aselect.value &&
|
|
!aselect.value.$el.contains(e.target) &&
|
|
!document.querySelector('.nc-dropdown-user-select-cell.active')?.contains(e.target as Node)
|
|
) {
|
|
// loose focus when clicked outside
|
|
isEditable.value = false
|
|
isOpen.value = false
|
|
}
|
|
}
|
|
|
|
useEventListener(document, 'click', handleClose, true)
|
|
|
|
// search with email or display_name
|
|
const filterOption = (input: string, option: any): boolean => {
|
|
const opt = options.value.find((o) => o.id === option.value)
|
|
if (!opt) return false
|
|
|
|
return searchCompare([opt.display_name, opt.email], input)
|
|
}
|
|
|
|
// check if user is part of the base
|
|
const isCollaborator = (userIdOrEmail) => {
|
|
return !idUserMap.value?.[userIdOrEmail]?.deleted
|
|
}
|
|
|
|
const onKeyDown = (e: KeyboardEvent) => {
|
|
// Tab
|
|
if (e.key === 'Tab') {
|
|
isOpen.value = false
|
|
return
|
|
} else if (e.key === 'Escape' && isForm.value) {
|
|
isOpen.value = false
|
|
return
|
|
}
|
|
|
|
e.stopPropagation()
|
|
}
|
|
|
|
const onFocus = () => {
|
|
isFocusing.value = true
|
|
|
|
setTimeout(() => {
|
|
isFocusing.value = false
|
|
}, 250)
|
|
|
|
if (isSurveyForm.value && vModel.value?.length) return
|
|
|
|
isOpen.value = true
|
|
}
|
|
const isExpandedForm = inject(IsExpandedFormOpenInj, ref(false))
|
|
const isCanvasInjected = inject(IsCanvasInjectionInj, false)
|
|
const isGrid = inject(IsGridInj, ref(false))
|
|
const isUnderLookup = inject(IsUnderLookupInj, ref(false))
|
|
const canvasCellEventData = inject(CanvasCellEventDataInj, reactive<CanvasCellEventDataInjType>({}))
|
|
onMounted(() => {
|
|
if (isGrid.value && isCanvasInjected && !isExpandedForm.value && !isEditColumn.value && !isUnderLookup.value) {
|
|
forcedNextTick(() => {
|
|
onFocus()
|
|
const key = canvasCellEventData.keyboardKey
|
|
if (key && isSinglePrintableKey(key)) {
|
|
searchVal.value = key
|
|
}
|
|
})
|
|
}
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
class="nc-cell-field nc-user-select h-full w-full flex items-center"
|
|
:class="{ 'read-only': readOnly }"
|
|
@click="toggleMenu"
|
|
>
|
|
<div v-if="isFormListView" class="w-full max-w-full">
|
|
<div class="rounded-lg border-1 border-nc-border-gray-medium w-full max-w-full">
|
|
<div v-if="nonDeletedOptions.length > 6" class="border-b-1 border-nc-border-gray-medium pl-1 group" @click.stop>
|
|
<a-input
|
|
ref="inputRef"
|
|
v-model:value="searchVal"
|
|
:placeholder="$t('general.search')"
|
|
class="nc-user-select-search-field-input !pl-2 !pr-1.5 flex-1 !py-2"
|
|
allow-clear
|
|
autocomplete="off"
|
|
:bordered="false"
|
|
>
|
|
<template #prefix>
|
|
<GeneralIcon icon="search" class="nc-search-icon text-nc-content-gray-muted opacity-50 h-4 w-4 mr-1.5" /> </template
|
|
></a-input>
|
|
</div>
|
|
<div class="w-full max-w-full max-h-[328px] nc-scrollbar-thin p-1">
|
|
<component
|
|
:is="isMultiple ? CheckboxGroup : RadioGroup"
|
|
:value="vModelListLayout"
|
|
:disabled="readOnly || !editAllowed"
|
|
class="nc-field-layout-list nc-user-select-list"
|
|
@update:value="
|
|
(value) => {
|
|
vModel = isMultiple ? value : [value]
|
|
}
|
|
"
|
|
>
|
|
<!-- If we have not rendered all options then update value will not work properly so we have to use display: hidden on option which is not visible -->
|
|
<template v-for="op of options" :key="op.id || op.email">
|
|
<component
|
|
:is="isMultiple ? Checkbox : Radio"
|
|
v-if="!op.deleted"
|
|
:key="op.id || op.email"
|
|
:value="op.id"
|
|
:data-testid="`select-option-${column.title}-${location === 'filter' ? 'filter' : rowIndex}`"
|
|
:class="[
|
|
`nc-select-option-${column.title}-${op.email}`,
|
|
{
|
|
'!hidden nc-hidden-option': !searchCompare([op.display_name, op.email], searchVal ?? ''),
|
|
},
|
|
]"
|
|
>
|
|
<div class="w-full flex gap-2 items-center py-1.5">
|
|
<GeneralUserIcon :user="op" size="base" class="flex-none" :disabled="!isCollaborator(op.id || op.email)" />
|
|
|
|
<div class="flex flex-col w-[calc(100%_-_40px)]">
|
|
<div class="w-full flex gap-3">
|
|
<NcTooltip
|
|
class="text-bodyDefaultSmBold !leading-5 capitalize truncate max-w-full text-nc-content-gray"
|
|
:class="{
|
|
'!text-nc-content-gray-muted': !isCollaborator(op.id || op.email),
|
|
}"
|
|
show-on-truncate-only
|
|
placement="bottom"
|
|
>
|
|
<template #title>
|
|
{{ extractUserDisplayNameOrEmail(op) }}
|
|
</template>
|
|
{{ extractUserDisplayNameOrEmail(op) }}
|
|
</NcTooltip>
|
|
</div>
|
|
<NcTooltip
|
|
class="text-xs !leading-4 text-nc-content-gray-muted truncate max-w-full"
|
|
show-on-truncate-only
|
|
placement="bottom"
|
|
>
|
|
<template #title>
|
|
{{ op.email === CURRENT_USER_TOKEN ? $t('title.filteredByLoggedInUser') : op.email }}
|
|
</template>
|
|
|
|
{{ op.email === CURRENT_USER_TOKEN ? $t('title.filteredByLoggedInUser') : op.email }}
|
|
</NcTooltip>
|
|
</div>
|
|
</div>
|
|
</component>
|
|
</template>
|
|
</component>
|
|
|
|
<div
|
|
v-if="nonDeletedOptions.length && searchVal && !filteredOptions.length"
|
|
class="h-full text-center flex items-center justify-center gap-3 mt-4"
|
|
>
|
|
<a-empty :image="Empty.PRESENTED_IMAGE_SIMPLE" :description="$t('labels.noData')" class="!my-0" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-if="!readOnly && !isMultiple && vModel.length"
|
|
class="inline-block px-2 pt-2 cursor-pointer text-xs text-nc-content-gray-muted hover:text-nc-content-gray"
|
|
@click="vModel = []"
|
|
>
|
|
{{ $t('labels.clearSelection') }}
|
|
</div>
|
|
</div>
|
|
|
|
<a-select
|
|
v-else
|
|
ref="aselect"
|
|
v-model:value="vModel"
|
|
mode="multiple"
|
|
class="w-full overflow-hidden"
|
|
:placeholder="isEditColumn ? $t('labels.optional') : ''"
|
|
:bordered="false"
|
|
clear-icon
|
|
:show-search="!isMobileMode"
|
|
:show-arrow="editAllowed && !readOnly"
|
|
:open="isOpen && editAllowed"
|
|
:disabled="readOnly || !editAllowed"
|
|
:class="{ 'caret-transparent': !hasEditRoles }"
|
|
:dropdown-class-name="`nc-dropdown-user-select-cell !min-w-256px ${isInFilter ? '!min-w-256px' : '!min-w-180px'} ${
|
|
isOpen ? 'active' : ''
|
|
}`"
|
|
:filter-option="filterOption"
|
|
:search-value="searchVal ?? ''"
|
|
@search="search"
|
|
@focus="onFocus"
|
|
@blur="isOpen = false"
|
|
@keydown="onKeyDown"
|
|
>
|
|
<template #suffixIcon>
|
|
<GeneralIcon icon="arrowDown" class="text-nc-content-gray-subtle nc-select-expand-btn" />
|
|
</template>
|
|
<template v-for="op of options" :key="op.id || op.email">
|
|
<a-select-option
|
|
v-if="!op.deleted"
|
|
:value="op.id"
|
|
:data-testid="`select-option-${column.title}-${location === 'filter' ? 'filter' : rowIndex}`"
|
|
:class="[
|
|
`nc-select-option-${column.title}-${op.email}`,
|
|
{
|
|
'nc-select-option-current-user mb-2': op.email === CURRENT_USER_TOKEN,
|
|
},
|
|
]"
|
|
@click.stop
|
|
>
|
|
<div
|
|
v-if="op.email === CURRENT_USER_TOKEN"
|
|
class="absolute -bottom-1 w-[calc(100%_+_16px)] border-b-1 border-nc-border-gray-medium -ml-4"
|
|
></div>
|
|
<div class="w-full flex gap-2 items-center">
|
|
<GeneralUserIcon :user="op" size="base" class="flex-none" :show-placeholder-icon="op.email === CURRENT_USER_TOKEN" />
|
|
|
|
<div
|
|
class="flex flex-col"
|
|
:class="{
|
|
'w-[calc(100%_-_32px)]': !vModel.find((i) => i.email === op.email),
|
|
'w-[calc(100%_-_68px)]': !!vModel.find((i) => i.email === op.email),
|
|
}"
|
|
>
|
|
<div class="w-full flex gap-3">
|
|
<NcTooltip
|
|
class="text-bodyDefaultSmBold !leading-5 capitalize truncate max-w-full"
|
|
:class="{
|
|
'text-nc-content-brand': op.email === CURRENT_USER_TOKEN,
|
|
'text-nc-content-gray': op.email !== CURRENT_USER_TOKEN,
|
|
}"
|
|
show-on-truncate-only
|
|
placement="bottom"
|
|
>
|
|
<template #title>
|
|
{{ extractUserDisplayNameOrEmail(op) }}
|
|
</template>
|
|
{{ extractUserDisplayNameOrEmail(op) }}
|
|
</NcTooltip>
|
|
</div>
|
|
<NcTooltip
|
|
class="text-xs !leading-4 text-nc-content-gray-muted truncate max-w-full"
|
|
show-on-truncate-only
|
|
placement="bottom"
|
|
>
|
|
<template #title>
|
|
{{ op.email === CURRENT_USER_TOKEN ? $t('title.filteredByLoggedInUser') : op.email }}
|
|
</template>
|
|
|
|
{{ op.email === CURRENT_USER_TOKEN ? $t('title.filteredByLoggedInUser') : op.email }}
|
|
</NcTooltip>
|
|
</div>
|
|
<GeneralIcon
|
|
v-if="!!vModel.find((i) => i.email === op.email)"
|
|
id="nc-selected-item-icon"
|
|
icon="check"
|
|
class="flex-none text-primary w-4 h-4"
|
|
/>
|
|
</div>
|
|
</a-select-option>
|
|
</template>
|
|
|
|
<template #tagRender="{ label, value: val, onClose }">
|
|
<a-tag
|
|
v-if="options.find((el) => el.id === val)"
|
|
class="rounded-tag nc-selected-option !pl-0"
|
|
:class="{
|
|
'!my-0': !rowHeight || rowHeight === 1,
|
|
}"
|
|
:style="{ display: 'flex', alignItems: 'center' }"
|
|
:color="
|
|
val === CURRENT_USER_TOKEN ? themeV4Colors.brand[50] : getColor('var(--nc-bg-gray-medium)', 'var(--nc-bg-gray-light)')
|
|
"
|
|
:closable="editAllowed && ((vModel?.length ?? 0) > 1 || !column?.rqd)"
|
|
:close-icon="h(MdiCloseCircle, { class: ['ms-close-icon'] })"
|
|
@click="onTagClick($event, onClose)"
|
|
@close="onClose"
|
|
>
|
|
<span
|
|
:style="{
|
|
color: getSelectTypeOptionTextColor(getColor('var(--nc-bg-gray-medium)', 'var(--nc-bg-gray-light)'), getColor),
|
|
}"
|
|
class="flex items-stretch gap-2"
|
|
:class="{ 'text-sm': isKanban, 'text-small': !isKanban }"
|
|
>
|
|
<div>
|
|
<GeneralUserIcon
|
|
size="auto"
|
|
:user="{
|
|
display_name: !label?.includes('@') ? label.trim() : '',
|
|
email: label,
|
|
meta: options.find((el) => el.id === val)?.meta,
|
|
}"
|
|
class="!text-[0.5rem] !h-[16.8px]"
|
|
:class="{
|
|
'!bg-nc-bg-default': val === CURRENT_USER_TOKEN,
|
|
}"
|
|
:is-deleted="!isCollaborator(val)"
|
|
:disabled="!isCollaborator(val)"
|
|
:show-placeholder-icon="val === CURRENT_USER_TOKEN"
|
|
/>
|
|
</div>
|
|
<span
|
|
:class="{
|
|
'text-nc-content-gray-muted': !isCollaborator(val) && val !== CURRENT_USER_TOKEN,
|
|
'text-nc-content-brand': val === CURRENT_USER_TOKEN,
|
|
'font-600': isInFilter,
|
|
}"
|
|
>
|
|
{{ label }}
|
|
</span>
|
|
</span>
|
|
</a-tag>
|
|
</template>
|
|
</a-select>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped lang="scss">
|
|
.ms-close-icon {
|
|
color: rgba(var(--rgb-base), 0.25);
|
|
cursor: pointer;
|
|
display: flex;
|
|
font-size: 12px;
|
|
font-style: normal;
|
|
height: 12px;
|
|
line-height: 1;
|
|
text-align: center;
|
|
text-transform: none;
|
|
transition: color 0.3s ease, opacity 0.15s ease;
|
|
width: 12px;
|
|
z-index: 1;
|
|
margin-right: -6px;
|
|
margin-left: 3px;
|
|
}
|
|
|
|
.ms-close-icon:before {
|
|
display: block;
|
|
}
|
|
|
|
.ms-close-icon:hover {
|
|
color: rgba(var(--rgb-base), 0.45);
|
|
}
|
|
|
|
.read-only {
|
|
.ms-close-icon {
|
|
display: none;
|
|
}
|
|
}
|
|
|
|
.rounded-tag {
|
|
@apply bg-nc-bg-gray-medium px-2 rounded-[12px];
|
|
}
|
|
|
|
:deep(.ant-tag) {
|
|
@apply "rounded-tag" my-[1px];
|
|
}
|
|
|
|
:deep(.ant-select-selection-overflow-item) {
|
|
@apply "flex overflow-hidden";
|
|
}
|
|
|
|
:deep(.ant-select-selection-overflow) {
|
|
@apply flex-nowrap overflow-hidden max-w-[fit-content];
|
|
}
|
|
|
|
.nc-user-select:not(.read-only) {
|
|
:deep(.ant-select-selector),
|
|
:deep(.ant-select-selector input) {
|
|
@apply "!cursor-pointer";
|
|
}
|
|
}
|
|
|
|
:deep(.ant-select-selector) {
|
|
@apply !pl-0 flex-nowrap;
|
|
}
|
|
|
|
:deep(.ant-select-selection-search-input) {
|
|
@apply !text-small;
|
|
}
|
|
|
|
:deep(.nc-user-avatar) {
|
|
@apply min-h-4.2;
|
|
}
|
|
|
|
:deep(.nc-select-option-current-user) {
|
|
@apply relative;
|
|
}
|
|
</style>
|
|
|
|
<style lang="scss">
|
|
.ant-select-dropdown.nc-dropdown-user-select-cell {
|
|
.ant-select-item-option-content {
|
|
@apply w-full;
|
|
}
|
|
|
|
.ant-select-item-option-state {
|
|
@apply !hidden;
|
|
}
|
|
}
|
|
</style>
|