feat: Brainstorm Topics with GSC + Issue #518 fixes + Blog Editor enhancements

Issue #518 - Subscription not updating after checkout:
- Fix stale closure in SubscriptionContext checkout polling (use subscriptionRef)
- Move checkout success polling from InitialRouteHandler into SubscriptionContext
- Remove redundant polling code from InitialRouteHandler
- Fix plan label: 'Free' instead of 'No Plan', proper capitalization
- Add plan refresh button in UserBadge
- Add 'View Costing Details' to UserBadge dropdown
- Rename 'ALwrity Podcast Maker' to 'Podcast Creator' across UI
- Clean subscription=success URL param after verification

Blog Writer WYSIWYG Editor enhancements:
- Per-section preview toggle (view/edit icons)
- Enhanced hover-based toolbar
- Circular SVG progress stats bar with detailed tooltip
- Research tool chips in stats bar footer
- Per-section TTS with useTextToSpeech hook (browser native)
- Full blog preview modal with print/PDF support
- PlayAllTTSButton: sequential playback with progress bar
- OnThisPageNav: floating sidebar with scroll tracking
- Section data attributes for scroll anchoring

GSC Brainstorm Topics feature:
- Backend: gsc_brainstorm_service.py (rule-based + LLM recommendations)
- Backend: POST /gsc/brainstorm endpoint with 3-word minimum validation
- Frontend: gscBrainstorm.ts API client
- Frontend: useGSCBrainstormConnection hook (popup OAuth, no /onboarding redirect)
- Frontend: useGSCBrainstorm hook (connect check + brainstorm call)
- Frontend: GSCBrainstormModal (3-tab results: Opportunities, Gaps, AI Recs)
- Frontend: BrainstormButton (visible at 3+ words, GSC connect overlay)
- Wire BrainstormButton into ManualResearchForm and ResearchAction
- Add blog_writer to gsc_auth router features for ALWRITY_ENABLED_FEATURES
This commit is contained in:
ajaysi
2026-05-20 22:34:37 +05:30
parent 68190dedb3
commit 644e72d289
98 changed files with 16137 additions and 2501 deletions

View File

@@ -93,6 +93,8 @@ export const useBlogWriterState = () => {
return result;
}, []);
const [restoreAttempted, setRestoreAttempted] = useState(false);
// Cache recovery - restore most recent research on page load
useEffect(() => {
const restoreState = async () => {
@@ -135,10 +137,31 @@ export const useBlogWriterState = () => {
}
console.log('Restored outline, content, and title data from localStorage');
} catch (error) {
// Restore seoAnalysis and seoMetadata from localStorage
const savedSeoAnalysis = localStorage.getItem('blog_seo_analysis');
if (savedSeoAnalysis) {
try { setSeoAnalysis(JSON.parse(savedSeoAnalysis)); } catch {}
}
const savedSeoMetadata = localStorage.getItem('blog_seo_metadata');
if (savedSeoMetadata) {
try { setSeoMetadata(JSON.parse(savedSeoMetadata)); } catch {}
}
// Restore outlineConfirmed - if outline exists and was previously confirmed, mark as confirmed.
// The user had to confirm outline to reach content/SEO/publish phases.
const savedOutlineConfirmed = localStorage.getItem('blog_outline_confirmed');
if (savedOutlineConfirmed === 'true') {
setOutlineConfirmed(true);
} else if (savedOutline) {
// Backward compatibility: if outline exists but outline_confirmed wasn't saved,
// assume it was confirmed (user wouldn't have progressed without confirming).
setOutlineConfirmed(true);
}
} catch (error) {
console.error('Error restoring outline data:', error);
}
}
setRestoreAttempted(true);
};
restoreState();
@@ -151,11 +174,42 @@ export const useBlogWriterState = () => {
} catch {}
}, [contentConfirmed]);
// Persist outlineConfirmed to localStorage whenever it changes
useEffect(() => {
try {
localStorage.setItem('blog_outline_confirmed', String(outlineConfirmed));
} catch {}
}, [outlineConfirmed]);
// Persist seoAnalysis to localStorage whenever it changes
useEffect(() => {
try {
if (seoAnalysis) {
localStorage.setItem('blog_seo_analysis', JSON.stringify(seoAnalysis));
}
} catch {}
}, [seoAnalysis]);
// Persist seoMetadata to localStorage whenever it changes
useEffect(() => {
try {
if (seoMetadata) {
localStorage.setItem('blog_seo_metadata', JSON.stringify(seoMetadata));
}
} catch {}
}, [seoMetadata]);
// Persist sections to blogWriterCache whenever they change
useEffect(() => {
const outlineIds = outline.map(s => String(s.id));
if (outlineIds.length > 0 && Object.keys(sections).length > 0) {
blogWriterCache.cacheContent(sections, outlineIds);
const normalized: Record<string, string> = {};
const values = Object.values(sections);
outline.forEach((s, idx) => {
const id = String(s.id);
normalized[id] = sections[id] ?? values[idx] ?? '';
});
blogWriterCache.cacheContent(normalized, outlineIds);
}
}, [sections, outline]);
@@ -316,6 +370,7 @@ export const useBlogWriterState = () => {
flowAnalysisCompleted,
flowAnalysisResults,
sectionImages,
restoreAttempted,
// Setters
setResearch,