Repository
https://github.com/to-the-sun/amanuensis
This development contribution is submitted in association with task request by @to-the-sun:
Link to the task request
What is Amanuensis?
The Amanuensis is an automated songwriting and recording system created by @to-the-sun aimed at ridding the process of anything left-brained, so one need never leave a creative, spontaneous and improvisational state of mind, from the inception of the song until its final master. The program will construct a cohesive song structure, using the best of what you give it, looping around you and growing in real-time as you play. All you have to do is jam and fully written songs will flow out behind you wherever you go.
New Feature
The graph
I implemented a graph for plotting the values in the dictionary rhythm
. Rhythm dictionary acted as a sparse array which gets updated as the time passes when the user starts recording. I had to plot the plot the data stored in the rhythm
dictionary from the moment referred in spout[0]
. The graph should also have to be updated in real time so I was needed to clear the plotted data after the plotting has been done. Matplotlib was used to plot the graph.
Implementation
Python script: https://github.com/to-the-sun/amanuensis/blob/master/consciousness.py
To make the graph real time I created a separate process using Python's multiprocessing
library. Also, a queue
was created to pass the data to be plotted to the drawing_process
in real time. The Queue was made non-blocking to prevent the graph window from freezing, waiting for new data.
Relevant codes
Funtion for graph drawing process
def draw(queue):
"""
Process to draw & update the graph.
"""
# Initializing Graph
xlim = stats['wake'] + 25
ylim = 2
plt.ion()
figure = plt.figure()
figure.subplots_adjust(bottom=0.22)
figure.patch.set_facecolor('#eff3b9')
figure.canvas.set_window_title('Rhythm')
axes = figure.add_subplot(111)
axes.set_facecolor('#eff3b9')
axes.set_xlim([0, xlim])
axes.set_ylim([0, ylim])
axes.set_title('Rhythm')
axes.set_xlabel('ms')
axes.set_ylabel('cumulative beat intervals')
try:
x_data, y_data = queue.get(False)
except que.Empty:
x_data, y_data = new_data()
l, = axes.plot(x_data, y_data, '#757981')
manager = figure.canvas.manager
manager.window.wm_geometry("+{}-{}".format(10, 20))
manager.resize(550, 200)
figure.canvas.draw()
figure.canvas.flush_events()
# Update Loop
while True:
try:
x_data, y_data = queue.get(False)
ylim = np.amax(y_data) + 1
axes.set_ylim([0, ylim])
if stats['wake'] > xlim:
xlim = stats['wake'] + 25
axes.set_xlim([0, xlim])
l.set_xdata(x_data)
l.set_ydata(y_data)
figure.canvas.draw()
figure.canvas.flush_events()
except que.Empty:
try:
figure.canvas.start_event_loop(0.5)
except TclError:
print("Graph closed!")
break
except TypeError:
pass
except TclError:
break
Function to prepare new data for plotting.
def new_data(keys=None, values=None):
"""
Function to prepare data for plotting the graph.
"""
ydata = np.zeros(stats['wake'] + 25, dtype='int')
xdata = np.linspace(0, stats['wake'] + 25, stats['wake'] + 25, dtype='int')
if keys is not None:
i = 0
j = 0
while j < len(values):
try:
if i in keys:
ydata[i] = values[j]
j += 1
i += 1
except Exception:
break
return xdata, ydata
Removing plotted values from the rhythm
dictionary after plotting.
def remove_past_moments(r_dict, spout):
"""
Removing keys from the rhythm dict which denotes moments in the past
Returns new rhythm dict.
"""
for key in r_dict.copy():
if key - spout < 0:
r_dict.pop(key)
return r_dict
Preparing new plot data and sending it via the queue
try:
# Stripping off past moments from the rhythm dict.
rhythm = remove_past_moments(rhythm, spout[0])
lists = sorted(rhythm.items())
x, y = zip(*lists)
x = np.array(x) - spout[0]
x, y = new_data(x, y)
except ValueError as v:
pass
except Exception as e:
print(e)
finally:
try:
q.put((x, y))
except Exception as e:
print(e)
A working implementation of the graph can be seen in this video: Video Link
Thank you for your contribution.Its better to show the commits you have worked on. Also for this commit https://github.com/to-the-sun/amanuensis/commit/ddc558e0e2879e6fd9dee60dc1cce46759e11655, I can see a lot of changes so do try to explain that too if you have done that change.
Your contribution has been evaluated according to Utopian policies and guidelines, as well as a predefined set of questions pertaining to the category.
To view those questions and the relevant answers related to your post, click here.
Need help? Write a ticket on https://support.utopian.io/.
Chat with us on Discord.
[utopian-moderator]
Hey @ajmaln
Thanks for contributing on Utopian.
We’re already looking forward to your next contribution!
Want to chat? Join us on Discord https://discord.gg/h52nFrV.
Vote for Utopian Witness!
That is amazing. Following.
Thank you so much.
Congratulations! This post has been upvoted from the communal account, @minnowsupport, by ajmaln from the Minnow Support Project. It's a witness project run by aggroed, ausbitbank, teamsteem, theprophet0, someguy123, neoxian, followbtcnews, and netuoso. The goal is to help Steemit grow by supporting Minnows. Please find us at the Peace, Abundance, and Liberty Network (PALnet) Discord Channel. It's a completely public and open space to all members of the Steemit community who voluntarily choose to be there.
If you would like to delegate to the Minnow Support Project you can do so by clicking on the following links: 50SP, 100SP, 250SP, 500SP, 1000SP, 5000SP.
Be sure to leave at least 50SP undelegated on your account.
What can I say, excellent work yet again! Great to be working with you.
Thank you
@resteemator is a new bot casting votes for its followers. Follow @resteemator and vote this comment to increase your chance to be voted in the future!