-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlook.py
106 lines (106 loc) · 3.53 KB
/
look.py
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
"""
The interpreter of the sound-based esoteric programming language Look!
Requires: mido, pygame
See detailed documentation of the language on esolang wiki: https://esolangs.org/wiki/Look!
"""
from os import environ
environ['PYGAME_HIDE_SUPPORT_PROMPT'] = '1'
from mido import MidiFile,MidiTrack,Message,MetaMessage,bpm2tempo
from random import *
import pygame
import sys
def bf(code):
s=[]
matches={}
tape=[0]*1000000
for i,j in enumerate(code):
if j=='[':
s.append(i)
if j==']':
m=s.pop()
matches[m]=i
matches[i]=m
cp=0
p=0
while cp<len(code):
if code[cp]=='+':
tape[p]=(tape[p]+1)%256
if code[cp]=='-':
tape[p]=(tape[p]-1)%256
if code[cp]==',':
ch=sys.stdin.read(1)
tape[p]=(ord(ch) if ch else 0)%256
if code[cp]=='.':
print(chr(tape[p]),end='')
if code[cp]=='<':
p-=1
if code[cp]=='>':
p+=1
if code[cp]=='[':
if not tape[p]:
cp=matches[cp]
if code[cp]==']':
if tape[p]:
cp=matches[cp]
cp+=1
notes=['C','C#','D','D#','E','F','F#','G','G#','A','A#','B']
note2fuck='+]-+,.[>><,['
fuck2note={}
for j,i in enumerate(note2fuck):
if i in fuck2note:
fuck2note[i].append(notes[j])
else:
fuck2note[i]=[notes[j]]
def brainfuck2notes(brainfuck):
return [choice(fuck2note[i]) for i in brainfuck]
def notes2brainfuck(notes_):
return ''.join([note2fuck[notes.index(i)] for i in notes_])
def notes2mid(notes_,fn,bpm=120,velocity=127):
file=MidiFile()
track=MidiTrack()
file.tracks.append(track)
track.append(MetaMessage('set_tempo', tempo=bpm2tempo(bpm)))
for i in notes_:
track.append(Message('note_on',note=60+notes.index(i),velocity=velocity,time=0))
track.append(Message('note_off', note=60 + notes.index(i), time=480))
file.save(fn)
def playmid(fn):
pygame.mixer.init()
pygame.mixer.music.load(fn)
pygame.mixer.music.play()
while pygame.mixer.music.get_busy():
pass
def mid2notes(fn):
file=MidiFile(fn)
track=file.tracks[0]
result=[]
for msg in track:
if msg.type=='note_on':
result.append(notes[msg.note%12])
return result
def interpret(fn):
bf(notes2brainfuck(mid2notes(fn)))
if __name__=='__main__':
def helper():
print(f'Usage: {sys.executable} {sys.argv[0]} <command> <filename>\nCommands:\ninterpret: Interprets a Look! program.\nfuck: Converts a Look! program to brainfuck code and prints it.\nplay: Plays a Look! program as audio\nnotes: outputs the notes in a Look! program.\nunfuck: input some brainfuck code and encode it into a Look! program.\nunnotes: input some notes and write them into a Look! program.')
a=sys.argv
if len(a)==3:
argument=a[1]
if argument=='interpret':
interpret(a[2])
if argument=='fuck':
print(notes2brainfuck(mid2notes(a[2])))
if argument=='play':
playmid(a[2])
if argument=='notes':
print(' '.join(mid2notes(a[2])))
if argument=='unfuck':
code=input('Please enter some brainfuck code:')
notes2mid(brainfuck2notes(code),a[2])
if argument=='unnotes':
code=input('Please enter some notes separated by spaces:')
notes2mid(code.split(),a[2])
if argument not in 'interpret fuck play notes unfuck unnotes'.split():
helper()
else:
helper()