mirror of
https://github.com/towfiqi/serpbear
synced 2025-06-26 18:15:54 +00:00
151 lines
7.7 KiB
TypeScript
151 lines
7.7 KiB
TypeScript
import React, { useState, useMemo } from 'react';
|
|
import TimeAgo from 'react-timeago';
|
|
import dayjs from 'dayjs';
|
|
import Icon from '../common/Icon';
|
|
import countries from '../../utils/countries';
|
|
import ChartSlim from '../common/ChartSlim';
|
|
import { generateTheChartData } from '../common/generateChartData';
|
|
|
|
type KeywordProps = {
|
|
keywordData: KeywordType,
|
|
selected: boolean,
|
|
refreshkeyword: Function,
|
|
favoriteKeyword: Function,
|
|
removeKeyword: Function,
|
|
selectKeyword: Function,
|
|
manageTags: Function,
|
|
showKeywordDetails: Function,
|
|
lastItem?:boolean
|
|
}
|
|
|
|
const Keyword = (props: KeywordProps) => {
|
|
const { keywordData, refreshkeyword, favoriteKeyword, removeKeyword, selectKeyword, selected, showKeywordDetails, manageTags, lastItem } = props;
|
|
const {
|
|
keyword, domain, ID, position, url = '', lastUpdated, country, sticky, history = {}, updating = false, lastUpdateError = 'false',
|
|
} = keywordData;
|
|
const [showOptions, setShowOptions] = useState(false);
|
|
const [showPositionError, setPositionError] = useState(false);
|
|
const turncatedURL = useMemo(() => {
|
|
return url.replace(`https://${domain}`, '').replace(`https://www.${domain}`, '').replace(`http://${domain}`, '');
|
|
}, [url, domain]);
|
|
|
|
const chartData = useMemo(() => {
|
|
return generateTheChartData(history, '7');
|
|
}, [history]);
|
|
|
|
const positionChange = useMemo(() => {
|
|
let status = 0;
|
|
if (Object.keys(history).length >= 2) {
|
|
const historyArray = Object.keys(history).map((dateKey:string) => {
|
|
return { date: new Date(dateKey).getTime(), dateRaw: dateKey, position: history[dateKey] };
|
|
});
|
|
const historySorted = historyArray.sort((a, b) => a.date - b.date);
|
|
const previousPos = historySorted[historySorted.length - 2].position;
|
|
status = previousPos === 0 ? position : previousPos - position;
|
|
}
|
|
return status;
|
|
}, [history, position]);
|
|
|
|
const optionsButtonStyle = 'block px-2 py-2 cursor-pointer hover:bg-indigo-50 hover:text-blue-700';
|
|
|
|
const renderPosition = () => {
|
|
if (position === 0) {
|
|
return <span title='Not in Top 100'>{'-'}</span>;
|
|
}
|
|
if (updating) {
|
|
return <span title='Updating Keyword Position'><Icon type="loading" /></span>;
|
|
}
|
|
return position;
|
|
};
|
|
|
|
return (
|
|
<div
|
|
key={keyword}
|
|
className={`keyword relative py-5 px-4 text-gray-600 border-b-[1px] border-gray-200 lg:py-4 lg:px-6 lg:border-0
|
|
lg:flex lg:justify-between lg:items-center ${selected ? ' bg-indigo-50 keyword--selected' : ''} ${lastItem ? 'border-b-0' : ''}`}>
|
|
<div className=' w-3/4 lg:flex-1 lg:basis-20 lg:w-auto font-semibold cursor-pointer'>
|
|
<button
|
|
className={`p-0 mr-2 leading-[0px] inline-block rounded-sm pt-0 px-[1px] pb-[3px] border
|
|
${selected ? ' bg-blue-700 border-blue-700 text-white' : 'text-transparent'}`}
|
|
onClick={() => selectKeyword(ID)}
|
|
>
|
|
<Icon type="check" size={10} />
|
|
</button>
|
|
<a
|
|
className='py-2 hover:text-blue-600'
|
|
onClick={() => showKeywordDetails()}>
|
|
<span className={`fflag fflag-${country} w-[18px] h-[12px] mr-2`} title={countries[country][0]} />{keyword}
|
|
</a>
|
|
{sticky && <button className='ml-2 relative top-[2px]' title='Favorite'><Icon type="star-filled" size={16} color="#fbd346" /></button>}
|
|
{lastUpdateError !== 'false'
|
|
&& <button className='ml-2 relative top-[2px]' onClick={() => setPositionError(true)}>
|
|
<Icon type="error" size={18} color="#FF3672" />
|
|
</button>
|
|
}
|
|
</div>
|
|
<div
|
|
className={`keyword_position absolute bg-[#f8f9ff] w-fit min-w-[50px] h-12 p-2 text-base mt-[-20px] rounded right-5 lg:relative
|
|
lg:bg-transparent lg:w-auto lg:h-auto lg:mt-0 lg:p-0 lg:text-sm lg:flex-1 lg:basis-40 lg:grow-0 lg:right-0 text-center font-semibold`}>
|
|
{renderPosition()}
|
|
{!updating && positionChange > 0 && <i className=' not-italic ml-1 text-xs text-[#5ed7c3]'>▲ {positionChange}</i>}
|
|
{!updating && positionChange < 0 && <i className=' not-italic ml-1 text-xs text-red-300'>▼ {positionChange}</i>}
|
|
</div>
|
|
{chartData.labels.length > 0 && (
|
|
<div className='lg:flex-1 hidden lg:block'>
|
|
<ChartSlim labels={chartData.labels} sreies={chartData.sreies} />
|
|
</div>
|
|
)}
|
|
<div
|
|
className={`keyword_url inline-block mt-4 mr-5 ml-5 lg:flex-1 text-gray-400 lg:m-0 max-w-[70px]
|
|
overflow-hidden text-ellipsis whitespace-nowrap lg:max-w-none lg:pr-5`}>
|
|
<span className='mr-3 lg:hidden'><Icon type="link-alt" size={14} color="#999" /></span>{turncatedURL || '-'}</div>
|
|
<div
|
|
className='inline-block mt-[4] top-[-5px] relative lg:flex-1 lg:m-0'>
|
|
<span className='mr-2 lg:hidden'><Icon type="clock" size={14} color="#999" /></span>
|
|
<TimeAgo title={dayjs(lastUpdated).format('DD-MMM-YYYY, hh:mm:ss A')} date={lastUpdated} />
|
|
</div>
|
|
<div className='absolute right-7 mt-[-10px] lg:flex-1 lg:basis-5 lg:grow-0 lg:shrink-0 lg:relative lg:mt-0 lg:right-auto'>
|
|
<button
|
|
className={`keyword_dots rounded px-1 text-indigo-300 hover:bg-indigo-50 ${showOptions ? 'bg-indigo-50 text-indigo-600 ' : ''}`}
|
|
onClick={() => setShowOptions(!showOptions)}>
|
|
<Icon type="dots" size={20} />
|
|
</button>
|
|
{showOptions && (
|
|
<ul className='keyword_options customShadow absolute w-[180px] right-0 bg-white rounded border z-20'>
|
|
<li>
|
|
<a className={optionsButtonStyle} onClick={() => { refreshkeyword([ID]); setShowOptions(false); }}>
|
|
<span className=' bg-indigo-100 text-blue-700 px-1 rounded'><Icon type="reload" size={11} /></span> Refresh Keyword</a>
|
|
</li>
|
|
<li>
|
|
<a className={optionsButtonStyle}
|
|
onClick={() => { favoriteKeyword({ keywordID: ID, sticky: !sticky }); setShowOptions(false); }}>
|
|
<span className=' bg-yellow-300/30 text-yellow-500 px-1 rounded'>
|
|
<Icon type="star" size={14} />
|
|
</span> { sticky ? 'Unfavorite Keyword' : 'Favorite Keyword'}
|
|
</a>
|
|
</li>
|
|
<li><a className={optionsButtonStyle} onClick={() => { manageTags(); setShowOptions(false); }}>
|
|
<span className=' bg-green-100 text-green-500 px-1 rounded'><Icon type="tags" size={14} /></span> Add/Edit Tags</a>
|
|
</li>
|
|
<li><a className={optionsButtonStyle} onClick={() => { removeKeyword([ID]); setShowOptions(false); }}>
|
|
<span className=' bg-red-100 text-red-600 px-1 rounded'><Icon type="trash" size={14} /></span> Remove Keyword</a>
|
|
</li>
|
|
</ul>
|
|
)}
|
|
</div>
|
|
{lastUpdateError !== 'false' && showPositionError
|
|
&& <div className=' absolute mt-[-70px] p-2 bg-white z-30 border border-red-200 rounded w-[220px] left-4 shadow-sm text-xs'>
|
|
Error Updating Keyword position (Tried <TimeAgo
|
|
title={dayjs(lastUpdateError).format('DD-MMM-YYYY, hh:mm:ss A')}
|
|
date={lastUpdateError} />)
|
|
<i className='absolute top-0 right-0 ml-2 p-2 font-semibold not-italic cursor-pointer' onClick={() => setPositionError(false)}>
|
|
<Icon type="close" size={16} color="#999" />
|
|
</i>
|
|
</div>
|
|
}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default Keyword;
|