Blame Lib/turtledemo/sorting_animate.py

rpm-build 2bd099
#!/usr/bin/env python3
rpm-build 2bd099
"""
rpm-build 2bd099
rpm-build 2bd099
         sorting_animation.py
rpm-build 2bd099
rpm-build 2bd099
A minimal sorting algorithm animation:
rpm-build 2bd099
Sorts a shelf of 10 blocks using insertion
rpm-build 2bd099
sort, selection sort and quicksort.
rpm-build 2bd099
rpm-build 2bd099
Shelfs are implemented using builtin lists.
rpm-build 2bd099
rpm-build 2bd099
Blocks are turtles with shape "square", but
rpm-build 2bd099
stretched to rectangles by shapesize()
rpm-build 2bd099
 ---------------------------------------
rpm-build 2bd099
       To exit press space button
rpm-build 2bd099
 ---------------------------------------
rpm-build 2bd099
"""
rpm-build 2bd099
from turtle import *
rpm-build 2bd099
import random
rpm-build 2bd099
rpm-build 2bd099
rpm-build 2bd099
class Block(Turtle):
rpm-build 2bd099
rpm-build 2bd099
    def __init__(self, size):
rpm-build 2bd099
        self.size = size
rpm-build 2bd099
        Turtle.__init__(self, shape="square", visible=False)
rpm-build 2bd099
        self.pu()
rpm-build 2bd099
        self.shapesize(size * 1.5, 1.5, 2) # square-->rectangle
rpm-build 2bd099
        self.fillcolor("black")
rpm-build 2bd099
        self.st()
rpm-build 2bd099
rpm-build 2bd099
    def glow(self):
rpm-build 2bd099
        self.fillcolor("red")
rpm-build 2bd099
rpm-build 2bd099
    def unglow(self):
rpm-build 2bd099
        self.fillcolor("black")
rpm-build 2bd099
rpm-build 2bd099
    def __repr__(self):
rpm-build 2bd099
        return "Block size: {0}".format(self.size)
rpm-build 2bd099
rpm-build 2bd099
rpm-build 2bd099
class Shelf(list):
rpm-build 2bd099
rpm-build 2bd099
    def __init__(self, y):
rpm-build 2bd099
        "create a shelf. y is y-position of first block"
rpm-build 2bd099
        self.y = y
rpm-build 2bd099
        self.x = -150
rpm-build 2bd099
rpm-build 2bd099
    def push(self, d):
rpm-build 2bd099
        width, _, _ = d.shapesize()
rpm-build 2bd099
        # align blocks by the bottom edge
rpm-build 2bd099
        y_offset = width / 2 * 20
rpm-build 2bd099
        d.sety(self.y + y_offset)
rpm-build 2bd099
        d.setx(self.x + 34 * len(self))
rpm-build 2bd099
        self.append(d)
rpm-build 2bd099
rpm-build 2bd099
    def _close_gap_from_i(self, i):
rpm-build 2bd099
        for b in self[i:]:
rpm-build 2bd099
            xpos, _ = b.pos()
rpm-build 2bd099
            b.setx(xpos - 34)
rpm-build 2bd099
rpm-build 2bd099
    def _open_gap_from_i(self, i):
rpm-build 2bd099
        for b in self[i:]:
rpm-build 2bd099
            xpos, _ = b.pos()
rpm-build 2bd099
            b.setx(xpos + 34)
rpm-build 2bd099
rpm-build 2bd099
    def pop(self, key):
rpm-build 2bd099
        b = list.pop(self, key)
rpm-build 2bd099
        b.glow()
rpm-build 2bd099
        b.sety(200)
rpm-build 2bd099
        self._close_gap_from_i(key)
rpm-build 2bd099
        return b
rpm-build 2bd099
rpm-build 2bd099
    def insert(self, key, b):
rpm-build 2bd099
        self._open_gap_from_i(key)
rpm-build 2bd099
        list.insert(self, key, b)
rpm-build 2bd099
        b.setx(self.x + 34 * key)
rpm-build 2bd099
        width, _, _ = b.shapesize()
rpm-build 2bd099
        # align blocks by the bottom edge
rpm-build 2bd099
        y_offset = width / 2 * 20
rpm-build 2bd099
        b.sety(self.y + y_offset)
rpm-build 2bd099
        b.unglow()
rpm-build 2bd099
rpm-build 2bd099
def isort(shelf):
rpm-build 2bd099
    length = len(shelf)
rpm-build 2bd099
    for i in range(1, length):
rpm-build 2bd099
        hole = i
rpm-build 2bd099
        while hole > 0 and shelf[i].size < shelf[hole - 1].size:
rpm-build 2bd099
            hole = hole - 1
rpm-build 2bd099
        shelf.insert(hole, shelf.pop(i))
rpm-build 2bd099
    return
rpm-build 2bd099
rpm-build 2bd099
def ssort(shelf):
rpm-build 2bd099
    length = len(shelf)
rpm-build 2bd099
    for j in range(0, length - 1):
rpm-build 2bd099
        imin = j
rpm-build 2bd099
        for i in range(j + 1, length):
rpm-build 2bd099
            if shelf[i].size < shelf[imin].size:
rpm-build 2bd099
                imin = i
rpm-build 2bd099
        if imin != j:
