Ir al contenido principal

Ralsina.Me — El sitio web de Roberto Alsina

Afeitando Yaks: septiembre 21, 2009

¡Y mis yaks son muy pe­lu­dos!

Em­pe­cé tra­tan­do de ha­cer un edi­tor de ho­jas de es­ti­lo pa­ra rs­t2­pdf (ver aquí).

Una co­sa lle­va a otra, y aho­ra ten­go por lo me­nos tres mi­ni-­pro­yec­tos in­te­re­san­tes por cul­pa de ése.

El de ho­y: (a­b)u­sar pyg­men­ts co­mo re­sal­ta­dor ge­né­ri­co de sin­ta­xis en una in­ter­fa­ce Qt.

¿Por­qué pyg­men­ts? Por­que es el úni­co re­sal­ta­dor de reS­truc­tu­red Text que en­contré. ¡E­so es pro­ba­ble­men­te por­que el reSt es muy di­fí­cil de re­sal­ta­r!

Has­ta adon­de sé es­ta es la pri­me­ra vez que al­guien hi­zo an­dar pyg­men­ts pa­ra es­to, en una ven­ta­na edi­ta­ble. Y hay muy bue­nos mo­ti­vos pa­ra eso:

  • Es pu­­ro py­­tho­­n, tal vez uno es­­pe­­ra que sea de­­ma­­sia­­do len­­to.

  • No ha­­ce le­­xing pa­r­­cial ni pro­­­gre­­si­­vo, hay que "lexea­­r" to­­­do el tex­­to jun­­to (de nue­­vo, eso pue­­de pa­­re­­cer muy len­­to)

  • Tie­­ne una API orien­­ta­­da a ar­­chi­­vo­­s, ge­­ne­­ra un ar­­chi­­vo con to­­­do el tex­­to fo­r­­ma­­tea­­do aden­­tro, y pa­­ra es­­ta cla­­se de uso se ne­­ce­­si­­ta ac­­ce­­der a da­­tos "del me­­dio" so­­­la­­men­­te.

Y, por su­pues­to, re­sul­ta que fun­cio­na bas­tan­te bien, co­mo se ve en es­te vi­deo:

Lec­ción del día: las com­pu­ta­do­ras aho­ra son rá­pi­das.

Acá es­tá el có­di­go pa­ra hi­gh­li­gh­te­r.­py con mu­chos co­men­ta­rio­s.

Se pue­de co­rrer y se ob­tie­ne la mis­ma de­mo que se ve en el vi­deo (s­al­vo el ti­peo ;-)

Jugando con poppler, PyQt y rst2pdf

Tam­bién pue­de lle­gar a ser úti­l, por ejem­plo, co­mo una for­ma de pro­bar cam­bios de sty­les­hee­t, ha­cien­do rs­t2­pdf más fá­cil de usar.

Des­pués de un par de ho­ras de ha­cking tran­qui­lo­... no es­tá na­da mal. Im­ple­men­té el (rús­ti­co) vi­sor PDF usan­do un bin­ding py­tho­n/­po­pple­r/­Qt que en­contré en google, la in­ter­fa­ce es Py­Q­t.

Acá es­tá el vi­deo:

Una no­ta: el vi­deo lo gra­bé usa­do qt-­re­cord­m­y­desk­to­p, y ese pro­gra­ma es ge­nia­l. Fue fa­ci­lí­si­mo de ha­ce­r.

No creo que fun­cio­ne muy bien con do­cu­men­tos lar­go­s, pe­ro el ma­nual de rs­t2­pdf (u­nas 25 pá­gi­na­s) se pro­ce­sa en 5 se­gun­do­s.

Una pregunta inocente...

To­do em­pe­zó cuan­do "Ga­li­leo Ga­li­lei" pre­gun­tó co­mo ha­cer una co­sa muy sim­ple. Él mos­tró es­te có­di­go:

age = int(raw_input('Cual es tu edad?'))
if age<18:
    print 'Sos menor'
else:
    print 'Sos mayor'

En rea­li­dad el ori­gi­na­l... era un po­co más gua­ran­go, pe­ro el có­di­go es bá­si­ca­men­te el mis­mo. has­ta ahí na­da ra­ro. Pe­ro en­ton­ces pre­gun­tó es­to:

Có­mo pue­do ha­cer que si el usua­rio en­tra al­go que no es un nú­me­ro, ha­ga al­go ti­po:

print 'No tengo poderes de supervaca'

o

print 'Error de tipeo'

Uno se ima­gi­na­ría que esa cla­se de pre­gun­ta pue­de pro­du­cir una o dos res­pues­ta­s. ¿No?

Efec­ti­va­men­te así es, y se ven las res­pues­tas de Fa­cun­do Ba­tis­ta o Eze­quiel.

Pe­ro­... que pa­sa­ría si que­re­mos se­guir pre­gun­tan­do cuan­do el usua­rio en­tra un no­-­nú­me­ro?

En­ton­ces ami­go­s... es una cues­tión de gus­to, y es to­do cul­pa de Juan Pe­dro Fi­sano­tti.

Acá es­tá mi idea:

while True:
    edad=raw_input('¿Cuantos años tenes?')
    if edad.isdigit():
        break
    print 'No ingresaste un numero!'

Sí, lo ad­mi­to, un po­co a la an­ti­gua. Y hu­bu gri­tos de "no, break es una por­que­ría, no es­tá bien", lo que lle­va a és­to, de Ma­nuel Aráoz

age = raw_input('Tu edad?')
while not age.isdigit():
    print "No es un número!"
    age = raw_input('Tu edad?')

