Mac workflow: klikkmentes to do

Régen nagy örömmel használtam OS X alatt a Things alkalmazást. Ez egész odáig tartott, míg a fejlesztő srácok el nem kezdték ígérgetni a cloud sync támogatást, ami csak nem akart megérkezni - én pedig továbbléptem, előbb a The Hit List-re, aztán az OS X / iOS gyári Reminders alkalmazására, majd a német Wunderlist appra. Mindegyikkel volt valami bajom, aztán nemrég szembejött egy review, amiben újra nyakig dícsérték a Things-et, ezért úgy gondoltam, visszanézek - és nem bántam meg :)

Amivel már jó ideje kísérletezek az az, hogy a munka közben eszembe jutó teendőket minél kevesebb erőfeszítéssel rögzíthessem. Erre optimális lenne, ha a feladatkezelő alkalmazás korrektül scriptelhető lenne, én meg egy Alfred workflow-val beletolhatnám a mondandómat - ezt fogjuk most megcsinálni a Things mellé.

A Things-hez létezik már többféle Alfred workflow, én viszont egyikkel sem voltam elégedett. Olyat szerettem volna, amihez egy sorban kitölthetem a taskhoz tartozó összes paramétert mindenféle egerezés nélkül. Ehhez még hozzáálmodtam azt is, hogy felszedhetné az aktív Safari ablak URL-jét és az oldal címét, hátha csak épp azt akarom elmenteni. Tudom, van Reading list Safariban, amit viszont nem tudok taggelni és főleg fejlesztői doksiknál sokszor élem át azt, hogy emiatt nem találok meg egy korábban látott oldalt.

Szóval legyen benne task description meg task note kitaláló, de gyorsan állíthassak határidőt, feladatcsoportot és tageket is. Kitaláltam ehhez a következő szintaxist:

things-workflow-sample

Az eleje a feladat leírása, a @ után álló "3d print" a feladatcsoport (a Things nyelvén "project vagy area of responsibility"), a # utáni "Ultimaker" a feladatra akasztott címke, a > karaktert követő 1 érték pedig azt mondja a workflownak, hogy ez a munka holnap határidős. Mindezt látod is on-the-fly, ahogy gépeled be a feladatot. Csak elsőre hangzik bonyolultan, 5 task kézzel és már ott is van az izommemóriában :)

Ez pedig a Safari autofill - így néz ki, amikor felszedi az aktív Safari ablak címét és URL-jét:

things-workflow-safari-autofill-sample

A > karakter utáni dátummal tud minimálisat matekozni. Megadhatsz neki 8 (=YYYYMMDD), 6 (=YYMMDD), 4 (=MMDD), vagy 2 (=DD) jegyű dátumot - ha nem teljes a dátum, akkor a hiányzó darabokat az aktuális napi dátumból kiegészíti. Magyarul a >1201 ma 2014 december elsejét jelenti a workflownak.
Ezen túlmenően tud hozzáadni illetve elvenni napokat az aktuális dátumból: az első karakter az előjel, azt követi egy szám, majd az utolsó karakter opcionálisan jelölhet nap helyett hetet (w), hónapot (m) vagy évet (y). Egy példán keresztül: a mai napon gyártott to do-n a >+3w dátummatek 2014 december 17-ét állítja be due date-nek, mivel az van 3 hét múlva.

Ha tetszik és kipróbálnád, a workflow-t szedheted tőlem vagy a packal.org-ról is.

Update: kicsit szellősebbre vettem a parser által kitalált to do elemeket, illetve most már felismeri a @today és @someday targeteket is és a megfelelő Things listákba pakolja az így címzett to do itemeket:

things-alfred-workflow-v1_1

Mac workflow: iTunes playlist tagger

A probléma

Elég régóta használok tageket OS X alatt - a gyári óperenciás rendszerben még nem léteztek, már gyártottam őket valami toollal, ami openmeta kompatibilis volt. Amióta a Finderrel is lehet taggelni, azóta nem értem, hogy az iTunes médiaszörnyeteg miért nincs ugyanerre felkészítve. Számtalan alkalommal került már elő, hogy mondjuk hangulat alapján hallgatnék zenét, vagy mondjuk csak egy szál vékony női énekhangot akarok hallani - ilyesfajta keresésre pedig semmi nem képes.

