IOBinding.py 20.2 KB
Newer Older
David Scherer's avatar
David Scherer committed
1 2 3 4 5 6 7 8
# changes by dscherer@cmu.edu
#   - IOBinding.open() replaces the current window with the opened file,
#     if the current window is both unmodified and unnamed
#   - IOBinding.loadfile() interprets Windows, UNIX, and Macintosh
#     end-of-line conventions, instead of relying on the standard library,
#     which will only understand the local convention.

import os
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
9 10 11
import types
import sys
import codecs
12
import tempfile
David Scherer's avatar
David Scherer committed
13 14 15
import tkFileDialog
import tkMessageBox
import re
16 17
from Tkinter import *
from SimpleDialog import SimpleDialog
David Scherer's avatar
David Scherer committed
18

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
19
from configHandler import idleConf
David Scherer's avatar
David Scherer committed
20

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
21 22 23 24 25 26 27 28 29 30 31
try:
    from codecs import BOM_UTF8
except ImportError:
    # only available since Python 2.3
    BOM_UTF8 = '\xef\xbb\xbf'

# Try setting the locale, so that we can find out
# what encoding to use
try:
    import locale
    locale.setlocale(locale.LC_CTYPE, "")
32
except (ImportError, locale.Error):
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
33 34
    pass

35 36 37
# Encoding for file names
filesystemencoding = sys.getfilesystemencoding()

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
encoding = "ascii"
if sys.platform == 'win32':
    # On Windows, we could use "mbcs". However, to give the user
    # a portable encoding name, we need to find the code page
    try:
        encoding = locale.getdefaultlocale()[1]
        codecs.lookup(encoding)
    except LookupError:
        pass
else:
    try:
        # Different things can fail here: the locale module may not be
        # loaded, it may not offer nl_langinfo, or CODESET, or the
        # resulting codeset may be unknown to Python. We ignore all
        # these problems, falling back to ASCII
        encoding = locale.nl_langinfo(locale.CODESET)
54
        if encoding is None or encoding is '':
55 56
            # situation occurs on Mac OS X
            encoding = 'ascii'
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
57 58 59 60 61 62 63
        codecs.lookup(encoding)
    except (NameError, AttributeError, LookupError):
        # Try getdefaultlocale well: it parses environment variables,
        # which may give a clue. Unfortunately, getdefaultlocale has
        # bugs that can cause ValueError.
        try:
            encoding = locale.getdefaultlocale()[1]
64
            if encoding is None or encoding is '':
65 66
                # situation occurs on Mac OS X
                encoding = 'ascii'
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
67 68 69 70 71 72 73 74
            codecs.lookup(encoding)
        except (ValueError, LookupError):
            pass

encoding = encoding.lower()

coding_re = re.compile("coding[:=]\s*([-\w_.]+)")

75 76 77 78
class EncodingMessage(SimpleDialog):
    "Inform user that an encoding declaration is needed."
    def __init__(self, master, enc):
        self.should_edit = False
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
        self.root = top = Toplevel(master)
        top.bind("<Return>", self.return_event)
        top.bind("<Escape>", self.do_ok)
        top.protocol("WM_DELETE_WINDOW", self.wm_delete_window)
        top.wm_title("I/O Warning")
        top.wm_iconname("I/O Warning")
        self.top = top

        l1 = Label(top,
            text="Non-ASCII found, yet no encoding declared. Add a line like")
        l1.pack(side=TOP, anchor=W)
        l2 = Entry(top, font="courier")
        l2.insert(0, "# -*- coding: %s -*-" % enc)
        # For some reason, the text is not selectable anymore if the
        # widget is disabled.
        # l2['state'] = DISABLED
        l2.pack(side=TOP, anchor = W, fill=X)
        l3 = Label(top, text="to your file\n"
                   "Choose OK to save this file as %s\n"
                   "Edit your general options to silence this warning" % enc)
        l3.pack(side=TOP, anchor = W)

        buttons = Frame(top)
        buttons.pack(side=TOP, fill=X)
        # Both return and cancel mean the same thing: do nothing
        self.default = self.cancel = 0
        b1 = Button(buttons, text="Ok", default="active",
                    command=self.do_ok)
        b1.pack(side=LEFT, fill=BOTH, expand=1)
        b2 = Button(buttons, text="Edit my file",
                    command=self.do_edit)
        b2.pack(side=LEFT, fill=BOTH, expand=1)
