-
Notifications
You must be signed in to change notification settings - Fork 83
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: new backend dashboard [wip][ref Codeinwp/neve-pro-addon#2914]
- Loading branch information
Showing
47 changed files
with
2,237 additions
and
589 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
122 changes: 122 additions & 0 deletions
122
assets/apps/dashboard/src/Components/Common/Multiselect.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,122 @@ | ||
import { useRef, useState } from '@wordpress/element'; | ||
import { __ } from '@wordpress/i18n'; | ||
import cn from 'classnames'; | ||
import { Check, ChevronDown } from 'lucide-react'; | ||
import { useEffect } from 'react'; | ||
const MultiSelect = ({ value, label, disabled, choices = {}, onChange }) => { | ||
const [isOpen, setIsOpen] = useState(false); | ||
|
||
const dropdownRef = useRef(null); | ||
|
||
const closeDropdown = (e) => { | ||
if (dropdownRef.current && !dropdownRef.current.contains(e.target)) { | ||
setIsOpen(false); | ||
} | ||
}; | ||
|
||
useEffect(() => { | ||
if (isOpen) { | ||
document.addEventListener('click', closeDropdown); | ||
} else { | ||
document.removeEventListener('click', closeDropdown); | ||
} | ||
|
||
return () => { | ||
document.removeEventListener('click', closeDropdown); | ||
}; | ||
}, [isOpen]); | ||
|
||
const handleChange = (optionValue) => { | ||
const nextValues = value.includes(optionValue) | ||
? value.filter((v) => v !== optionValue) | ||
: [...value, optionValue]; | ||
onChange(nextValues); | ||
}; | ||
|
||
return ( | ||
<div className="grid gap-1"> | ||
{label && ( | ||
<span className="text-sm text-gray-600 font-medium"> | ||
{label} | ||
</span> | ||
)} | ||
<div className="relative"> | ||
<button | ||
onClick={() => !disabled && setIsOpen(!isOpen)} | ||
className={cn( | ||
'relative w-full py-1.5 px-2 text-sm rounded border flex items-center gap-3 min-w-[200px]', | ||
'border-gray-300 hover:border-gray-500', | ||
{ | ||
'bg-gray-100': disabled, | ||
} | ||
)} | ||
disabled={disabled} | ||
> | ||
<div className="flex flex-wrap gap-1 max-w-[160px] overflow-hidden"> | ||
{Object.entries(value).length === 0 ? ( | ||
<span className="text-gray-500"> | ||
{__('Select options', 'neve')}... | ||
</span> | ||
) : ( | ||
value.map((choice, index) => { | ||
if (!choices[choice]) { | ||
return null; | ||
} | ||
|
||
return ( | ||
<span | ||
key={index} | ||
className="inline-flex items-center gap-1 bg-blue-100 text-blue-800 rounded px-2 py-0.5 text-xs" | ||
> | ||
{choices[choice]} | ||
</span> | ||
); | ||
}) | ||
)} | ||
</div> | ||
|
||
<ChevronDown | ||
size={18} | ||
className={`ml-auto transition-transform ${ | ||
isOpen ? 'transform rotate-180' : '' | ||
}`} | ||
aria-hidden="true" | ||
/> | ||
</button> | ||
|
||
{isOpen && ( | ||
<div | ||
ref={dropdownRef} | ||
className="absolute z-10 w-full mt-1 bg-white border rounded shadow-lg" | ||
> | ||
<div className="max-h-60 overflow-y-auto p-1"> | ||
{Object.entries(choices).map( | ||
([optionValue, optionLabel]) => ( | ||
<button | ||
key={optionValue} | ||
className="flex w-full items-center gap-2 text-sm rounded py-1.5 px-3 hover:bg-blue-100 hover:text-blue-700 cursor-pointer" | ||
onClick={() => | ||
handleChange(optionValue) | ||
} | ||
> | ||
<div className="w-4 h-4 border bg-white rounded flex items-center justify-center"> | ||
{value.includes(optionValue) && ( | ||
<Check | ||
size={12} | ||
className="text-blue-600" | ||
/> | ||
)} | ||
</div> | ||
{optionLabel} | ||
</button> | ||
) | ||
)} | ||
</div> | ||
</div> | ||
)} | ||
</div> | ||
</div> | ||
); | ||
}; | ||
|
||
export default MultiSelect; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
import cn from 'classnames'; | ||
import { LoaderCircle, LucideChevronDown } from 'lucide-react'; | ||
|
||
import { | ||
Field, | ||
Label, | ||
Listbox, | ||
ListboxButton, | ||
ListboxOption, | ||
ListboxOptions, | ||
} from '@headlessui/react'; | ||
import { __ } from '@wordpress/i18n'; | ||
|
||
export default ({ | ||
label, | ||
value, | ||
onChange, | ||
disabled = false, | ||
loading, | ||
choices, | ||
}) => { | ||
return ( | ||
<Field className="grid gap-1"> | ||
{label && ( | ||
<Label className="text-sm text-gray-600 font-medium"> | ||
{label} | ||
</Label> | ||
)} | ||
<div className="flex items-center gap-3"> | ||
{loading && ( | ||
<LoaderCircle size={18} className="animate-spin shrink-0" /> | ||
)} | ||
<Listbox | ||
value={value} | ||
onChange={onChange} | ||
disabled={loading || disabled} | ||
> | ||
{({ open }) => ( | ||
<> | ||
<ListboxButton | ||
className={cn( | ||
'relative w-full rounded py-1.5 px-2 text-left', | ||
'rounded border border-gray-300 hover:border-gray-500', | ||
'flex items-center gap-3 min-w-[200px]', | ||
{ | ||
'bg-gray-100': disabled || loading, | ||
} | ||
)} | ||
> | ||
{choices[value] || | ||
__('Select an option', 'neve')} | ||
|
||
<LucideChevronDown | ||
size={18} | ||
className={cn( | ||
'ml-auto transition-transform', | ||
{ | ||
'transform rotate-180': open, | ||
} | ||
)} | ||
aria-hidden="true" | ||
/> | ||
</ListboxButton> | ||
|
||
<ListboxOptions | ||
anchor="bottom" | ||
transition | ||
className={cn( | ||
'text-sm font-normal shadow-lg', | ||
'rounded border bg-white p-1 my-1 min-w-[200px]', | ||
'transition duration-100 ease-in data-[leave]:data-[closed]:opacity-0 antialiased' | ||
)} | ||
> | ||
{Object.entries(choices).map( | ||
([optionValue, optionLabel]) => ( | ||
<ListboxOption | ||
key={optionValue} | ||
value={optionValue} | ||
className="flex w-full items-center gap-2 text-sm rounded py-1.5 px-3 hover:bg-blue-100 hover:text-blue-700 data-[focus]:bg-blue-100 data-[focus]:text-blue-700 cursor-pointer" | ||
> | ||
{optionLabel} | ||
</ListboxOption> | ||
) | ||
)} | ||
</ListboxOptions> | ||
</> | ||
)} | ||
</Listbox> | ||
</div> | ||
</Field> | ||
); | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
import { Description, Field, Input, Label } from '@headlessui/react'; | ||
import { Fragment } from '@wordpress/element'; | ||
import cn from 'classnames'; | ||
|
||
const TextInput = ({ | ||
value, | ||
label, | ||
disabled, | ||
onChange, | ||
name, | ||
className = '', | ||
type = 'text', | ||
description, | ||
}) => { | ||
let TagName = 'input'; | ||
|
||
if (type === 'textarea') { | ||
TagName = 'textarea'; | ||
} | ||
|
||
return ( | ||
<Field className="grid gap-1"> | ||
{label && ( | ||
<Label className="text-sm text-gray-600 font-medium"> | ||
{label} | ||
</Label> | ||
)} | ||
{description && ( | ||
<Description className="text-xs text-gray-600"> | ||
{description} | ||
</Description> | ||
)} | ||
<Input type={type} as={Fragment} name={name}> | ||
<TagName | ||
type={type} | ||
value={value} | ||
onChange={onChange} | ||
disabled={disabled} | ||
className={cn( | ||
'block w-full px-3 !py-1 text-sm border rounded disabled:bg-gray-100 !border-gray-300 focus:!border-gray-500 focus:!shadow-none', | ||
{ | ||
'cursor-not-allowed': disabled, | ||
}, | ||
className | ||
)} | ||
/> | ||
</Input> | ||
</Field> | ||
); | ||
}; | ||
|
||
export default TextInput; |
Oops, something went wrong.