Léteznek mindenféle toolok, amik ezt próbálják segíteni valahogyan: a Tangerine pl. ütemet próbál számolni, a Moody pedig egy 4x4 osztású 2D sebesség vs. hangulat mátrixba soroltatná veled a trackeket.

A megoldás

Mindkét fent említett játékos a trackekbe irkálna valami custom metaadatot, én viszont kitaláltam valami sokkal ügyesebbet! Egy track tetszőleges számú playlistben szerepelhet - legyen hát a playlist neve a tag, amit a trackre akasztunk! Már csak egy olyan eszköz kell, ami az aktuálisan játszott tracket gyorsan bedobálja ezekbe a "tag" playlistekbe anélkül, hogy ehhez egerezni kellene meló közben.
Szerencsére a platformon létezik Alfred, a csodás productivity tool, ami szénné scriptelhető, az iTunes meg nagy haverja az AppleScriptnek, úgyhogy már csak össze kell engedni őket.

Azt találtam ki, hogy ezek a speciális, tag-ként funkcionáló playlistek egy közös, "Tags" nevű playlist folderben fognak lakni és a nevük @ prefixxel kezdődik majd, így különböztetve meg őket a többi listától. Az Alfred workflow-nak pedig a következőket kell szolgáltatnia:

  • mondja meg, hogy épp mi szól
  • lássam egy listában az éppen lejátszott trackra összes releváns és irreleváns taget
  • egyszerre akárhány taget rápakolhassak az aktuális trackre

Nos, ilyen lett:

iTunes playlist tagger

Keyword vagy shortcut indítja. A listában zöld + ikonnal szerepelnek az aktuális trackre még nem ráakasztott, de egyébként létező tag-ek, piros - ikonnal pedig azok, amelyek már hozzá vannak rendelve. Értelemszerűen a pirosak törlik az adott tag-et, a zöldek hozzáadják.

Ez azonban a rendes power usernek még kevés lenne, mert mi van, ha több új tag-re van szükségünk? Ilyenkor jön az, hogy a "plst " kulcsszó után space-szel elválasztva irkálhatjuk a tageket. Ha a tag neve - prefixszel kezdődik, akkor az a tag törlődni fog, ha + karakterrel (a + nem kötelező, a nem mínusz első karakter automatikusan +-at jelent), akkor pedig hozzácsapjuk az adott tracket. Így akármennyi taget írhatunk egymás után, a tagger script szép sorban feldolgozza őket egyesével.

Na de nem lennék rendes kocka, ha nem néznénk meg a forrást. Itt van a script filter rész, ami felépíti a kiválasztó listát:

on run argv
	set q to argv as text
 
(*
	if not (itunes_is_running()) then 
		set xml to xml & linefeed & "<item uid=\"itunes_not_running\" arg=\"itunes_not_running\" valid=\"no\">" & linefeed
		set xml to xml & "<title>iTunes is not running</title>" & linefeed
		set xml to xml & "<subtitle>Start iTunes and retry</subtitle>" & linefeed
		set xml to xml & "<icon>now_paused.png</icon>" & linefeed
		set xml to xml & "</item>" & linefeed
 
		return xml & "</items>" & linefeed
	end if
*)
 
	-- pick up info from iTunes
	tell application "iTunes"
		set dbid to the database ID of the current track
		set artist_name to (get artist of the current track)
		set track_name to the name of the current track
		set pls to (get name of user playlists)
	end tell
 
	-- xml header
	set xml to "<?xml version=\"1.0\"?>" & linefeed
	set xml to xml & "<items>" & linefeed
 
	-- track info list item
	set xml to xml & linefeed & "<item uid=\"trackinfo\" arg=\"trackinfo\" valid=\"yes\">" & linefeed
	set xml to xml & "<title>" & track_name & "</title>" & linefeed
	set xml to xml & "<subtitle>" & artist_name & "</subtitle>" & linefeed
	set xml to xml & "<icon>now_playing.png</icon>" & linefeed
	set xml to xml & "</item>" & linefeed
 
	-- add new tag list item
	set xml to xml & linefeed & "<item uid=\"newtag\" arg=\"" & dbid & "|" & q & "\" valid=\"yes\">" & linefeed
	set xml to xml & "<title>New Playlist Tag</title>" & linefeed
	if q is equal to "" then 
		set xml to xml & "<subtitle>To remove tags use - prefix, e.g: slow -fast</subtitle>" & linefeed
	else
		set xml to xml & "<subtitle>" & q & "</subtitle>" & linefeed
	end if
	set xml to xml & "<icon>playlist_new.png</icon>" & linefeed
	set xml to xml & "</item>" & linefeed
 
	return xml & getTags(dbid) & "</items>"
