-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathhemibrain_axons_dendrites.Rmd
257 lines (192 loc) · 12.3 KB
/
hemibrain_axons_dendrites.Rmd
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
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
---
title: "axons and dendrites"
output: rmarkdown::html_vignette
vignette: >
%\VignetteIndexEntry{axons and dendrites}
%\VignetteEngine{knitr::rmarkdown}
%\VignetteEncoding{UTF-8}
---
```{r setup, include = FALSE}
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>"
)
```
> "This history of civilization proves beyond doubt just how sterile the repeated attempts of metaphysics to guess at nature' s laws have been. Instead, there is every reason to believe that when the human intellect ignores reality and concentrates within, it can no longer explain the simplest inner workings of life' s machinery or of the world around us."
([Santiago Ramón y Cajal](https://en.wikiquote.org/wiki/Santiago_Ram%C3%B3n_y_Cajal)
# Axons, dendrite and other compartments
The philosophy of `hemibrainr` is to try and treat axons and dendrites separately, where we can. Generally, the division between axon and dendrite is ignored when people use graph theoretic approach on connectome data, or otherwise build their models. However, dendrite and axons may function differently - we just don't yet know in insects, how differently.
Other compartments include the 'primary neurite' or 'cell body fibre' (purple, below) [(Ito et al., 2014)](https://www.sciencedirect.com/science/article/pii/S0896627313011781), the non-synaptic cable that attaches the neuron's cell body, its 'soma' to the rest of the neuron. There is also a 'primary dendrite', or 'linker' (green, below), which is non-synaptic cable that connects a neuron's dendrite and axon, through which an action potential presumably travels.
## Splitting a neuron
Neurons can be split into a principle input zone, a dendrite, and a principle output zone, an axon. Dendrites tend to be more spindly and axons bulkier, with boutons packed with presynapses, ready to release neurotransmitter.
<center>
![neuron_split](https://raw.githubusercontent.com/flyconnectome/hemibrainr/master/inst/images/neuron_split.png)
</center>
But axons and dendrites are not a totally clear division for most neurons. A neuron may have more than one of each. It may also be more 'amacrine-like' and lack this division, though this is less common. Dendrites can have outputs, and axons can have inputs. Axons may connect with axons, even dendrites may input and axon.
However, all fly neurons examined can be split into an axon-like and a dendrite-like compartment [(Schneider-Mizell et al., 2016)](https://elifesciences.org/articles/12059)). The density of outputs in the axon is higher than in the dendrite. The density of inputs to the dendrite, are higher than in the axon [(Bates & Schlegel et al., 2020)](https://www.sciencedirect.com/science/article/pii/S0960982220308587)). Interestingly, the 'start' of a dendrite is almost always closer to the neuron's cell body than the start of its axon. See our recent data [(Bates & Schlegel et al., 2020)](https://www.sciencedirect.com/science/article/pii/S0960982220308587)) for three broad classes of olfactory neuron:
<center>
![split_validation](https://raw.githubusercontent.com/flyconnectome/hemibrainr/master/inst/images/split_validation.png)
</center>
The biophysical impact of axo-axonic connections is unclear – they can have non-intuitive effects depending on the timing of action potentials in connected partners [(Burrows and Laurent, 1993; Burrows and Matheson, 1994; Cuntz et al., 2007; Haag and Borst, 2004)](https://en.wikipedia.org/wiki/Axo-axonic_synapse).
## Splitting a neuron by flow centrality
We can split neurons algorithmically by calculating a neuron's 'flow centrality' [(Schneider-Mizell et al., 2016)](https://elifesciences.org/articles/12059)).
From [(Schneider-Mizell et al., 2016)](https://elifesciences.org/articles/12059)):
> "We use flow centrality for four purposes. First, to split an arbor into axon and dendrite at the maximum centrifugal SFC, which is a preliminary step for computing the segregation index, for expressing all kinds of connectivity edges (e.g. axo-axonic, dendro-dendritic) in the wiring diagram, or for rendering the arbor in 3d with differently colored regions. Second, to quantitatively estimate the cable distance between the axon terminals and dendritic arbor by measuring the amount of cable with the maximum centrifugal SFC value. Third, to measure the cable length of the main dendritic shafts using centripetal SFC, which applies only to insect neurons with at least one output syn- apse in their dendritic arbor. And fourth, to weigh the color of each skeleton node in a 3d view, providing a characteristic signature of the arbor that enables subjective evaluation of its identity."
Explore their methods for more information.
# Tutorial
First, let us get and load the packages we will need:
```{r load, eval = FALSE}
# install
if (!require("remotes")) install.packages("remotes")
remotes::install_github("natverse/hemibrainr")
# use
library(hemibrainr)
```
## Split neurons from the hemibrain
Let us examine some third-order olfactory neurons (TONs), which we have stored in `hemibrainr` as `hemibrainr::ton.ids`.
```{r split, eval = FALSE}
# Let's examine a random set of tons
ids = sample(hemibrainr::ton.ids,10)
# Read neurons from neuprint
neurons = neuprintr::neuprint_read_neurons(ids)
# Now make sure the neurons have a soma marked
## Some hemibrain neurons do not, as the soma was chopped off
neurons.checked = hemibrain_skeleton_check(neurons, meshes = hemibrain.surf)
# Split neuron
## These are the recommended parameters for hemibrain neurons
neurons.flow = flow_centrality(neurons.checked,
polypre = TRUE,
mode = "centrifugal",
split = "synapses")
## What has this returned?
## It is still a neuronlist object, but now more infromation has been added
## To each neuron in the neuron list. If neuron = neurons.flow[[1]], then
## the neuron has flow centrality information added to neuron$d and a segregation index score.
## The neuron$d$Label now gives the compartment,
## where axon is Label = 2, dendrite Label = 3, primary dendrite Label = 9
## and primary neurite Label = 7. Soma is Label = 1.
## (note, some outputs from various function give names rather than numbers)
## We can see this, as so:
neuron = neurons.flow[[1]]
head(neuron$d)
### standard_compartments is a helper function to change numbers to words
### for the different compartment types.
```
However, to make things easier, we can do this all in one step:
```{r hemibrain_read_neurons, eval = FALSE}
# Get the split neurons
neurons.flow = hemibrain_read_neurons(ids, remote = FALSE)
## This function reads neurons using neuptinr
## If remote == TRUE, then neurons read from linked google drive. See relevant article for more information.
```
## Get already split neurons
Splitting neurons can take a little time. We can also load thousands of pre-split neurons from the `hemibrainr` Google team drive. In order to connect R to this Google drive, you have a few options. Please see [this article](https://flyconnectome.github.io/hemibrainr/articles/google_filestream.html). Once you have access and have mounted the drive, you should be able to do:
```{r hemi.db, eval = FALSE}
# All split hemibrain neurons
db = hemibrain_neurons()
length(db)
## This is a nat::neuronlistfh object.
### This means that a data frame for the neurons and information
### specifying where to find each neuron's data is read into R -
### but not the whole, huge nat::neuronlist object. This saves on memory
### for your R session. When an operation that requires the actual
### neuron data is performed, neurons are read into R.
# Or just load the neurons you want
neurons.flow = hemibrain_read_neurons(ids, remote = TRUE)
```
### Hemibrain select cable
If we want, we can select just the cable and synapses that constitute one of the neuron's compartments.
For example:
```{r cable, eval = FALSE}
# Extract individual bits of cable
axons = axonic_cable(neurons.flow)
dendrites = dendritic_cable(neurons.flow)
pnts = primary_neurite(neurons.flow)
pds = primary_dendrite_cable(neurons.flow)
tracts = tract_cable(neurons.flow)
# Plot
nat::nopen3d()
hemibrain_view()
plot3d(axons, col = "orange")
plot3d(dendrites, col = "cyan")
plot3d(pnts, col = "purple", soma = 500)
plot3d(pds, col = "green")
plot3d(tracts, col = "black", lwd = 5)
# We can see, for example, just axon output synapses:
axon.synapses = hemibrain_extract_synapses(axons, prepost = "PRE")
head(axon.synapses)
```
<center>
![tracts](https://raw.githubusercontent.com/flyconnectome/hemibrainr/master/inst/images/tracts.png)
</center>
### Visualise the split
We can see the neruon splits, using `hemibrainr::plot3d_split`:
```{r plot3d_split, eval = FALSE}
# Plot 3D
nat::nopen3d()
hemibrain_view()
plot3d_split(neurons.flow)
plot3d(hemibrain.surf, col = "lightgrey", alpha = 0.1, add = TRUE)
```
<center>
![plot3d_split](https://raw.githubusercontent.com/flyconnectome/hemibrainr/master/inst/images/plot3d_split.png)
</center>
Here, lines: blue = dendrite, orange = axon, green = linker, purple = cell body fibre.
Balls: red = presynapse (output), navy blue = postsynapse (input), purple = soma.
We can also scan through split to see them individually, using `hemibrainr::nlscan_split`:
```{r nlscan_split, eval = FALSE}
# Plot 3D
clear3d()
nlscan_split(neurons.flow)
```
<center>
![nlscan_split](https://raw.githubusercontent.com/flyconnectome/hemibrainr/master/inst/images/nlscan_split.png)
</center>
So I think most of those splits are pretty convincing.
### Split connectivity
It will likely be important, when looking at neuron connectivity, to know whether we are examining a axo-dendritic, axo-axonic, dendro-dendritic, etc., connection.
We can see where synapses on these neurons lie, using:
```{r comp.synapses, eval = FALSE}
neuron.synapses = hemibrain_extract_synapses(neurons.flow, prepost = "BOTH")
table(neuron.synapses$prepost, neuron.synapses$Label)
```
Here, there is one entry per synapse. The prepost column tells us whether the synapse is presynaptic (0, an output synapse) or postsynaptic (1, an input synapse).
We can also easily get one entry per target-source connection, by:
```{r comp.connections, eval = FALSE}
# Extract connections
neuron.connections = hemibrain_extract_connections(neurons.flow, prepost = "BOTH")
neuron.connections.strong = subset(neuron.connections, count >= 5)
# Connections from neurons' dendrites
hist(subset(neuron.connections.strong, Label == "dendrite" & prepost == 1)$count,
main="Connections from neurons' dendrites",
xlab="Strength of neuron->partner synaptic connection",
col="cyan",
freq=TRUE,
breaks = max(subset(neuron.connections.strong, Label == "dendrite")$count)
)
# Connections from neurons' axons
hist(subset(neuron.connections.strong, Label == "axon" & prepost == 1)$count,
main="Maximum daily temperature at La Guardia Airport",
xlab="Strength of neuron->partner synaptic connection",
col="orange",
freq=TRUE,
breaks = max(subset(neuron.connections.strong, Label == "axon")$count)
)
```
Here, the prepost column tells us whether the partner is presynaptic (0, upstream source) or postsynaptic (1, downstream target).
We might also want to know the 'Label' of the partner cell. We can get an edgelist with this information:
```{r comp.elist, eval = FALSE}
## The function hemibrain_extract_compartment_edgelist will give an edgelist
## for connections between the neruons you provide to it.
## To make sure some of our randomly selected neurons connect
## Let's also grab some PNs
pns.flow = hemibrain_read_neurons(hemibrainr::upn.ids, remote= TRUE)
### Set remote to FALSE, if you do not have the google drive mounted.
# Extract connections
pns.elist = hemibrain_extract_compartment_edgelist(pns.flow, .progress = FALSE)
pns.elist.strong = subset(pns.elist, count >= 10 | norm > 0.01)
sort(table(pns.elist.strong$connection))
## A lot of axo-axonic and dendro-dendritic action amongst the PNs
```
Here, pre is the `bodyid` of the presynaptic (upstream source) neuron and post is the `bodyid` of the postsynaptic (downstream target) neuron. The `norm` column gives a synaptic weight normalised by the total inputs of the downstream neuron (post), i.e. count/total postsynapses on 'post'.
Please see [this article](https://flyconnectome.github.io/hemibrainr/articles/hemibrain_connectivity.html) to see a more in-depth example of how to efficiently work neuron connectivity data from `hemibrainr`.