Press n or j to go to the next uncovered block, b, p or k for the previous block.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 | 99x 99x 99x 2x 2x 97x 97x 99x 99x 99x 178x 178x 178x 117x 99x 20x 20x 20x 20x 20x 97x 97x 4x 4x 4x 4x 4x 4x 4x 4x 4x 6x 6x 97x 99x 95x 2x 95x 95x 172x 172x 172x 172x 95x 95x 71x 71x 146x 146x 147x 147x 147x 25x 97x 76x 76x 97x 36x 36x 36x 36x 36x 56x 56x 56x 56x 36x 97x 99x 97x | /**
* @module data-transformers/content-generators/motions
* @description Generator for "motions" article content. Renders opposition motions
* grouped by party with strategy analysis.
*
* @author Hack23 AB
* @license Apache-2.0
*/
import { generateDeepAnalysisSection, analyzeDocumentsForContent } from './shared.js';
import { escapeHtml } from '../../html-utils.js';
import type { Language } from '../../types/language.js';
import type { ArticleContentData, RawDocument } from '../types.js';
import { getPillarTransition } from '../../editorial-pillars.js';
import {
L,
svSpan,
normalizePartyKey,
} from '../helpers.js';
import { detectPolicyDomains } from '../policy-analysis.js';
import {
groupMotionsByProposition,
generateOppositionStrategySection,
renderMotionEntry,
PROP_TITLE_SUFFIX_REGEX,
} from '../document-analysis.js';
export function generateMotionsContent(data: ArticleContentData, lang: Language | string): string {
const motions = data.motions || [];
let content = `<h2>${L(lang, 'oppMotions')}</h2>\n`;
if (motions.length === 0) {
content += `<p>${L(lang, 'noMotions')}</p>\n`;
return content;
}
// Analytical lede paragraph
const breakdownFn = L(lang, 'motionsBreakdown') as string | ((n: number) => string);
const breakdownText = typeof breakdownFn === 'function'
? breakdownFn(motions.length)
: `${motions.length} new opposition motions filed.`;
content += `<p class="article-lede">${escapeHtml(String(breakdownText))}</p>\n`;
// Group motions by party for strategic analysis
const byParty: Record<string, RawDocument[]> = {};
motions.forEach(motion => {
const party = normalizePartyKey(motion.parti);
if (!byParty[party]) byParty[party] = [];
byParty[party].push(motion);
});
// Opposition strategy section with per-party analysis
const partyCount = Object.keys(byParty).filter(p => p !== 'other').length;
if (partyCount > 1) {
content += `\n <h2>${L(lang, 'oppositionStrategy')}</h2>\n`;
const strategyFn = L(lang, 'oppositionStrategyContext') as string | ((n: number) => string);
const strategyContext = typeof strategyFn === 'function'
? strategyFn(partyCount)
: `Motions from ${partyCount} different parties reveal the breadth of opposition political criticism and alternative policy agendas.`;
content += ` <p>${escapeHtml(String(strategyContext))}</p>\n`;
// Per-party analysis with domain focus
content += generateOppositionStrategySection(motions, lang);
}
// Group "med anledning av prop." motions by parent proposition to eliminate duplicate headings
const { grouped: groupedByProp, independent: independentMotions } = groupMotionsByProposition(motions);
if (groupedByProp.size > 0) {
content += `\n <h2>${L(lang, 'responsesToProp')}</h2>\n`;
groupedByProp.forEach((propMotions, propRef) => {
// Extract the descriptive title portion that follows the prop ID
const firstTitle = propMotions[0]?.titel || propMotions[0]?.title || '';
const suffixMatch = firstTitle.match(PROP_TITLE_SUFFIX_REGEX);
const propTitle = suffixMatch?.[1]?.trim() || String(propRef);
const safePropRef = escapeHtml(String(propRef));
const safePropTitle = escapeHtml(propTitle);
content += ` <h3>Prop. ${safePropRef}: ${svSpan(safePropTitle, lang)}</h3>\n`;
// Individual motions inside a prop group use h4 to maintain h2→h3→h4 hierarchy
propMotions.forEach(m => {
const html = renderMotionEntry(m, lang);
content += html.replace(/<h3(\b[^>]*)?>/g, '<h4$1>').replace(/<\/h3>/g, '</h4>');
});
});
}
// Motions to render with thematic analysis:
// - when proposition groups exist: only independent motions (non-"med anledning av")
// - when no proposition groups: all motions (preserves existing thematic behaviour)
const thematicMotions = groupedByProp.size > 0 ? independentMotions : motions;
if (thematicMotions.length > 0) {
if (groupedByProp.size > 0) {
content += `\n <h2>${L(lang, 'independentMotions')}</h2>\n`;
}
// Group motions by primary policy theme for thematic analysis
const byTheme: Record<string, RawDocument[]> = {};
thematicMotions.forEach(motion => {
const domains = detectPolicyDomains(motion, lang);
const theme = domains[0] || String(L(lang, 'generalMatters'));
if (!byTheme[theme]) byTheme[theme] = [];
byTheme[theme].push(motion);
});
const themeCount = Object.keys(byTheme).length;
if (themeCount > 1 && groupedByProp.size === 0) {
// Suppress "Thematic Analysis" h2 when already under an "Independent Motions" h2
// (groupedByProp.size === 0 means we are NOT in the split-section layout, so it is
// safe to emit the additional h2 without creating two consecutive section headers)
content += `\n <h2>${L(lang, 'thematicAnalysis')}</h2>\n`;
Object.entries(byTheme).forEach(([theme, themeMotions]) => {
content += `\n <h3>${escapeHtml(theme)} (${themeMotions.length})</h3>\n`;
themeMotions.forEach(motion => {
// Demote motion entry headings one level when inside a themed section
const entryHtml = renderMotionEntry(motion, lang);
const demotedHtml = entryHtml
.replace(/<h3(\b[^>]*)?>/g, '<h4$1>')
.replace(/<\/h3>/g, '</h4>');
content += demotedHtml;
});
});
} else {
// Single theme, no detection, or alongside proposition groups: flat list
thematicMotions.forEach(motion => { content += renderMotionEntry(motion, lang); });
}
}
// Deep Analysis section (5W framework + document analysis framework)
if (motions.length >= 2) {
const { frameworkAnalysis, perspectiveAnalysis } = analyzeDocumentsForContent(motions, lang, data.ciaContext);
content += generateDeepAnalysisSection({
documents: motions,
lang,
cia: data.ciaContext,
articleType: 'motions',
frameworkAnalysis,
perspectiveAnalysis,
});
}
// Party activity breakdown
if (partyCount > 0) {
// Narrative bridge before cross-party analysis (inter-pillar transition)
const watchTransition = getPillarTransition(lang, 'watchToOpposition');
Iif (watchTransition) {
content += ` <p class="pillar-transition">${escapeHtml(watchTransition)}</p>\n`;
}
content += `\n <h2>${L(lang, 'coalitionDynamics')}</h2>\n`;
content += ` <div class="context-box">\n <ul>\n`;
Object.entries(byParty).forEach(([party, partyMotions]) => {
Eif (party !== 'other') {
const detailFn = L(lang, 'partyMotionsFiled') as string | ((party: string, n: number) => string);
const detail = typeof detailFn === 'function'
? detailFn(party, partyMotions.length)
: `${party}: ${partyMotions.length} motions filed`;
content += ` <li>${escapeHtml(String(detail))}</li>\n`;
}
});
content += ` </ul>\n </div>\n`;
}
// Government department engagement section (from analyze_g0v_by_department)
const govDeptData = data.govDeptData ?? [];
Iif (govDeptData.length > 0) {
content += `\n <h2>${L(lang, 'govEngagement')}</h2>\n`;
content += ` <div class="context-box">\n <ul>\n`;
govDeptData.slice(0, 5).forEach(dept => {
const deptName = escapeHtml(String(dept['name'] ?? dept['departement'] ?? dept['department'] ?? ''));
const deptCount = dept['count'] ?? dept['total'] ?? dept['document_count'];
if (deptName) {
const hasDeptCount = deptCount !== null && deptCount !== undefined;
content += hasDeptCount
? ` <li><strong>${deptName}</strong> (${escapeHtml(String(deptCount))})</li>\n`
: ` <li><strong>${deptName}</strong></li>\n`;
}
});
content += ` </ul>\n </div>\n`;
}
return content;
}
|