end run
 
on getTags(dbid)
	-- assemble tag list
 
	set tags to ""
	tell application "iTunes"
		set artist_name to (get artist of the current track)
		set track_name to the name of the current track
		set pls to (get name of user playlists)
	end tell
 
	set tagsToRemove to {}
	set tagsToAdd to {}
 
	-- collect used/unused tags into lists
	repeat with pl in pls
		if pl starts with "@" then
			tell application "iTunes"
				set results to (every file track of user playlist pl whose database id is dbid)
			end tell
 
			if (count of results) > 0 then
				set end of tagsToRemove to pl
			else
				set end of tagsToAdd to pl
			end if
		end if
	end repeat
 
	repeat with pl in tagsToRemove
		set tags to tags & getItem("-", pl, dbid)
	end repeat
 
	repeat with pl in tagsToAdd
		set tags to tags & getItem("+", pl, dbid)
	end repeat
 
	return tags
 
end getTags
 
on getItem(prefix, myID, dbid)
	-- create tag list item
 
	set myIcon to ""
	tell application "iTunes"
		set track_name to the name of the current track
		set results to (every file track of playlist myID whose database id is dbid)
	end tell
	if (count of results) > 0 then
		set icon to "playlist_remove.png"
	else
		set icon to "playlist_add.png"
	end if
 
	set AppleScript's text item delimiters to "@"
	set myID to (text item 2 of myID)
 
	set myItem to ""
	set myItem to myItem & linefeed & "<item uid=\"" & myID & "\" arg=\"" & dbid & "|" & prefix & myID & "\" valid=\"YES\" autocomplete=\"" & myID & "\" type=\"default\">" & linefeed
	set myItem to myItem & "<title>" & myID & "</title>" & linefeed
	if prefix is equal to "+" then 
		set myItem to myItem & "<subtitle>Add this tag</subtitle>" & linefeed
	else
		set myItem to myItem & "<subtitle>Remove this tag</subtitle>" & linefeed
	end if
	set myItem to myItem & "<icon>" & icon & "</icon>" & linefeed
	set myItem to myItem & "</item>" & linefeed
 
	return myItem
 
end getItem
 
on itunes_is_running()
	-- iTunes detect - too slow to use
	tell application "System Events" to return (exists process "iTunes")
end itunes_is_running

Ez pedig a tagger, ami a playlist managementet csinálja:

on run argv
	set q to argv as text
 
	-- create folder playlist for tags
	tell application "iTunes"
		if not (exists folder playlist "Tags") then make folder playlist with properties {name:"Tags"}
	end tell
 
	set tid to AppleScript's text item delimiters
 
	-- separate dbid and tags
	set AppleScript's text item delimiters to "|"
	set dbid to (item 1 of (text items of q))
	set tags to (item 2 of (text items of q))
	set AppleScript's text item delimiters to tid
 
	-- get list of tags
	set AppleScript's text item delimiters to " "
	set qList to text items of tags -- not text of, text items of
	set AppleScript's text item delimiters to tid
 
	-- collect add/remove tags
	set addTags to {}
	set removeTags to {}
 
	repeat with qItem in qList
		set prefix to (character 1 of qItem)
		if (prefix is not equal to "+") and (prefix is not equal to "-") then
			set qItem to "+" & qItem
		end if
 
		if character 1 of qItem is equal to "-" then
			set end of removeTags to getToken(qItem)
		else
			set end of addTags to getToken(qItem)
		end if
 
	end repeat
 
	-- add tags
	repeat with addTag in addTags
		tell application "iTunes"
			set tagFolder to folder playlist "Tags"
			if (exists playlist addTag) then
				set newTag to (playlist addTag)
			else
				set newTag to (make playlist at tagFolder with properties {name:addTag})
			end if
 
 
			set results to (every file track of playlist addTag whose database ID is dbid)
			if (count of results) = 0 then
				duplicate (current track) to newTag
			end if
		end tell
	end repeat
 
	-- remove tags
	repeat with removeTag in removeTags
		tell application "iTunes"
			set tagFolder to folder playlist "Tags"
			if not (exists playlist removeTag) then exit repeat
 
			set results to (every file track of playlist removeTag whose database ID is dbid)
			if (count of results) > 0 then
				delete (some track of playlist (removeTag as text) whose database ID is dbid)
			end if
		end tell
	end repeat
