This vignette introduces the concept of a performance analysis and demonstrates how
eiCompare can be used to conduct one given voter files and districting maps.
A successful voting rights case may result in a local jurisdiction’s districting map being thrown out, prompting the need for a new map. Thus, it’s necessary assess whether the new map provides sufficient representation for minority groups. It’s not desirable to simply wait for elections to happen and see what the results might be. Instead, we can look at past elections, and observe how candidates would perform if the proposed map was used at the time of the election. This is the basis of a performance analysis.
Ultimately, we need to assess the demographic breakdown across racial groups for each district in the new map. What data source do we use to calculate the percentage of each racial group? We could use total population or citizen voting age population (CVAP), as provided by the Census Bureau. However, these measures may not be accurate, because they assume that voter turnout is equal across racial groups. This is not always true, especially in cases of racial gerrymandering, where turnout for minority groups may be depressed. Instead, we argue that going to the level of the voter file is necessary, as the voter file actually informs who turned out to vote.
To conduct a performance analysis, we need to join the voters in a voter file to the new districts and determine the turnout by race, per district. This includes the following steps:
eiCompare provides functions to perform steps 2-4 of the analysis, as well as a function that completes the entire pipeline. Note that Gecoding (step 1) must be performed separately (
eiCompare provides tools to aid in geocoding: see the Geocoding vignette). In the following section, we walk through each of the steps.
The example we’ll use to demonstrate the performance analysis is East Ramapo School District (ERSD), located in Rockland County of the New York City suburbs. ERSD is highly segregated, with the majority of Black and Hispanic students attending public schools and white students attending private schools. Furthermore, ERSD uses an at-large voting system for School Board elections, where all voters could vote for all seats on the school board. This system favored the white families whose students largely attended private schools, resulting in redistribution of funds that had adverse impacts on public school students.
In May 2020, the at-large voting system was struck down, and a ward system with a new set of districting maps was required. In a ward system, voters elect a representative for their own geographically compact ward. Two maps were proposed: one by the plaintiffs (New York Civil Liberties Union, NYCLU) and the defendants (ERSD). Ideally, the map should allow sufficient representation for the minority aggregate population (in this case, Black and Hispanic/Latino voters).
In this case study, we’ll focus on the defendant map. We’ll demonstrate how a performance analysis reveals that simply using CVAP to assess the minority constituency may overestimate the number of seats won by minority supported candidates.
Let’s take a look at a map proposed by the defendants, ERSD. The district map is composed of nine wards. To assess representation, we could examine the CVAP (the number of people who can vote) by race across the wards. Thus, let’s take a look at the fraction of CVAP voters that are in the minority aggregate, by ward:
# Load the map data("ersd_maps") sf::st_crs(ersd_maps) <- 4326
# Plot the map, using a fill that depends on Citizen Voting Age Population (CVAP) options(repr.plot.width = 7.2, repr.plot.height = 6) cvap_map <- ggplot() + geom_sf(data = ersd_maps, aes(fill = MIN_AGG_FRAC)) + geom_sf_label(data = ersd_maps, aes(label = WARD), size = 5) + scale_fill_continuous(limits = c(0, 1)) + xlab("Latitude") + ylab("Longitude") + theme_bw(base_size = 10) + theme( axis.title.x = element_text(size = 15, face = "bold", margin = margin(t = 5)), axis.title.y = element_text(size = 15, face = "bold", margin = margin(r = 5)), legend.key.width = unit(0.4, "cm"), legend.key.height = unit(1, "cm") ) + guides(fill = guide_legend( title = "Fraction\nMinority", title.position = "top", title.size = 10 )) show(cvap_map)
Examining this map reveals that, according to CVAP, four wards (1-4) have potential for minority voters to elect a representative of their choice (they have a plurality or majority). However, due to turnout differences across racial groups, this does not imply that the minority voters would have actually turned out sufficiently enough to elect four representatives. In other words, this map may not have “performed” well enough to guarantee representation, due to turnout. Now, we’ll walk through how to conduct a performance analysis and test this hypothesis"
Since the entire pipeline is contained within the function
performance_analysis(), we’ll first use a toy voter file to demonstrate the individual steps. This toy voter file already is already “geocoded”, implying that step 1 is complete.
voter_file <- data.frame( voter_id = c(1, 2, 3, 4, 5, 5), surname = c( "ROSENBERG", "JACKSON", "HERNANDEZ", "LEE", "SMITH", "SMITH" ), lat = c(41.168, 41.1243, 41.089, 41.14, 41.12, 41.123), lon = c(-74.02, -74.039, -74.08, -74.05, -74.045, -74.046) )
The voter file consists of 5 example voters whose surnames are actually found in the East Ramapo voter file, but locations are randomly assigned. The file depicts the bare necessities for conducting a performance analysis: a voter ID column to identify unique voters, a surname column for identifying race, and latitude/longitude columns for identifying location.
Observe that the above voter file contains a duplicate: voter “SMITH” appears twice, with the same voter ID (but different locations). This is a common occurrence in voter files, particularly when voters request a change of address. In these cases, the voter ID remains the same, but both the old and new addresses remain on the voter file for some time. Thus, voter files need to be de-duplicated.
To handle this,
eiCompare has a
dedupe_voter_file function which will automatically take the most recent entry in the voter file for repeated voter IDs. Voter files are typically sorted by registration date, so de-duplicating automatically takes the latest rows. Let’s apply this function to the toy voter file:
voter_file <- eiCompare::dedupe_voter_file( voter_file = voter_file, voter_id = "voter_id" ) print(voter_file)
## voter_id surname lat lon ## 1 1 ROSENBERG 41.17 -74.02 ## 2 2 JACKSON 41.12 -74.04 ## 3 3 HERNANDEZ 41.09 -74.08 ## 4 4 LEE 41.14 -74.05 ## 6 5 SMITH 41.12 -74.05
Now, there’s only one row for voter “SMITH” with voter ID “5”. Importantly, it’s the second row in the voter file corresponding to that voter, implying we have the most recent information.
Next, we need to identify which wards these voters are located in. Performing this spatial join is abstracted in the
merge_voter_file_to_shape() function, which can convert the voter file to a geometry object and join on location. This function, in addition to requiring the voter and shape files, also uses a Coordinate Reference System (CRS) to use (as a string or integer) as well as the column names for the longitude, latitude, and voter ID (in order to de-duplicate the voter file after the spatial join). The CRS can be ommitted if the shape file comes with its own CRS, which is the case.
voter_file_w_ward <- eiCompare::merge_voter_file_to_shape( voter_file = voter_file, shape_file = ersd_maps, coords = c("lon", "lat"), voter_id = "voter_id" ) print(as.data.frame(voter_file_w_ward)[, c("surname", "WARD")])
## surname WARD ## 1 ROSENBERG 8 ## 2 JACKSON 2 ## 3 HERNANDEZ 5 ## 4 LEE 1 ## 6 SMITH 3
We can double check that the correct wards were identified by plotting the voters on the ward map:
# Plot the map with no fill and voters as points options(repr.plot.width = 7.2, repr.plot.height = 6) map <- ggplot() + geom_sf(data = ersd_maps, fill = "white") + geom_sf_label(data = ersd_maps, aes(label = WARD), size = 3) + geom_sf(data = voter_file_w_ward, size = 4, color = "black") + xlab("Latitude") + ylab("Longitude") + theme_bw(base_size = 10) + theme( axis.title.x = element_text(size = 15, face = "bold", margin = margin(t = 5)), axis.title.y = element_text(size = 15, face = "bold", margin = margin(r = 5)) ) show(map)