plugin updates
This commit is contained in:
@@ -1,311 +0,0 @@
|
||||
import React, { useRef, useState, useEffect } from 'react';
|
||||
import { styled } from '@linaria/react';
|
||||
import {
|
||||
CALYPSO,
|
||||
CALYPSO_LIGHT,
|
||||
CALYPSO_MEDIUM,
|
||||
OBSIDIAN,
|
||||
} from '../UIComponents/colors';
|
||||
import UISpinner from '../UIComponents/UISpinner';
|
||||
import LoadState, { LoadStateType } from '../enums/loadState';
|
||||
|
||||
const Container = styled.div`
|
||||
color: ${OBSIDIAN};
|
||||
font-family: 'Lexend Deca', Helvetica, Arial, sans-serif;
|
||||
font-size: 14px;
|
||||
position: relative;
|
||||
`;
|
||||
|
||||
interface IControlContainerProps {
|
||||
focused: boolean;
|
||||
}
|
||||
|
||||
const ControlContainer = styled.div<IControlContainerProps>`
|
||||
align-items: center;
|
||||
background-color: hsl(0, 0%, 100%);
|
||||
border-color: hsl(0, 0%, 80%);
|
||||
border-radius: 4px;
|
||||
border-style: solid;
|
||||
border-width: ${props => (props.focused ? '0' : '1px')};
|
||||
cursor: default;
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
justify-content: space-between;
|
||||
min-height: 38px;
|
||||
outline: 0 !important;
|
||||
position: relative;
|
||||
transition: all 100ms;
|
||||
box-sizing: border-box;
|
||||
box-shadow: ${props =>
|
||||
props.focused ? `0 0 0 2px ${CALYPSO_MEDIUM}` : 'none'};
|
||||
&:hover {
|
||||
border-color: hsl(0, 0%, 70%);
|
||||
}
|
||||
`;
|
||||
|
||||
const ValueContainer = styled.div`
|
||||
align-items: center;
|
||||
display: flex;
|
||||
flex: 1;
|
||||
flex-wrap: wrap;
|
||||
padding: 2px 8px;
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
box-sizing: border-box;
|
||||
`;
|
||||
|
||||
const Placeholder = styled.div`
|
||||
color: hsl(0, 0%, 50%);
|
||||
margin-left: 2px;
|
||||
margin-right: 2px;
|
||||
position: absolute;
|
||||
top: 50%;
|
||||
transform: translateY(-50%);
|
||||
box-sizing: border-box;
|
||||
font-size: 16px;
|
||||
`;
|
||||
|
||||
const SingleValue = styled.div`
|
||||
color: hsl(0, 0%, 20%);
|
||||
margin-left: 2px;
|
||||
margin-right: 2px;
|
||||
max-width: calc(100% - 8px);
|
||||
overflow: hidden;
|
||||
position: absolute;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
top: 50%;
|
||||
transform: translateY(-50%);
|
||||
box-sizing: border-box;
|
||||
`;
|
||||
|
||||
const IndicatorContainer = styled.div`
|
||||
align-items: center;
|
||||
align-self: stretch;
|
||||
display: flex;
|
||||
flex-shrink: 0;
|
||||
box-sizing: border-box;
|
||||
`;
|
||||
|
||||
const DropdownIndicator = styled.div`
|
||||
border-top: 8px solid ${CALYPSO};
|
||||
border-left: 6px solid transparent;
|
||||
border-right: 6px solid transparent;
|
||||
width: 0px;
|
||||
height: 0px;
|
||||
margin: 10px;
|
||||
`;
|
||||
|
||||
const InputContainer = styled.div`
|
||||
margin: 2px;
|
||||
padding-bottom: 2px;
|
||||
padding-top: 2px;
|
||||
visibility: visible;
|
||||
color: hsl(0, 0%, 20%);
|
||||
box-sizing: border-box;
|
||||
`;
|
||||
|
||||
const Input = styled.input`
|
||||
box-sizing: content-box;
|
||||
background: rgba(0, 0, 0, 0) none repeat scroll 0px center;
|
||||
border: 0px none;
|
||||
font-size: inherit;
|
||||
opacity: 1;
|
||||
outline: currentcolor none 0px;
|
||||
padding: 0px;
|
||||
color: inherit;
|
||||
font-family: inherit;
|
||||
`;
|
||||
|
||||
const InputShadow = styled.div`
|
||||
position: absolute;
|
||||
opacity: 0;
|
||||
font-size: inherit;
|
||||
`;
|
||||
|
||||
const MenuContainer = styled.div`
|
||||
position: absolute;
|
||||
top: 100%;
|
||||
background-color: #fff;
|
||||
border-radius: 4px;
|
||||
margin-bottom: 8px;
|
||||
margin-top: 8px;
|
||||
z-index: 9999;
|
||||
box-shadow: 0 0 0 1px hsla(0, 0%, 0%, 0.1), 0 4px 11px hsla(0, 0%, 0%, 0.1);
|
||||
width: 100%;
|
||||
`;
|
||||
|
||||
const MenuList = styled.div`
|
||||
max-height: 300px;
|
||||
overflow-y: auto;
|
||||
padding-bottom: 4px;
|
||||
padding-top: 4px;
|
||||
position: relative;
|
||||
`;
|
||||
|
||||
const MenuGroup = styled.div`
|
||||
padding-bottom: 8px;
|
||||
padding-top: 8px;
|
||||
`;
|
||||
|
||||
const MenuGroupHeader = styled.div`
|
||||
color: #999;
|
||||
cursor: default;
|
||||
font-size: 75%;
|
||||
font-weight: 500;
|
||||
margin-bottom: 0.25em;
|
||||
text-transform: uppercase;
|
||||
padding-left: 12px;
|
||||
padding-left: 12px;
|
||||
`;
|
||||
|
||||
interface IMenuItemProps {
|
||||
selected: boolean;
|
||||
}
|
||||
|
||||
const MenuItem = styled.div<IMenuItemProps>`
|
||||
display: block;
|
||||
background-color: ${props =>
|
||||
props.selected ? CALYPSO_MEDIUM : 'transparent'};
|
||||
color: ${props => (props.selected ? '#fff' : 'inherit')};
|
||||
cursor: default;
|
||||
font-size: inherit;
|
||||
width: 100%;
|
||||
padding: 8px 12px;
|
||||
&:hover {
|
||||
background-color: ${props =>
|
||||
props.selected ? CALYPSO_MEDIUM : CALYPSO_LIGHT};
|
||||
}
|
||||
`;
|
||||
|
||||
interface IAsyncSelectProps {
|
||||
placeholder: string;
|
||||
value: any;
|
||||
loadOptions?: Function;
|
||||
defaultOptions?: any[];
|
||||
onChange: Function;
|
||||
}
|
||||
|
||||
export default function AsyncSelect({
|
||||
placeholder,
|
||||
value,
|
||||
loadOptions,
|
||||
onChange,
|
||||
defaultOptions,
|
||||
}: IAsyncSelectProps) {
|
||||
const inputEl = useRef<HTMLInputElement>(null);
|
||||
const inputShadowEl = useRef<HTMLDivElement>(null);
|
||||
const [isFocused, setFocus] = useState(false);
|
||||
const [loadState, setLoadState] = useState<LoadStateType>(
|
||||
LoadState.NotLoaded
|
||||
);
|
||||
const [localValue, setLocalValue] = useState('');
|
||||
const [options, setOptions] = useState(defaultOptions);
|
||||
|
||||
const inputSize = `${
|
||||
inputShadowEl.current ? inputShadowEl.current.clientWidth + 10 : 2
|
||||
}px`;
|
||||
|
||||
useEffect(() => {
|
||||
if (loadOptions && loadState === LoadState.NotLoaded) {
|
||||
loadOptions('', (result: any) => {
|
||||
setOptions(result);
|
||||
setLoadState(LoadState.Idle);
|
||||
});
|
||||
}
|
||||
}, [loadOptions, loadState]);
|
||||
|
||||
const renderItems = (items: any[] = [], parentKey?: number) => {
|
||||
return items.map((item, index) => {
|
||||
if (item.options) {
|
||||
return (
|
||||
<MenuGroup key={`async-select-item-${index}`}>
|
||||
<MenuGroupHeader id={`${index}-heading`}>
|
||||
{item.label}
|
||||
</MenuGroupHeader>
|
||||
<div>{renderItems(item.options, index)}</div>
|
||||
</MenuGroup>
|
||||
);
|
||||
} else {
|
||||
const key = `async-select-item-${
|
||||
parentKey !== undefined ? `${parentKey}-${index}` : index
|
||||
}`;
|
||||
return (
|
||||
<MenuItem
|
||||
key={key}
|
||||
id={key}
|
||||
selected={value && item.value === value.value}
|
||||
onClick={() => {
|
||||
onChange(item);
|
||||
setFocus(false);
|
||||
}}
|
||||
>
|
||||
{item.label}
|
||||
</MenuItem>
|
||||
);
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
return (
|
||||
<Container>
|
||||
<ControlContainer
|
||||
id="leadin-async-selector"
|
||||
focused={isFocused}
|
||||
onClick={() => {
|
||||
if (isFocused) {
|
||||
if (inputEl.current) {
|
||||
inputEl.current.blur();
|
||||
}
|
||||
setFocus(false);
|
||||
setLocalValue('');
|
||||
} else {
|
||||
if (inputEl.current) {
|
||||
inputEl.current.focus();
|
||||
}
|
||||
setFocus(true);
|
||||
}
|
||||
}}
|
||||
>
|
||||
<ValueContainer>
|
||||
{localValue === '' &&
|
||||
(!value ? (
|
||||
<Placeholder>{placeholder}</Placeholder>
|
||||
) : (
|
||||
<SingleValue>{value.label}</SingleValue>
|
||||
))}
|
||||
<InputContainer>
|
||||
<Input
|
||||
ref={inputEl}
|
||||
onFocus={() => {
|
||||
setFocus(true);
|
||||
}}
|
||||
onChange={e => {
|
||||
setLocalValue(e.target.value);
|
||||
setLoadState(LoadState.Loading);
|
||||
loadOptions &&
|
||||
loadOptions(e.target.value, (result: any) => {
|
||||
setOptions(result);
|
||||
setLoadState(LoadState.Idle);
|
||||
});
|
||||
}}
|
||||
value={localValue}
|
||||
width={inputSize}
|
||||
id="asycn-select-input"
|
||||
/>
|
||||
<InputShadow ref={inputShadowEl}>{localValue}</InputShadow>
|
||||
</InputContainer>
|
||||
</ValueContainer>
|
||||
<IndicatorContainer>
|
||||
{loadState === LoadState.Loading && <UISpinner />}
|
||||
<DropdownIndicator />
|
||||
</IndicatorContainer>
|
||||
</ControlContainer>
|
||||
{isFocused && (
|
||||
<MenuContainer>
|
||||
<MenuList>{renderItems(options)}</MenuList>
|
||||
</MenuContainer>
|
||||
)}
|
||||
</Container>
|
||||
);
|
||||
}
|
||||
@@ -1,55 +0,0 @@
|
||||
import React from 'react';
|
||||
import UIButton from '../UIComponents/UIButton';
|
||||
import UIContainer from '../UIComponents/UIContainer';
|
||||
import HubspotWrapper from './HubspotWrapper';
|
||||
import { adminUrl, redirectNonce } from '../../constants/leadinConfig';
|
||||
import { pluginPath } from '../../constants/leadinConfig';
|
||||
import { __ } from '@wordpress/i18n';
|
||||
|
||||
interface IErrorHandlerProps {
|
||||
status: number;
|
||||
resetErrorState?: React.MouseEventHandler<HTMLButtonElement>;
|
||||
errorInfo?: {
|
||||
header: string;
|
||||
message: string;
|
||||
action: string;
|
||||
};
|
||||
}
|
||||
|
||||
function redirectToPlugin() {
|
||||
window.location.href = `${adminUrl}admin.php?page=leadin&leadin_expired=${redirectNonce}`;
|
||||
}
|
||||
|
||||
export default function ErrorHandler({
|
||||
status,
|
||||
resetErrorState,
|
||||
errorInfo = { header: '', message: '', action: '' },
|
||||
}: IErrorHandlerProps) {
|
||||
const isUnauthorized = status === 401 || status === 403;
|
||||
const errorHeader = isUnauthorized
|
||||
? __("Your plugin isn't authorized", 'leadin')
|
||||
: errorInfo.header;
|
||||
const errorMessage = isUnauthorized
|
||||
? __('Reauthorize your plugin to access your free HubSpot tools', 'leadin')
|
||||
: errorInfo.message;
|
||||
|
||||
return (
|
||||
<HubspotWrapper pluginPath={pluginPath}>
|
||||
<UIContainer textAlign="center">
|
||||
<h4>{errorHeader}</h4>
|
||||
<p>
|
||||
<b>{errorMessage}</b>
|
||||
</p>
|
||||
{isUnauthorized ? (
|
||||
<UIButton data-test-id="authorize-button" onClick={redirectToPlugin}>
|
||||
{__('Go to plugin', 'leadin')}
|
||||
</UIButton>
|
||||
) : (
|
||||
<UIButton data-test-id="retry-button" onClick={resetErrorState}>
|
||||
{errorInfo.action}
|
||||
</UIButton>
|
||||
)}
|
||||
</UIContainer>
|
||||
</HubspotWrapper>
|
||||
);
|
||||
}
|
||||
@@ -1,26 +0,0 @@
|
||||
import { styled } from '@linaria/react';
|
||||
|
||||
interface IHubspotWrapperProps {
|
||||
pluginPath: string;
|
||||
padding?: string;
|
||||
}
|
||||
|
||||
export default styled.div<IHubspotWrapperProps>`
|
||||
background-image: ${props =>
|
||||
`url(${props.pluginPath}/public/assets/images/hubspot.svg)`};
|
||||
background-color: #f5f8fa;
|
||||
background-repeat: no-repeat;
|
||||
background-position: center 25px;
|
||||
background-size: 120px;
|
||||
color: #33475b;
|
||||
font-family: 'Lexend Deca', Helvetica, Arial, sans-serif;
|
||||
font-size: 14px;
|
||||
|
||||
padding: ${(props: any) => props.padding || '90px 20% 25px'};
|
||||
|
||||
p {
|
||||
font-size: inherit !important;
|
||||
line-height: 24px;
|
||||
margin: 4px 0;
|
||||
}
|
||||
`;
|
||||
@@ -1,12 +0,0 @@
|
||||
import React from 'react';
|
||||
import HubspotWrapper from './HubspotWrapper';
|
||||
import UISpinner from '../UIComponents/UISpinner';
|
||||
import { pluginPath } from '../../constants/leadinConfig';
|
||||
|
||||
export default function LoadingBlock() {
|
||||
return (
|
||||
<HubspotWrapper pluginPath={pluginPath}>
|
||||
<UISpinner size={50} />
|
||||
</HubspotWrapper>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user