112

113 114 115 116 117 118 119 120
        self._set_transient(master)

    def do_ok(self):
        self.done(0)

    def do_edit(self):
        self.done(1)

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
121
def coding_spec(str):
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
122 123
    """Return the encoding declaration according to PEP 263.

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
124 125
    Raise LookupError if the encoding is declared but unknown.
    """
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
    # Only consider the first two lines
    str = str.split("\n")[:2]
    str = "\n".join(str)

    match = coding_re.search(str)
    if not match:
        return None
    name = match.group(1)
    # Check whether the encoding is known
    import codecs
    try:
        codecs.lookup(name)
    except LookupError:
        # The standard encoding error does not indicate the encoding
        raise LookupError, "Unknown encoding "+name
    return name
David Scherer's avatar
David Scherer committed
142

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
143

David Scherer's avatar
David Scherer committed
144 145 146 147 148 149 150 151 152 153 154
class IOBinding:

    def __init__(self, editwin):
        self.editwin = editwin
        self.text = editwin.text
        self.__id_open = self.text.bind("<<open-window-from-file>>", self.open)
        self.__id_save = self.text.bind("<<save-window>>", self.save)
        self.__id_saveas = self.text.bind("<<save-window-as-file>>",
                                          self.save_as)
        self.__id_savecopy = self.text.bind("<<save-copy-of-window-as-file>>",
                                            self.save_a_copy)
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
155
        self.fileencoding = None
156
        self.__id_print = self.text.bind("<<print-window>>", self.print_window)
157

David Scherer's avatar
David Scherer committed
158 159 160 161 162 163
    def close(self):
        # Undo command bindings
        self.text.unbind("<<open-window-from-file>>", self.__id_open)
        self.text.unbind("<<save-window>>", self.__id_save)
        self.text.unbind("<<save-window-as-file>>",self.__id_saveas)
        self.text.unbind("<<save-copy-of-window-as-file>>", self.__id_savecopy)
164
        self.text.unbind("<<print-window>>", self.__id_print)
David Scherer's avatar
David Scherer committed
165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184
        # Break cycles
        self.editwin = None
        self.text = None
        self.filename_change_hook = None

    def get_saved(self):
        return self.editwin.get_saved()

    def set_saved(self, flag):
        self.editwin.set_saved(flag)

    def reset_undo(self):
        self.editwin.reset_undo()

    filename_change_hook = None

    def set_filename_change_hook(self, hook):
        self.filename_change_hook = hook

    filename = None
185
    dirname = None
David Scherer's avatar
David Scherer committed
186 187

    def set_filename(self, filename):
188 189 190 191 192 193 194 195 196
        if filename and os.path.isdir(filename):
            self.filename = None
            self.dirname = filename
        else:
            self.filename = filename
            self.dirname = None
            self.set_saved(1)
            if self.filename_change_hook:
                self.filename_change_hook()
David Scherer's avatar
David Scherer committed
197

198
    def open(self, event=None, editFile=None):
David Scherer's avatar
David Scherer committed
199
        if self.editwin.flist:
200 201 202 203
            if not editFile:
                filename = self.askopenfile()
            else:
                filename=editFile
David Scherer's avatar
David Scherer committed
204
            if filename:
205 206 207 208 209 210 211 212 213 214
                # If the current window has no filename and hasn't been
                # modified, we replace its contents (no loss).  Otherwise
                # we open a new window.  But we won't replace the
                # shell window (which has an interp(reter) attribute), which
                # gets set to "not modified" at every new prompt.
                try:
                    interp = self.editwin.interp
                except:
                    interp = None
                if not self.filename and self.get_saved() and not interp:
David Scherer's avatar
David Scherer committed
215 216 217 218 219 220
                    self.editwin.flist.open(filename, self.loadfile)
                else:
                    self.editwin.flist.open(filename)
            else:
                self.text.focus_set()
            return "break"
221
        #
David Scherer's avatar
David Scherer committed
222 223 224 225 226 227
        # Code for use outside IDLE:
        if self.get_saved():
            reply = self.maybesave()
            if reply == "cancel":
                self.text.focus_set()
                return "break"
228 229 230 231
        if not editFile:
            filename = self.askopenfile()
        else:
            filename=editFile
