mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-06 22:40:14 -05:00
AdminX Labs static UI (#17754)
refs. https://github.com/TryGhost/Product/issues/3745 - added static UI for Labs in AdminX
This commit is contained in:
parent
629aa08262
commit
b70f58860e
4 changed files with 179 additions and 1 deletions
|
@ -17,6 +17,7 @@ interface ListItemProps {
|
|||
separator?: boolean;
|
||||
|
||||
bgOnHover?: boolean;
|
||||
paddingRight?: boolean;
|
||||
onClick?: (e: React.MouseEvent<HTMLDivElement>) => void;
|
||||
children?: React.ReactNode;
|
||||
}
|
||||
|
@ -32,6 +33,7 @@ const ListItem: React.FC<ListItemProps> = ({
|
|||
testId,
|
||||
separator = true,
|
||||
bgOnHover = true,
|
||||
paddingRight = true,
|
||||
onClick,
|
||||
children
|
||||
}) => {
|
||||
|
@ -58,7 +60,7 @@ const ListItem: React.FC<ListItemProps> = ({
|
|||
</div>
|
||||
}
|
||||
{action &&
|
||||
<div className={`px-6 py-3 ${hideActions ? 'invisible group-hover:visible' : ''}`}>
|
||||
<div className={`py-3 pl-6 ${paddingRight && 'pr-6'} ${hideActions ? 'invisible group-hover:visible' : ''}`}>
|
||||
{action}
|
||||
</div>
|
||||
}
|
||||
|
|
8
apps/admin-x-settings/src/assets/images/labs-bg.svg
Normal file
8
apps/admin-x-settings/src/assets/images/labs-bg.svg
Normal file
|
@ -0,0 +1,8 @@
|
|||
<svg width="267" height="285" viewBox="0 0 267 285" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<circle cx="165" cy="183" r="102" fill="#F1F3F4"/>
|
||||
<circle cx="68" cy="71" r="31" fill="#F1F3F4"/>
|
||||
<circle cx="99" cy="17" r="17" fill="#F1F3F4"/>
|
||||
<circle cx="118" cy="70" r="7" fill="#F1F3F4"/>
|
||||
<circle cx="45" cy="32" r="3" fill="#F1F3F4"/>
|
||||
<circle cx="12.5" cy="83.5" r="12.5" fill="#F1F3F4"/>
|
||||
</svg>
|
After Width: | Height: | Size: 403 B |
|
@ -1,12 +1,14 @@
|
|||
import CodeInjection from './CodeInjection';
|
||||
import History from './History';
|
||||
import Integrations from './Integrations';
|
||||
import Labs from './Labs';
|
||||
import React from 'react';
|
||||
import SettingSection from '../../../admin-x-ds/settings/SettingSection';
|
||||
|
||||
const searchKeywords = {
|
||||
integrations: ['integration', 'zapier', 'slack', 'amp', 'unsplash', 'first promoter', 'firstpromoter', 'pintura', 'disqus', 'analytics', 'ulysses', 'typeform', 'buffer', 'plausible', 'github'],
|
||||
codeInjection: ['newsletter', 'enable', 'disable', 'turn on'],
|
||||
labs: ['labs', 'alpha', 'beta', 'flag', 'import', 'export', 'migrate', 'routes', 'redirects', 'translation', 'delete'],
|
||||
history: ['history', 'log', 'events', 'user events', 'staff']
|
||||
};
|
||||
|
||||
|
@ -15,6 +17,7 @@ const AdvancedSettings: React.FC = () => {
|
|||
<SettingSection keywords={Object.values(searchKeywords).flat()} title='Advanced'>
|
||||
<Integrations keywords={searchKeywords.integrations} />
|
||||
<CodeInjection keywords={searchKeywords.codeInjection} />
|
||||
<Labs keywords={searchKeywords.labs} />
|
||||
<History keywords={searchKeywords.history} />
|
||||
</SettingSection>
|
||||
);
|
||||
|
|
165
apps/admin-x-settings/src/components/settings/advanced/Labs.tsx
Normal file
165
apps/admin-x-settings/src/components/settings/advanced/Labs.tsx
Normal file
|
@ -0,0 +1,165 @@
|
|||
import Button from '../../../admin-x-ds/global/Button';
|
||||
import LabsBubbles from '../../../assets/images/labs-bg.svg';
|
||||
import List from '../../../admin-x-ds/global/List';
|
||||
import ListItem from '../../../admin-x-ds/global/ListItem';
|
||||
import React, {useState} from 'react';
|
||||
import SettingGroup from '../../../admin-x-ds/settings/SettingGroup';
|
||||
import SettingGroupHeader from '../../../admin-x-ds/settings/SettingGroupHeader';
|
||||
import TabView from '../../../admin-x-ds/global/TabView';
|
||||
import Toggle from '../../../admin-x-ds/global/form/Toggle';
|
||||
|
||||
const LabItem: React.FC<{
|
||||
title?: React.ReactNode;
|
||||
detail?: React.ReactNode;
|
||||
action?: React.ReactNode;
|
||||
}> = ({
|
||||
title,
|
||||
detail,
|
||||
action
|
||||
}) => {
|
||||
return (
|
||||
<ListItem
|
||||
action={action}
|
||||
bgOnHover={false}
|
||||
detail={detail}
|
||||
paddingRight={false}
|
||||
title={title}
|
||||
/>
|
||||
);
|
||||
};
|
||||
|
||||
const MigrationOptions: React.FC = () => {
|
||||
return (
|
||||
<List titleSeparator={false}>
|
||||
<LabItem
|
||||
action={<Button color='grey' label='Open importer' size='sm' />}
|
||||
detail='Import posts from a JSON or zip file'
|
||||
title='Import content'
|
||||
/>
|
||||
<LabItem
|
||||
action={<Button color='grey' label='Export' size='sm' />}
|
||||
detail='Download all of your posts and settings in a single, glorious JSON file'
|
||||
title='Export your content'
|
||||
/>
|
||||
<LabItem
|
||||
action={<Button color='red' label='Delete' size='sm' />}
|
||||
detail='Permanently delete all posts and tags from the database, a hard reset'
|
||||
title='Delete all content'
|
||||
/>
|
||||
</List>
|
||||
);
|
||||
};
|
||||
|
||||
/*
|
||||
<a className='text-green' href="" rel="noopener noreferrer" target="_blank"></a>
|
||||
*/
|
||||
|
||||
const AlphaFeatures: React.FC = () => {
|
||||
return (
|
||||
<List titleSeparator={false}>
|
||||
<LabItem
|
||||
action={<Toggle />}
|
||||
detail={<>Try out <a className='text-green' href="https://ghost.org/changelog/editor-beta/" rel="noopener noreferrer" target="_blank">Ghost{`'`}s brand new editor</a>, and get early access to the latest features and improvements</>}
|
||||
title='Ghost editor (beta)'
|
||||
/>
|
||||
<LabItem
|
||||
action={<Button color='grey' label='Open' size='sm' />}
|
||||
detail={<>A <a className='text-green' href="https://ghost.org/help/importing-from-substack/" rel="noopener noreferrer" target="_blank">step-by-step tool</a> to easily import all your content, members and paid subscriptions</>}
|
||||
title='Substack migrator'
|
||||
/>
|
||||
<LabItem
|
||||
action={<Toggle />}
|
||||
detail={<>Translate your membership flows into your publication language (<a className='text-green' href="https://github.com/TryGhost/Ghost/tree/main/ghost/i18n/locales" rel="noopener noreferrer" target="_blank">supported languages</a>). Don’t see yours? <a className='text-green' href="https://forum.ghost.org/t/help-translate-ghost-beta/37461" rel="noopener noreferrer" target="_blank">Get involved</a></>}
|
||||
title='Portal translation'
|
||||
/>
|
||||
<LabItem
|
||||
action={
|
||||
<div className='flex flex-col items-end gap-1'>
|
||||
<Button color='grey' label='Upload redirects file' size='sm' />
|
||||
<Button color='green' label='Download current redirects' link />
|
||||
</div>
|
||||
}
|
||||
detail={<>Configure redirects for old or moved content, <br /> more info in the <a className='text-green' href="https://ghost.org/tutorials/implementing-redirects/" rel="noopener noreferrer" target="_blank">docs</a></>}
|
||||
title='Redirects'
|
||||
/>
|
||||
<LabItem
|
||||
action={
|
||||
<div className='flex flex-col items-end gap-1'>
|
||||
<Button color='grey' label='Upload routes file' size='sm' />
|
||||
<Button color='green' label='Download current routes' link />
|
||||
</div>
|
||||
}
|
||||
detail='Configure dynamic routing by modifying the routes.yaml file'
|
||||
title='Routes'
|
||||
/>
|
||||
</List>
|
||||
);
|
||||
};
|
||||
|
||||
const BetaFeautres: React.FC = () => {
|
||||
return (
|
||||
<List titleSeparator={false}>
|
||||
<LabItem
|
||||
action={<Toggle />}
|
||||
detail='This is dynamic'
|
||||
title='Example beta feature'
|
||||
/>
|
||||
</List>
|
||||
);
|
||||
};
|
||||
|
||||
const Labs: React.FC<{ keywords: string[] }> = ({keywords}) => {
|
||||
const [selectedTab, setSelectedTab] = useState<'labs-migration-options' | 'labs-alpha-features' | 'labs-beta-features'>('labs-migration-options');
|
||||
const [isOpen, setIsOpen] = useState(false);
|
||||
|
||||
const tabs = [
|
||||
{
|
||||
id: 'labs-migration-options',
|
||||
title: 'Migration options',
|
||||
contents: <MigrationOptions />
|
||||
},
|
||||
{
|
||||
id: 'labs-alpha-features',
|
||||
title: 'Alpha features',
|
||||
contents: <AlphaFeatures />
|
||||
},
|
||||
{
|
||||
id: 'labs-beta-features',
|
||||
title: 'Beta features',
|
||||
contents: <BetaFeautres />
|
||||
}
|
||||
] as const;
|
||||
|
||||
return (
|
||||
<SettingGroup
|
||||
customHeader={
|
||||
<div className='z-10 flex items-start justify-between'>
|
||||
<SettingGroupHeader description='This is a testing ground for new or experimental features. They may change, break or inexplicably disappear at any time.' title='Labs' />
|
||||
{
|
||||
!isOpen ?
|
||||
<Button color='green' label='Open' link onClick={() => {
|
||||
setIsOpen(true);
|
||||
}} /> :
|
||||
<Button color='green' label='Close' link onClick={() => {
|
||||
setIsOpen(false);
|
||||
}} />
|
||||
}
|
||||
</div>
|
||||
}
|
||||
isEditing={isOpen}
|
||||
keywords={keywords}
|
||||
navid='labs'
|
||||
testId='labs'
|
||||
>
|
||||
{isOpen ?
|
||||
<TabView<'labs-migration-options' | 'labs-alpha-features' | 'labs-beta-features'> selectedTab={selectedTab} tabs={tabs} onTabChange={setSelectedTab} />
|
||||
:
|
||||
<div className='absolute inset-0 z-0 overflow-hidden opacity-70'>
|
||||
<img className='absolute -right-6 -top-6' src={LabsBubbles} />
|
||||
</div>
|
||||
}
|
||||
</SettingGroup>
|
||||
);
|
||||
};
|
||||
|
||||
export default Labs;
|
Loading…
Reference in a new issue