rpm-build 2bd099
            shelf.insert(j, shelf.pop(imin))
rpm-build 2bd099
rpm-build 2bd099
def partition(shelf, left, right, pivot_index):
rpm-build 2bd099
    pivot = shelf[pivot_index]
rpm-build 2bd099
    shelf.insert(right, shelf.pop(pivot_index))
rpm-build 2bd099
    store_index = left
rpm-build 2bd099
    for i in range(left, right): # range is non-inclusive of ending value
rpm-build 2bd099
        if shelf[i].size < pivot.size:
rpm-build 2bd099
            shelf.insert(store_index, shelf.pop(i))
rpm-build 2bd099
            store_index = store_index + 1
rpm-build 2bd099
    shelf.insert(store_index, shelf.pop(right)) # move pivot to correct position
rpm-build 2bd099
    return store_index
rpm-build 2bd099
rpm-build 2bd099
def qsort(shelf, left, right):
rpm-build 2bd099
    if left < right:
rpm-build 2bd099
        pivot_index = left
rpm-build 2bd099
        pivot_new_index = partition(shelf, left, right, pivot_index)
rpm-build 2bd099
        qsort(shelf, left, pivot_new_index - 1)
rpm-build 2bd099
        qsort(shelf, pivot_new_index + 1, right)
rpm-build 2bd099
rpm-build 2bd099
def randomize():
rpm-build 2bd099
    disable_keys()
rpm-build 2bd099
    clear()
rpm-build 2bd099
    target = list(range(10))
rpm-build 2bd099
    random.shuffle(target)
rpm-build 2bd099
    for i, t in enumerate(target):
rpm-build 2bd099
        for j in range(i, len(s)):
rpm-build 2bd099
            if s[j].size == t + 1:
rpm-build 2bd099
                s.insert(i, s.pop(j))
rpm-build 2bd099
    show_text(instructions1)
rpm-build 2bd099
    show_text(instructions2, line=1)
rpm-build 2bd099
    enable_keys()
rpm-build 2bd099
rpm-build 2bd099
def show_text(text, line=0):
rpm-build 2bd099
    line = 20 * line
rpm-build 2bd099
    goto(0,-250 - line)
rpm-build 2bd099
    write(text, align="center", font=("Courier", 16, "bold"))
rpm-build 2bd099
rpm-build 2bd099
def start_ssort():
rpm-build 2bd099
    disable_keys()
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text("Selection Sort")
rpm-build 2bd099
    ssort(s)
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text(instructions1)
rpm-build 2bd099
    show_text(instructions2, line=1)
rpm-build 2bd099
    enable_keys()
rpm-build 2bd099
rpm-build 2bd099
def start_isort():
rpm-build 2bd099
    disable_keys()
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text("Insertion Sort")
rpm-build 2bd099
    isort(s)
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text(instructions1)
rpm-build 2bd099
    show_text(instructions2, line=1)
rpm-build 2bd099
    enable_keys()
rpm-build 2bd099
rpm-build 2bd099
def start_qsort():
rpm-build 2bd099
    disable_keys()
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text("Quicksort")
rpm-build 2bd099
    qsort(s, 0, len(s) - 1)
rpm-build 2bd099
    clear()
rpm-build 2bd099
    show_text(instructions1)
rpm-build 2bd099
    show_text(instructions2, line=1)
rpm-build 2bd099
    enable_keys()
rpm-build 2bd099
rpm-build 2bd099
def init_shelf():
rpm-build 2bd099
    global s
rpm-build 2bd099
    s = Shelf(-200)
rpm-build 2bd099
    vals = (4, 2, 8, 9, 1, 5, 10, 3, 7, 6)
rpm-build 2bd099
    for i in vals:
rpm-build 2bd099
        s.push(Block(i))
rpm-build 2bd099
rpm-build 2bd099
def disable_keys():
rpm-build 2bd099
    onkey(None, "s")
rpm-build 2bd099
    onkey(None, "i")
rpm-build 2bd099
    onkey(None, "q")
rpm-build 2bd099
    onkey(None, "r")
rpm-build 2bd099
rpm-build 2bd099
def enable_keys():
rpm-build 2bd099
    onkey(start_isort, "i")
rpm-build 2bd099
    onkey(start_ssort, "s")
rpm-build 2bd099
    onkey(start_qsort, "q")
rpm-build 2bd099
    onkey(randomize, "r")
rpm-build 2bd099
    onkey(bye, "space")
rpm-build 2bd099
rpm-build 2bd099
def main():
rpm-build 2bd099
    getscreen().clearscreen()
rpm-build 2bd099
    ht(); penup()
rpm-build 2bd099
    init_shelf()
rpm-build 2bd099
    show_text(instructions1)
rpm-build 2bd099
    show_text(instructions2, line=1)
rpm-build 2bd099
    enable_keys()
rpm-build 2bd099
    listen()
rpm-build 2bd099
    return "EVENTLOOP"
rpm-build 2bd099
rpm-build 2bd099
instructions1 = "press i for insertion sort, s for selection sort, q for quicksort"
rpm-build 2bd099
instructions2 = "spacebar to quit, r to randomize"
rpm-build 2bd099
rpm-build 2bd099
if __name__=="__main__":
rpm-build 2bd099
    msg = main()
rpm-build 2bd099
    mainloop()