Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a Snackbar notice to inform users that settings have been saved successfully. #838

Merged
merged 4 commits into from
Jan 13, 2025
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -174,7 +174,8 @@ export const SaveSettingsButton = ( {
setSettings,
onSaveSuccess = () => {},
} ) => {
const { createErrorNotice, removeNotices } = useDispatch( noticesStore );
const { createErrorNotice, createSuccessNotice, removeNotices } =
useDispatch( noticesStore );
const notices = useSelect( ( select ) =>
select( noticesStore ).getNotices()
);
Expand Down Expand Up @@ -205,6 +206,12 @@ export const SaveSettingsButton = ( {

setSettings( res.settings );
onSaveSuccess();
createSuccessNotice(
__( 'Settings saved successfully.', 'classifai' ),
{
type: 'snackbar',
}
);
setIsSaving( false );
} )
.catch( ( error ) => {
Expand Down
34 changes: 32 additions & 2 deletions src/js/settings/components/classifai-settings/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,12 @@ import {
/**
* WordPress dependencies
*/
import { useDispatch } from '@wordpress/data';
import { SlotFillProvider } from '@wordpress/components';
import { useSelect, useDispatch } from '@wordpress/data';
import { SlotFillProvider, SnackbarList } from '@wordpress/components';
import { __, sprintf } from '@wordpress/i18n';
import { useEffect, useState } from '@wordpress/element';
import apiFetch from '@wordpress/api-fetch';
import { store as noticeStore } from '@wordpress/notices';

/**
* Internal dependencies
Expand All @@ -28,6 +29,7 @@ import { STORE_NAME } from '../../data/store';
import { FeatureContext } from '../feature-settings/context';
import { ClassifAIRegistration } from '../classifai-registration';
import { ClassifAIWelcomeGuide } from './welcome-guide';
import { Notices } from '../feature-settings/notices';

const { services, features } = window.classifAISettings;

Expand Down Expand Up @@ -121,6 +123,32 @@ export const ServiceNavigation = () => {
);
};

/**
* Snackbar component to render the snackbar notifications.
*
* @return {React.ReactElement} The Snackbar component.
*/
export const SnackbarNotifications = () => {
const { removeNotice } = useDispatch( noticeStore );

const { notices } = useSelect( ( select ) => {
const allNotices = select( noticeStore ).getNotices();
return {
notices: allNotices.filter(
( notice ) => notice.type === 'snackbar'
),
};
}, [] );

return (
<SnackbarList
className="classifai-settings-snackbar-notices"
notices={ notices }
onRemove={ ( notice ) => removeNotice( notice ) }
/>
);
};

/**
* Main ClassifAI Settings Component.
*
Expand Down Expand Up @@ -188,6 +216,7 @@ export const ClassifAISettings = () => {
<Header />
<div className="classifai-settings-wrapper">
<div className="classifai-admin-notices wrap"></div>
<Notices feature={ 'generic-notices' } />
<ServiceNavigation />
<Routes>
<Route
Expand All @@ -211,6 +240,7 @@ export const ClassifAISettings = () => {
/>
</Routes>
</div>
<SnackbarNotifications />
</HashRouter>
</SlotFillProvider>
);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,8 +33,12 @@ export const SaveSettingsButton = ( {
label = __( 'Save Settings', 'classifai' ),
} ) => {
const { featureName } = useFeatureSettings();
const { createErrorNotice, removeNotices, removeNotice } =
useDispatch( noticesStore );
const {
createSuccessNotice,
createErrorNotice,
removeNotices,
removeNotice,
} = useDispatch( noticesStore );
const notices = useSelect( ( select ) =>
select( noticesStore ).getNotices()
);
Expand Down Expand Up @@ -82,6 +86,12 @@ export const SaveSettingsButton = ( {
return;
}
onSaveSuccess();
createSuccessNotice(
__( 'Settings saved successfully.', 'classifai' ),
{
type: 'snackbar',
}
);
setSettings( res.settings );
setIsSaving( false );
} )
Expand Down
69 changes: 57 additions & 12 deletions src/js/settings/components/service-settings/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,9 @@
Notice,
Icon,
} from '@wordpress/components';
import { useEffect, useRef } from '@wordpress/element';
import { useEffect, useRef, useState } from '@wordpress/element';
import apiFetch from '@wordpress/api-fetch';
import { store as noticesStore } from '@wordpress/notices';

/**
* Internal dependencies
Expand Down Expand Up @@ -54,8 +55,18 @@
* @return {Object} The ServiceSettings component.
*/
export const ServiceSettings = () => {
const [ enabled, setEnabled ] = useState( false );
const { setCurrentService, setIsSaving, setSettings } =
useDispatch( STORE_NAME );
const {
createSuccessNotice,
createErrorNotice,
removeNotices,
removeNotice,
} = useDispatch( noticesStore );
const notices = useSelect( ( select ) =>
select( noticesStore ).getNotices()
);

const { service } = useParams();
const isInitialPageLoad = useRef( true );
Expand All @@ -77,6 +88,13 @@
const serviceFeatures = features[ service ] || {};

const saveSettings = () => {
// Remove existing notices.
if ( removeNotices ) {
removeNotices( notices.map( ( { id } ) => id ) );
} else if ( removeNotice ) {
notices.forEach( ( { id } ) => removeNotice( id ) );
}

setIsSaving( true );
apiFetch( {
path: '/classifai/v1/settings/',
Expand All @@ -86,15 +104,41 @@
is_setup: true,
step: 'enable_features',
},
} ).then( ( res ) => {
if ( res.errors && res.errors.length ) {
setIsSaving( false );
return;
}
} )
.then( ( res ) => {
if ( res.errors && res.errors.length ) {
res.errors.forEach( ( error ) => {
createErrorNotice( error.message, {
id: 'error-generic-notices',
} );
} );
setIsSaving( false );
return;
}

setSettings( res.settings );
setIsSaving( false );
} );
const message = enabled
? __( 'Feature enabled successfully.', 'classifai' )
: __( 'Feature disabled successfully.', 'classifai' );

createSuccessNotice( message, {
type: 'snackbar',
} );
setSettings( res.settings );
setIsSaving( false );
} )
.catch( ( error ) => {
createErrorNotice(
error.message ||
__(
'An error occurred while saving settings.',
'classifai'
),
{
id: 'error-generic-notices',
}
);
setIsSaving( false );
} );
};

const statuses = Object.keys( settings )
Expand All @@ -107,7 +151,7 @@
return;
}
saveSettings();
}, [ statuses ] );

Check warning on line 154 in src/js/settings/components/service-settings/index.js

View workflow job for this annotation

GitHub Actions / eslint

React Hook useEffect has a missing dependency: 'saveSettings'. Either include it or remove the dependency array

return (
<div className="classifai-settings-dashboard">
Expand All @@ -126,7 +170,8 @@
feature
)
}
onChange={ ( value ) =>
onChange={ ( value ) => {
setEnabled( value );
wp.data
.dispatch( STORE_NAME )
.setFeatureSettings(
Expand All @@ -136,8 +181,8 @@
: '0',
},
feature
)
}
);
} }
__nextHasNoMarginBottom
/>
</Flex>
Expand Down
10 changes: 10 additions & 0 deletions src/scss/settings.scss
Original file line number Diff line number Diff line change
Expand Up @@ -430,6 +430,16 @@
.classifai-admin-notices {
margin-right: 0px;
}

.classifai-settings-snackbar-notices {
position: fixed;
left: auto;
bottom: 40px;

@media screen and (max-width: 600px) {
padding: 0px 20px 0px 0px;
}
}
}

.classifai-onboarding {
Expand Down
Loading