Lo que lle­va a llan­tos de "Te­ner dos raw_i­n­put es feo­!", lo que a su vez pro­vo­ca es­to (nue­va­men­te, Ma­nuel Aráoz:

get_age = lambda: raw_input('Tu edad?')
age = get_age()
while not age.isdigit():
    print 'No es un numero!'
    age = get_age()

Acá Pa­tri­cio Mo­li­na pe­la la PEP 315.

Y en­ton­ces Ale­jan­dro San­tos di­ce al­go co­mo "Es­to es más fa­cil en C por­que po­de­mos asig­nar va­lo­res a edad en la con­di­ción del whi­le". Acuér­den­se de es­to.

Aho­ra Pa­blo Zi­llia­ni da su ver­sió­n, que, de­bo de­ci­r, es per­fec­ta en cier­ta for­ma:

age = reset = msg = 'Edad?: '
while not age.isdigit():
    age = raw_input(msg)
    msg = "%r no es un numero!, %s" % (age, reset)

print age

En­ton­ces Ga­briel Ge­ne­lli­na de­ci­de de­fen­der el uso de break pe­gán­do­nos a to­dos en la ca­be­za con Knu­th lo que de­be­ría te­ner un efec­to mu­cho más po­ten­te que men­cio­nar a Hi­tle­r.

Y va­mos lle­gan­do a aguas po­co na­ve­ga­ble­s. Acá es­tá la pro­pues­ta de news , que ad­mi­ro. A una res­pe­tuo­sa dis­tan­cia.

Pri­me­ro el có­di­go re­le­van­te:

edad = "0" # Entra igual la primera vez

while firstTrue (not edad.isdigit()):
    edad = raw_input ("¿Cuantos años tenes? ")
    if not edad.isdigit():
        print "No ingresaste un nro!"

¿Pe­ro qué, exac­ta­men­te, es firs­tTrue?

import inspect

def firstTrue(cond):
    """ devuelve True siempre la primera vez que se la ejecuta,
    las veces subsiguientes evalua la condicion """
    stack = inspect.stack()[1] # El stack del programa llamador
    line = stack[2] # Nro de linea desde la que llame a firstTrue
    del stack

    if not "line" in firstTrue.__dict__:
        # Primera vez que llamo a la funcion
        firstTrue.line = line
        return True
    elif firstTrue.line != line:
        # Llame a la funcion desde otro punto del programa
        firstTrue.line = line
    return True

    return cond

En­ton­ces yo men­ciono ge­ne­ra­do­res, lo que lle­va a es­to, por Clau­dio Frei­re, que ca­si fun­cio­na:

age = ''
def invalidAge():
    yield True
    while not age.isdigit():
        print "Not a number"
        yield True
    yield False

for i in invalidAge():
    age = raw_input("Age please: ")

print age

Y en­ton­ces Fa­bian Ga­lli­na es el se­gun­do en men­cio­nar que en C po­dés asig­nar en con­di­cio­nes.

No pien­so acep­tar­lo. C no pue­de ser más fá­cil pa­ra es­to!

Así que con una ayu­di­ta del cook­book...

edad=[1]

while not edad |asig| raw_input('Edad? '):
    print u'Poné un número!'

print u'Tenes %s años'%edad[0]

¿Pero qué es |asig|? ¡Qué buena pregunta!

class Infix:
    def __init__(self, function):
        self.function = function
    def __ror__(self, other):
        return Infix(lambda x, self=self, other=other: self.function(other, x))
    def __or__(self, other):
        return self.function(other)
    def __rlshift__(self, other):
        return Infix(lambda x, self=self, other=other: self.function(other, x))
    def __rshift__(self, other):
        return self.function(other)
    def __call__(self, value1, value2):
        return self.function(value1, value2)

def opasigna (x,y):
    x[0]=y
    return y.isdigit()

asig=Infix(opasigna)

Y en­ton­ce­s, Pa­blo pos­tea es­ta jo­yi­ta:

import inspect

def assign(var, value):
    stack = inspect.stack()[1][0]
    stack.f_locals [var] = value
    del stack
    return value

while not assign("edad", raw_input('Edad? ')).isdigit():
    print u'No es un numero!'

print u'Tenes %s años' % edad

Que es, me pa­re­ce a mí, lo me­nos tri­vial que se pue­de lle­gar con es­te pro­ble­ma. Cla­ro que el hi­lo no se mu­rió to­da­vía ;-)

Unas fotos más de Pycon Argentina 2009

Es­tas son del pa­nel Py­Q­t+W­x­P­y­tho­n+­P­y­Gtk (py­q­t.­pdf):

Charla PyQt+Wx+PyGtk 1Charla PyQt+Wx+PyGtk 2Charla PyQt+Wx+PyGtk 3

Y es­ta es de la fa­lli­da char­la re­lám­pa­go so­bre la "ho­ja de cál­cu­lo" de He­ttin­ger (ho­ja.­pdf):

Charla Relampago Hoja de Calculo

To­das las fo­tos por Ariel Kan­te­rewi­cz (Kan­t) li­cen­cia CC-b­y.

Machetes: RST y Python

Si te pa­sa eso, o al­go pa­re­ci­do, pro­bá el nue­vo ma­che­te de rst, son dos ca­ri­llas A4.

Es una ver­sión pre­li­mi­na­r, las co­lum­nas iz­quier­das les fal­ta mu­cho tra­ba­jo.

Tam­bién, ya que es­ta­mo­s, ca­paz que quie­ras mi­rar el ma­che­te de py­thon.

Los dos es­tán crea­dos usan­do ReS­truc­tu­red Text y rs­t2­pdf, y li­cen­cia­dos ba­jo Crea­ti­ve Co­m­mon­s.


Contents © 2000-2023 Roberto Alsina