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 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 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 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 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 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 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571
|
## Fonty Python Copyright (C) 2006, 2007, 2008, 2009 Donn.C.Ingle
## Contact: donn.ingle@gmail.com - I hope this email lasts.
##
## This file is part of Fonty Python.
## Fonty Python is free software: you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation, either version 3 of the License, or
## (at your option) any later version.
##
## Fonty Python is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
##
## You should have received a copy of the GNU General Public License
## along with Fonty Python. If not, see <http://www.gnu.org/licenses/>.
## fpsys : fonty python system.
## I debated calling it fpglobals.
## This is a common-ground for variables and defs that will be used from
## other modules - so they are global to everything.
import sys, os, pickle
import linux_safe_path_library
LSP = linux_safe_path_library.linuxSafePath()
import pathcontrol
import strings
import fontcontrol
import charmaps
import wx
import subprocess
## Oct 2009 Default Font Family (System font)
DFAM=None # Set in wxgui.py in class App()
## Ensure we have a .fontypython folder and a .fonts folder.
iPC = pathcontrol.PathControl() #Make an instance - hence the small 'i' (Boy this convention *sure* lasted....)
## Borrowed from wxglade.py
## The reason for this is to find the path of this file
## when it's called from an import somewhere else.
## There is no sys.argv[0] in this case.
root = __file__
if os.path.islink(root):
root = os.path.realpath(root)
fontyroot = os.path.dirname(os.path.abspath(root))
## Where my images and things are.
mythingsdir = os.path.join(sys.prefix, 'share', 'fontypython', 'things') + '/'
## Sept 2009
class Overlaperize(object):
'''
If a single font is in many pogs, then we count each 'overlap' and
control the removal of them until there are no overlaps anymore.
i.e. When no other installed pogs are using the font, it is safe to
remove the link (should that last pog be removed by the user).
'''
def __init__(self):
self.OVERLAP_COUNT_DICT = {}
self.DISABLE_THIS = False # In case all this makes a horrible mess : users can flip this to True....
def inc(self,key):
if self.DISABLE_THIS: return True
if key in self.OVERLAP_COUNT_DICT:
self.OVERLAP_COUNT_DICT[key] += 1
else:
self.OVERLAP_COUNT_DICT[key] = 2 #starts at 2 because there is already one installed.
#self.report(key)
return True
def dec(self,key):
'''
Return True means : This font overlaps
Return False means : This font can be uninstalled
'''
if self.DISABLE_THIS: return False
if key in self.OVERLAP_COUNT_DICT:
self.OVERLAP_COUNT_DICT[key] -= 1
if self.OVERLAP_COUNT_DICT[key] == 0:
del self.OVERLAP_COUNT_DICT[key]
return False # it does NOT overlap anymore.
else:
#self.report(key)
return True # It still overlaps
# It gets if the font is totally unknown to the OVERLAP_COUNT_DICT
return False #It therefore does NOT overlap.
def report(self,key):
print "%s has overlap count of %d" % (key, self.OVERLAP_COUNT_DICT[key])
def sleep(self):
'''Save the OVERLAP_COUNT_DICT to a file (if it has content). Called when app closes.'''
if self.DISABLE_THIS: return
if not self.OVERLAP_COUNT_DICT:
self.OVERLAP_COUNT_DICT={} # Ensure there is a blank overlap_counts file!
paf = os.path.join(iPC.appPath(),"overlap_counts")
fr = open( paf, 'wb' ) # pickle says use 'binary' files, but only Windows makes this distinction. I use it to be safe...
pickle.dump( self.OVERLAP_COUNT_DICT, fr, protocol=pickle.HIGHEST_PROTOCOL )
fr.close()
def wakeup(self):
'''Restore the OVERLAP_COUNT_DICT from a file (if any). Called as app starts.'''
if self.DISABLE_THIS: return
paf = os.path.join(iPC.appPath(),"overlap_counts")
if os.path.exists( paf ):
fr = open( paf, "rb" )
self.OVERLAP_COUNT_DICT = pickle.load( fr )
fr.close()
## start it up!
Overlap = Overlaperize()
Overlap.wakeup()
## Jan 18 2008
segfonts = []# Global var
def getSegfontsList():
"""Runs (below) on startup"""
## On startup, open the 'segfonts' file and keep a list in RAM
## This file is written by the 'check' routine.
global segfonts
paf = os.path.join(iPC.appPath(),"segfonts")
try:
if os.path.exists( paf ):
fr = open( paf, 'r' ) # byte string only ascii file
segfonts = fr.read().split("\n")
fr.close()
except:
## CORNER CASE: Some error or other.
raise
## Call it.
getSegfontsList()
def checkFonts( dirtocheck, printer ):
"""
Jan 18 2008
Scan a tree for fonts that can cause segfaults.
Write a file 'segfonts' and create a list 'segfonts'
that gets checked to exclude them.
printer is a function of some kind.
Can be called from the cli or the gui.
"""
global segfonts
code = """
import ImageFont
try:
font=ImageFont.truetype("%s", 24, 0)
dud=font.getname()
except:
pass
"""
def checkForSegfault( pafbytestring ):
## Uses Ajaksu's idea : 17 Jan 2008. Thanks!
segfault_I_hope = False
## I have ignored ALL catchable errors (see code var)
## This is because I want to (try to) only catch SEGFAULTS
## and leave all other flavours of font-related errors to
## the fontcontrol module -- where fonts are still useable
## if not visible.
retval = subprocess.call( ["python", '-c', code % pafbytestring] )
if retval != 0:
segfault_I_hope = True
return segfault_I_hope
printer ( _("Checking fonts, this could take some time.") )
printer ( _("Starting in %s:") % dirtocheck )
printer ()
## dirtocheck comes-in as unicode - let's stick to byte strings:
dirtocheck = LSP.to_bytes( dirtocheck )
seglist = [] # our local list of newly found bad fonts
gotsome = False
for cwd, dirs, files in os.walk( dirtocheck ):
printer(_("Looking in %s...") % os.path.basename(cwd) )
## We only want certain font files:
fontfiles = [f for f in files if f.upper().endswith( ("TTF","TTC","PFA","PFB","OTF")) ]
if len(fontfiles) < 1:
printer (_("No supported fonts found there..."))
printer()
for file in fontfiles:
paf = os.path.join( cwd, file )
bad = checkForSegfault( paf )
if bad:
gotsome = True
seglist.append( paf )
printer ( " " + file ) # show it on-screen somewhere.
if not gotsome:
printer(_("I could not find any bad fonts."))
## Now write the segfonts file:
if seglist:
## Add the new fonts found to the ones in global segfonts list
for bf in seglist:
segfonts.append(bf)
## Now remove duplicates
tmp = list( set( segfonts ) )
segfonts = tmp
del (tmp)
## Now save it.
paf = os.path.join(iPC.appPath(),"segfonts")
fw = open( paf, "w" ) # byte string ascii
bytestring = "".join([line + "\n" for line in segfonts if line != ""])
#print "about to write bytestring:"
#print [bytestring]
#print
fw.write( bytestring )
fw.close()
printer()
printer(_("The process is complete."))
def isFolder(thing):
"""True if a folder. False if not - but that does not mean it's a pog."""
if os.path.isdir(thing): return True
return False
def isPog(thing):
"""True if a Pog. False if not."""
## thing comes in as UNICODE
## iPC.getPogNames() is a list of BYTE STRINGS
## We must encode thing to a byte string to avoid warnings:
if LSP.to_bytes( thing ) in iPC.getPogNames(): #getPogNames contains byte strings!
return True
if thing == "EMPTY": return True #Special case
return False
class FPState:
"""The global vars to hold the state of the situation."""
def __init__(self):
## Contains the Pog or Folder being viewed
self.viewobject = None
## Refs the view object *after* the filter has been applied
self.filteredViewObject= None
## Contains a Pog (or None) that is the Target
self.targetobject = None
## Represents the situation in a letter code
## P for Pog, F for Folder, E for Empty, N for None
self.viewpattern = ""
self.targetpattern = ""
## Will be "NOTHING_TO_DO", "REMOVE" or "APPEND" (Add fonts to Pog)
self.action = ""
## Can an item be ticked
self.cantick = None
## The View and Target pogs chosen are the same.
self.samepogs = False
## How many tick marks.
self.numticks = 0
state = FPState() #The only instance of the state object -- app-wide
####
## Save and Load the conf file
class Configure:
"""Makes/Loads the conf file.
Supplies size, pos, numinpage, text string and point size to other objects."""
def __init__(self) :
## Private vars
self.__dontSaveNumInPage = False
## PUBLIC vars : Set some defaults:
self.size = (800,600)
self.pos = (10, 10)
self.numinpage = 10
self.text = _("Jump the lazy dog fox")
self.points = 64
self.lastview = "EMPTY" # a pog name or a folder path.
self.usegui = "wxgui"
self.max = True
self.lastdir = iPC.home()
## Added Dec 2007
self.leftSash = 200
self.rightSash = 128
## Added June 2009
self.recurseFolders = False
## Added Sept 2009
self.ignore_adjustments = False
## Added 3 Oct 2009
self.app_char_map = "UNSET" # A string of an app name.
self.__setData()
## Oct 2009 -- The Character Map Controller.
self.CMC = charmaps.CharMapController( self.app_char_map_set )
if os.path.exists(iPC.appConf()):
try:
pf = open(iPC.appConf(), "rb" ) # Binary for new pickle protocol.
self.__data = pickle.load( pf )
pf.close()
except:
## Dec 2007 : Let's try erase and rewind
os.unlink(iPC.appConf())
if not os.path.exists(iPC.appConf()):
print _("No config file found, creating it with defaults.")
self.Save()
## Now get them into the instance vars:
try:
self.size = self.__data['size']
self.pos = self.__data['pos']
self.numinpage = self.__data['numinpage']
self.text = self.__data['text']
self.points= self.__data['points']
self.lastview = self.__data['lastview']
self.usegui = self.__data['usegui']
self.max = self.__data['max']
self.lastdir = self.__data['lastdir']
self.leftSash = self.__data['leftSash']
self.rightSash = self.__data['rightSash']
self.recurseFolders = self.__data['recurseFolders']
self.ignore_adjustments = self.__data['ignore_adjustments']
self.app_char_map = self.__data['app_char_map']
## We must also set our instance of the Char Map Controller:
## This can be "UNSET" (default first run) or an appname
## That appname may be valid or not (it may have been uninstalled...)
self.CMC.SET_CURRENT_APPNAME(self.app_char_map)
except KeyError:
## The conf file has keys that don't work for this version, chances are it's old.
## Let's delete and re-make it.
try:
os.unlink(iPC.appConf())
except:
print _("The fontypython config file is damaged.\nPlease remove it and start again")
raise SystemExit
self.Save()
def dontSaveNumInPage(self, flag):
self.__dontSaveNumInPage = flag
def __setData(self):
self.__data = {"size" : self.size,
"pos" : self.pos,
"numinpage" : self.numinpage,
"text" : self.text,
"points" : self.points,
"lastview" : self.lastview,
"usegui" : self.usegui,
"max" : self.max,
"lastdir" : self.lastdir,
"leftSash" : self.leftSash,
"rightSash" : self.rightSash,
"recurseFolders": self.recurseFolders,
"ignore_adjustments": self.ignore_adjustments,
"app_char_map" : self.app_char_map
}
def app_char_map_set( self, x ):
'''
A callback from the CharMapController: when the CURRENT_APPNAME is set,
this gets called to keep the config version of the appname current.
'''
self.app_char_map = x
def Save(self) :
#If we are NOT to save the numinpage, then fetch it from what was there before.
if self.__dontSaveNumInPage:
self.numinpage = self.__data["numinpage"]
self.__setData()
try:
pf = open( iPC.appConf(), "wb" )
pickle.dump(self.__data, pf, protocol = pickle.HIGHEST_PROTOCOL )
pf.close()
except IOError:
print _("Could not write to the config file.")
Overlap.sleep() #sept 2009 : Save the OVERLAP_COUNT_DICT
## Our config instance - it will have one instance across
## all the modules that use it.
config = Configure()
def instantiateViewFolder( foldername, recurse=None ):
"""
Creates a Folder object and fills it with FontItem objects
according to what's in the folder's path.
This is the VIEW - i.e. what you are looking at.
"""
if state.viewobject: del state.viewobject
## Default assumptions in case of raised error.
state.viewobject = fontcontrol.EmptyView()
state.viewpattern = "E"
#July 2016
#=========
# Made recurse default to None in the def sig.
# This has the effect of allowing THREE states to enter:
# None, True, False
# None means the call came from cli.py
# If so, we want to fetch the recurse from config -
# so it becomes either T or F, depending on last state.
# This has stopped that initial Schroedinger's Cat state
# of the recurse setting.
if recurse is None:
recurse=config.recurseFolders
else:
config.recurseFolders = recurse
#print "recurse:", recurse
ifolder = fontcontrol.Folder(foldername, recurse) #raises : fontybugs.FolderHasNoFonts : BENIGN ERROR.
## Only continues if there is no problem.
state.viewobject = ifolder
## Because we have a new view object, we must reset the last filteredViewObject
state.filteredViewObject = None
config.lastview = foldername
state.viewpattern = "F"
markInactive()
flushTicks()
def instantiateViewPog( newpog_name ):
"""
Given a Pog Name string, make a Pog object.
This is the VIEW - i.e. what you are looking at.
A VIEW Pog can be EMPTY. This happens on the first run when there is no config file.
There are other arcane situations too, but I forget.
"""
## print "COMES IN to instantiateViewPog:"
## print "newpog_name:", newpog_name
## print "type(newpog_name):", type(newpog_name)
#if state.viewobject: del state.viewobject
if state.viewobject: state.viewobject = None
if newpog_name == "EMPTY":
ipog = fontcontrol.EmptyView()
else:
ipog = fontcontrol.Pog( newpog_name )
## Test TARGETPOG to see if this is the same pogname
## The not None test is for first run - there is no targetobject yet just after cli.py calls us, so we
## do not want to access it or we get NoneType errors.
if state.targetobject is not None and state.targetobject.name == newpog_name:
state.samepogs = True
else:
state.samepogs = False
## Must gen the Pog to get a count of items:
## Errors raised in genList (and company):
## fontybugs.PogInvalid (only valid from cli pov)
##
## We 'handle' this by NOT catching it, pass it up.
ipog.genList()
## Continue if all ok.
state.viewobject = ipog
## Because we have a new view object, we must reset the last filteredViewObject
state.filteredViewObject = None
config.lastview = newpog_name
if len(state.viewobject) == 0:
empty = True
state.viewpattern = "E"
else:
empty = False
state.viewpattern = "P"
markInactive()
flushTicks()
#print "instantiateViewPog says viewpattern is:", state.viewpattern
return empty # this return is only used in cli.py
def instantiateTargetPog( newpog_name ):
"""
The app could begin with NO TARGET POG chosen.
After that (in the gui) either a pog is chosen or NO POG is chosen (i.e. None)
Therefore - there can NEVER BE a targetobject called EMPTY
The CLI install/uninstall/purge DO NOT use this routine.
"""
if state.targetobject: del state.targetobject
ipog = fontcontrol.Pog(newpog_name)
## Must gen the Pog to get a count of items:
ipog.genList() # Raises fontybugs.PogInvalid error THIS ENDS THE APP.
## TEST the viewobject which is the stuff being
## LOOKED AT IN THE MIDDLE OF THE SCREEN (which could be a Pog OR a Folder)
## If it's a Pog then we may have chosen the same Pog (on the right)
## that we are looking at, so check that:
state.samepogs = False
if isinstance( state.viewobject, fontcontrol.Pog ):
if state.viewobject.name == newpog_name:
## The pog clicked in the TARGET is the same as what's ALREADY selected in the VIEW
state.samepogs = True
quickinstalledflag = False
if ipog.isInstalled(): quickinstalledflag = True
state.targetpattern = "P"
state.targetobject = ipog
markInactive()
flushTicks()
return quickinstalledflag
def markInactive():
"""
INACTIVE means the font displayed is already inside the
chosen target pog. So, it's not 'active', not clickable etc.
Mark each font item as inactive, as needs-be.
Clear the ticks.
Sets the message to display in the fontmap.
"""
if state.viewobject: state.viewobject.clearInactiveflags()
if state.viewobject and state.targetobject:
## What's in TARGET must be inactive in VIEW
## pafBlist is a list of UNICODEs
## glyphpaf_unicode is UNICODE, so I will use it instead
## because we compare it to pafBlist
pafBlist = [i.glyphpaf_unicode for i in state.targetobject]
for iA in state.viewobject:
if iA.glyphpaf_unicode in pafBlist:
iA.activeInactiveMsg = _("This font is in %s") % state.targetobject.name
iA.inactive = True
del pafBlist
def SetTargetPogToNone():
state.targetobject = None
state.targetpattern = "N"
def SetViewPogToEmpty():
state.viewobject = fontcontrol.EmptyView()
state.viewpattern = "E"
def flushTicks():
for fi in state.viewobject:
fi.ticked = False
state.numticks = 0
def logSegfaulters( lastPaf ):
"""
Writes a string to ~/.fontypython/lastFontBeforeSegfault
"""
paf = os.path.join( iPC.appPath(),"lastFontBeforeSegfault")
try:
f = open( paf, "w" )
lastPaf = LSP.ensure_bytes( lastPaf )
f.write( lastPaf + "\n" )
f.close()
except:
raise
|