404

[ Avaa Bypassed ]




Upload:

Command:

elspacio@18.117.11.13: ~ $
"""
An auto-completion window for IDLE, used by the AutoComplete extension
"""
from Tkinter import *
from idlelib.MultiCall import MC_SHIFT
from idlelib.AutoComplete import COMPLETE_FILES, COMPLETE_ATTRIBUTES

HIDE_VIRTUAL_EVENT_NAME = "<<autocompletewindow-hide>>"
HIDE_SEQUENCES = ("<FocusOut>", "<ButtonPress>")
KEYPRESS_VIRTUAL_EVENT_NAME = "<<autocompletewindow-keypress>>"
# We need to bind event beyond <Key> so that the function will be called
# before the default specific IDLE function
KEYPRESS_SEQUENCES = ("<Key>", "<Key-BackSpace>", "<Key-Return>", "<Key-Tab>",
                      "<Key-Up>", "<Key-Down>", "<Key-Home>", "<Key-End>",
                      "<Key-Prior>", "<Key-Next>")
KEYRELEASE_VIRTUAL_EVENT_NAME = "<<autocompletewindow-keyrelease>>"
KEYRELEASE_SEQUENCE = "<KeyRelease>"
LISTUPDATE_SEQUENCE = "<B1-ButtonRelease>"
WINCONFIG_SEQUENCE = "<Configure>"
DOUBLECLICK_SEQUENCE = "<B1-Double-ButtonRelease>"

