one-click-accessibility/modules/scanner/assets/js/components/manual-fix-form/resolve-with-ai.js
VasylD 4c1546784b
[APP-1747] add parent selector (#360)
* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* Add mixpanel events

* Add mixpanel events

* Add mixpanel events

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Update modules/remediation/assets/js/actions/styles.js

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Update modules/scanner/assets/js/hooks/use-color-contrast-form.js

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix linter

---------

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>
2025-08-05 15:27:40 +02:00

277 lines
7.8 KiB
JavaScript

import AIIcon from '@elementor/icons/AIIcon';
import CopyIcon from '@elementor/icons/CopyIcon';
import EditIcon from '@elementor/icons/EditIcon';
import InfoCircleIcon from '@elementor/icons/InfoCircleIcon';
import Box from '@elementor/ui/Box';
import Button from '@elementor/ui/Button';
import Card from '@elementor/ui/Card';
import CardActions from '@elementor/ui/CardActions';
import CardContent from '@elementor/ui/CardContent';
import IconButton from '@elementor/ui/IconButton';
import Infotip from '@elementor/ui/Infotip';
import Tooltip from '@elementor/ui/Tooltip';
import Typography from '@elementor/ui/Typography';
import PropTypes from 'prop-types';
import { mixpanelEvents, mixpanelService } from '@ea11y-apps/global/services';
import { UpgradeInfoTip } from '@ea11y-apps/scanner/components/upgrade-info-tip';
import { AI_QUOTA_LIMIT, IS_AI_ENABLED } from '@ea11y-apps/scanner/constants';
import { useScannerWizardContext } from '@ea11y-apps/scanner/context/scanner-wizard-context';
import { useCopyToClipboard } from '@ea11y-apps/scanner/hooks/use-copy-to-clipboard';
import { useManualFixForm } from '@ea11y-apps/scanner/hooks/use-manual-fix-form';
import { StyledAlert } from '@ea11y-apps/scanner/styles/app.styles';
import {
AIHeader,
AITitle,
InfotipBox,
ManualTextField,
StyledSnippet,
} from '@ea11y-apps/scanner/styles/manual-fixes.styles';
import { scannerItem } from '@ea11y-apps/scanner/types/scanner-item';
import { useEffect, useState } from '@wordpress/element';
import { __ } from '@wordpress/i18n';
export const ResolveWithAi = ({ item, current }) => {
const { manualData, openedBlock } = useScannerWizardContext();
const { copied, copyToClipboard } = useCopyToClipboard();
const { getAISuggestion, resolveIssue, aiResponseLoading, resolving } =
useManualFixForm({
item,
current,
});
const [openUpgrade, setOpenUpgrade] = useState(false);
const [isEdit, setIsEdit] = useState(false);
const [manualEdit, setManualEdit] = useState(false);
const [aiSuggestion, setAiSuggestion] = useState(null);
useEffect(() => {
setAiSuggestion({
...manualData[openedBlock][current]?.aiSuggestion,
submitted: false,
});
setManualEdit(manualData[openedBlock][current]?.aiSuggestion?.snippet);
}, [manualData[openedBlock][current]?.aiSuggestion]);
const openEdit = () => {
setIsEdit(true);
mixpanelService.sendEvent(mixpanelEvents.editSnippetClicked, {
snippet_content: aiSuggestion.snippet,
category_name: openedBlock,
source: 'assistant',
});
};
const closeEdit = () => setIsEdit(false);
const handleButtonClick = async () => {
if (IS_AI_ENABLED && AI_QUOTA_LIMIT) {
await getAISuggestion();
mixpanelService.sendEvent(mixpanelEvents.fixWithAiButtonClicked, {
issue_type: item.message,
rule_id: item.ruleId,
wcag_level: item.reasonCategory.match(/\((AAA?|AA?|A)\)/)?.[1] || '',
category_name: openedBlock,
page_url: window.ea11yScannerData?.pageData?.url,
// ai_text_response: text,
});
} else {
setOpenUpgrade(true);
mixpanelService.sendEvent(mixpanelEvents.upgradeSuggestionViewed, {
current_plan: window.ea11yScannerData?.planData?.plan?.name,
action_trigger: 'fix_with_ai',
feature_locked: 'AI manual',
});
}
};
const closeUpgrade = () => setOpenUpgrade(false);
const disabled =
aiResponseLoading ||
resolving ||
(isEdit && !manualEdit) ||
(isEdit && manualEdit === aiSuggestion?.snippet);
const onResolve = async () => {
await resolveIssue(isEdit ? manualEdit : null);
setAiSuggestion({
...aiSuggestion,
snippet: manualEdit,
submitted: !isEdit,
});
};
const onManualEdit = (e) => {
setManualEdit(e.target.value);
setAiSuggestion({ ...aiSuggestion, submitted: false });
};
return aiSuggestion?.snippet ? (
<Box>
<Card variant="outlined" sx={{ overflow: 'visible' }}>
<CardContent sx={{ pt: 1.5, pb: 0 }}>
<AIHeader>
<AITitle>
<AIIcon />
<Typography variant="subtitle2">
{__('Resolve with AI', 'pojo-accessibility')}
</Typography>
{aiSuggestion?.explanation && (
<Infotip
placement="top"
PopperProps={{
disablePortal: true,
}}
content={
<InfotipBox>
<Typography
variant="subtitle1"
sx={{ mb: 1, textTransform: 'none' }}
>
{__('How AI resolves this?', 'pojo-accessibility')}
</Typography>
<Typography variant="body2">
{aiSuggestion.explanation}
</Typography>
</InfotipBox>
}
>
<InfoCircleIcon fontSize="small" />
</Infotip>
)}
</AITitle>
{!isEdit && (
<Tooltip
placement="top"
title={__('Edit', 'pojo-accessibility')}
PopperProps={{
disablePortal: true,
}}
>
<IconButton size="tiny" onClick={openEdit}>
<EditIcon fontSize="small" />
</IconButton>
</Tooltip>
)}
</AIHeader>
{isEdit ? (
<ManualTextField
value={manualEdit}
color="secondary"
size="small"
multiline
fullWidth
onChange={onManualEdit}
inputProps={{
'aria-label': __('Manual edit', 'pojo-accessibility'),
}}
/>
) : (
<StyledAlert color="info" icon={false} disabled={disabled}>
<Box display="flex" gap={0.5} alignItems="start">
<StyledSnippet variant="body2" sx={{ width: '290px' }}>
{aiSuggestion.snippet}
</StyledSnippet>
<Box>
<Tooltip
arrow
placement="bottom"
title={
copied
? __('Copied!', 'pojo-accessibility')
: __('Copy', 'pojo-accessibility')
}
PopperProps={{
disablePortal: true,
}}
>
<IconButton
size="tiny"
onClick={copyToClipboard(
aiSuggestion.snippet,
'fixed_snippet',
'assistant',
)}
>
<CopyIcon fontSize="tiny" />
</IconButton>
</Tooltip>
</Box>
</Box>
</StyledAlert>
)}
</CardContent>
<CardActions sx={{ p: 2 }}>
{isEdit ? (
<Button
size="small"
color="secondary"
variant="text"
onClick={closeEdit}
>
{__('Cancel', 'pojo-accessibility')}
</Button>
) : (
<Tooltip
arrow
placement="bottom"
title={__('Generate another solution', 'pojo-accessibility')}
PopperProps={{
disablePortal: true,
}}
>
<Button
size="small"
color="secondary"
variant="text"
onClick={handleButtonClick}
disabled={disabled}
loading={aiResponseLoading}
loadingPosition="start"
aria-label={__('Retry', 'pojo-accessibility')}
>
{__('Retry', 'pojo-accessibility')}
</Button>
</Tooltip>
)}
<Button
size="small"
color="info"
variant="contained"
disabled={disabled || aiSuggestion?.submitted}
loading={resolving}
loadingPosition="start"
onClick={onResolve}
>
{isEdit
? __('Apply changes', 'pojo-accessibility')
: __('Apply fix', 'pojo-accessibility')}
</Button>
</CardActions>
</Card>
</Box>
) : (
<Box>
<UpgradeInfoTip closeUpgrade={closeUpgrade} openUpgrade={openUpgrade}>
<Button
variant="contained"
size="small"
color={IS_AI_ENABLED && AI_QUOTA_LIMIT ? 'info' : 'promotion'}
startIcon={<AIIcon />}
fullWidth
disabled={aiResponseLoading}
loading={aiResponseLoading}
onClick={handleButtonClick}
>
{__('Let AI resolve it for you', 'pojo-accessibility')}
</Button>
</UpgradeInfoTip>
</Box>
);
};
ResolveWithAi.propTypes = {
item: scannerItem,
current: PropTypes.number.isRequired,
};