end run
 
on getToken(the_string)
	return "@" & text 2 thru length of the_string
end getToken

Ha idáig eljutsz, be kell írjak egy pirosat, annyira száraz az egész. Azonban ha hozzám hasonlóan dolgozol és neked is hiányzik az iTunes-ből a tagging funkcionalitás, akkor szeretnéd a toolt! Ha tetszik és használnád, innen vagy a packal.org-ról vihető a komplett workflow.

Cicás wallpaperek fds-nek

Fazekas "fds" Dani a barátom. Rengeteget segített már mindenféle szoftveres dologban. Az utóbbi időben sajna nagyon keveset látom, de szerencsére online napi kapcsolatban vagyunk.

Mikor megfixál nekem valamit, felírok neki kedvem szerint ajándék csülköt vagy sört, mikor melyik jut épp eszembe. Jelenleg így áll Dani listája:

  • csülök: 2012.01.17
  • csülök: 2013.03.21
  • csülök: 2013.10.21
  • csülök: 2014.02.12
  • sör: 2014.10.23
  • sörpótlék: 2014.10.29

A lista közel nem teljes, csak akkor kerül rá újabb item, ha épp nem feledkezem meg róla.

OS X 10.10 óta nálam van egy zavaró probléma: a Mac reboot után elfelejti, hogy melyik wallpaper volt beállítva. A hiba oka még nincs meg, csak épp beszélgettünk róla chat-en, amikor kiderült, hogy fds háttérkép nélkül hajtja az iMac-et, nehogy beégjen. Innen jutott eszembe, hogy pár mozdulatból csinálhatnék neki egy csomó háttérképet, amiket aztán az OS X a beégés ellen periodikusan váltogathatna nála magától (System Preferences/Desktop & Screen Saver/Change picture checkbox). Ebből lett ez a "cicás wallpaperek fds-nek" poszt, meg a 37 cicás wallpaper:

  • 20101214-5340.jpg
  • 20101224-5423.jpg
  • 20110101-IMG_5500.jpg
  • 20110101-IMG_5546.jpg
  • 20110101-Lick.jpg
  • 20110105-IMG_5735.jpg
  • 20110118-IMG_5977 - Version 2.jpg
  • 20110118-IMG_5995.jpg
  • 20110119-Favorites.jpg
  • 20110129-IMG_6132.jpg
  • 20110203-IMG_6164.jpg
  • 20110218-IMG_6578.jpg
  • 20110219-IMG_6869.jpg
  • 20110220-IMG_6872.jpg
  • 20110226-Climber.jpg
  • 20110226-IMG_7054.jpg
  • 20110226-IMG_7133.jpg
  • 20110226-IMG_7136.jpg
  • 20110226-IMG_7202.jpg
  • 20110325-8882.jpg
  • 20110402-9369.jpg
  • 20110403-9559.jpg
  • Morning rest
  • 20110423-1771.jpg
  • 20110502-2153.jpg
  • 20110502-2157.jpg
  • 20110508-2970.jpg
  • 20110603-3389.jpg
  • 20110616-3722.jpg
  • 20110719-7069.jpg
  • 20110719-7118.jpg
  • 20111015-0948.jpg
  • 20111110-1248.jpg
  • 20111115-1349.jpg
  • Not Kim Jong Il but definitely looking at things
  • 20120309-4262.jpg
  • 20130103-1657.jpg

Ha letöltenéd az fds 2560x1440-es felbontású iMac-jére optimalizált setet, akkor eressz el egy wgetet erre.

Adóhivatal, Magyarország, 2014 november