class AutoCompleteWindow:

    def __init__(self, widget):
        # The widget (Text) on which we place the AutoCompleteWindow
        self.widget = widget
        # The widgets we create
        self.autocompletewindow = self.listbox = self.scrollbar = None
        # The default foreground and background of a selection. Saved because
        # they are changed to the regular colors of list items when the
        # completion start is not a prefix of the selected completion
        self.origselforeground = self.origselbackground = None
        # The list of completions
        self.completions = None
        # A list with more completions, or None
        self.morecompletions = None
        # The completion mode. Either AutoComplete.COMPLETE_ATTRIBUTES or
        # AutoComplete.COMPLETE_FILES
        self.mode = None
        # The current completion start, on the text box (a string)
        self.start = None
        # The index of the start of the completion
        self.startindex = None
        # The last typed start, used so that when the selection changes,
        # the new start will be as close as possible to the last typed one.
        self.lasttypedstart = None
        # Do we have an indication that the user wants the completion window
        # (for example, he clicked the list)
        self.userwantswindow = None
        # event ids
        self.hideid = self.keypressid = self.listupdateid = self.winconfigid \
        = self.keyreleaseid = self.doubleclickid                         = None
        # Flag set if last keypress was a tab
        self.lastkey_was_tab = False

    def _change_start(self, newstart):
        min_len = min(len(self.start), len(newstart))
        i = 0
        while i < min_len and self.start[i] == newstart[i]:
            i += 1
        if i < len(self.start):
            self.widget.delete("%s+%dc" % (self.startindex, i),
                               "%s+%dc" % (self.startindex, len(self.start)))
        if i < len(newstart):
            self.widget.insert("%s+%dc" % (self.startindex, i),
                               newstart[i:])
        self.start = newstart

    def _binary_search(self, s):
        """Find the first index in self.completions where completions[i] is
        greater or equal to s, or the last index if there is no such
        one."""
        i = 0; j = len(self.completions)
        while j > i:
            m = (i + j) // 2
            if self.completions[m] >= s:
                j = m
            else:
                i = m + 1
        return min(i, len(self.completions)-1)

    def _complete_string(self, s):
        """Assuming that s is the prefix of a string in self.completions,
        return the longest string which is a prefix of all the strings which
        s is a prefix of them. If s is not a prefix of a string, return s."""
        first = self._binary_search(s)
        if self.completions[first][:len(s)] != s:
            # There is not even one completion which s is a prefix of.
            return s
        # Find the end of the range of completions where s is a prefix of.
        i = first + 1
        j = len(self.completions)
        while j > i:
            m = (i + j) // 2
            if self.completions[m][:len(s)] != s:
                j = m
            else:
                i = m + 1
        last = i-1

        if first == last: # only one possible completion
            return self.completions[first]

        # We should return the maximum prefix of first and last
        first_comp = self.completions[first]
        last_comp = self.completions[last]
        min_len = min(len(first_comp), len(last_comp))
        i = len(s)
        while i < min_len and first_comp[i] == last_comp[i]:
            i += 1
        return first_comp[:i]

    def _selection_changed(self):
        """Should be called when the selection of the Listbox has changed.
        Updates the Listbox display and calls _change_start."""
        cursel = int(self.listbox.curselection()[0])

        self.listbox.see(cursel)

        lts = self.lasttypedstart
        selstart = self.completions[cursel]
        if self._binary_search(lts) == cursel:
            newstart = lts
        else:
            min_len = min(len(lts), len(selstart))
            i = 0
            while i < min_len and lts[i] == selstart[i]:
                i += 1
            newstart = selstart[:i]
        self._change_start(newstart)

        if self.completions[cursel][:len(self.start)] == self.start:
            # start is a prefix of the selected completion
            self.listbox.configure(selectbackground=self.origselbackground,
                                   selectforeground=self.origselforeground)
        else:
            self.listbox.configure(selectbackground=self.listbox.cget("bg"),
                                   selectforeground=self.listbox.cget("fg"))
            # If there are more completions, show them, and call me again.
            if self.morecompletions:
                self.completions = self.morecompletions
                self.morecompletions = None
                self.listbox.delete(0, END)
                for item in self.completions:
                    self.listbox.insert(END, item)
                self.listbox.select_set(self._binary_search(self.start))
                self._selection_changed()

    def show_window(self, comp_lists, index, complete, mode, userWantsWin):
        """Show the autocomplete list, bind events.
        If complete is True, complete the text, and if there is exactly one
        matching completion, don't open a list."""
        # Handle the start we already have
        self.completions, self.morecompletions = comp_lists
        self.mode = mode
        self.startindex = self.widget.index(index)
        self.start = self.widget.get(self.startindex, "insert")
        if complete:
            completed = self._complete_string(self.start)
            start = self.start
            self._change_start(completed)
            i = self._binary_search(completed)
            if self.completions[i] == completed and \
               (i == len(self.completions)-1 or
                self.completions[i+1][:len(completed)] != completed):
                # There is exactly one matching completion
                return completed == start
        self.userwantswindow = userWantsWin
        self.lasttypedstart = self.start

        # Put widgets in place
        self.autocompletewindow = acw = Toplevel(self.widget)
        # Put it in a position so that it is not seen.
        acw.wm_geometry("+10000+10000")
        # Make it float
        acw.wm_overrideredirect(1)
        try:
            # This command is only needed and available on Tk >= 8.4.0 for OSX
            # Without it, call tips intrude on the typing process by grabbing
            # the focus.
            acw.tk.call("::tk::unsupported::MacWindowStyle", "style", acw._w,
                        "help", "noActivates")
        except TclError:
            pass
        self.scrollbar = scrollbar = Scrollbar(acw, orient=VERTICAL)
        self.listbox = listbox = Listbox(acw, yscrollcommand=scrollbar.set,
                                         exportselection=False, bg="white")
        for item in self.completions:
            listbox.insert(END, item)
        self.origselforeground = listbox.cget("selectforeground")
        self.origselbackground = listbox.cget("selectbackground")
        scrollbar.config(command=listbox.yview)
        scrollbar.pack(side=RIGHT, fill=Y)
        listbox.pack(side=LEFT, fill=BOTH, expand=True)
        acw.lift()  # work around bug in Tk 8.5.18+ (issue #24570)

        # Initialize the listbox selection
        self.listbox.select_set(self._binary_search(self.start))
        self._selection_changed()

        # bind events
        self.hideid = self.widget.bind(HIDE_VIRTUAL_EVENT_NAME,
                                       self.hide_event)
        for seq in HIDE_SEQUENCES:
            self.widget.event_add(HIDE_VIRTUAL_EVENT_NAME, seq)
        self.keypressid = self.widget.bind(KEYPRESS_VIRTUAL_EVENT_NAME,
                                           self.keypress_event)
        for seq in KEYPRESS_SEQUENCES:
            self.widget.event_add(KEYPRESS_VIRTUAL_EVENT_NAME, seq)
        self.keyreleaseid = self.widget.bind(KEYRELEASE_VIRTUAL_EVENT_NAME,
                                             self.keyrelease_event)
        self.widget.event_add(KEYRELEASE_VIRTUAL_EVENT_NAME,KEYRELEASE_SEQUENCE)
        self.listupdateid = listbox.bind(LISTUPDATE_SEQUENCE,
                                         self.listselect_event)
        self.winconfigid = acw.bind(WINCONFIG_SEQUENCE, self.winconfig_event)
        self.doubleclickid = listbox.bind(DOUBLECLICK_SEQUENCE,
                                          self.doubleclick_event)

    def winconfig_event(self, event):
        if not self.is_active():
            return
        # Position the completion list window
        text = self.widget
        text.see(self.startindex)
        x, y, cx, cy = text.bbox(self.startindex)
        acw = self.autocompletewindow
        acw_width, acw_height = acw.winfo_width(), acw.winfo_height()
        text_width, text_height = text.winfo_width(), text.winfo_height()
        new_x = text.winfo_rootx() + min(x, max(0, text_width - acw_width))
        new_y = text.winfo_rooty() + y
        if (text_height - (y + cy) >= acw_height # enough height below
            or y < acw_height): # not enough height above
            # place acw below current line
            new_y += cy
        else:
            # place acw above current line
            new_y -= acw_height
        acw.wm_geometry("+%d+%d" % (new_x, new_y))

    def hide_event(self, event):
        if not self.is_active():
            return
        self.hide_window()

    def listselect_event(self, event):
        if not self.is_active():
            return
        self.userwantswindow = True
        cursel = int(self.listbox.curselection()[0])
        self._change_start(self.completions[cursel])

    def doubleclick_event(self, event):
        # Put the selected completion in the text, and close the list
        cursel = int(self.listbox.curselection()[0])
        self._change_start(self.completions[cursel])
        self.hide_window()

    def keypress_event(self, event):
        if not self.is_active():
            return
        keysym = event.keysym
        if hasattr(event, "mc_state"):
            state = event.mc_state
        else:
            state = 0
        if keysym != "Tab":
            self.lastkey_was_tab = False
        if (len(keysym) == 1 or keysym in ("underscore", "BackSpace")
            or (self.mode == COMPLETE_FILES and keysym in
                ("period", "minus"))) \
           and not (state & ~MC_SHIFT):
            # Normal editing of text
            if len(keysym) == 1:
                self._change_start(self.start + keysym)
            elif keysym == "underscore":
                self._change_start(self.start + '_')
            elif keysym == "period":
                self._change_start(self.start + '.')
            elif keysym == "minus":
                self._change_start(self.start + '-')
            else:
                # keysym == "BackSpace"
                if len(self.start) == 0:
                    self.hide_window()
                    return
                self._change_start(self.start[:-1])
            self.lasttypedstart = self.start
            self.listbox.select_clear(0, int(self.listbox.curselection()[0]))
            self.listbox.select_set(self._binary_search(self.start))
            self._selection_changed()
            return "break"

        elif keysym == "Return":
            self.hide_window()
            return

        elif (self.mode == COMPLETE_ATTRIBUTES and keysym in
              ("period", "space", "parenleft", "parenright", "bracketleft",
               "bracketright")) or \
             (self.mode == COMPLETE_FILES and keysym in
              ("slash", "backslash", "quotedbl", "apostrophe")) \
             and not (state & ~MC_SHIFT):
            # If start is a prefix of the selection, but is not '' when
            # completing file names, put the whole
            # selected completion. Anyway, close the list.
            cursel = int(self.listbox.curselection()[0])
            if self.completions[cursel][:len(self.start)] == self.start \
               and (self.mode == COMPLETE_ATTRIBUTES or self.start):
                self._change_start(self.completions[cursel])
            self.hide_window()
            return

        elif keysym in ("Home", "End", "Prior", "Next", "Up", "Down") and \
             not state:
            # Move the selection in the listbox
            self.userwantswindow = True
            cursel = int(self.listbox.curselection()[0])
            if keysym == "Home":
                newsel = 0
            elif keysym == "End":
                newsel = len(self.completions)-1
            elif keysym in ("Prior", "Next"):
                jump = self.listbox.nearest(self.listbox.winfo_height()) - \
                       self.listbox.nearest(0)
                if keysym == "Prior":
                    newsel = max(0, cursel-jump)
                else:
                    assert keysym == "Next"
                    newsel = min(len(self.completions)-1, cursel+jump)
            elif keysym == "Up":
                newsel = max(0, cursel-1)
            else:
                assert keysym == "Down"
                newsel = min(len(self.completions)-1, cursel+1)
            self.listbox.select_clear(cursel)
            self.listbox.select_set(newsel)
            self._selection_changed()
            self._change_start(self.completions[newsel])
            return "break"

        elif (keysym == "Tab" and not state):
            if self.lastkey_was_tab:
                # two tabs in a row; insert current selection and close acw
                cursel = int(self.listbox.curselection()[0])
                self._change_start(self.completions[cursel])
                self.hide_window()
                return "break"
            else:
                # first tab; let AutoComplete handle the completion
                self.userwantswindow = True
                self.lastkey_was_tab = True
                return

        elif any(s in keysym for s in ("Shift", "Control", "Alt",
                                       "Meta", "Command", "Option")):
            # A modifier key, so ignore
            return

        else:
            # Unknown event, close the window and let it through.
            self.hide_window()
            return

    def keyrelease_event(self, event):
        if not self.is_active():
            return
        if self.widget.index("insert") != \
           self.widget.index("%s+%dc" % (self.startindex, len(self.start))):
            # If we didn't catch an event which moved the insert, close window
            self.hide_window()

    def is_active(self):
        return self.autocompletewindow is not None

    def complete(self):
        self._change_start(self._complete_string(self.start))
        # The selection doesn't change.

    def hide_window(self):
        if not self.is_active():
            return

        # unbind events
        for seq in HIDE_SEQUENCES:
            self.widget.event_delete(HIDE_VIRTUAL_EVENT_NAME, seq)
        self.widget.unbind(HIDE_VIRTUAL_EVENT_NAME, self.hideid)
        self.hideid = None
        for seq in KEYPRESS_SEQUENCES:
            self.widget.event_delete(KEYPRESS_VIRTUAL_EVENT_NAME, seq)
        self.widget.unbind(KEYPRESS_VIRTUAL_EVENT_NAME, self.keypressid)
        self.keypressid = None
        self.widget.event_delete(KEYRELEASE_VIRTUAL_EVENT_NAME,
                                 KEYRELEASE_SEQUENCE)
        self.widget.unbind(KEYRELEASE_VIRTUAL_EVENT_NAME, self.keyreleaseid)
        self.keyreleaseid = None
        self.listbox.unbind(LISTUPDATE_SEQUENCE, self.listupdateid)
        self.listupdateid = None
        self.autocompletewindow.unbind(WINCONFIG_SEQUENCE, self.winconfigid)
        self.winconfigid = None

        # destroy widgets
        self.scrollbar.destroy()
        self.scrollbar = None
        self.listbox.destroy()
        self.listbox = None
        self.autocompletewindow.destroy()
        self.autocompletewindow = None

