-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.jl
152 lines (137 loc) · 4.01 KB
/
main.jl
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
## Config (changes allowed)
latlen = 220 # lattice length
iters = 220 # iterations
start_rule = 113 # starting rule
order2_rule = ">|+" # the rule of changing the rule
# - no changes (regular auomata)
# > - shift one to the right
# >> - shift two to the right
# < - shift one to the left
# << - shift two to the left
# + - next rule (increases Wolfram's number)
# - - previous rule (decreases Wolfram's number)
# | - flip the rule by bits (mirror)
# ! - perform "not" operation on each bit
# +>> - combination (increase rule and move two rules to the right). + operation cometh first
init = "r" # initial distribution of cells
# . - dot
# r - random
# i - inverse
# : - alternating black and white cells
# .i - combination (inverse dot)
# a - get from init_arr
init_arr = [] # initial distribution of cells (leave blank if init is not "a").
# watch out, init_arr needs to be of length latlen
## Some interesting configurations
#=
starting with dot:
# 60 >>
# 60 >>|<<
# 60 >|<
# 60 <|>
# 60 >+<
# 101
# 101 >
# 101 >>
# 101 >|
# 101 |
# 105 >>
# 105 >|>
# 105 >|<
random start:
# 40 +
# 40 >+< any even?
# 50 >|<
# 60 >|<
# 105 >>
# 105 >>+
# 110 +<
# 110 <
# 110 +
# 110 -
# 110 --
# 110 >+++<
=#
## Config processing (changes not recommended)
# rule is a vector of integers (1s and 0s)
rule = [Int(i)-48 for i::Char in Base.bin(Unsigned(start_rule), 8, false)]
# a function to change the rule number by one
function next_rule!(rulevec::Vector{Int}, step::Int)::Nothing
num::Int = parse(Int, join(rulevec), base=2)
num += step
if num > 255
num -= 255
elseif num < 0
num = 256 + num
end
b::String = Base.bin(Unsigned(num), 8, false)
for i in 1:length(rulevec)
rulevec[i] = Int(b[i])-48
end
nothing
end
# a function to change the rule with a bitwise "not"
function not!(rulevec::Vector{Int})::Nothing
for i in 1:length(rulevec)
rulevec[i] = Int(rulevec[i] == 0)
end
nothing
end
# a mapping from characters to actions (order2 rules)
char2rule = Dict([
'+' => r -> next_rule!(r, 1),
'-' => r -> next_rule!(r, -1),
'>' => r -> circshift!(r, copy(r), 1),
'<' => r -> circshift!(r, copy(r), -1),
'|' => r -> reverse!(r),
'!' => r -> not!(r)
])
## Import and setup for visualization
using PyCall
import PyPlot; const plt = PyPlot
rcParams = PyDict(plt.matplotlib."rcParams")
rcParams["image.cmap"] = "binary"
## Data init (first line)
data = [] # comment this line to append the new auomaton graph to the previous one
if occursin("r", init)
push!(data, [rand((0, 1)) for i::Int in 1:latlen])
elseif occursin(".", init)
push!(data, [((i==round(latlen/2)) ? 1 : 0) for i::Int in 1:latlen])
elseif occursin("a", init)
push!(data, init_arr)
elseif occursin(":", init)
push!(data, [i%2 for i::Int in 1:latlen])
end
if occursin("i", init)
data = [[abs(i-1) for i::Int in data[1]]]
end
## The magic
past = length(data)-1
for iter::Int in 1:iters
println(parse(Int, join(rule), base=2), rule)
new_line::Vector{Int} = []
for i::Int in 1:length(data[past+iter])
neigh::String = ""
for j::Int in i-1:i+1
aj::Unsigned = j # actual index to look at
if (j <= 0)
aj = length(data[past+iter])-1+j
elseif (j > length(data[past+iter]))
aj = j - length(data[past+iter])
end
neigh *= string(data[past+iter][aj])
end
push!(new_line, rule[8-parse(Int, neigh, base=2)])
end
push!(data, new_line)
# change the rule according to the order2_rule
for c::Char in order2_rule
char2rule[c](rule)
end
end
## Visualization
fig, ax = plt.subplots(figsize=(80, 80))
ax.matshow(data)
ax.axis(false)
display(plt.gcf()) # for Juno Plots window
plt.show() # for terminal usage