David Scherer's avatar
David Scherer committed
232 233 234 235 236 237
        if filename:
            self.loadfile(filename)
        else:
            self.text.focus_set()
        return "break"

238 239 240 241
    eol = r"(\r\n)|\n|\r"  # \r\n (Windows), \n (UNIX), or \r (Mac)
    eol_re = re.compile(eol)
    eol_convention = os.linesep # Default

David Scherer's avatar
David Scherer committed
242 243 244 245 246 247 248 249 250
    def loadfile(self, filename):
        try:
            # open the file in binary mode so that we can handle
            #   end-of-line convention ourselves.
            f = open(filename,'rb')
            chars = f.read()
            f.close()
        except IOError, msg:
            tkMessageBox.showerror("I/O Error", str(msg), master=self.text)
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
251
            return False
David Scherer's avatar
David Scherer committed
252

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
253
        chars = self.decode(chars)
David Scherer's avatar
David Scherer committed
254
        # We now convert all end-of-lines to '\n's
255 256 257
        firsteol = self.eol_re.search(chars)
        if firsteol:
            self.eol_convention = firsteol.group(0)
258 259 260
            if isinstance(self.eol_convention, unicode):
                # Make sure it is an ASCII string
                self.eol_convention = self.eol_convention.encode("ascii")
261
            chars = self.eol_re.sub(r"\n", chars)
David Scherer's avatar
David Scherer committed
262 263 264 265 266 267 268 269

        self.text.delete("1.0", "end")
        self.set_filename(None)
        self.text.insert("1.0", chars)
        self.reset_undo()
        self.set_filename(filename)
        self.text.mark_set("insert", "1.0")
        self.text.see("insert")
270
        self.updaterecentfileslist(filename)
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
271 272 273
        return True

    def decode(self, chars):
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
274
        """Create a Unicode string
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
275

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
276 277
        If that fails, let Tcl try its best
        """
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316
        # Check presence of a UTF-8 signature first
        if chars.startswith(BOM_UTF8):
            try:
                chars = chars[3:].decode("utf-8")
            except UnicodeError:
                # has UTF-8 signature, but fails to decode...
                return chars
            else:
                # Indicates that this file originally had a BOM
                self.fileencoding = BOM_UTF8
                return chars
        # Next look for coding specification
        try:
            enc = coding_spec(chars)
        except LookupError, name:
            tkMessageBox.showerror(
                title="Error loading the file",
                message="The encoding '%s' is not known to this Python "\
                "installation. The file may not display correctly" % name,
                master = self.text)
            enc = None
        if enc:
            try:
                return unicode(chars, enc)
            except UnicodeError:
                pass
        # If it is ASCII, we need not to record anything
        try:
            return unicode(chars, 'ascii')
        except UnicodeError:
            pass
        # Finally, try the locale's encoding. This is deprecated;
        # the user should declare a non-ASCII encoding
        try:
            chars = unicode(chars, encoding)
            self.fileencoding = encoding
        except UnicodeError:
            pass
        return chars
David Scherer's avatar
David Scherer committed
317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342

    def maybesave(self):
        if self.get_saved():
            return "yes"
        message = "Do you want to save %s before closing?" % (
            self.filename or "this untitled document")
        m = tkMessageBox.Message(
            title="Save On Close",
            message=message,
            icon=tkMessageBox.QUESTION,
            type=tkMessageBox.YESNOCANCEL,
            master=self.text)
        reply = m.show()
        if reply == "yes":
            self.save(None)
            if not self.get_saved():
                reply = "cancel"
        self.text.focus_set()
        return reply

    def save(self, event):
        if not self.filename:
            self.save_as(event)
        else:
            if self.writefile(self.filename):
                self.set_saved(1)
343 344 345 346
                try:
                    self.editwin.store_file_breaks()
                except AttributeError:  # may be a PyShell
                    pass
David Scherer's avatar
David Scherer committed
347 348 349 350 351 352 353 354 355
        self.text.focus_set()
        return "break"

    def save_as(self, event):
        filename = self.asksavefile()
        if filename:
            if self.writefile(filename):
                self.set_filename(filename)
                self.set_saved(1)
356 357 358 359
                try:
                    self.editwin.store_file_breaks()
                except AttributeError:
                    pass
David Scherer's avatar
David Scherer committed
360
        self.text.focus_set()