Filemanager

Name Type Size Permission Actions
Icons Folder 0755
idle_test Folder 0755
AutoComplete.py File 8.75 KB 0644
AutoComplete.pyc File 7.99 KB 0644
AutoComplete.pyo File 7.99 KB 0644
AutoCompleteWindow.py File 16.91 KB 0644
AutoCompleteWindow.pyc File 12.42 KB 0644
AutoCompleteWindow.pyo File 12.36 KB 0644
AutoExpand.py File 3.32 KB 0644
AutoExpand.pyc File 3.5 KB 0644
AutoExpand.pyo File 3.5 KB 0644
Bindings.py File 2.91 KB 0644
Bindings.pyc File 4.6 KB 0644
Bindings.pyo File 4.6 KB 0644
CREDITS.txt File 1.82 KB 0644
CallTipWindow.py File 5.92 KB 0644
CallTipWindow.pyc File 6.14 KB 0644
CallTipWindow.pyo File 6.14 KB 0644
CallTips.py File 7.56 KB 0644
CallTips.pyc File 8.16 KB 0644
CallTips.pyo File 8.16 KB 0644
ChangeLog File 55.07 KB 0644
ClassBrowser.py File 6.83 KB 0644
ClassBrowser.pyc File 9.66 KB 0644
ClassBrowser.pyo File 9.66 KB 0644
CodeContext.py File 8.15 KB 0644
CodeContext.pyc File 6.63 KB 0644
CodeContext.pyo File 6.59 KB 0644
ColorDelegator.py File 9.53 KB 0644
ColorDelegator.pyc File 8.91 KB 0644
ColorDelegator.pyo File 8.91 KB 0644
Debugger.py File 17.81 KB 0644
Debugger.pyc File 17.72 KB 0644
Debugger.pyo File 17.72 KB 0644
Delegator.py File 665 B 0644
Delegator.pyc File 1.31 KB 0644
Delegator.pyo File 1.31 KB 0644
EditorWindow.py File 63.96 KB 0644
EditorWindow.pyc File 57.15 KB 0644
EditorWindow.pyo File 57.05 KB 0644
FileList.py File 3.63 KB 0644
FileList.pyc File 4.07 KB 0644
FileList.pyo File 4.04 KB 0644
FormatParagraph.py File 7.12 KB 0644
FormatParagraph.pyc File 7.14 KB 0644
FormatParagraph.pyo File 7.14 KB 0644
GrepDialog.py File 5.02 KB 0644
GrepDialog.pyc File 6.45 KB 0644
GrepDialog.pyo File 6.45 KB 0644
HISTORY.txt File 10.08 KB 0644
HyperParser.py File 10.25 KB 0644
HyperParser.pyc File 6.64 KB 0644
HyperParser.pyo File 6.64 KB 0644
IOBinding.py File 21.4 KB 0644
IOBinding.pyc File 18.62 KB 0644
IOBinding.pyo File 18.62 KB 0644
IdleHistory.py File 3.96 KB 0644
IdleHistory.pyc File 4.05 KB 0644
IdleHistory.pyo File 4.05 KB 0644
MultiCall.py File 17.29 KB 0644
MultiCall.pyc File 16.4 KB 0644
MultiCall.pyo File 16.33 KB 0644
MultiStatusBar.py File 1.32 KB 0644
MultiStatusBar.pyc File 2.3 KB 0644
MultiStatusBar.pyo File 2.3 KB 0644
NEWS.txt File 46.14 KB 0644
ObjectBrowser.py File 4.27 KB 0644
ObjectBrowser.pyc File 7.26 KB 0644
ObjectBrowser.pyo File 7.26 KB 0644
OutputWindow.py File 4.47 KB 0644
OutputWindow.pyc File 5.3 KB 0644
OutputWindow.pyo File 5.3 KB 0644
ParenMatch.py File 6.56 KB 0644
ParenMatch.pyc File 7.18 KB 0644
ParenMatch.pyo File 7.18 KB 0644
PathBrowser.py File 2.94 KB 0644
PathBrowser.pyc File 4.57 KB 0644
PathBrowser.pyo File 4.57 KB 0644
Percolator.py File 3.15 KB 0644
Percolator.pyc File 4.69 KB 0644
Percolator.pyo File 4.51 KB 0644
PyParse.py File 19.05 KB 0644
PyParse.pyc File 9.99 KB 0644
PyParse.pyo File 9.56 KB 0644
PyShell.py File 57.48 KB 0755
PyShell.pyc File 53 KB 0644
PyShell.pyo File 52.9 KB 0644
README.txt File 7.71 KB 0644
RemoteDebugger.py File 11.36 KB 0644
RemoteDebugger.pyc File 16.73 KB 0644
RemoteDebugger.pyo File 16.58 KB 0644
RemoteObjectBrowser.py File 942 B 0644
RemoteObjectBrowser.pyc File 2.23 KB 0644
RemoteObjectBrowser.pyo File 2.23 KB 0644
ReplaceDialog.py File 6.48 KB 0644
ReplaceDialog.pyc File 7.83 KB 0644
ReplaceDialog.pyo File 7.83 KB 0644
RstripExtension.py File 1.03 KB 0644
RstripExtension.pyc File 1.63 KB 0644
RstripExtension.pyo File 1.63 KB 0644
ScriptBinding.py File 8.26 KB 0644
ScriptBinding.pyc File 8.17 KB 0644
ScriptBinding.pyo File 8.17 KB 0644
ScrolledList.py File 4.27 KB 0644
ScrolledList.pyc File 6.61 KB 0644
ScrolledList.pyo File 6.61 KB 0644
SearchDialog.py File 2.57 KB 0644
SearchDialog.pyc File 4.04 KB 0644
SearchDialog.pyo File 4.04 KB 0644
SearchDialogBase.py File 6.93 KB 0644
SearchDialogBase.pyc File 8.43 KB 0644
SearchDialogBase.pyo File 8.43 KB 0644
SearchEngine.py File 7.29 KB 0644
SearchEngine.pyc File 8.37 KB 0644
SearchEngine.pyo File 8.37 KB 0644
StackViewer.py File 4.33 KB 0644
StackViewer.pyc File 6.52 KB 0644
StackViewer.pyo File 6.52 KB 0644
TODO.txt File 8.28 KB 0644
ToolTip.py File 3.1 KB 0644
ToolTip.pyc File 4.78 KB 0644
ToolTip.pyo File 4.78 KB 0644
TreeWidget.py File 14.68 KB 0644
TreeWidget.pyc File 17.95 KB 0644
TreeWidget.pyo File 17.95 KB 0644
UndoDelegator.py File 10.53 KB 0644
UndoDelegator.pyc File 13.88 KB 0644
UndoDelegator.pyo File 13.88 KB 0644
WidgetRedirector.py File 6.74 KB 0644
WidgetRedirector.pyc File 7.76 KB 0644
WidgetRedirector.pyo File 7.76 KB 0644
WindowList.py File 2.42 KB 0644
WindowList.pyc File 3.74 KB 0644
WindowList.pyo File 3.74 KB 0644
ZoomHeight.py File 1.27 KB 0644
ZoomHeight.pyc File 1.67 KB 0644
ZoomHeight.pyo File 1.67 KB 0644
__init__.py File 288 B 0644
__init__.pyc File 444 B 0644
__init__.pyo File 444 B 0644
aboutDialog.py File 6.85 KB 0644
aboutDialog.pyc File 6.85 KB 0644
aboutDialog.pyo File 6.85 KB 0644
config-extensions.def File 2.9 KB 0644
config-highlight.def File 2.46 KB 0644
config-keys.def File 7.59 KB 0644
config-main.def File 2.5 KB 0644
configDialog.py File 64.41 KB 0644
configDialog.pyc File 53.16 KB 0644
configDialog.pyo File 53.16 KB 0644
configHandler.py File 31.72 KB 0644
configHandler.pyc File 29.28 KB 0644
configHandler.pyo File 29.28 KB 0644
configHelpSourceEdit.py File 6.53 KB 0644
configHelpSourceEdit.pyc File 6.55 KB 0644
configHelpSourceEdit.pyo File 6.55 KB 0644
configSectionNameDialog.py File 3.95 KB 0644
configSectionNameDialog.pyc File 4.4 KB 0644
configSectionNameDialog.pyo File 4.4 KB 0644
dynOptionMenuWidget.py File 1.94 KB 0644
dynOptionMenuWidget.pyc File 2.8 KB 0644
dynOptionMenuWidget.pyo File 2.8 KB 0644
extend.txt File 3.56 KB 0644
help.html File 41.42 KB 0644
help.py File 10.78 KB 0644
help.pyc File 12.25 KB 0644
help.pyo File 12.25 KB 0644
help.txt File 11.86 KB 0644
idle.py File 453 B 0644
idle.pyc File 423 B 0644
idle.pyo File 423 B 0644
idle.pyw File 563 B 0644
idlever.py File 415 B 0644
idlever.pyc File 591 B 0644
idlever.pyo File 591 B 0644
keybindingDialog.py File 12.18 KB 0644
keybindingDialog.pyc File 12.08 KB 0644
keybindingDialog.pyo File 12.08 KB 0644
macosxSupport.py File 8.24 KB 0644
macosxSupport.pyc File 8.38 KB 0644
macosxSupport.pyo File 8.25 KB 0644
rpc.py File 19.68 KB 0644
rpc.pyc File 21.97 KB 0644
rpc.pyo File 21.86 KB 0644
run.py File 12.61 KB 0644
run.pyc File 13.45 KB 0644
run.pyo File 13.39 KB 0644
tabbedpages.py File 18.01 KB 0644
tabbedpages.pyc File 18.63 KB 0644
tabbedpages.pyo File 18.63 KB 0644
textView.py File 3.44 KB 0644
textView.pyc File 4.02 KB 0644
textView.pyo File 4.02 KB 0644