Passed
Pull Request — master (#1902)
by Struan
06:25
created

member_interests()   A

Complexity

Conditions 4
Paths 8

Size

Total Lines 19
Code Lines 11

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 4
eloc 11
c 0
b 0
f 0
nc 8
nop 1
dl 0
loc 19
rs 9.9
1
<?php
2
3
/*
4
 * index.php
5
 *
6
 * For displaying info about a person for a postcode or constituency.
7
 *
8
 * This page accepts either 'm' (a member_id), 'pid' (a person_id),
9
 * 'c' (a postcode or constituency), or 'n' (a name).
10
 *
11
 * First, we check to see if a person_id's been submitted.
12
 * If so, we display that person.
13
 *
14
 * Else, we check to see if a member_id's been submitted.
15
 * If so, we display that person.
16
 *
17
 * Otherwise, we then check to see if a postcode's been submitted.
18
 * If it's valid we put it in a cookie.
19
 *
20
 * If no postcode, we check to see if a constituency's been submitted.
21
 *
22
 * If neither has been submitted, we see if either the user is logged in
23
 * and has a postcode set or the user has a cookied postcode from a previous
24
 * search.
25
 *
26
 * If we have a valid constituency after all this, we display its MP.
27
 *
28
 * Either way, we print the forms.
29
 */
30
31
// Disable the old PAGE class.
32
33
use MySociety\TheyWorkForYou\PolicyDistributionCollection;
34
use MySociety\TheyWorkForYou\PolicyComparisonPeriod;
35
36
$new_style_template = true;
37
38
// Include all the things this page needs.
39
include_once '../../includes/easyparliament/init.php';
40
include_once INCLUDESPATH . 'easyparliament/member.php';
41
include_once INCLUDESPATH . '../../commonlib/phplib/random.php';
42
include_once INCLUDESPATH . '../../commonlib/phplib/auth.php';
43
include_once '../api/api_getGeometry.php';
44
include_once '../api/api_getConstituencies.php';
45
46
// Ensure that page type is set
47
$allowed_page_types = ['divisions', 'votes', 'policy_set_svg', 'policy_set_png', 'recent', 'register', 'election_register', 'member_interests'];
48
49
if (get_http_var('pagetype')) {
50
    $pagetype = get_http_var('pagetype');
51
} else {
52
    $pagetype = 'profile';
53
}
54
if (!in_array($pagetype, $allowed_page_types)) {
55
    $pagetype = 'profile';
56
}
57
if ($pagetype == 'profile') {
58
    $pagetype = '';
59
}
60
61
// list of years for which we have WTT response stats in
62
// reverse chronological order. Add new years here as we
63
// get them.
64
// NB: also need to update ./mpinfoin.pl to import the stats
65
$wtt_stats_years = [2015, 2014, 2013, 2008, 2007, 2006, 2005];
66
67
// Set the PID, name and constituency.
68
$pid = get_http_var('pid') != '' ? get_http_var('pid') : get_http_var('p');
69
$name = strtolower(str_replace('_', ' ', get_http_var('n')));
70
$constituency = strtolower(str_replace('_', ' ', get_http_var('c')));
71
72
// Fix for names with non-ASCII characters
73
if ($name == 'sion simon') {
74
    $name = 'si\xf4n simon';
75
}
76
if ($name == 'sian james') {
77
    $name = 'si\xe2n james';
78
}
79
if ($name == 'lembit opik') {
80
    $name = 'lembit \xf6pik';
81
}
82
if ($name == 'bairbre de brun') {
83
    $name = 'bairbre de br\xfan';
84
}
85
if ($name == 'daithi mckay') {
86
    $name = 'daith\xed mckay';
87
}
88
if ($name == 'caral ni chuilin') {
89
    $name = 'car\xe1l n\xed chuil\xedn';
90
}
91
if ($name == 'caledon du pre') {
92
    $name = 'caledon du pr\xe9';
93
}
94
if ($name == 'sean etchingham') {
95
    $name = 'se\xe1n etchingham';
96
}
97
if ($name == 'john tinne') {
98
    $name = 'john tinn\xe9';
99
}
100
if ($name == 'renee short') {
101
    $name = 'ren\xe9e short';
102
}
103
104
// Fix for common misspellings, name changes etc
105
$name_fix = [
106
    'a j beith' => 'alan beith',
107
    'micky brady' => 'mickey brady',
108
    'daniel rogerson' => 'dan rogerson',
109
    'andrew slaughter' => 'andy slaughter',
110
    'robert wilson' => ['rob wilson', 'reading east'],
111
    'james mcgovern' => 'jim mcgovern',
112
    'patrick mcfadden' => 'pat mcfadden',
113
    'chris leslie' => 'christopher leslie',
114
    'joseph meale' => 'alan meale',
115
    'james sheridan' => 'jim sheridan',
116
    'chinyelu onwurah' => 'chi onwurah',
117
    'steve rotherham' => 'steve rotheram',
118
    'michael weatherley' => 'mike weatherley',
119
    'louise bagshawe' => 'louise mensch',
120
    'andrew sawford' => 'andy sawford',
121
];
122
123
if (array_key_exists($name, $name_fix)) {
124
    if (is_array($name_fix[$name])) {
125
        if ($constituency == $name_fix[$name][1]) {
126
            $name = $name_fix[$name][0];
127
        }
128
    } else {
129
        $name = $name_fix[$name];
130
    }
131
}
132
133
// Fixes for Ynys Mon, and a Unicode URL
134
if ($constituency == 'ynys mon') {
135
    $constituency = "ynys m\xf4n";
136
}
137
if (preg_match("#^ynys m\xc3\xb4n#i", $constituency)) {
138
    $constituency = "ynys m\xf4n";
139
}
140
141
// If this is a request for recent appearances, redirect to search results
142
if (get_http_var('recent')) {
143
    if ($THEUSER->postcode_is_set() && !$pid) {
144
        $MEMBER = new MySociety\TheyWorkForYou\Member(['postcode' => $THEUSER->postcode(), 'house' => HOUSE_TYPE_COMMONS]);
145
        if ($MEMBER->person_id()) {
146
            $pid = $MEMBER->person_id();
147
        }
148
    }
149
    if ($pid) {
150
        $URL = new \MySociety\TheyWorkForYou\Url('search');
151
        $URL->insert(['pid' => $pid, 'pop' => 1]);
152
        header('Location: ' . $URL->generate('none'));
153
        exit;
154
    }
155
}
156
157
/////////////////////////////////////////////////////////
158
// DETERMINE TYPE OF REPRESENTITIVE
159
160
switch (get_http_var('representative_type')) {
161
    case 'peer':
162
        $this_page = 'peer';
163
        break;
164
    case 'royal':
165
        $this_page = 'royal';
166
        break;
167
    case 'mla':
168
        $this_page = 'mla';
169
        break;
170
    case 'msp':
171
        $this_page = 'msp';
172
        break;
173
    case 'ms':
174
        $this_page = 'ms';
175
        break;
176
    case 'london-assembly-member':
177
        $this_page = 'london-assembly-member';
178
        break;
179
    default:
180
        $this_page = 'mp';
181
        break;
182
}
183
184
try {
185
    if (is_numeric($pid)) {
186
        $MEMBER = get_person_by_id($pid);
187
    } elseif (is_numeric(get_http_var('m'))) {
188
        get_person_by_member_id(get_http_var('m'));
189
    } elseif (get_http_var('pc')) {
190
        get_person_by_postcode(get_http_var('pc'));
191
    } elseif ($name) {
192
        $MEMBER = get_person_by_name($name, $constituency);
193
    } elseif ($constituency) {
194
        get_mp_by_constituency($constituency);
195
    } elseif (($this_page == 'msp' || $this_page == 'mla' || $this_page == 'ms') && $THEUSER->postcode_is_set()) {
196
        get_regional_by_user_postcode($THEUSER->postcode(), $this_page);
197
        exit;
198
    } elseif ($THEUSER->postcode_is_set()) {
199
        get_mp_by_user_postcode($THEUSER->postcode());
200
    } else {
201
        twfy_debug('MP', "We don't have any way of telling what MP to display");
202
        throw new MySociety\TheyWorkForYou\MemberException(gettext('Sorry, but we can’t tell which representative to display.'));
203
    }
204
    if (!isset($MEMBER) || !$MEMBER->valid) {
205
        throw new MySociety\TheyWorkForYou\MemberException(gettext('You haven’t provided a way of identifying which representative you want'));
206
    }
207
} catch (MySociety\TheyWorkForYou\MemberMultipleException $e) {
208
    person_list_page($e->ids);
209
    exit;
210
} catch (MySociety\TheyWorkForYou\MemberException $e) {
211
    person_error_page($e->getMessage());
212
    exit;
213
}
214
215
# We have successfully looked up one person to show now.
216
217
if (!DEVSITE) {
218
    header('Cache-Control: max-age=900');
219
}
220
221
twfy_debug_timestamp("before load_extra_info");
222
$MEMBER->load_extra_info(true);
223
twfy_debug_timestamp("after load_extra_info");
224
225
// Basic name, title and description
226
$member_name = ucfirst($MEMBER->full_name());
227
$title = $member_name;
228
$desc = "Read $member_name's contributions to Parliament, including speeches and questions";
229
230
// Enhance description if this is a current member
231
if ($MEMBER->current_member_anywhere()) {
232
    $desc .= ', investigate their voting record, and get email alerts on their activity';
233
}
234
235
// Enhance title if this is a member of the Commons
236
if ($MEMBER->house(HOUSE_TYPE_COMMONS)) {
237
    if (!$MEMBER->current_member(1)) {
238
        $title .= ', former';
239
    }
240
    $title .= ' MP';
241
    if ($MEMBER->constituency()) {
242
        $title .= ', ' . $MEMBER->constituency();
243
    }
244
}
245
246
// Enhance title if this is a member of NIA
247
if ($MEMBER->house(HOUSE_TYPE_NI)) {
248
    if ($MEMBER->house(HOUSE_TYPE_COMMONS) || $MEMBER->house(HOUSE_TYPE_LORDS)) {
249
        $desc = str_replace('Parliament', 'Parliament and the Northern Ireland Assembly', $desc);
250
    } else {
251
        $desc = str_replace('Parliament', 'the Northern Ireland Assembly', $desc);
252
    }
253
    if (!$MEMBER->current_member(HOUSE_TYPE_NI)) {
254
        $title .= ', former';
255
    }
256
    $title .= ' MLA';
257
    if ($MEMBER->constituency()) {
258
        $title .= ', ' . $MEMBER->constituency();
259
    }
260
}
261
262
// Enhance title if this is a member of Scottish Parliament
263
if ($MEMBER->house(HOUSE_TYPE_SCOTLAND)) {
264
    if ($MEMBER->house(HOUSE_TYPE_COMMONS) || $MEMBER->house(HOUSE_TYPE_LORDS)) {
265
        $desc = str_replace('Parliament', 'the UK and Scottish Parliaments', $desc);
266
    } else {
267
        $desc = str_replace('Parliament', 'the Scottish Parliament', $desc);
268
    }
269
    $desc = str_replace(', and get email alerts on their activity', '', $desc);
270
    if (!$MEMBER->current_member(HOUSE_TYPE_SCOTLAND)) {
271
        $title .= ', former';
272
    }
273
    $title .= ' MSP, ' . $MEMBER->constituency();
274
}
275
276
// Enhance title if this is a member of Welsh Parliament
277
if ($MEMBER->house(HOUSE_TYPE_WALES)) {
278
    if ($MEMBER->house(HOUSE_TYPE_COMMONS) || $MEMBER->house(HOUSE_TYPE_LORDS)) {
279
        $desc = str_replace('Parliament', 'the UK and Welsh Parliaments', $desc);
280
    } else {
281
        $desc = str_replace('Parliament', 'the Senedd', $desc);
282
    }
283
    $desc = str_replace(', and get email alerts on their activity', '', $desc);
284
    if (!$MEMBER->current_member(HOUSE_TYPE_WALES)) {
285
        $title .= ', former';
286
    }
287
    $title .= ' MS, ' . $MEMBER->constituency();
288
}
289
290
$known_for = '';
291
$current_offices_ignoring_committees = $MEMBER->offices('current', true);
292
if (count($current_offices_ignoring_committees) > 0) {
293
    $known_for = $current_offices_ignoring_committees[0];
294
}
295
296
// Finally, if this is a Votes page, replace the page description with
297
// something more descriptive of the actual data on the page.
298
if ($pagetype == 'votes') {
299
    $title = "Voting record - " . $title;
300
    $desc = 'See how ' . $member_name . ' voted on topics like Employment, Social Issues, Foreign Policy, and more.';
301
}
302
303
// Set page metadata
304
$DATA->set_page_metadata($this_page, 'title', $title);
305
$DATA->set_page_metadata($this_page, 'meta_description', $desc);
306
307
// Build the RSS link and add it to page data.
308
$feedurl = $DATA->page_metadata('mp_rss', 'url') . $MEMBER->person_id() . '.rdf';
309
if (file_exists(BASEDIR . '/' . $feedurl)) {
310
    $DATA->set_page_metadata($this_page, 'rss', $feedurl);
311
}
312
313
// Prepare data for the template
314
$data["pagetype"] = $pagetype;
315
$data['full_name'] = $MEMBER->full_name();
316
$data['person_id'] = $MEMBER->person_id();
317
$data['member_id'] = $MEMBER->member_id();
318
319
$data['known_for'] = $known_for;
320
$data['latest_membership'] = $MEMBER->getMostRecentMembership();
321
322
$data['constituency'] = $MEMBER->constituency();
323
$data['party'] = $MEMBER->party_text();
324
$data['current_party_comparison'] = $MEMBER->currentPartyComparison();
325
$data['current_member_anywhere'] = $MEMBER->current_member_anywhere();
326
$data['current_member'] = $MEMBER->current_member();
327
$data['the_users_mp'] = $MEMBER->the_users_mp();
328
$data['user_postcode'] = $THEUSER->postcode;
329
$data['houses'] = $MEMBER->houses();
330
$data['member_url'] = $MEMBER->url();
331
$data['abs_member_url'] = $MEMBER->url(true);
332
// If there's photo attribution information, copy it into data
333
foreach (['photo_attribution_text', 'photo_attribution_link'] as $key) {
334
    if (isset($MEMBER->extra_info[$key])) {
335
        $data[$key] = $MEMBER->extra_info[$key];
336
    }
337
}
338
$data['profile_message'] = $MEMBER->extra_info['profile_message'] ?? '';
339
$data['image'] = $MEMBER->image();
340
$data['member_summary'] = person_summary_description($MEMBER);
341
$data['enter_leave'] = $MEMBER->getEnterLeaveStrings();
342
$data['entry_date'] = $MEMBER->getEntryDate(HOUSE_TYPE_COMMONS);
343
$data['leave_date'] = $MEMBER->getLeftDate(HOUSE_TYPE_COMMONS);
344
$data['is_new_mp'] = $MEMBER->isNew();
345
$data['other_parties'] = $MEMBER->getOtherPartiesString();
346
$data['other_constituencies'] = $MEMBER->getOtherConstituenciesString();
347
$data['rebellion_rate'] = person_rebellion_rate($MEMBER);
348
$data['recent_appearances'] = person_recent_appearances($MEMBER);
349
$data['useful_links'] = person_useful_links($MEMBER);
350
$data['social_links'] = person_social_links($MEMBER);
351
$data['topics_of_interest'] = person_topics($MEMBER);
352
$data['current_offices'] = $MEMBER->offices('current', true);
353
$data['previous_offices'] = $MEMBER->offices('previous', true);
354
$data['register_interests'] = person_register_interests($MEMBER, $MEMBER->extra_info);
355
$data['register_2024_enriched'] = person_register_interests_from_key('person_regmem_enriched2024_en', $MEMBER->extra_info);
356
$data['eu_stance'] = $MEMBER->getEUStance();
357
$data['standing_down_2024'] = $MEMBER->extra_info['standing_down_2024'] ?? '';
358
$data['member_interests'] = member_interests($MEMBER);
359
360
# People who are or were MPs and Lords potentially have voting records, except Sinn Fein MPs
361
$data['has_voting_record'] = (($MEMBER->house(HOUSE_TYPE_COMMONS) && $MEMBER->party() != 'Sinn Féin') || $MEMBER->house(HOUSE_TYPE_LORDS));
362
# Everyone who is currently somewhere has email alert signup, apart from current Sinn Fein MPs who are not MLAs
363
$data['has_email_alerts'] = ($MEMBER->current_member_anywhere() && !($MEMBER->current_member(HOUSE_TYPE_COMMONS) && $MEMBER->party() == 'Sinn Féin' && !$MEMBER->current_member(HOUSE_TYPE_NI)));
364
$data['has_expenses'] = $data['leave_date'] > '2004-01-01';
365
366
$data['pre_2010_expenses'] = false;
367
$data['post_2010_expenses'] = $data['leave_date'] > '2010-05-05' ? ($MEMBER->extra_info['datadotparl_id'] ?? '') : '';
368
369
if ($data['entry_date'] < '2010-05-05') {
370
    $data['pre_2010_expenses'] = true;
371
    // Set the expenses URL if we know it
372
    $data['expenses_url_2004'] = $MEMBER->extra_info['expenses_url'] ?? 'https://mpsallowances.parliament.uk/mpslordsandoffices/hocallowances/allowances%2Dby%2Dmp/';
373
}
374
375
$data['constituency_previous_mps'] = constituency_previous_mps($MEMBER);
376
$data['constituency_future_mps'] = constituency_future_mps($MEMBER);
377
$data['public_bill_committees'] = person_pbc_membership($MEMBER);
378
379
$data['this_page'] = $this_page;
380
$country = MySociety\TheyWorkForYou\Utility\House::getCountryDetails($data['latest_membership']['house']);
381
$data['current_assembly'] = $country[2];
382
383
$data['policy_last_update'] = MySociety\TheyWorkForYou\Divisions::getMostRecentDivisionDate();
384
385
$data['comparison_party'] = $MEMBER->cohortParty();
386
$data['unslugified_comparison_party'] = ucwords(str_replace('-', ' ', $data['comparison_party']));
387
388
// is the party we're comparing this MP to different from the party they're currently in?
389
$data['party_switcher'] = (slugify($data['current_party_comparison']) != slugify($data["comparison_party"]));
390
391
// Update the social image URL generation logic
392
switch ($pagetype) {
393
    case 'votes':
394
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Voting Summaries', $data['current_assembly']);
395
        $policy_set = get_http_var('policy');
396
397
        $policiesList = new MySociety\TheyWorkForYou\Policies();
398
        $divisions = new MySociety\TheyWorkForYou\Divisions($MEMBER);
399
        // Generate voting segments
400
        $set_descriptions = $policiesList->getSetDescriptions();
401
        if ($policy_set && array_key_exists($policy_set, $set_descriptions)) {
402
            $sets = [$policy_set];
403
            $data['page_title'] = $set_descriptions[$policy_set] . ' ' . $title . ' - TheyWorkForYou';
404
            $data['meta_description'] = 'See how ' . $data['full_name'] . ' voted on ' . $set_descriptions[$policy_set];
405
            $data['single_policy_page'] = true;
406
        } else {
407
            $data['single_policy_page'] = false;
408
            $sets = [
409
                'social', 'foreignpolicy', 'welfare', 'taxation', 'business',
410
                'health', 'education', 'reform', 'home', 'environment',
411
                'transport', 'housing', 'misc',
412
            ];
413
            $sets = array_filter($sets, function ($v) use ($set_descriptions) {
414
                return array_key_exists($v, $set_descriptions);
415
            });
416
            shuffle($sets);
417
        }
418
        $house = HOUSE_TYPE_COMMONS;
419
        $party = new MySociety\TheyWorkForYou\Party($MEMBER->party());
420
        $voting_comparison_period_slug = get_http_var('comparison_period') ?: 'all_time';
421
        $voting_comparison_period = new PolicyComparisonPeriod($voting_comparison_period_slug, $house);
422
        $cohort_party = $MEMBER->cohortParty();
423
424
        // this comes up if the votes page is being accessed for an old MP/Lord without party information.
425
        // by definition, not covered by our voting comparisons so just return an empty array.
426
        if ($cohort_party == null) {
427
            $data['key_votes_segments'] = [];
428
        } else {
429
            $data['key_votes_segments'] = PolicyDistributionCollection::getPersonDistributions($sets, $MEMBER->person_id(), $cohort_party, $voting_comparison_period->slug, $house);
430
        }
431
432
        $data["comparison_period"] = $voting_comparison_period;
433
        $data['available_periods'] = PolicyComparisonPeriod::getComparisonPeriodsForPerson($MEMBER->person_id(), $house);
434
        // shuffle the key_votes_segments for a random order
435
        shuffle($data['key_votes_segments']);
436
        $data["sig_diff_policy"] = PolicyDistributionCollection::getSignificantDistributions($data['key_votes_segments']);
437
        $data['party_member_count'] = $party->getCurrentMemberCount($house);
438
439
        // Send the output for rendering
440
        MySociety\TheyWorkForYou\Renderer::output('mp/votes', $data);
441
442
        break;
443
444
    case 'recent':
445
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Recent Votes', $data['current_assembly']);
446
        $divisions = new MySociety\TheyWorkForYou\Divisions($MEMBER);
447
        $data['divisions'] = $divisions->getRecentMemberDivisions();
448
        MySociety\TheyWorkForYou\Renderer::output('mp/recent', $data);
449
        break;
450
451
    case 'member_interests':
452
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Profile', $data['current_assembly']);
453
        MySociety\TheyWorkForYou\Renderer::output('mp/interests', $data);
454
        break;
455
456
    case 'election_register':
457
        // Send the output for rendering
458
459
        $memcache = new \MySociety\TheyWorkForYou\Memcache();
460
        $mem_key = "highlighted_interests" . $MEMBER->person_id();
461
462
        $highlighted_for_this_mp = $memcache->get($mem_key);
463
464
        if (!$highlighted_for_this_mp) {
0 ignored issues
show
introduced by
The condition $highlighted_for_this_mp is always false.
Loading history...
465
            $highlighted_register = MySociety\TheyWorkForYou\DataClass\Regmem\Register::getMisc("highlighted_interests.json");
466
            $str_id = "uk.org.publicwhip/person/" . $MEMBER->person_id();
467
            $highlighted_for_this_mp = $highlighted_register->getPersonFromId($str_id);
0 ignored issues
show
Bug introduced by
Are you sure the assignment to $highlighted_for_this_mp is correct as $highlighted_register->getPersonFromId($str_id) targeting MySociety\TheyWorkForYou...ster::getPersonFromId() seems to always return null.

This check looks for function or method calls that always return null and whose return value is assigned to a variable.

class A
{
    function getObject()
    {
        return null;
    }

}

$a = new A();
$object = $a->getObject();

The method getObject() can return nothing but null, so it makes no sense to assign that value to a variable.

The reason is most likely that a function or method is imcomplete or has been reduced for debug purposes.

Loading history...
468
            $memcache->set($mem_key, $highlighted_for_this_mp, 60 * 60 * 24);
469
        }
470
471
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Election Register', $data['current_assembly']);
472
        $data['mp_has_highlighted_interests'] = (bool) $highlighted_for_this_mp;
473
        $overlapping_interests = [];
474
475
        MySociety\TheyWorkForYou\Renderer::output('mp/election_register', $data);
476
477
        // no break
478
    case 'register':
479
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Register of Interests', $data['current_assembly']);
480
        // Send the output for rendering
481
        MySociety\TheyWorkForYou\Renderer::output('mp/register', $data);
482
483
        // no break
484
    case '':
485
    default:
486
        $data['og_image'] = \MySociety\TheyWorkForYou\Url::generateSocialImageUrl($member_name, 'Profile', $data['current_assembly']);
487
        // if extra detail needed for overview page in future
488
489
        // Send the output for rendering
490
        MySociety\TheyWorkForYou\Renderer::output('mp/profile', $data);
491
492
        break;
493
494
}
495
496
497
/////////////////////////////////////////////////////////
498
// SUPPORTING FUNCTIONS
499
500
/* Person lookup functions */
501
502
function get_person_by_id($pid) {
503
    global $pagetype, $this_page;
504
    $MEMBER = new MySociety\TheyWorkForYou\Member(['person_id' => $pid]);
505
    if (!$MEMBER->valid) {
506
        throw new MySociety\TheyWorkForYou\MemberException('Sorry, that ID number wasn&rsquo;t recognised.');
507
    }
508
    // Ensure that we're actually at the current, correct and canonical URL for the person. If not, redirect.
509
    // No need to worry about other URL syntax forms for vote pages, they shouldn't happen.
510
    $at = str_replace('/mp/', "/$this_page/", get_http_var('url'));
511
    $shouldbe = urldecode($MEMBER->url());
512
    if ($pagetype) {
513
        $shouldbe .= "/$pagetype";
514
    }
515
    if ($at !== $shouldbe) {
516
        member_redirect($MEMBER, 301, $pagetype);
517
    }
518
    return $MEMBER;
519
}
520
521
function get_person_by_member_id($member_id) {
522
    // Got a member id, redirect to the canonical MP page, with a person id.
523
    $MEMBER = new MySociety\TheyWorkForYou\Member(['member_id' => $member_id]);
524
    member_redirect($MEMBER);
525
}
526
527
function get_person_by_postcode($pc) {
528
    global $THEUSER;
529
    $pc = preg_replace('#[^a-z0-9]#i', '', $pc);
530
    if (!validate_postcode($pc)) {
531
        twfy_debug('MP', "Can't display an MP because the submitted postcode wasn't of a valid form.");
532
        throw new MySociety\TheyWorkForYou\MemberException(sprintf(gettext('Sorry, %s isn’t a valid postcode'), _htmlentities($pc)));
533
    }
534
    twfy_debug('MP', "MP lookup by postcode");
535
    $constituency = strtolower(MySociety\TheyWorkForYou\Utility\Postcode::postcodeToConstituency($pc));
536
    if ($constituency == "connection_timed_out") {
537
        throw new MySociety\TheyWorkForYou\MemberException(gettext('Sorry, we couldn’t check your postcode right now, as our postcode lookup server is under quite a lot of load.'));
538
    } elseif ($constituency == "") {
539
        twfy_debug('MP', "Can't display an MP, as submitted postcode didn't match a constituency");
540
        throw new MySociety\TheyWorkForYou\MemberException(sprintf(gettext('Sorry, %s isn’t a known postcode'), _htmlentities($pc)));
541
    } else {
542
        // Redirect to the canonical MP page, with a person id.
543
        $MEMBER = new MySociety\TheyWorkForYou\Member(['constituency' => $constituency, 'house' => HOUSE_TYPE_COMMONS]);
544
        if ($MEMBER->person_id()) {
545
            // This will cookie the postcode.
546
            $THEUSER->set_postcode_cookie($pc);
547
        }
548
        member_redirect($MEMBER, 302);
549
    }
550
}
551
552
function get_person_by_name($name, $const = '') {
553
    $MEMBER = new MySociety\TheyWorkForYou\Member(['name' => $name, 'constituency' => $const]);
554
    // Edge case, only attempt further detection if this isn't the Queen.
555
    if (($name !== 'elizabeth the second' && $name !== 'prince charles') || $const) {
556
        twfy_debug('MP', 'Redirecting for MP found by name/constituency');
557
        member_redirect($MEMBER);
558
    }
559
    return $MEMBER;
560
}
561
562
function get_mp_by_constituency($constituency) {
563
    $MEMBER = new MySociety\TheyWorkForYou\Member(['constituency' => $constituency, 'house' => HOUSE_TYPE_COMMONS]);
564
    member_redirect($MEMBER);
565
}
566
567
function get_regional_by_user_postcode($pc, $page) {
568
    global $this_page;
569
    $this_page = "your$page";
570
    $areas = \MySociety\TheyWorkForYou\Utility\Postcode::postcodeToConstituencies($pc);
571
    if ($page == 'msp' && isset($areas['SPC'])) {
572
        regional_list($pc, 'SPC', $page);
573
    } elseif ($page == 'ms' && isset($areas['WAC'])) {
574
        regional_list($pc, 'WAC', $page);
575
    } elseif ($page == 'mla' && isset($areas['NIE'])) {
576
        regional_list($pc, 'NIE', $page);
577
    } else {
578
        throw new MySociety\TheyWorkForYou\MemberException('Your set postcode is not in the right region.');
579
    }
580
}
581
582
function get_mp_by_user_postcode($pc) {
583
    $MEMBER = new MySociety\TheyWorkForYou\Member(['postcode' => $pc, 'house' => HOUSE_TYPE_COMMONS]);
584
    member_redirect($MEMBER, 302);
585
}
586
587
/**
588
 * Member Redirect
589
 *
590
 * Redirect to the canonical page for a member.
591
 */
592
593
function member_redirect(&$MEMBER, $code = 301, $pagetype = null) {
594
    // We come here after creating a MEMBER object by various methods.
595
    // Now we redirect to the canonical MP page, with a person_id.
596
    if ($MEMBER->person_id()) {
597
        $url = $MEMBER->url();
598
        $params = [];
599
        foreach ($_GET as $key => $value) {
600
            if (substr($key, 0, 4) == 'utm_' || $key == 'gclid') {
601
                $params[] = urlencode($key) . "=" . urlencode($value);
602
            }
603
        }
604
        if ($pagetype) {
605
            $url .= '/' . $pagetype;
606
        }
607
        if (count($params)) {
608
            $url .= '?' . join('&', $params);
609
        }
610
        header('Location: ' . $url, true, $code);
611
        exit;
612
    }
613
}
614
615
/* Error list page */
616
617
function person_list_page($ids) {
618
    global $name;
619
    if (!DEVSITE) {
620
        header('Cache-Control: max-age=900');
621
    }
622
    $data = ['mps' => []];
623
    foreach ($ids as $id => $constituency) {
624
        $data['mps'][] = [
625
            'url'  => WEBPATH . 'mp/?pid=' . $id,
626
            'name' => ucwords(strtolower($name)) . ', ' . $constituency,
627
        ];
628
    }
629
    $MPSURL = new \MySociety\TheyWorkForYou\Url('mps');
630
    $data['all_mps_url'] = $MPSURL->generate();
631
    MySociety\TheyWorkForYou\Renderer::output('mp/list', $data);
632
}
633
634
/* Error page */
635
636
function person_error_page($message) {
637
    global $this_page;
638
    $SEARCHURL = '';
639
    switch($this_page) {
640
        case 'peer':
641
            $people = new MySociety\TheyWorkForYou\People\Peers();
642
            $MPSURL = new \MySociety\TheyWorkForYou\Url('peers');
643
            break;
644
        case 'mla':
645
            $people = new MySociety\TheyWorkForYou\People\MLAs();
646
            $SEARCHURL = '/postcode/';
647
            $MPSURL = new \MySociety\TheyWorkForYou\Url('mlas');
648
            break;
649
        case 'msp':
650
            $people = new MySociety\TheyWorkForYou\People\MSPs();
651
            $SEARCHURL = '/postcode/';
652
            $MPSURL = new \MySociety\TheyWorkForYou\Url('msps');
653
            break;
654
        case 'ms':
655
            $people = new MySociety\TheyWorkForYou\People\MSs();
656
            $SEARCHURL = '/postcode/';
657
            $MPSURL = new \MySociety\TheyWorkForYou\Url('mss');
658
            break;
659
        case 'london-assembly-member':
660
            $people = new MySociety\TheyWorkForYou\People\LondonAssemblyMembers();
661
            $MPSURL = new \MySociety\TheyWorkForYou\Url('london-assembly-members');
662
            break;
663
        default:
664
            $people = new MySociety\TheyWorkForYou\People\MPs();
665
            $SEARCHURL = new \MySociety\TheyWorkForYou\Url('mp');
666
            $SEARCHURL = $SEARCHURL->generate();
667
            $MPSURL = new \MySociety\TheyWorkForYou\Url('mps');
668
    }
669
670
    $data = [
671
        'error' => $message,
672
        'rep_name' => $people->rep_name,
673
        'rep_name_plural' => $people->rep_plural,
674
        'all_mps_url' => $MPSURL->generate(),
675
        'rep_search_url' => $SEARCHURL,
676
    ];
677
    MySociety\TheyWorkForYou\Renderer::output('mp/error', $data);
678
}
679
680
/**
681
 * Person Positions Summary
682
 *
683
 * Generate the summary of this person's held positions.
684
 */
685
686
function person_summary_description($MEMBER) {
687
    $entered_house = $MEMBER->entered_house();
688
    $current_member = $MEMBER->current_member();
689
    $left_house = $MEMBER->left_house();
690
691
    if (in_array(HOUSE_TYPE_ROYAL, $MEMBER->houses())) {
692
        # Royal short-circuit
693
        if (substr($entered_house[HOUSE_TYPE_ROYAL]['date'], 0, 4) == 1952) {
694
            return '<strong>Acceded on ' . $entered_house[HOUSE_TYPE_ROYAL]['date_pretty']
695
                . '<br>Coronated on 2 June 1953</strong></li>';
696
        } else {
697
            return '';
698
        }
699
    }
700
    $desc = '';
701
    foreach ($MEMBER->houses() as $house) {
702
        if ($house == HOUSE_TYPE_COMMONS && isset($entered_house[HOUSE_TYPE_LORDS])) {
703
            # Same info is printed further down
704
            continue;
705
        }
706
707
        $party = $left_house[$house]['party'];
708
        $party_br = '';
709
        if (preg_match('#^(.*?)\s*\((.*?)\)$#', $party, $m)) {
710
            $party_br = " ($m[2])";
711
            $party = $m[1];
712
        }
713
        $pparty = $party != 'unknown' ? _htmlentities($party) : '';
714
715
        if ($house != HOUSE_TYPE_LORDS) {
716
            if ($house == HOUSE_TYPE_COMMONS) {
717
                $type = gettext('<abbr title="Member of Parliament">MP</abbr>');
718
            } elseif ($house == HOUSE_TYPE_NI) {
719
                $type = gettext('<abbr title="Member of the Legislative Assembly">MLA</abbr>');
720
            } elseif ($house == HOUSE_TYPE_SCOTLAND) {
721
                $type = gettext('<abbr title="Member of the Scottish Parliament">MSP</abbr>');
722
            } elseif ($house == HOUSE_TYPE_WALES) {
723
                $type = gettext('<abbr title="Member of the Senedd">MS</abbr>');
724
            } elseif ($house == HOUSE_TYPE_LONDON_ASSEMBLY) {
725
                $type = gettext('Member of the London Assembly');
726
            }
727
728
            if ($party == 'Speaker' || $party == 'Deputy Speaker') {
729
                # XXX: Might go horribly wrong if something odd happens
730
                if ($party == 'Deputy Speaker') {
731
                    $last = end($MEMBER->other_parties);
732
                    $oparty = $last['from'];
733
                } else {
734
                    $oparty = '';
735
                }
736
                if ($current_member[$house]) {
737
                    $line = sprintf(gettext('%s, and %s %s for %s'), $pparty, $oparty, $type, $left_house[$house]['constituency']);
0 ignored issues
show
Comprehensibility Best Practice introduced by
The variable $type does not seem to be defined for all execution paths leading up to this point.
Loading history...
738
                } else {
739
                    $line = sprintf(gettext('Former %s, and %s %s for %s'), $pparty, $oparty, $type, $left_house[$house]['constituency']);
740
                }
741
            } elseif ($current_member[$house]) {
742
                $line = sprintf(gettext('%s %s %s for %s'), $pparty, $type, $party_br, $left_house[$house]['constituency']);
743
            } else {
744
                $line = sprintf(gettext('Former %s %s %s for %s'), $pparty, $type, $party_br, $left_house[$house]['constituency']);
745
            }
746
        } elseif ($house == HOUSE_TYPE_LORDS && $party != 'Bishop') {
747
            if ($current_member[$house]) {
748
                $line = sprintf(gettext('%s Peer'), $pparty);
749
            } else {
750
                $line = sprintf(gettext('Former %s Peer'), $pparty);
751
            }
752
        } else {
753
            if ($current_member[$house]) {
754
                $line = $pparty;
755
            } else {
756
                $line = sprintf(gettext('Former %s'), $pparty);
757
            }
758
        }
759
        $desc .= $line . ', ';
760
    }
761
    $desc = preg_replace('#, $#', '', $desc);
762
    return $desc;
763
}
764
765
/**
766
 * Person Rebellion Rate
767
 *
768
 * How often has this person rebelled against their party?
769
 *
770
 * @param MEMBER $member The member to calculate rebellion rate for.
771
 *
772
 * @return string A HTML summary of this person's rebellion rate.
773
 */
774
775
function person_rebellion_rate($member) {
776
777
    // Rebellion string may be empty.
778
    $rebellion_string = '';
779
780
    if (isset($member->extra_info['party_vote_alignment_last_year'])) {
781
782
        // unserialise the data from json
783
        $data = json_decode($member->extra_info['party_vote_alignment_last_year'], true);
784
        $total_votes = $data['total_votes'];
785
        $avg_diff_from_party = $data['avg_diff_from_party'];
786
787
        // as int %
788
        $avg_diff_str = number_format((1 - $avg_diff_from_party) * 100, 0) . '%';
789
790
        if ($total_votes == 0) {
791
            return '';
792
        }
793
        $votes_help_url = TWFY_VOTES_URL . "/help/about#voting-breakdowns-and-party-alignment";
0 ignored issues
show
Bug introduced by
The constant TWFY_VOTES_URL was not found. Maybe you did not declare it correctly or list all dependencies?
Loading history...
794
795
        $rebellion_string .= 'In the last year, ' . $member->full_name() . ' has an alignment score of ' . $avg_diff_str . ' with other MPs of their party (over ' . $total_votes . ' votes).';
796
        $rebellion_string .= ' <small><a title="More about party alignment" href="' . $votes_help_url . '">Find out more</a>.</small>';
797
    }
798
    return $rebellion_string;
799
}
800
801
function person_recent_appearances($member) {
802
    global $DATA, $SEARCHENGINE, $this_page;
803
804
    $out = [];
805
    $out['appearances'] = [];
806
807
    //$this->block_start(array('id'=>'hansard', 'title'=>$title));
808
    // This is really far from ideal - I don't really want $PAGE to know
809
    // anything about HANSARDLIST / DEBATELIST / WRANSLIST.
810
    // But doing this any other way is going to be a lot more work for little
811
    // benefit unfortunately.
812
    twfy_debug_timestamp();
813
814
    $person_id = $member->person_id();
815
816
    $memcache = new MySociety\TheyWorkForYou\Memcache();
817
    $recent = $memcache->get("recent_appear:$person_id:" . LANGUAGE);
818
819
    if (!$recent) {
0 ignored issues
show
introduced by
The condition $recent is always false.
Loading history...
820
        // Initialise the search engine
821
        $searchstring = "speaker:$person_id";
822
        $SEARCHENGINE = new \SEARCHENGINE($searchstring);
823
824
        $hansard = new MySociety\TheyWorkForYou\Hansard();
825
        $args =  [
826
            's' => $searchstring,
827
            'p' => 1,
828
            'num' => 3,
829
            'pop' => 1,
830
            'o' => 'd',
831
        ];
832
        $results = $hansard->search($searchstring, $args);
833
        $recent = serialize($results['rows']);
834
        $memcache->set('recent_appear:' . $person_id, $recent);
835
    }
836
    $out['appearances'] = unserialize($recent);
837
    twfy_debug_timestamp();
838
839
    $MOREURL = new \MySociety\TheyWorkForYou\Url('search');
840
    $MOREURL->insert(['pid' => $person_id, 'pop' => 1]);
841
842
    $out['more_href'] = $MOREURL->generate() . '#n4';
843
    $out['more_text'] = sprintf(gettext('More of %s’s recent appearances'), ucfirst($member->full_name()));
844
845
    if ($rssurl = $DATA->page_metadata($this_page, 'rss')) {
846
        // If we set an RSS feed for this page.
847
        $HELPURL = new \MySociety\TheyWorkForYou\Url('help');
848
        $out['additional_links'] = '<a href="' . WEBPATH . $rssurl . '" title="XML version of this person&rsquo;s recent appearances">RSS feed</a> (<a href="' . $HELPURL->generate() . '#rss" title="An explanation of what RSS feeds are for">?</a>)';
849
    }
850
851
    return $out;
852
853
}
854
855
function person_useful_links($member) {
856
857
    $links = $member->extra_info();
858
859
    $out = [];
860
861
    if (isset($links['maiden_speech'])) {
862
        $maiden_speech = fix_gid_from_db($links['maiden_speech']);
863
        $out[] = [
864
            'href' => WEBPATH . 'debate/?id=' . $maiden_speech,
865
            'text' => 'Maiden speech',
866
        ];
867
    }
868
869
    // BIOGRAPHY.
870
    global $THEUSER;
871
    if (isset($links['mp_website'])) {
872
        $out[] = [
873
            'href' => $links['mp_website'],
874
            'text' => 'Personal website',
875
        ];
876
    }
877
878
    if (isset($links['sp_url'])) {
879
        $out[] = [
880
            'href' => $links['sp_url'],
881
            'text' => 'Page on the Scottish Parliament website',
882
        ];
883
    }
884
885
    if (isset($links['wikipedia_url'])) {
886
        $out[] = [
887
            'href' => $links['wikipedia_url'],
888
            'text' => 'Wikipedia page',
889
        ];
890
    }
891
892
    if (isset($links['bbc_profile_url'])) {
893
        $out[] = [
894
            'href' => $links['bbc_profile_url'],
895
            'text' => 'BBC News profile',
896
        ];
897
    }
898
899
    if (isset($links['diocese_url'])) {
900
        $out[] = [
901
            'href' => $links['diocese_url'],
902
            'text' => 'Diocese website',
903
        ];
904
    }
905
906
    return $out;
907
}
908
909
function person_social_links($member) {
910
911
    $links = $member->extra_info();
912
913
    $out = [];
914
915
916
    if (isset($links['bluesky_handle'])) {
917
        $out[] = [
918
            'href' => 'https://bsky.app/profile/' . _htmlentities($links['bluesky_handle']),
919
            'text' => '@' . _htmlentities($links['bluesky_handle']),
920
            'type' => 'bluesky',
921
        ];
922
    }
923
924
    if (isset($links['twitter_username'])) {
925
        $out[] = [
926
            'href' => 'https://twitter.com/' . _htmlentities($links['twitter_username']),
927
            'text' => '@' . _htmlentities($links['twitter_username']),
928
            'type' => 'twitter',
929
        ];
930
    }
931
932
    if (isset($links['facebook_page'])) {
933
        $out[] = [
934
            'href' => _htmlentities($links['facebook_page']),
935
            'text' => _htmlentities("Facebook"),
936
            'type' => 'facebook',
937
        ];
938
    }
939
940
    $official_keys = [
941
        'profile_url_uk_parl' => 'UK Parliament Profile',
942
        'profile_url_scot_parl' => 'Scottish Parliament Profile',
943
        'profile_url_ni_assembly' => 'Northern Ireland Assembly Profile',
944
    ];
945
946
    if (LANGUAGE == 'cy') {
0 ignored issues
show
introduced by
The condition LANGUAGE == 'cy' is always false.
Loading history...
947
        $official_keys['profile_url_senedd_cy'] = 'Proffil Senedd';
948
    } else {
949
        $official_keys['profile_url_senedd_en'] = 'Senedd Profile';
950
    }
951
952
    foreach ($official_keys as $key => $text) {
953
        if (isset($links[$key])) {
954
            $out[] = [
955
                'href' => $links[$key],
956
                'text' => $text,
957
                'type' => 'official',
958
            ];
959
        }
960
    }
961
962
    return $out;
963
}
964
965
function person_topics($member) {
966
    $out = [];
967
968
    $extra_info = $member->extra_info();
969
970
    if (isset($extra_info['wrans_departments'])) {
971
        $subjects = explode(',', $extra_info['wrans_departments']);
972
        $out = array_merge($out, $subjects);
973
    }
974
975
    if (isset($extra_info['wrans_subjects'])) {
976
        $subjects = explode(',', $extra_info['wrans_subjects']);
977
        $out = array_merge($out, $subjects);
978
    }
979
980
    return $out;
981
}
982
983
function member_interests($member) {
984
    $out = [];
985
986
    $topics = person_topics($member);
987
    if ($topics) {
988
        $out['topics'] = $topics;
989
    }
990
991
    $posts = $member->offices('current', false, true);
992
    if ($posts) {
993
        $out['posts'] = $posts;
994
    }
995
996
    $posts = $member->offices('previous', false, true);
997
    if ($posts) {
998
        $out['previous_posts'] = $posts;
999
    }
1000
1001
    return $out;
1002
}
1003
1004
function constituency_previous_mps($member) {
1005
    if ($member->house(HOUSE_TYPE_COMMONS)) {
1006
        return $member->previous_mps();
1007
    } else {
1008
        return [];
1009
    }
1010
}
1011
1012
function constituency_future_mps($member) {
1013
    if ($member->house(HOUSE_TYPE_COMMONS)) {
1014
        return $member->future_mps();
1015
    } else {
1016
        return [];
1017
    }
1018
}
1019
1020
function person_pbc_membership($member) {
1021
1022
    $extra_info = $member->extra_info();
1023
    $out = ['info' => '', 'data' => []];
1024
1025
    # Public Bill Committees
1026
    if (count($extra_info['pbc'])) {
1027
        if ($member->party() == 'Scottish National Party') {
1028
            $out['info'] = 'SNP MPs only attend sittings where the legislation pertains to Scotland.';
1029
        }
1030
        foreach ($extra_info['pbc'] as $bill_id => $arr) {
1031
            $text = '';
1032
            if ($arr['chairman']) {
1033
                $text .= 'Chairman, ';
1034
            }
1035
            $text .= $arr['title'] . ' Committee';
1036
            $out['data'][] = [
1037
                'href'      => '/pbc/' . $arr['session'] . '/' . urlencode($arr['title']),
1038
                'text'      => $text,
1039
                'attending' => $arr['attending'] . ' out of ' . $arr['outof'],
1040
            ];
1041
        }
1042
    }
1043
1044
    return $out;
1045
}
1046
1047
function person_register_interests_from_key($key, $extra_info): ?MySociety\TheyWorkForYou\DataClass\Regmem\Person {
1048
    $lang = LANGUAGE;
0 ignored issues
show
Unused Code introduced by
The assignment to $lang is dead and can be removed.
Loading history...
1049
    $reg = null;
1050
    if (isset($extra_info[$key])) {
1051
        $reg = MySociety\TheyWorkForYou\DataClass\Regmem\Person::fromJson($extra_info[$key]);
1052
    }
1053
    return $reg;
0 ignored issues
show
Bug Best Practice introduced by
The expression return $reg could return the type MySociety\TheyWorkForYou\DataClass\BaseInterface which includes types incompatible with the type-hinted return MySociety\TheyWorkForYou...lass\Regmem\Person|null. Consider adding an additional type-check to rule them out.
Loading history...
1054
}
1055
1056
function person_register_interests($member, $extra_info) {
0 ignored issues
show
Unused Code introduced by
The parameter $member is not used and could be removed. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-unused  annotation

1056
function person_register_interests(/** @scrutinizer ignore-unused */ $member, $extra_info) {

This check looks for parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
1057
1058
    $valid_chambers = ['house-of-commons', 'scottish-parliament', 'northern-ireland-assembly', 'senedd'];
1059
1060
    $lang = LANGUAGE;
1061
1062
    $reg = ['chamber_registers' => [] ];
1063
1064
    foreach ($valid_chambers as $chamber) {
1065
        $key = 'person_regmem_' . $chamber . '_' . $lang;
1066
        $chamber_register = person_register_interests_from_key($key, $extra_info);
1067
        if ($chamber_register) {
1068
            $reg['chamber_registers'][$chamber] = $chamber_register;
1069
        }
1070
    }
1071
    // if chamber_registers is empty, we don't have any data
1072
    if (empty($reg['chamber_registers'])) {
1073
        return;
1074
    }
1075
1076
    // sort chamber registers by published_date
1077
    uasort($reg['chamber_registers'], function ($a, $b) {
1078
        return $a->published_date <=> $b->published_date;
1079
    });
1080
1081
    return $reg;
1082
}
1083
1084
function regional_list($pc, $area_type, $rep_type) {
1085
    $constituencies = MySociety\TheyWorkForYou\Utility\Postcode::postcodeToConstituencies($pc);
1086
    if ($constituencies == 'CONNECTION_TIMED_OUT') {
1087
        throw new MySociety\TheyWorkForYou\MemberException('Sorry, we couldn&rsquo;t check your postcode right now, as our postcode lookup server is under quite a lot of load.');
1088
    } elseif (!$constituencies) {
1089
        throw new MySociety\TheyWorkForYou\MemberException('Sorry, ' . htmlentities($pc) . ' isn&rsquo;t a known postcode');
1090
    } elseif (!isset($constituencies[$area_type])) {
1091
        throw new MySociety\TheyWorkForYou\MemberException(htmlentities($pc) . ' does not appear to be a valid postcode');
1092
    }
1093
    global $PAGE;
1094
    $a = array_values($constituencies);
0 ignored issues
show
Bug introduced by
It seems like $constituencies can also be of type string; however, parameter $array of array_values() does only seem to accept array, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

1094
    $a = array_values(/** @scrutinizer ignore-type */ $constituencies);
Loading history...
1095
    $db = new ParlDB();
1096
    $query_base = "SELECT member.person_id, given_name, family_name, constituency, house
1097
        FROM member, person_names pn
1098
        WHERE constituency IN ('" . join("','", $a) . "')
1099
            AND member.person_id = pn.person_id AND pn.type = 'name'
1100
            AND pn.end_date = (SELECT MAX(end_date) FROM person_names WHERE person_names.person_id = member.person_id)";
1101
    $q = $db->query($query_base . " AND left_reason = 'still_in_office' AND house in (" . HOUSE_TYPE_NI . "," . HOUSE_TYPE_SCOTLAND . "," . HOUSE_TYPE_WALES . ")");
1102
    $current = true;
1103
    if (!$q->rows() && ($dissolution = MySociety\TheyWorkForYou\Dissolution::db())) {
1104
        $current = false;
1105
        $q = $db->query(
1106
            $query_base . " AND $dissolution[query]",
1107
            $dissolution['params']
1108
        );
1109
    }
1110
    $mcon = [];
1111
    $mreg = [];
1112
    foreach ($q as $row) {
1113
        $house = $row['house'];
1114
        $cons = $row['constituency'];
1115
        if ($house == HOUSE_TYPE_COMMONS) {
1116
            continue;
1117
        } elseif ($house == HOUSE_TYPE_NI) {
1118
            $mreg[] = $row;
1119
        } elseif ($house == HOUSE_TYPE_SCOTLAND) {
1120
            if ($cons == $constituencies['SPC']) {
1121
                $mcon = $row;
1122
            } elseif ($cons == $constituencies['SPE']) {
1123
                $mreg[] = $row;
1124
            }
1125
        } elseif ($house == HOUSE_TYPE_WALES) {
1126
            if ($cons == $constituencies['WAC']) {
1127
                $mcon = $row;
1128
            } elseif ($cons == $constituencies['WAE']) {
1129
                $mreg[] = $row;
1130
            }
1131
        } else {
1132
            throw new MySociety\TheyWorkForYou\MemberException('Odd result returned!' . $house);
1133
        }
1134
    }
1135
    if ($rep_type == 'msp') {
1136
        $name = $mcon['given_name'] . ' ' . $mcon['family_name'];
1137
        $cons = $mcon['constituency'];
1138
        $reg = $constituencies['SPE'];
1139
        $url = '/msp/?p=' . $mcon['person_id'];
1140
        if ($current) {
1141
            $data['members_statement'] = '<p>You have one constituency MSP (Member of the Scottish Parliament) and multiple region MSPs.</p>';
0 ignored issues
show
Comprehensibility Best Practice introduced by
$data was never initialized. Although not strictly required by PHP, it is generally a good practice to add $data = array(); before regardless.
Loading history...
1142
            $data['members_statement'] .= '<p>' . sprintf('Your <strong>constituency MSP</strong> is <a href="%s">%s</a>, MSP for %s.', $url, $name, $cons) . '</p>';
1143
            $data['members_statement'] .= '<p>' . sprintf('Your <strong>%s region MSPs</strong> are:', $reg) . '</p>';
1144
        } else {
1145
            $data['members_statement'] = '<p>' . 'You had one constituency MSP (Member of the Scottish Parliament) and multiple region MSPs.' . '</p>';
1146
            $data['members_statement'] .= '<p>' . sprintf('Your <strong>constituency MSP</strong> was <a href="%s">%s</a>, MSP for %s.', $url, $name, $cons) . '</p>';
1147
            $data['members_statement'] .= '<p>' . sprintf('Your <strong>%s region MSPs</strong> were:', $reg) . '</p>';
1148
        }
1149
    } elseif ($rep_type == 'ms') {
1150
        $name = $mcon['given_name'] . ' ' . $mcon['family_name'];
1151
        $cons = gettext($mcon['constituency']);
1152
        $reg = gettext($constituencies['WAE']);
1153
        $url = '/ms/?p=' . $mcon['person_id'];
1154
        if ($current) {
1155
            $data['members_statement'] = '<p>' . gettext('You have one constituency MS (Member of the Senedd) and multiple region MSs.') . '</p>';
1156
            $data['members_statement'] .= '<p>' . sprintf(gettext('Your <strong>constituency MS</strong> is <a href="%s">%s</a>, MS for %s.'), $url, $name, $cons) . '</p>';
1157
            $data['members_statement'] .= '<p>' . sprintf(gettext('Your <strong>%s region MSs</strong> are:'), $reg) . '</p>';
1158
        } else {
1159
            $data['members_statement'] = '<p>' . gettext('You had one constituency MS (Member of the Senedd) and multiple region MSs.') . '</p>';
1160
            $data['members_statement'] .= '<p>' . sprintf(gettext('Your <strong>constituency MS</strong> was <a href="%s">%s</a>, MS for %s.'), $url, $name, $cons) . '</p>';
1161
            $data['members_statement'] .= '<p>' . sprintf(gettext('Your <strong>%s region MSs</strong> were:'), $reg) . '</p>';
1162
        }
1163
    } else {
1164
        if ($current) {
1165
            $data['members_statement'] = '<p>You have multiple MLAs (Members of the Legislative Assembly) who represent you in ' . $constituencies['NIE'] . '. They are:</p>';
1166
        } else {
1167
            $data['members_statement'] = '<p>You had multiple MLAs (Members of the Legislative Assembly) who represented you in ' . $constituencies['NIE'] . '. They were:</p>';
1168
        }
1169
    }
1170
1171
    foreach($mreg as $reg) {
1172
        $data['members'][] =  [
1173
            'url' => '/' . $rep_type . '/?p=' . $reg['person_id'],
1174
            'name' => $reg['given_name'] . ' ' . $reg['family_name'],
1175
        ];
1176
1177
    }
1178
1179
    // Send the output for rendering
1180
    MySociety\TheyWorkForYou\Renderer::output('mp/regional_list', $data);
1181
1182
}
1183