361
        self.updaterecentfileslist(filename)
David Scherer's avatar
David Scherer committed
362 363 364 365 366 367 368
        return "break"

    def save_a_copy(self, event):
        filename = self.asksavefile()
        if filename:
            self.writefile(filename)
        self.text.focus_set()
369
        self.updaterecentfileslist(filename)
David Scherer's avatar
David Scherer committed
370 371 372 373
        return "break"

    def writefile(self, filename):
        self.fixlastline()
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
374
        chars = self.encode(self.text.get("1.0", "end-1c"))
375 376
        if self.eol_convention != "\n":
            chars = chars.replace("\n", self.eol_convention)
David Scherer's avatar
David Scherer committed
377
        try:
378
            f = open(filename, "wb")
David Scherer's avatar
David Scherer committed
379
            f.write(chars)
380
            f.flush()
David Scherer's avatar
David Scherer committed
381
            f.close()
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
382
            return True
David Scherer's avatar
David Scherer committed
383 384 385
        except IOError, msg:
            tkMessageBox.showerror("I/O Error", str(msg),
                                   master=self.text)
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431
            return False

    def encode(self, chars):
        if isinstance(chars, types.StringType):
            # This is either plain ASCII, or Tk was returning mixed-encoding
            # text to us. Don't try to guess further.
            return chars
        # See whether there is anything non-ASCII in it.
        # If not, no need to figure out the encoding.
        try:
            return chars.encode('ascii')
        except UnicodeError:
            pass
        # If there is an encoding declared, try this first.
        try:
            enc = coding_spec(chars)
            failed = None
        except LookupError, msg:
            failed = msg
            enc = None
        if enc:
            try:
                return chars.encode(enc)
            except UnicodeError:
                failed = "Invalid encoding '%s'" % enc
        if failed:
            tkMessageBox.showerror(
                "I/O Error",
                "%s. Saving as UTF-8" % failed,
                master = self.text)
        # If there was a UTF-8 signature, use that. This should not fail
        if self.fileencoding == BOM_UTF8 or failed:
            return BOM_UTF8 + chars.encode("utf-8")
        # Try the original file encoding next, if any
        if self.fileencoding:
            try:
                return chars.encode(self.fileencoding)
            except UnicodeError:
                tkMessageBox.showerror(
                    "I/O Error",
                    "Cannot save this as '%s' anymore. Saving as UTF-8" \
                    % self.fileencoding,
                    master = self.text)
                return BOM_UTF8 + chars.encode("utf-8")
        # Nothing was declared, and we had not determined an encoding
        # on loading. Recommend an encoding line.
432 433 434 435 436 437 438 439 440 441
        config_encoding = idleConf.GetOption("main","EditorWindow",
                                             "encoding")
        if config_encoding == 'utf-8':
            # User has requested that we save files as UTF-8
            return BOM_UTF8 + chars.encode("utf-8")
        ask_user = True
        try:
            chars = chars.encode(encoding)
            enc = encoding
            if config_encoding == 'locale':
442
                ask_user = False
443 444 445 446
        except UnicodeError:
            chars = BOM_UTF8 + chars.encode("utf-8")
            enc = "utf-8"
        if not ask_user:
447
            return chars
448 449 450 451 452 453 454 455 456 457 458 459 460
        dialog = EncodingMessage(self.editwin.top, enc)
        dialog.go()
        if dialog.num == 1:
            # User asked us to edit the file
            encline = "# -*- coding: %s -*-\n" % enc
            firstline = self.text.get("1.0", "2.0")
            if firstline.startswith("#!"):
                # Insert encoding after #! line
                self.text.insert("2.0", encline)
            else:
                self.text.insert("1.0", encline)
            return self.encode(self.text.get("1.0", "end-1c"))
        return chars
461

Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
462 463 464 465 466
    def fixlastline(self):
        c = self.text.get("end-2c")
        if c != '\n':
            self.text.insert("end-1c", "\n")

467 468
    def print_window(self, event):
        tempfilename = None
469 470
        saved = self.get_saved()
        if saved:
471
            filename = self.filename
472 473 474
        # shell undo is reset after every prompt, looks saved, probably isn't
        if not saved or filename is None:
            # XXX KBK 08Jun03 Wouldn't it be better to ask the user to save?