Fiam küld egy scannelt papermailt, melyet az adóhivataltól kaptam, melyben a NAV arról tájékoztat, hogy 9600 HUF importtermék ÁFA adótartozásom van. Ugyanilyen levelet kaptam már vagy öt alkalommal és minden egyes esetben megpróbáltam kideríteni, hogy mire rótták ki az adót, eleddig sikertelenül. Ma viszont feladtam és megkértem Somát, hogy hívja fel az adóhivatalt és kérje meg őket, hogy adjanak egy számlaszámot, ahova a sarcot elutalhatom. Soma felhívta őket, a válasz MEGDÖBBENTŐ.

Nem elég, hogy az ügyintéző nem tudott utalási számlaszámot mondani, a saját rendszerükből az adószámom és az általuk vámolási azonosítóként használt, engem azonosító VPID ismeretében _sem_ tudta megmondani, hogy mire vetették ki az adót. Ehelyett azt tudta javasolni a gyereknek, hogy írjak egy papermailt. 2014-ben, papír levelet, MINT AZ ÁLLATOK.

Szerencsére a nav.gov.hu-n van egy email cím a NAV Váminformációs központjához, így az alábbi szerelmetes levelet küldtem el nekik:

Tisztelt adóhivatal,

Pár napja kaptam egy levelet, melyben jelzik, hogy 9600 HUF importtermék ÁFA adótartozásom van. Ilyen levelet már több alkalommal kaptam, többször próbáltam kinyomozni, hogy mi indukálta a kirótt adót és elintézni, hogy egy másik adónemről, melyen közel 50000 HUF többletem van, jóváírják az említett 9600 HUF tartozást. Továbbra sem tudta nekem senki megadni, hogy milyen import árura rótták ki a 9600 HUF vám+ÁFA adót, de ezen már túl vagyok, csupán az adótartozást szeretném kiegyenlíteni.

Mivel úgy tűnik, valami általam ismeretlen okból kifolyólag az átkönyvelés nem működik, így kérem, adjanak egy számlaszámot, amelyre elutalhatom a 9600 HUF adótartozást. Kérem egyúttal szíveskedjenek azt is megadni, hogy mit írjak az utalás közlemény rovatába annak érdekében, hogy az utalásom önök által egyértelműen azonosítható legyen.

Üdvözlettel,
Penoff Gábor
szerencsétlen hülye jogkövető állampolgár

Amint történik érdemi változás, frissítem a posztot. Stay tuned!

Update 20141107

Kaptam egy választ, amelynek az alján az alábbi szöveg szerepelt:

A jelen üzenetben található információk bizalmasak és üzleti titoknak minősülhetnek. Kizárólag a címzett, illetve az általa meghatalmazott(ak) használhatják fel. Ha a jelen üzenetnek nem Ön a címzettje, felhívjuk figyelmét, hogy annak jogosulatlan felhasználása, másolása, terjesztése vagy a tartalmával való visszaélés szigorúan tilos. Ha jelen üzenetet tévedésből kapta, kérjük, haladéktalanul értesítsen bennünket az alábbi telefonszámon vagy e-mail címen. Köszönöm.

Mivel a címzett én vagyok, úgy gondolom, bátran beszélhetek bárkinek a mi kis "üzleti titkunkról" - íme a válaszlevél eleje:

A Nemzeti Adó- és Vámhivatalhoz írt, elektronikus levelével kapcsolatban az alábbi általános tájékoztatást adom:

A Nemzeti Adó- és Vámhivatal (a továbbiakban: NAV) Váminformációs Központja általános váminformációt szolgáltat.

Az Ön által feltett kérdésre a válaszom a következő:

Sajnos adózási kérdésekben nem tudok konkrétan segíteni, de kérem a scannelt kép alapján járjon el: a képet csatoltam!

Továbbá a kitöltési útmutató áttanulmányozása után annak megfelelően a lakóhelye szerinti illetékes Adóigazgatósághoz fáradjon be, legyen szíves és munkaidőben a kollégáin segíteni fognak önnek.

A NAV munkatársa csatolt képként küldött nekem egy .docx kiterjesztésű dokumentumot (=makrómentes, zippelt Microsoft Office Word XML formátum). Szerencsére az ilyenek ellen megfelelő fegyverzettel rendelkezem, így viszonylag gyorsan előkerült a "csatolt kép":

nav-valasz-20141107-image1.png