475 476 477 478
            (tfd, tempfilename) = tempfile.mkstemp(prefix='IDLE_tmp_')
            filename = tempfilename
            os.close(tfd)
            if not self.writefile(tempfilename):
479 480 481 482 483
                os.unlink(tempfilename)
                return "break"
        platform=os.name
        printPlatform=1
        if platform == 'posix': #posix platform
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
484 485
            command = idleConf.GetOption('main','General',
                                         'print-command-posix')
486 487 488 489 490 491 492 493
            command = command + " 2>&1"
        elif platform == 'nt': #win32 platform
            command = idleConf.GetOption('main','General','print-command-win')
        else: #no printing for this platform
            printPlatform=0
        if printPlatform:  #we can try to print for this platform
            command = command % filename
            pipe = os.popen(command, "r")
494
            # things can get ugly on NT if there is no printer available.
495 496 497
            output = pipe.read().strip()
            status = pipe.close()
            if status:
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
498 499
                output = "Printing failed (exit status 0x%x)\n" % \
                         status + output
500 501 502 503
            if output:
                output = "Printing command: %s\n" % repr(command) + output
                tkMessageBox.showerror("Print status", output, master=self.text)
        else:  #no printing for this platform
504
            message="Printing is not enabled for this platform: %s" % platform
505
            tkMessageBox.showinfo("Print status", message, master=self.text)
506 507
        if tempfilename:
            os.unlink(tempfilename)
508
        return "break"
509

David Scherer's avatar
David Scherer committed
510 511 512 513 514 515 516 517 518 519 520 521 522 523
    opendialog = None
    savedialog = None

    filetypes = [
        ("Python and text files", "*.py *.pyw *.txt", "TEXT"),
        ("All text files", "*", "TEXT"),
        ("All files", "*"),
        ]

    def askopenfile(self):
        dir, base = self.defaultfilename("open")
        if not self.opendialog:
            self.opendialog = tkFileDialog.Open(master=self.text,
                                                filetypes=self.filetypes)
524 525 526 527
        filename = self.opendialog.show(initialdir=dir, initialfile=base)
        if isinstance(filename, unicode):
            filename = filename.encode(filesystemencoding)
        return filename
David Scherer's avatar
David Scherer committed
528 529 530 531

    def defaultfilename(self, mode="open"):
        if self.filename:
            return os.path.split(self.filename)
532 533
        elif self.dirname:
            return self.dirname, ""
David Scherer's avatar
David Scherer committed
534 535 536 537 538 539 540 541 542 543 544 545
        else:
            try:
                pwd = os.getcwd()
            except os.error:
                pwd = ""
            return pwd, ""

    def asksavefile(self):
        dir, base = self.defaultfilename("save")
        if not self.savedialog:
            self.savedialog = tkFileDialog.SaveAs(master=self.text,
                                                  filetypes=self.filetypes)
546 547 548 549
        filename = self.savedialog.show(initialdir=dir, initialfile=base)
        if isinstance(filename, unicode):
            filename = filename.encode(filesystemencoding)
        return filename
David Scherer's avatar
David Scherer committed
550

551
    def updaterecentfileslist(self,filename):
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
552
        "Update recent file list on all editor windows"
Kurt B. Kaiser's avatar
Kurt B. Kaiser committed
553
        self.editwin.update_recent_files_list(filename)
554

David Scherer's avatar
David Scherer committed
555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584
def test():
    root = Tk()
    class MyEditWin:
        def __init__(self, text):
            self.text = text
            self.flist = None
            self.text.bind("<Control-o>", self.open)
            self.text.bind("<Control-s>", self.save)
            self.text.bind("<Alt-s>", self.save_as)
            self.text.bind("<Alt-z>", self.save_a_copy)
        def get_saved(self): return 0
        def set_saved(self, flag): pass
        def reset_undo(self): pass
        def open(self, event):
            self.text.event_generate("<<open-window-from-file>>")
        def save(self, event):
            self.text.event_generate("<<save-window>>")
        def save_as(self, event):
            self.text.event_generate("<<save-window-as-file>>")
        def save_a_copy(self, event):
            self.text.event_generate("<<save-copy-of-window-as-file>>")
    text = Text(root)
    text.pack()
    text.focus_set()
    editwin = MyEditWin(text)
    io = IOBinding(editwin)
    root.mainloop()

if __name__ == "__main__":
    test()