Becsülettel végigolvastam mindent én hülye, azt gondolva, hogy majd csak előbújik az importtermék ÁFA alszámla száma valahol, de hiába.
Hogy kijöjjön a fejemből a gőz, elmentem befejezni a bablevest, aztán az illattól eltelve visszaültem a gép elé és 2 perc múlva előkerült ez a pdf a NAV oldaláról, amiben a 2. oldalon levő táblázat 2. sora az alábbi szöveget tartalmazza:

NAV importtermék általános forgalmi adó bevételi számla:
10032000-01037203-00000000

Ezt a 3*8 karaktert kértem volna, tisztelt adóhivatal. Ugyanakkor be kell látnom, hogy egyáltalán nem lehet egyszerű infot szolgáltatni az állampolgár számára, ha annak ennyi kézen kell keresztül mennie, míg elér egy kopipészt robotig:

NAV-original-mail-header-NEM_HISZED_EL

A 9600 HUF pénzt természetesen elutaltam. Végre.

No time to loose ebéd

Csak a szósszal van minimális macera:

01-szosz

Összevágsz apróra pár gerezd fokhagymát, egy fej vöröshagymát, egy csípős paprikát, felszeletelsz ~20 centi pizza mozzarellát. Széles serpenyőben olívaolajra fokhagyma, paprika, vöröshagyma, majd ha hagyma barnul, akkor egy adag darált hús. Kevergeted, mikor a hús megpörkölődött körben, akkor só, bors, pesto, majd rá egy adag darabos paradicsomkonzerv. Egyet rottyan és mehet a tésztára:

02-pasta

03-first-layer

Szósz tetejére mozzarella szeletek bőven, aztán megint tészta-szósz-mozzarella:

04-second-layer

05-third-layer

06-fourth-layer

Oldalról ilyen brutál:

07-all-layers-side

Felülről meg ilyen:

08-all-layers-top

45 perc sülés után meg ilyen:

09-meal

Mac workflow: arrange multiple windows

A probléma

  • több, egymás mellé pakolt ablakban szeretnénk dolgozni különböző szoftverekkel
  • az aktuális feladattól függően más-más alkalmazásablakokra van szükségünk más-más layouttal
  • az egyszer már jól beállított munkakörnyezetet a lehető legkevesebb pöcsöléssel szeretnénk újra beállítani

A megoldás

A layourok elmentéséhez Moom-ot fogunk használni. A Many Tricks alkalmazása rengeteg egyéb mellett képes arra, hogy az épp látható ablakok méretét és pozícióját úgynevezett Snapshotként elmentse és ezt az állapotot aztán visszaállítsa. Ezeket a snapshotokat aztán aktiválhatjuk a Moom menüjéből, vagy akár egy keyboard shortcut segítségével, de mi most inkább az Alfred nevű launchert fogjuk megkérni arra, hogy segítsen ebben.

Gyártunk Alfredben egy workflowt, ami egy script filterrel összeszedi nekünk a lementett snapshotokat, aztán pedig megkéri a user által kiválasztott snapshot beállítására a Moom-ot.

A nagyon szimpla workflownk így néz ki:

mooms-workflow-dataflow

A script filterünkbe egy Alfred által kívánt formátumú XML-t kell legyártanunk, majd azt átadnunk a launchernek, amiből ő megjeleníti a snapshot listát:

mooms-workflow-in-action

Így néz ki a script filterünk snapshot gyűjtést végző része:

tell application "Moom"
	set snaps to list of snapshots
 
	set xml to "<?xml version=\"1.0\"?>"
	set xml to xml & "<items>"
 
	repeat with snap in snaps
		set xml to xml & "<item uid=\"" & snap & "\" arg=\"" & snap & "\" valid=\"YES\" autocomplete=\"" & snap & "\" type=\"default\">"
		set xml to xml & "<title>" & snap & "</title>"
		set xml to xml & "<icon type=\"fileicon\">/Applications/Moom.app</icon>"
		set xml to xml & "</item>"
	end repeat
 
	set xml to xml & "</items>"
 
end tell
 
return xml

Ezt a selectort már csak egy Applescript futtató elemmel kell összedrótozni, amiben megkérjük a Moom-ot a kiválasztott snapshot beállítására:

on alfred_script(q)
	tell application "Moom"
		arrange windows according to snapshot named q
	end tell
end alfred_script

Voila - az eredmény az Alfredből aktiválható snapshot lista. Lusták szedhetik innen.