[{"data":1,"prerenderedAt":1157},["ShallowReactive",2],{"content-/jbagley/2023-06-01/universal_ffmpeg_custom_builds":3},{"article":4,"all":271},{"_path":5,"_dir":6,"_draft":7,"_partial":7,"_locale":8,"title":9,"description":10,"tags":11,"excerpt":10,"image":16,"publishDate":17,"body":18,"_type":262,"_id":263,"_source":264,"_file":265,"_stem":266,"_extension":267,"author":268},"/jbagley/2023-06-01/universal_ffmpeg_custom_builds","2023-06-01",false,"","Building Universal FFmpeg Custom Binaries","I am using a very pared down set of FFMpeg features for a macOS project that I\nbuild into a custom library. I had a script set up to configure the build which\nworked fine on my Intel based MacBook Pro. Then I upgraded to an Apple Silicon\nMacBookPro and wanted to run natively, or at least see what happened when I\ndid. To build, FFMpeg uses autoconf which produces a makefile that then handles\nthe build.",[12,13,14,15],"c","bash","ffmpeg","apple","/jbagley/2023-06-01/img/header.png","2024-04-01",{"type":19,"children":20,"toc":254},"root",[21,30,35,41,74,87,93,98,103,114,119,156,177,183,203,212,224,236,249],{"type":22,"tag":23,"props":24,"children":26},"element","h2",{"id":25},"motivation",[27],{"type":28,"value":29},"text","Motivation",{"type":22,"tag":31,"props":32,"children":33},"p",{},[34],{"type":28,"value":10},{"type":22,"tag":23,"props":36,"children":38},{"id":37},"the-simplest-thing-which-could-work-but-didnt",[39],{"type":28,"value":40},"The simplest thing which could work but didn't",{"type":22,"tag":31,"props":42,"children":43},{},[44,46,53,55,61,63,72],{"type":28,"value":45},"Autoconf supports passing flags for the compiler in the call to configure. I was\nalready passing ",{"type":22,"tag":47,"props":48,"children":50},"code",{"className":49},[],[51],{"type":28,"value":52},"-arch x86_64",{"type":28,"value":54}," as part of ",{"type":22,"tag":47,"props":56,"children":58},{"className":57},[],[59],{"type":28,"value":60},"--extra-cflags",{"type":28,"value":62},". Its\n",{"type":22,"tag":64,"props":65,"children":69},"a",{"href":66,"rel":67},"https://www.gnu.org/software/autoconf/manual/autoconf-2.69/html_node/Multiple-Architectures.html",[68],"nofollow",[70],{"type":28,"value":71},"documentation",{"type":28,"value":73},"\nsays that I may provide more than one architecture on macOS. That page notes\nthat this doesn't always work.",{"type":22,"tag":31,"props":75,"children":76},{},[77,79,85],{"type":28,"value":78},"Based on that, I tried ",{"type":22,"tag":47,"props":80,"children":82},{"className":81},[],[83],{"type":28,"value":84},"-arch x86_64 -arch arm64",{"type":28,"value":86},"', and as foreshadowed it\nindeed did not work. Autoconf's compiler tests would fail when trying the x86_64\nbuild with clang. I went through some more permutations of arguments, but in the\nend I had to do separate builds for each architecture.",{"type":22,"tag":23,"props":88,"children":90},{"id":89},"building",[91],{"type":28,"value":92},"Building",{"type":22,"tag":31,"props":94,"children":95},{},[96],{"type":28,"value":97},"I use a build directory rather than calling configure in the top of my FFmpeg\nworking copy and dirtying it up. I am running this from that build dir.",{"type":22,"tag":31,"props":99,"children":100},{},[101],{"type":28,"value":102},"Here is bash psuedocode of my build script:",{"type":22,"tag":104,"props":105,"children":109},"pre",{"className":106,"code":108,"language":28},[107],"language-text","for arch in x86_64 arm64; do\n\n   rm -rf Makefile config.h ffbuild lib* tests \n\n    ../configure --cc=clang \\\n                 --extra-cflags=\"-arch $arch\" \\\n                 --extra-ldflags=\"-arch $arch\"  \\\n                 --build_suffix=\"$install_dir\"  \\\n                 --prefix=\"path/to/where/I/want/the/libs/to/be/copied/by/make/install\" \\\n                 # other flags for ffmpeg configuration, see '../configure -h' output\n    make -j\n    make install\ndone\n",[110],{"type":22,"tag":47,"props":111,"children":112},{"__ignoreMap":8},[113],{"type":28,"value":108},{"type":22,"tag":31,"props":115,"children":116},{},[117],{"type":28,"value":118},"Looping through the desired architectures, first it cleans the build directory\nto get rid of the previous architecture output, then configures, building and\ninstalling the library.",{"type":22,"tag":31,"props":120,"children":121},{},[122,124,130,132,138,140,146,148,154],{"type":28,"value":123},"In the end, you have two sets of library files installed into the path passed\nwith ",{"type":22,"tag":47,"props":125,"children":127},{"className":126},[],[128],{"type":28,"value":129},"--prefix",{"type":28,"value":131}," with the suffixes ",{"type":22,"tag":47,"props":133,"children":135},{"className":134},[],[136],{"type":28,"value":137},"_x86-64",{"type":28,"value":139}," and ",{"type":22,"tag":47,"props":141,"children":143},{"className":142},[],[144],{"type":28,"value":145},"_arm64",{"type":28,"value":147}," provided by the\n",{"type":22,"tag":47,"props":149,"children":151},{"className":150},[],[152],{"type":28,"value":153},"--build_suffix",{"type":28,"value":155}," argument.",{"type":22,"tag":31,"props":157,"children":158},{},[159,161,167,169,175],{"type":28,"value":160},"I needed to pass the architecture again with ",{"type":22,"tag":47,"props":162,"children":164},{"className":163},[],[165],{"type":28,"value":166},"--extra-ldflags",{"type":28,"value":168}," to avoid the\nautoconf compiler test failing. The most basic test for compiler support would\nfail when an x86_64 ",{"type":22,"tag":47,"props":170,"children":172},{"className":171},[],[173],{"type":28,"value":174},".o",{"type":28,"value":176}," file couldn't be linked to an arm64 exectuable.",{"type":22,"tag":23,"props":178,"children":180},{"id":179},"linking-the-universal-libraries",[181],{"type":28,"value":182},"Linking The Universal Libraries",{"type":22,"tag":31,"props":184,"children":185},{},[186,188,194,196,201],{"type":28,"value":187},"The call to ",{"type":22,"tag":47,"props":189,"children":191},{"className":190},[],[192],{"type":28,"value":193},"lipo",{"type":28,"value":195}," turned out to be simple. The trickier part for me was\nbuilding the array of names to pass to ",{"type":22,"tag":47,"props":197,"children":199},{"className":198},[],[200],{"type":28,"value":193},{"type":28,"value":202}," as input files. I don't remember\nthe last time I used these.",{"type":22,"tag":104,"props":204,"children":207},{"className":205,"code":206,"language":28},[107],"lib_output_dir=\u003Cthe same path I used as the --prefix argument to configure above>\ncd \"$lib_output_dir\"\nfor lib_name in \u003Clib names I produced in my build without the file extension, e.g. libavcodec>; do\n    input_names=()\n    for arch in x86_64 arm64; do\n        input_names+=(${lib_name}_{$arch}.a)\n    done\n    /usr/bin/lipo -create -output ${lib_name}.a ${input_names[*]}\n    rm ${input_names}[*]}\ndone\ncd -\n",[208],{"type":22,"tag":47,"props":209,"children":210},{"__ignoreMap":8},[211],{"type":28,"value":206},{"type":22,"tag":31,"props":213,"children":214},{},[215,217,222],{"type":28,"value":216},"It changes to the directory where the first section installed the libraries. Of\ncourse, ",{"type":22,"tag":47,"props":218,"children":220},{"className":219},[],[221],{"type":28,"value":193},{"type":28,"value":223}," could be run on the libs within the build folder, but my script\nwas already installing them, and I found this more convenient and simple.",{"type":22,"tag":31,"props":225,"children":226},{},[227,229,234],{"type":28,"value":228},"For each architecture, it builds the array of library filenames, including the\narchitecture suffix of each. That array gives the typical DRY advantages and\nkeeps the invocation of ",{"type":22,"tag":47,"props":230,"children":232},{"className":231},[],[233],{"type":28,"value":193},{"type":28,"value":235}," simple.",{"type":22,"tag":31,"props":237,"children":238},{},[239,241,247],{"type":28,"value":240},"You can confirm it worked using the ",{"type":22,"tag":47,"props":242,"children":244},{"className":243},[],[245],{"type":28,"value":246},"file",{"type":28,"value":248}," utility. You should see a line for each\narchitecture it its output.",{"type":22,"tag":31,"props":250,"children":251},{},[252],{"type":28,"value":253},"That's it. If you have a better way, or see something I can do better here, let\nme know.",{"title":8,"searchDepth":255,"depth":255,"links":256},3,[257,259,260,261],{"id":25,"depth":258,"text":29},2,{"id":37,"depth":258,"text":40},{"id":89,"depth":258,"text":92},{"id":179,"depth":258,"text":182},"markdown","content:jbagley:2023-06-01:Universal_FFMPEG_custom_builds.md","content","jbagley/2023-06-01/Universal_FFMPEG_custom_builds.md","jbagley/2023-06-01/Universal_FFMPEG_custom_builds","md",{"user":269,"name":270},"jbagley","Jason Bagley",[272,285,294,307,321,333,341,351,362,371,383,394,405,414,424,435,448,459,468,478,486,493,502,510,519,529,539,548,557,565,576,584,592,601,609,619,627,635,643,651,659,667,675,685,693,703,711,721,731,742,752,762,773,783,796,805,813,824,834,837,849,863,877,892,900,908,917,926,935,944,953,962,971,980,989,1000,1013,1023,1031,1039,1047,1055,1067,1079,1089,1098,1107,1117,1126,1136,1145],{"_path":273,"title":274,"description":275,"image":276,"publishDate":277,"tags":278,"_id":281,"author":282},"/alalande/2023-1/escaping_text","Thinking Like a Programmer: Escaping Text","In this progressive and vivid explanation, we explore the HOW and WHY of quoting, nesting, and escaping text.","/alalande/2023-1/img/header.png","2024-02-15",[279,280],"series","educational","content:alalande:2023-1:escaping_text.md",{"user":283,"name":284},"alalande","Anthony Lalande",{"_path":286,"title":287,"description":288,"image":289,"publishDate":290,"tags":291,"_id":292,"author":293},"/alalande/2023-2/heuristics","Thinking Like a Programmer: Heuristics","We want to help you turbo-charge your decision making.","/alalande/2023-2/img/heuristics.png","2024-03-15",[279,280],"content:alalande:2023-2:heuristics.md",{"user":283,"name":284},{"_path":295,"title":296,"description":297,"tags":298,"publishDate":301,"image":302,"_id":303,"author":304},"/areichert/2023-10/rapid-testing-techniques-for-web-and-mobile","Rapid Testing Techniques for Web & Mobile Apps","Testing rapidly or the need to test quickly under shortened execution cycles is not new. Rapid testing has been around for as long as software testing has existed but increases when a product release is imminent, strained, or behind schedule. There are many reasons development gets behind schedule. Similar to testing, development is prone to surprise requirements changes and increases in scope and complexity.",[299,300],"software-testing","qa","2024-06-01","/areichert/2023-10/img/rapid_testing.png","content:areichert:2023-10:Rapid Testing techniques for web and mobile.md",{"user":305,"name":306},"areichert","Amy Reichert",{"_path":308,"title":309,"description":310,"image":311,"publishDate":312,"tags":313,"_id":317,"author":318},"/asherbrooke/2020-4/watchwah","Bringing an Idea to Life: WatchWah Proof of Concept","\"Wouldn't it be cool if...\"","/asherbrooke/2020-4/img/Guitar_and_Watch.jpeg","2020-04-01",[314,15,315,316],"ios","watch","juce","content:asherbrooke:2020-4:watchwah.md",{"user":319,"name":320},"asherbrooke","Andrew Sherbrooke",{"_path":322,"title":323,"description":324,"publishDate":325,"tags":326,"_id":329,"author":330},"/avogan/2012-07/salt","What Your Users Don't Know (Part 1)","What's wrong with this code?","2012-07-13",[327,328,279],"cryptography","security","content:avogan:2012-07:salt.md",{"user":331,"name":332},"avogan","Andrew Vogan",{"_path":334,"title":335,"description":336,"publishDate":337,"tags":338,"_id":339,"author":340},"/avogan/2012-07/salt-2","What Your Users Don't Know (Part 2)","In my last post we saw that what your users don't know can hurt them. In other words, how securely you handle your users' private data behind the scenes can have profound implications both for your business and your users' well being. To put it bluntly, it's bad for your business to be publicly shamed over your handling of sensitive data, and it's bad for your users to have their bank accounts pilfered -- those being some of the worse case scenarios.","2012-07-26",[327,328,279],"content:avogan:2012-07:salt-2.md",{"user":331,"name":332},{"_path":342,"title":343,"description":344,"publishDate":345,"image":346,"_id":347,"author":348},"/bporter/2012-5/cd_player","My First CD Player","I started college right about the time when the first CD players were coming onto the market -- there weren't many available, and they were all obscenely expensive. At the time, my dad was dong a lot of traveling to Japan for business, and he was able to bring me a really nice Yamaha CD player back from a shop in Akihabara for about 1/4th of what a similar unit would have cost me here in the US.","2012-05-01","/bporter/2012-5/img/cd_player.jpg","content:bporter:2012-5:cd_player.md",{"user":349,"name":350},"bporter","Brett Porter",{"_path":352,"title":353,"description":354,"publishDate":355,"tags":356,"image":359,"_id":360,"author":361},"/bporter/2012-5/improtech","ImproTech Paris-New York 2012","Last week, I took a vacation day to attend one day of workshops at NYU as part of ImproTech 2012 Paris-New York That website descibes the event as:","2012-05-24",[357,358],"improvisation","music","/bporter/2012-5/img/affichemartin.jpg","content:bporter:2012-5:improtech.md",{"user":349,"name":350},{"_path":363,"title":364,"description":365,"publishDate":366,"tags":367,"_id":369,"author":370},"/bporter/2012-5/learntocode","Yes, Do Learn To Code!","My usual pre-work routine is to walk the dog (working at home, this is my counterpart to a commute), pour my first cup of coffee, and then curl up for a little while with Google Reader. I don't know if it's because I've selected feeds that are too closely aligned with my values and personal agenda, but it's really rare that I'll read a post that is just so wrong that it makes me angry. Jeff Atwood wrote a post like that: Please Don't Learn To Code","2012-05-15",[368],"learn-to-code","content:bporter:2012-5:learntocode.md",{"user":349,"name":350},{"_path":372,"title":373,"description":374,"publishDate":375,"tags":376,"image":380,"_id":381,"author":382},"/bporter/2012-6/dsl","Watch Your Language","Interesting to see a theme emerge in my Pinboard account this week -- lots of stuff about the idea of 'programming language'. I've spent the last few weeks preparing to dive back into a personal interactive music project that I've been working on sporadically since I was in graduate school. I had recently realized that the conceptual roadblock I hit before my last hiatus was something that I'd need to address by adding some sort of little programming language into the system. After following Martin Fowler's many blog posts over the years discussing domain specific languages, I finally broke down and bought his book on the topic. It's too early yet for me to have much concrete to say about the book, but I remember getting enough out of those blog posts to be confident that it will be worth the money and time to read.","2012-06-29",[377,378,379],"dsl","erlang","go","/bporter/2012-6/img/dsl.jpg","content:bporter:2012-6:dsl.md",{"user":349,"name":350},{"_path":384,"title":385,"description":386,"image":387,"publishDate":388,"tags":389,"_id":392,"author":393},"/bporter/2019-3/animator","Friz: A flexible animation controller for JUCE","As is often the case, I found myself working on a personal project and had some UI elements that really wanted to have some life to them on the screen.","/bporter/2019-3/img/animator.png","2019-03-01",[316,390,391],"ui","c++","content:bporter:2019-3:animator.md",{"user":349,"name":350},{"_path":395,"title":396,"description":397,"image":398,"publishDate":399,"tags":400,"_id":403,"author":404},"/bporter/2019-4/aesannounce","Art+Logic In the Real World","There are a few events coming up in the next few weeks where A+L will have people in attendance. If you're going to be there or nearby, please get in touch and we'll meet up.","/bporter/2019-4/img/aesLogo.jpg","2019-04-01",[401,402],"a+l","event","content:bporter:2019-4:aesAnnounce.md",{"user":349,"name":350},{"_path":406,"title":407,"description":408,"image":409,"publishDate":410,"tags":411,"_id":412,"author":413},"/bporter/2020-10/reanimated","Re-animated","Last year, I posted here about an animation control framework called 'Friz' that works within the JUCE Application Framework.","/bporter/2020-10/img/module.png","2020-10-01",[316,390,391],"content:bporter:2020-10:reanimated.md",{"user":349,"name":350},{"_path":415,"title":416,"description":417,"image":418,"publishDate":388,"tags":419,"_id":420,"author":421},"/bstevens/2019-3","Coding the Impossible","As you can see on the Art+Logic website, our slogan is Coding the \"impossible.\"®","/bstevens/2019-3/img/impossible.png",[401],"content:bstevens:2019-3:index.md",{"user":422,"name":423},"bstevens","Ben Stevens",{"_path":425,"title":426,"description":427,"publishDate":428,"tags":429,"_id":431,"author":432},"/ckeefer/2013-1/misc","JS Hints & Shortcuts","During the course of any complex project (and even many simple ones), on the way to accomplish the actual goal, you're certain to encounter any number of small hurdles along the way - little problems which need to be resolved for the bigger picture to come into focus.","2013-09-01",[430],"js","content:ckeefer:2013-1:misc.md",{"user":433,"name":434},"ckeefer","Christopher Keefer",{"_path":436,"title":437,"description":438,"publishDate":439,"tags":440,"image":445,"_id":446,"author":447},"/ckeefer/2013-2/xslt","XML and XSLT","Not terribly long ago, XML was the darling of the web. HTML4 was reformulated as XHTML 1.0, SOAP messages were XML, and let us not forget XMLHttpRequest.","2013-10-08",[441,442,443,444],"data-formats","xml","xsl","xslt","/ckeefer/2013-2/img/xslt-processing.png","content:ckeefer:2013-2:xslt.md",{"user":433,"name":434},{"_path":449,"title":450,"description":451,"publishDate":452,"tags":453,"image":456,"_id":457,"author":458},"/ckeefer/2013-3/ajax-upload","Ajax Upload Part I: Framed (and jQuery Deferred)","Inevitably, people want their files on the Internet. If your project is about cute cats, someone will task you with allowing users to upload photos of their cats, videos of their cats, long rambling audio clips in which they attempt to convince their cat to stop attacking the microphone, etcetera. If your project is about the nature and proclivities of mold, someone, somewhere will want to share detailed photographic evidence of their mold problem. The need to upload files is a given.","2013-03-20",[430,454,455],"jquery","html5","/ckeefer/2013-3/img/upframe.jpg","content:ckeefer:2013-3:ajax-upload.md",{"user":433,"name":434},{"_path":460,"title":461,"description":462,"publishDate":463,"tags":464,"_id":466,"author":467},"/ckeefer/2013-4/teaching-programming","Can (and Should) Everyone Learn to Program?","Fair warning: The following article is long, rambly, and contains no code. It does, however, contain some rumination on the idea that everyone can and should learn to program.","2013-12-03",[465],"programming","content:ckeefer:2013-4:teaching-programming.md",{"user":433,"name":434},{"_path":469,"title":470,"description":471,"publishDate":472,"tags":473,"image":475,"_id":476,"author":477},"/ckeefer/2013-5/ajax-uploader","Ajax Upload XHR2, Take 2","It's a pleasure to be able to interact with files in the browser at long last, isn't it? Reading files in without needing to bounce them against the server first opens up a lot of possibilities - and getting progress from a chunked ajax upload is miles away from the indeterminate form uploads of days past.","2014-02-19",[454,430,474],"xhr2","/ckeefer/2013-5/img/html5.jpg","content:ckeefer:2013-5:ajax-uploader.md",{"user":433,"name":434},{"_path":479,"title":480,"description":481,"publishDate":482,"tags":483,"_id":484,"author":485},"/ckeefer/2013-07/anchors-hash","Anchors, Hash Sign, javascript:void(0)","So, you've got a link that, in reality, is just a click target for performing some javascript function. You want the appearance of a standard anchor link, but if it's not performing the intended function, should it really be an anchor? And if so, what should we fill that 'href' attribute in with?","2013-07-29",[430],"content:ckeefer:2013-07:anchors-hash.md",{"user":433,"name":434},{"_path":487,"title":488,"description":489,"publishDate":490,"_id":491,"author":492},"/ckeefer/2013-07/static-vmware-host","Static Hosting with VMWare","Virtualization is one of the many benefits of the excess (metaphorical) horsepower available to us with modern hardware. Need to test against (Windows XP/7/8/NT || Fedora || Mint || Ubuntu || FreeBSD || MacOSX || etc)? Fire up the VM. Need a Linux environment for the packages your server relies on, but need to test in the iPad simulator? VM's to the rescue.","2013-07-26","content:ckeefer:2013-07:static-vmware-host.md",{"user":433,"name":434},{"_path":494,"title":495,"description":496,"publishDate":497,"tags":498,"_id":500,"author":501},"/ckeefer/2013-08/fullproof-fulltext-search","Client-side Fulltext Searching with Fullproof","Recently, I was engaged in a genial argument with a friend of an older generation, each of us taking an opposing stance on some obscure trivia neither of us was entirely certain about - but which we were both ready to defend with all the wit and rhetoric at our disposal. When we had finally exhausted all attempts to make the other budge on the matter, we turned to an authoritative 3rd-party source to lay the matter to rest for us - a Google search.","2013-08-29",[499,444],"search","content:ckeefer:2013-08:fullproof-fulltext-search.md",{"user":433,"name":434},{"_path":503,"title":504,"description":505,"publishDate":506,"tags":507,"_id":508,"author":509},"/ckeefer/2013-11/jquery-ajax-blobs","jQuery Ajax Blobs and Array Buffers","A big part of what makes jQuery a regular part of so many web projects is the clean interface it offers us for a number of sometimes messy built-in aspects of javascript. The most obvious is the DOM interface; and in second place, jquery ajax and its various shorthand methods. Abstracting away the difference between ActiveXObject and XMLHttpRequest is one of the most obvious benefits - but even if you don't need to worry about supporting old versions of IE, you might well enjoy the clean, object-based, promise-returning interface that jquery ajax offers.","2013-11-21",[430,454],"content:ckeefer:2013-11:jquery-ajax-blobs.md",{"user":433,"name":434},{"_path":511,"title":512,"description":513,"publishDate":514,"tags":515,"_id":517,"author":518},"/ckeefer/2013-12/deploying-with-git","Deploying Websites with Git","Deploying your webapp is an important part of the web development equation - your client's site isn't going to attract a lot of attention sitting in your local dev directory. Deployment concerns tend to fall to the bottom of the priority list, though, and the end result tends to be kludgy, hastily thrown-together deployment scripts; and because they are so kludgy and, often, time consuming, when time crunches threaten, a developer may resort to making changes directly on the remote server that need to be (but sometimes never are) backported to the code living in your version control.","2013-12-23",[516],"git","content:ckeefer:2013-12:deploying-with-git.md",{"user":433,"name":434},{"_path":520,"title":521,"description":522,"publishDate":523,"tags":524,"image":526,"_id":527,"author":528},"/ckeefer/2014-1/still-using-php","Still Using PHP?","Poor PHP. It's so lonely and unloved these days.","2014-01-29",[525],"php","/ckeefer/2014-1/img/php.jpg","content:ckeefer:2014-1:still-using-php.md",{"user":433,"name":434},{"_path":530,"title":531,"description":532,"publishDate":533,"tags":534,"image":536,"_id":537,"author":538},"/ckeefer/2014-2/ajax-upload-2","Ajax Upload Part II: XHR2 (and FileReader)","So, the client has told you their users should be able to upload their drunken party pictures for all the internet to see. \"We want the very best experience possible,\" they tell you. \"Simple, seamless - maybe using that new html5 thing I've heard so much about.\"","2013-04-09",[430,535],"xmlhttprequest","/ckeefer/2014-2/img/html5.jpg","content:ckeefer:2014-2:ajax-upload-2.md",{"user":433,"name":434},{"_path":540,"title":541,"description":542,"publishDate":543,"tags":544,"_id":546,"author":547},"/ckeefer/2014-3/customgmapsinfowindow","Custom Google Maps Info Windows","When it comes time to relate the ephemeral world of data to the physical world, Maps are key in both enterprise and consumer applications. Whatever else you might think of it, Google Maps tends to be the default option - certainly, its the only one I've ever had clients ask for by name.","2014-02-26",[430,545],"google-maps","content:ckeefer:2014-3:customgmapsinfowindow.md",{"user":433,"name":434},{"_path":549,"title":550,"description":551,"publishDate":552,"tags":553,"_id":555,"author":556},"/ckeefer/2014-4/hidden-options","Hidden Options: A Workaround","Here's the situation: You've got a select. Maybe a whole bunch of selects, with a ton of options each (metric ton - let's keep our imaginary hyperbolic units straight here); and these are meant to be complex interactive elements, with options made visible or not as some programmatic condition dictates.","2014-04-23",[554,430,454],"how-to","content:ckeefer:2014-4:hidden-options.md",{"user":433,"name":434},{"_path":558,"title":559,"description":560,"publishDate":561,"tags":562,"_id":563,"author":564},"/ckeefer/2014-5/cgwin2","Custom Google Info Windows: Updated, Live","April 30, 2014 at 3:22 am Remy says:","2014-05-09",[430,545],"content:ckeefer:2014-5:cgwin2.md",{"user":433,"name":434},{"_path":566,"title":567,"description":568,"publishDate":569,"tags":570,"image":573,"_id":574,"author":575},"/ckeefer/2014-6/backbonesocketsync","Websockets for Backbone","Backbone's had some of its thunder stolen lately by trendier frameworks like Meteor and Angular; for good reason, in most cases, as without the prosthetic functionality offered by the likes of Marionette, Backbone's view handling (amongst a few other lacks and warts) is really just 'roughed in'.","2014-06-25",[430,571,572],"websockets","backbone","/ckeefer/2014-6/img/WebsocketsPlusBackbone.png","content:ckeefer:2014-6:backbonesocketsync.md",{"user":433,"name":434},{"_path":577,"title":578,"description":579,"publishDate":580,"tags":581,"_id":582,"author":583},"/ckeefer/2014-7/promises","It's a (jQuery-style) Promise","Way back when I brought up the topic of promises (particularly, jQuery Deferred), and I promised we would come back to the topic someday.","2014-10-16",[430,454],"content:ckeefer:2014-7:promises.md",{"user":433,"name":434},{"_path":585,"title":586,"description":587,"publishDate":588,"tags":589,"_id":590,"author":591},"/ckeefer/2014-8/behold-views","Behold! (JavaScript Views)","JavaScript has the propensity to be very untidy - if you let it, it will sprawl all over the place. Hundreds of global variables scattered across dozens of files, messy half-measures towards object-orientation, mixed in seemingly at random with ungrouped functions - anyone who's had a client bring them a failed project from some other development team knows just how bad it can get.","2015-01-07",[430],"content:ckeefer:2014-8:behold-views.md",{"user":433,"name":434},{"_path":593,"title":594,"description":595,"publishDate":596,"tags":597,"image":598,"_id":599,"author":600},"/ckeefer/2015-1/writeonce","Write Once, Debug Everywhere","It's pretty seldom that anyone mentions web pages these days, other than in historical reference to days long gone by (yes, a whole few years ago). Web sites, sure, but not if what is really wanted is to replace something that, not so long ago, would have been some native code for a smartphone (or a little further back still, a desktop computer). Generally speaking, the most common term tripping from client's lips these days is 'web applications' - or webapps, because who has time for spaces and proper spelling, amirite?","2015-02-02",[430],"/ckeefer/2015-1/img/html5java.jpg","content:ckeefer:2015-1:writeonce.md",{"user":433,"name":434},{"_path":602,"title":603,"description":604,"publishDate":605,"tags":606,"_id":607,"author":608},"/ckeefer/2015-2/js-frameworks","The What and Why of Javascript Frameworks","JavaScript has the propensity to be very untidy if you let it be. This isn't a problem unique to JavaScript, of course - many other languages suffer from a lack of native organization, especially for specific tasks.","2015-05-29",[430],"content:ckeefer:2015-2:js-frameworks.md",{"user":433,"name":434},{"_path":610,"title":611,"description":612,"publishDate":613,"tags":614,"_id":617,"author":618},"/ckeefer/2015-3/emailvalidation","Email Validation with Django and python-social-auth","When it comes to user accounts, the standard litmus test is email validation. Besides the immediate benefits - of offering us a straightforward unique identifier for users, and making it more difficult to automate creating a mass of accounts on our service - by requiring that each account have an email address and interact therewith to confirm the addresses validity, it also offers us the chance to associate a known-working email account with a user account. This is important for transactional emails such as password resets or for potential two-factor authentication use... and if you're a little less ethical, for sending marketing desirable and informative emails about interesting products and services.","2015-07-23",[615,616],"python","django","content:ckeefer:2015-3:EmailValidation.md",{"user":433,"name":434},{"_path":620,"title":621,"description":622,"publishDate":623,"tags":624,"_id":625,"author":626},"/ckeefer/2015-5/file-reader-chunking","FileReader Chunking and Base64 DataURLs","In a hurry? You can now use our HUp jquery plugin to read files in a chunked fashion as data URLs. Hooray!","2015-12-15",[430,454],"content:ckeefer:2015-5:file-reader-chunking.md",{"user":433,"name":434},{"_path":628,"title":629,"description":630,"publishDate":631,"tags":632,"_id":633,"author":634},"/ckeefer/2016-1/ajaxbinarycaching","Caching Binary Data With jQuery Ajax and IndexedDB","After long, grueling months (years? or does it only feel like years?), your web application nears completion. It is tightly coded, well documented, works across all modern browsers, and is well received by your beta testers. It's nearly time to go live, and a smile of pure relief plays upon your lips... and freezes into a rictus grin when your client turns to you, and asks, \"so, hey, can we speed up the dynamic cat pic loading? Especially when I close the browser and come back to it later. I think that's really key to the whole application.\"","2016-04-25",[430,454],"content:ckeefer:2016-1:ajaxBinaryCaching.md",{"user":433,"name":434},{"_path":636,"title":637,"description":638,"publishDate":639,"tags":640,"_id":641,"author":642},"/ckeefer/2016-2/paymentprocessing","Payment Processing with Braintree","You've built the web application of the century, and the users have rightly flooded to it. Cat pictures for everyone!","2016-05-11",[616,430,454,615],"content:ckeefer:2016-2:paymentprocessing.md",{"user":433,"name":434},{"_path":644,"title":645,"description":646,"publishDate":647,"tags":648,"_id":649,"author":650},"/ckeefer/2016-3/djangochannels1","Django Channels: From the Ground Up - Part 1","You stare mournfully into the mass of code you've inherited. At some point, it's clear, the requirements called for the server to push information to the client, because there's an unholy mix of Server-Side Events, long-polling, hidden iframes and even a Java applet in there, all supporting some level of long-term connectivity with the server. It's almost fascinating in its barely functional hideousness, and you would be inclined to leave well enough alone... except for the new feature specifications you've been assigned, which require the client to be able to send data back to the server in response to the received events, in as close to real-time as you can get.","2016-06-13",[616,615,571],"content:ckeefer:2016-3:djangoChannels1.md",{"user":433,"name":434},{"_path":652,"title":653,"description":654,"publishDate":655,"tags":656,"_id":657,"author":658},"/ckeefer/2016-4/djangochannels2","Django Channels: From the Ground Up - Part 2","Last time, we decided to embark on a brave new adventure and give our Django framework a big upgrade with the inclusion of Django Channels. We got just far enough to get the development server running, but while this may be an adequate start, it's better to develop against something like what we intend to deploy, right?","2016-06-15",[616,615,571],"content:ckeefer:2016-4:djangochannels2.md",{"user":433,"name":434},{"_path":660,"title":661,"description":662,"publishDate":663,"tags":664,"_id":665,"author":666},"/ckeefer/2016-6/gofetch1","Go Fetch! (JavaScript Fetch API)","Long ago, we briefly brushed upon the topic of what has made jQuery such a valuable part of the web developer's toolset for such a long time - namely, a cleaner interface for interacting with the DOM, and the $.ajax abstraction over XMLHttpRequest.","2016-10-03",[430,454,279],"content:ckeefer:2016-6:goFetch1.md",{"user":433,"name":434},{"_path":668,"title":669,"description":670,"publishDate":671,"tags":672,"_id":673,"author":674},"/ckeefer/2016-7/gofetch2","Go Fetch 2! (JavaScript Fetch API)","Last time we discussed the Fetch API in general, taking a look at how it differed from the XMLHttpRequest API, and some of its advantages. Today, we're going to take a look at a little library that you can include in your projects today that offers you localStorage caching for the Fetch API.","2016-10-10",[430,454,279],"content:ckeefer:2016-7:goFetch2.md",{"user":433,"name":434},{"_path":676,"title":677,"description":678,"publishDate":679,"tags":680,"_id":683,"author":684},"/ckeefer/2016-8/herokupdf","Generating PDFs: wkhtmltopdf & Heroku","So, it has come to this.","2016-12-21",[681,682,615],"heroku","pdf","content:ckeefer:2016-8:HerokuPDF.md",{"user":433,"name":434},{"_path":686,"title":687,"description":688,"publishDate":689,"tags":690,"_id":691,"author":692},"/ckeefer/2017-1/downloadingclientsidecontent","Downloading Client-side Generated Content","A young developer, new to the Tao of the client-side, comes to a Master of the way, and speaks thusly: \"Oh Master, our application nears completion; and lo, cat pics can be drawn upon, and captions fixated thereto, for the creation of humour and the bounteous enjoyment of our users.\"","2017-02-06",[430],"content:ckeefer:2017-1:downloadingclientsidecontent.md",{"user":433,"name":434},{"_path":694,"title":695,"description":696,"publishDate":697,"tags":698,"_id":701,"author":702},"/ckeefer/2017-2/morepwatoya-part1","More PWA to Ya! (Progressive Web Apps, Part 1)","It's project kickoff time, and you're having a conversation with your client about what form the application will take:","2017-02-01",[699,700,279],"pwa","mobile","content:ckeefer:2017-2:MorePWAToYa-Part1.md",{"user":433,"name":434},{"_path":704,"title":705,"description":706,"tags":707,"publishDate":708,"_id":709,"author":710},"/ckeefer/2017-3/morepwatoya-part2","More PWA to Ya! (Progressive Web Apps, Part 2)","Last time, we got into the nitty gritty on how to make your web application into a Progressive Web Application (PWA to it's friends). I promised we'd dig even deeper this time, and show you how to make your web app a little more 'native' on Android - and how to deal with iOS Safari's special snowflake syndrome.",[700,699,279],"2017-03-01","content:ckeefer:2017-3:MorePWAToYa-Part2.md",{"user":433,"name":434},{"_path":712,"title":713,"description":714,"image":715,"publishDate":716,"tags":717,"_id":719,"author":720},"/ckeefer/2019-1/unlockingwebaudio","Unlocking Web Audio","\"It's going to be the coolest thing ever.\"","/ckeefer/2019-1/img/featured_image.jpg","2019-01-01",[430,718],"audio","content:ckeefer:2019-1:UnlockingWebAudio.md",{"user":433,"name":434},{"_path":722,"title":723,"description":724,"publishDate":725,"image":726,"tags":727,"_id":729,"author":730},"/ckeefer/2020-1/why-vue","Why Vue","Why choose Vue over any other front-end framework?","2020-01-01","/ckeefer/2020-1/img/vue-wall.jpg",[430,728],"vue","content:ckeefer:2020-1:Why Vue.md",{"user":433,"name":434},{"_path":732,"title":733,"description":734,"tags":735,"image":738,"publishDate":739,"_id":740,"author":741},"/ckeefer/2024-3/e2e_testing","E2E Testing: To What End?","Friend, can we agree that tests are a good idea? I won't scorn you for sometimes omitting them - time and budget constraints are what they are, and even the best intentioned of us sometimes have to just give our projects a lick and a promise. \"Proper test coverage soon\", you sweetly croon as you rock it to sleep, the knowledge that you're telling a dark, terrible lie twisting you up inside. Maybe you could just scrape enough budget together for some simple unit tests? Then, at least, you'd have \"tests\", right?",[300,299,736,737],"e2e","playwright","/ckeefer/2024-3/img/E2E_Testing_2024.png","2024-06-15","content:ckeefer:2024-3:e2e_testing.md",{"user":433,"name":434},{"_path":743,"title":744,"description":745,"tags":746,"image":748,"publishDate":749,"_id":750,"author":751},"/ckeefer/2024-7/vpubsub","Vue 3 Pub / Sub: All aboard the (event) bus","We like Vue at A+L. We think it's one of the best frontend frameworks, and a great choice pretty much anywhere you might otherwise be tempted to use React.",[430,728,747],"pub/sub","/ckeefer/2024-7/img/event_bus.png","2024-08-15","content:ckeefer:2024-7:VPubSub.md",{"user":433,"name":434},{"_path":753,"title":754,"description":755,"publishDate":756,"tags":757,"_id":758,"author":759},"/cmacksey/2012-5/php-musings","PHP Musings","Ran into an interesting, but thorough, rant the other day - PHP: A Fractal of Bad Design. The part that grabbed me the most was the analogy at the beginning, which was all too perfect:","2012-05-07",[525],"content:cmacksey:2012-5:php-musings.md",{"user":760,"name":761},"cmacksey","Chris Macksey",{"_path":763,"title":764,"description":765,"publishDate":766,"tags":767,"_id":769,"author":770},"/dpopowich/2021-07-30/data-collector","Asynchronous Python - A Real World Example","A dive into a real example of async Python usage.","2021-07-30",[615,554,768],"async","content:dpopowich:2021-07-30:data-collector.md",{"user":771,"name":772},"dpopowich","Daniel Popowich",{"_path":774,"title":775,"description":776,"tags":777,"image":779,"publishDate":780,"_id":781,"author":782},"/dpopowich/2023-8/postgres-pubsub","Using PostgreSQL for Pub/Sub","A+L has been working on a Single Page Application (SPA) wherein our client's users take on the role of Staff Users (think: project managers) as they aid their Customer Users in using the application to complete a complex project.",[778,615,768],"postgresql","/dpopowich/2023-8/img/psql_pub_sub.png","2024-04-15","content:dpopowich:2023-8:postgres-pubsub.md",{"user":771,"name":772},{"_path":784,"title":785,"description":786,"tags":787,"image":790,"publishDate":791,"_id":792,"author":793},"/ewahl/2025-05/escape_deployment_hell","Escape Deployment Hell: IaC, CDK, Ephemeral Environments, and the Pragmatic Path to Platform Power","Another Friday afternoon, another deployment fire. If this sounds familiar, you're not alone. On too many projects, the chasm between application code and infrastructure management breeds manual configuration nightmares, crippling complexity, and agonizingly slow development cycles. But what if your team could sidestep this chaos, focusing on building features instead of constantly battling deployment gremlins?",[788,789,615],"devops","aws","/ewahl/2025-05/img/deployment_hell.png","2025-05-13","content:ewahl:2025-05:escape_deployment_hell.md",{"user":794,"name":795},"ewahl","Edward F. Wahl",{"_path":797,"title":798,"description":799,"publishDate":800,"image":801,"tags":802,"_id":803,"author":804},"/ewahl/2025-06/argued_with_ai","I Argued With an AI for 20 Minutes About Async Code &mdash; And I'm Surprisingly Happy","If you have ever spent twenty minutes debating an obscure AWS Lambda invocation pattern with an AI, you might question your life choices. But here I am: amused by the wasted time but ultimately happy with the outcome and understanding I gained.","2026-06-01","/ewahl/2025-06/img/argued_with_ai.png",[788,789,615],"content:ewahl:2025-06:argued_with_ai.md",{"user":794,"name":795},{"_path":806,"title":807,"description":808,"tags":809,"image":810,"publishDate":399,"_id":811,"author":812},"/jbagley/2019-4/makingspectrogramsinjuce","Making Spectrograms in JUCE","Art+Logic's Incubator project has made a lot of progress. In a previous post I mentioned that Dr. Scott Hawley's technique to classify audio involved converting audio to an image and using a Convolution Neural Network (CNN) to classify the audio based on this image. That image is a spectrogram. I'm going to go into some detail about what we do to create one, and why to the best of my ability.",[316,391,718],"/jbagley/2019-4/img/Fortissimo_Trumpet_Ensemble_Matrix_Swells_61.wav-2048x1700.png","content:jbagley:2019-4:MakingSpectrogramsInJUCE.md",{"user":269,"name":270},{"_path":814,"title":815,"description":816,"tags":817,"image":820,"publishDate":821,"_id":822,"author":823},"/jbagley/2021-07/softwaresenescence","Legacy Vulnerabilities AKA Software Senescence","Does your business still have an XT computer in the back office because it's\nrunning that one version of some database software that your business depends\non? Yeah, we know there is. Most modern software doesn't work like that.",[818,819],"legacy","project-management","/jbagley/2021-07/img/old_software_to_new.jpg","2021-07-01","content:jbagley:2021-07:SoftwareSenescence.md",{"user":269,"name":270},{"_path":825,"title":826,"description":827,"tags":828,"image":830,"publishDate":831,"_id":832,"author":833},"/jbagley/2021-08-01/accuratetiming","Accurate Timing","In many tasks we need to do something at given intervals of time. The most obvious ways may not give you the best results.",[391,829],"timing","/jbagley/2021-08-01/img/accurateTiming.jpg","2021-08-01","content:jbagley:2021-08-01:AccurateTiming.md",{"user":269,"name":270},{"_path":5,"title":9,"description":10,"tags":835,"image":16,"publishDate":17,"_id":263,"author":836},[12,13,14,15],{"user":269,"name":270},{"_path":838,"title":839,"description":840,"publishDate":841,"image":842,"tags":843,"_id":847,"author":848},"/jbagley/2025-08/a_developers_primer_on_apple_tracking_transparency","A Primer on Apple's App Tracking Transparency","If an app tracks user activity, Apple requires them to declare all information they collect as well as whether that data is linked or tracked. This includes collection by the app itself and any third parties the app uses. The app owner is responsible for knowing and correctly reporting privacy information for all components in the app.","2026-05-22","/jbagley/2025-08/img/apple_app_transparency.png",[844,845,314,846],"app tracking transparency","att","macos","content:jbagley:2025-08:a_developers_primer_on_apple_tracking_transparency.md",{"user":269,"name":270},{"_path":850,"title":851,"description":852,"tags":853,"image":857,"publishDate":858,"_id":859,"author":860},"/jestep/2023-3/fastapi","FastAPI: A High-Performance Python Framework for Rapid Web Development","FastAPI is a modern and high-performance Python web framework designed specifically for building APIs and web applications quickly and efficiently. Developed by Sebastián Ramírez and first released in 2018, FastAPI has rapidly gained traction in the developer community thanks to its focus on providing key features for API and web app development with excellent performance.",[615,854,616,855,856],"fastapi","flask","pyramid","/jestep/2023-3/img/header.png","2024-05-01","content:jestep:2023-3:fastapi.md",{"user":861,"name":862},"jestep","Jagger Estep",{"_path":864,"title":865,"description":866,"publishDate":867,"tags":868,"image":872,"_id":873,"author":874},"/nharrison/2012-07/core-data","Securing Your Core Data with Transformable Attributes","In order to store private data in an iOS Core Data database, there are several methods available for encryption, including:","2012-07-30",[869,870,314,328,871],"core-data","encryption","objective-c","/nharrison/2012-07/img/superman.jpg","content:nharrison:2012-07:core-data.md",{"user":875,"name":876},"nharrison","Noah Harrison",{"_path":878,"title":879,"description":880,"tags":881,"publishDate":886,"image":887,"_id":888,"author":889},"/phendry/2019-3/restfromthebottomup","REST from the Bottom Up","The RESTful API has a funny place in the software development world: it's widely regarded as the best general-purpose pattern for building web application APIs, and yet it's also nebulous enough of a concept to cause endless disagreements within teams over exactly how to implement one.",[882,883,884,885],"rest","api","web","architecture","2019-10-01","/phendry/2019-3/img/feature_image.png","content:phendry:2019-3:RestFromTheBottomUp.md",{"user":890,"name":891},"phendry","Paul Hendry",{"_path":893,"title":894,"description":895,"tags":896,"publishDate":821,"image":897,"_id":898,"author":899},"/phendry/2021-06/smoothupgradestovue3","Smooth Upgrades to Vue 3","This post assumes basic familiarity with Vue.js v2.x.",[430,728,554],"/phendry/2021-06/img/vue-transition.jpg","content:phendry:2021-06:SmoothUpgradesToVue3.md",{"user":890,"name":891},{"_path":901,"title":902,"description":903,"image":904,"tags":905,"publishDate":766,"_id":906,"author":907},"/phendry/2021-07-30/spotthevulndataranges","Spot the Vulnerability: Data Ranges and Untrusted Input","In 1997, a flaw was discovered in how Linux and Windows handled IP fragmentation, a Denial-of-Service vulnerability which allowed systems to be crashed remotely.","/phendry/2021-07-30/img/vulnerability.jpg",[328,279],"content:phendry:2021-07-30:SpotTheVulnDataRanges.md",{"user":890,"name":891},{"_path":909,"title":910,"description":911,"tags":912,"image":913,"publishDate":914,"_id":915,"author":916},"/phendry/2021-08-15/exploringdependenttypesinidris","Exploring Dependent Types in Idris","When I'm not coding the \"impossible\" at Art+Logic, I take a lot of interest in new programming technologies and paradigms; even if they're not yet viable for use in production, there can often be takeaways for improving your everyday code.",[465],"/phendry/2021-08-15/img/dependent-types.jpg","2021-08-15","content:phendry:2021-08-15:ExploringDependentTypesInIdris.md",{"user":890,"name":891},{"_path":918,"title":919,"description":920,"tags":921,"image":922,"publishDate":923,"_id":924,"author":925},"/phendry/2021-10-30/spotthevulnloopsandtermconditions","Spot the Vulnerability: Loops and Terminating Conditions","In memory-unsafe languages like C, special care must be taken when copying untrusted data, particularly when copying it to another buffer. In this post, we'll spot and mitigate a past vulnerability in Linux's NTP daemon.",[328,279],"/phendry/2021-10-30/img/vulnerability-2.jpg","2021-10-30","content:phendry:2021-10-30:SpotTheVulnLoopsAndTermConditions.md",{"user":890,"name":891},{"_path":927,"title":928,"description":929,"image":930,"tags":931,"publishDate":932,"_id":933,"author":934},"/phendry/2022-07-21/migratingfromexpresstofastifypart1","Migrating from Express to Fastify, Part 1","Express.js has for years been the dominant lightweight Web framework for Node.js, but over time its development has stalled, with its latest major version (5.0) still in pre-release nearly eight years after its first alpha release. There's a lot to be said for this sort of stability in a foundational dependency for a project, but it's worth assessing whether the added features of competing frameworks are worth making a switch. In this article we'll be looking at Fastify in particular, to understand what it has to offer compared to Express and how difficult it is to migrate an existing Express project.","/phendry/2022-07-21/img/Migrating from Express to Fastify, Part 1.png",[430,279],"2023-12-01","content:phendry:2022-07-21:MigratingFromExpressToFastifyPart1.md",{"user":890,"name":891},{"_path":936,"title":937,"description":938,"image":939,"tags":940,"publishDate":941,"_id":942,"author":943},"/phendry/2022-07-28/migratingfromexpresstofastifypart2","Migrating from Express to Fastify, Part 2","In Part 1, we looked at the features of the Fastify Node.js Web framework compared to Express.js. In Part 2, we'll work through migrating an example Express.js application to Fastify.","/phendry/2022-07-28/img/Migrating from Express to Fastify, Part 2.png",[430,279],"2023-12-31","content:phendry:2022-07-28:MigratingFromExpressToFastifyPart2.md",{"user":890,"name":891},{"_path":945,"title":946,"description":947,"tags":948,"image":949,"publishDate":950,"_id":951,"author":952},"/phendry/2023-01-19/badcode","\"Bad\" Code (Or, Why Software Development is Hard)","Recently, the Dutch government open-sourced the iOS application for their \"DigiD\" authentication service. A tweet with a snippet of that source code, presumably making fun of it, blew up into a debate about whether mocking it is even justified. The amount of debate over such a simple snippet of code highlights, in my mind, just how tricky software development can be.",[328,279],"/phendry/2023-01-19/img/Bad Code.png","2024-01-15","content:phendry:2023-01-19:BadCode.md",{"user":890,"name":891},{"_path":954,"title":955,"description":956,"image":957,"publishDate":958,"tags":959,"_id":960,"author":961},"/phendry/2023-01-31/forgetaboutcodestyle","Forget About [Code] Style","Good code style, being highly subjective, is something often debated among developers. After all, we spend more time reading code than writing it, so it's worth making sure our code is styled to be as easy as possible to read and to understand. On the other hand, deciding upon and continuously enforcing a style is also time-consuming, and the benefits are near-impossible to quantify. Given that modern code formatting tools can fully automate the process, is it still worth fretting about style?","/phendry/2023-01-31/img/forget_style_header.png","2024-02-01",[465],"content:phendry:2023-01-31:ForgetAboutCodeStyle.md",{"user":890,"name":891},{"_path":963,"title":964,"description":965,"image":966,"tags":967,"publishDate":968,"_id":969,"author":970},"/phendry/2023-04-02/semantichtml","Don't Give Up on Semantic HTML","Since the early days of the Web, there has been tension between the ideal of \"semantic HTML\" and the practical reality of designing complex page layouts, which often could not be achieved without inserting style concerns into the document. More recently, frameworks like Tailwind CSS have emerged which challenge the very idea that semantic HTML is an ideal to strive for, and which commit to thoroughly embedding style concerns into HTML documents. With modern CSS features however, semantic HTML is more achievable than ever, and I do think it remains a worthy goal.","/phendry/2023-04-02/img/Don't Give Up on Semantic HTML.png",[465],"2024-03-01","content:phendry:2023-04-02:SemanticHtml.md",{"user":890,"name":891},{"_path":972,"title":973,"description":974,"image":975,"tags":976,"publishDate":977,"_id":978,"author":979},"/phendry/2023-05-16/doyouneedacsspreprocessor","Do You Need a CSS Preprocessor in 2023?","CSS preprocessors like Less, Sass and Stylus have long provided powerful features that vanilla CSS lacked: variables, nesting of rulesets, mixins, control flow constructs, etc. These days however, the feature gap is considerably narrower, and it's not so clear that the benefits of a preprocessor outweight the burdens of setting it up.","/phendry/2023-05-16/img/css_preprocessor_header.png",[465],"2023-01-01","content:phendry:2023-05-16:DoYouNeedACSSPreprocessor.md",{"user":890,"name":891},{"_path":981,"title":982,"description":983,"image":984,"publishDate":985,"tags":986,"_id":987,"author":988},"/phendry/2023-07-28/dependencymanagement","Software Dependency Management: Best Practices","Leveraging third-party libraries and frameworks is essential in most modern software projects, and the projects we build at Art+Logic are no exception. The pressure on developers to rapidly deliver features is high, and there are so many commonalities in the details of each project (particularly in Web development) that a lot of development time can be saved by using well-designed libraries that handle the details.","/phendry/2023-07-28/img/dependency_header.png","2023-01-02",[465],"content:phendry:2023-07-28:DependencyManagement.md",{"user":890,"name":891},{"_path":990,"title":991,"description":992,"tags":993,"publishDate":996,"image":997,"_id":998,"author":999},"/phendry/2023-11-06/frontendframeworksin2024","Frontend Frameworks in 2024: React, Svelte and Vue","Several years ago, Art+Logic settled on Vue.js as our preferred frontend Web framework. Now, in 2024, we feel it's time to revisit the frontend framework landscape to see how things have (or haven't) changed.",[465,994,995,728],"react","svelte","2024-05-15","/phendry/2023-11-06/img/frontend_frameworks_2024.png","content:phendry:2023-11-06:FrontendFrameworksIn2024.md",{"user":890,"name":891},{"_path":1001,"title":1002,"description":1003,"publishDate":1004,"tags":1005,"image":1008,"_id":1009,"author":1010},"/rbrubaker/2012-06/arduino-thermometer","Turn Your Mac into a Thermometer with Arduino","The topic of the Arduino came up around A&L's \"virtual water cooler\" last week. About a year and a half ago, I purchased a SparkFun Inventor's Kit for Arduino. The kit is a fun way for a hardware novice like me to get started and learn some basics. It comes with more than a dozen sample projects such as lighting LEDs, spinning a motor and generating audio.","2012-06-28",[1006,1007],"arduino","java","/rbrubaker/2012-06/img/arduino1.jpg","content:rbrubaker:2012-06:arduino-thermometer.md",{"user":1011,"name":1012},"rbrubaker","Ryan Brubaker",{"_path":1014,"title":1015,"description":1016,"publishDate":1017,"tags":1018,"_id":1021,"author":1022},"/rbrubaker/2012-06/coffe-backbone-1","Fun with CoffeeScript and Backbone.js : Part 1","CoffeeScript has been all the rage lately and I've been wanting to hop on board the bandwagon. I've also seen Backbone.js mentioned quite a bit and was even more intrigued after listening to this .NET Rocks podcast. I decided to convert some plain JavaScript code I had in a side project to use both CoffeeScript and Backbone.js and see how things went.","2012-06-06",[1019,1020,455,279],"backbone-js","coffeescript","content:rbrubaker:2012-06:coffe-backbone-1.md",{"user":1011,"name":1012},{"_path":1024,"title":1025,"description":1026,"publishDate":1027,"tags":1028,"_id":1029,"author":1030},"/rbrubaker/2012-06/coffee-backbone-2","Fun with CoffeeScript and Backbone.js : Part 2","In this post I’ll discuss the code that handles updating the UI.","2012-06-07",[1019,1020,455,279],"content:rbrubaker:2012-06:coffee-backbone-2.md",{"user":1011,"name":1012},{"_path":1032,"title":1033,"description":1034,"publishDate":1035,"tags":1036,"_id":1037,"author":1038},"/rbrubaker/2012-06/coffee-backbone-3","Fun with CoffeeScript and Backbone.js : Part 3","In this post I’ll discuss my thoughts on CoffeeScript and Backbone.js.","2012-06-08",[1019,1020,455,279],"content:rbrubaker:2012-06:coffee-backbone-3.md",{"user":1011,"name":1012},{"_path":1040,"title":1041,"description":1042,"publishDate":1043,"tags":1044,"_id":1045,"author":1046},"/rbrubaker/2012-07/prototypal-js","Prototypal vs. Functional Inheritance in JavaScript","If you ever found JavaScript's prototypal inheritance confusing, do yourself a favor and open this article, open a JavaScript console and code each example in the article. You will definitely come away with a better understanding of how prototypal inheritance works in JavaScript.","2012-07-11",[1020,430],"content:rbrubaker:2012-07:prototypal-js.md",{"user":1011,"name":1012},{"_path":1048,"title":1049,"description":1050,"publishDate":1051,"tags":1052,"_id":1053,"author":1054},"/rbrubaker/2012-07/whither-pm","Whither Project Management?","When I was first asked to manage a project at Art & Logic, I had my reservations. Did I really want to start down a career path that led to less development? Would my skills as a developer go stale? My first few projects as a manager were solo projects so I still had plenty of development work and fortunately, I found myself to be a pretty easy person to manage. As time went on I started managing larger projects and with them came the responsibility to manage other developers. To my surprise I found project management to be rewarding and dare I say, even fun. It's very satisfying to work with clients, helping them define their visions and seeing those visions come to life.","2012-07-25",[819],"content:rbrubaker:2012-07:whither-pm.md",{"user":1011,"name":1012},{"_path":1056,"title":1057,"description":1058,"tags":1059,"image":1062,"publishDate":399,"_id":1063,"author":1064},"/scharette/2019-4/discover_machine_learning","Discover Machine Learning","Computers have been around for less than 100 years.  In that short period of time, some incredible things have happened:  they've been universally adopted so quickly that we have them in our houses.  In our cars.  Even in our pockets.  In the last 40 years, there have been many significant events when it comes to computers:",[1060,1061,391],"machine-learning","neural-networks","/scharette/2019-4/img/discover_machine_learning.png","content:scharette:2019-4:discover_machine_learning.md",{"user":1065,"name":1066},"scharette","Stéphane Charette",{"_path":1068,"title":1069,"description":1070,"publishDate":1071,"image":1072,"tags":1073,"_id":1075,"author":1076},"/shuey/2012-05/baas","BaaS Offerings Continue to Grow","The makers of Simplenote recently introduced their Backend as a Service (BaaS) offering called Simperium that looks to compete in an increasingly crowded space with services like CloudMine, Kinvey, and Parse and to some extent with iCloud for iOS and OS X only apps. So just how crowded is this space? Back in February, Kinvey published their own map of the BaaS ecosystem that highlights different tiers of the ecosystem and various relationships between them.","2012-05-10","/shuey/2012-05/img/header.png",[1074],"baas","content:shuey:2012-05:baas.md",{"user":1077,"name":1078},"shuey","Steven Huey",{"_path":1080,"title":1081,"description":1082,"publishDate":1083,"tags":1084,"image":1086,"_id":1087,"author":1088},"/shuey/2012-05/cloud","Under the Sheets with iCloud and Core Data","Drew McCormack is writing a great series (Part 1, Part 2, Part 3) of posts about using iCloud for syncing Core Data managed data. It's harder than Apple lets on and Drew has done a great job of uncovering how this actually works.","2012-05-28",[15,1085],"icloud","/shuey/2012-05/img/icloud.jpg","content:shuey:2012-05:cloud.md",{"user":1077,"name":1078},{"_path":1090,"title":1091,"description":1092,"publishDate":1093,"tags":1094,"_id":1096,"author":1097},"/shuey/2012-05/economics-android","The Economics of Android","If you haven't already do yourself a favor and head over to asymco.com to catch Horace Deidu's multi-post series on \"The Economics of Android\". Horace and Dan Benjamin discuss the series during this week's Critical Path podcast as well. Horace is a former analyst for Nokia and has been writing Asymco for a few years now. His analysis of the mobile industry and Apple's place within it in particular has been featured in publications such as Bloomberg and Forbes.","2012-05-17",[1095],"android","content:shuey:2012-05:economics-android.md",{"user":1077,"name":1078},{"_path":1099,"title":1100,"description":1101,"publishDate":756,"tags":1102,"_id":1105,"author":1106},"/shuey/2012-05/iot","The Internet of Things and Big Data","I've been following the developments in the \"Internet of Things\" and Big Data / Open Data markets as new apps and tools are released and they look to be two exciting technologies on a collision course. With the advent of internet connected home appliances like Wattvision and Nest that provide real utility to the average home owner at reasonable prices along with crowd funded projects like Air Quality Egg or Twine we should see an explosion in the kinds and amount of useful and real-time or near real-time data that is available to anyone with a smartphone. Health metric or \"quantitative self\" tracking devices such as Fitbit, Jawbone Up, and the Pebble watch will fuel this data explosion as well.",[1103,1104],"big-data","iot","content:shuey:2012-05:iot.md",{"user":1077,"name":1078},{"_path":1108,"title":1109,"description":1110,"publishDate":1111,"tags":1112,"image":1114,"_id":1115,"author":1116},"/shuey/2012-05/rubymotion","RubyMotion Brings Ruby to iOS","RubyMotion is a new development toolchain that allows you to build iOS apps using Ruby created by Laurent Sansonetti, a former Apple engineer and contributor to the MacRuby project. It has garnered a lot of attention the past few weeks and some detailed reviews have already been written:","2012-05-14",[314,1113],"ruby","/shuey/2012-05/img/logotype-icon.png","content:shuey:2012-05:rubymotion.md",{"user":1077,"name":1078},{"_path":1118,"title":1119,"description":1120,"publishDate":1121,"tags":1122,"image":1123,"_id":1124,"author":1125},"/shuey/2012-06/thoughts-ios6","A few thoughts on iOS 6","Apple made their session videos from WWDC 2012 available earlier this week in record time. It's nice to see since tickets for this years event sold out in under two hours. Apple has an iOS 6 Preview page touting some of the new features such as Siri's new abilities, tighter integration with Facebook, Photo Stream sharing, and things like iCloud tabs for Safari all of which look great.","2012-06-21",[15,314],"/shuey/2012-06/img/ios6.jpg","content:shuey:2012-06:thoughts-ios6.md",{"user":1077,"name":1078},{"_path":1127,"title":1128,"description":1129,"publishDate":1130,"tags":1131,"image":1133,"_id":1134,"author":1135},"/shuey/2012-07/mixer","A Simple Mixer Using AVFoundation","In iOS 4.0 Apple introduced the AV Foundation APIs that made working with audio and video media much easier than it had been in previous versions of iOS. Apple then brought these APIs to Mac OS X in OS X 10.7 \"Lion\". In this post I'll show how to use some of the APIs to create a simple four track mixer.","2012-07-02",[15,1132,314],"cocoa","/shuey/2012-07/img/mixer-screenshot.jpg","content:shuey:2012-07:mixer.md",{"user":1077,"name":1078},{"_path":1137,"title":1138,"description":1139,"publishDate":1140,"tags":1141,"image":1142,"_id":1143,"author":1144},"/shuey/2012-07/reset-button","The Reset Button","Horace Dediu of Asymco has been publishing some fantastic insights and analysis of the mobile market in the past few weeks. I linked to some of Dediu's analysis of the Economics of Android in an earlier post, and since then he's updated his work with the latest data and is studying RIM and Microsoft's efforts in the space as well.","2012-07-19",[1095,15,314],"/shuey/2012-07/img/kevin.jpg","content:shuey:2012-07:reset-button.md",{"user":1077,"name":1078},{"_path":1146,"title":1147,"description":1148,"publishDate":1149,"tags":1150,"image":1152,"_id":1153,"author":1154},"/tfarrel/2012-07","Looking at Steganography","With the help of one of my favorite news aggregators, I discovered this article on using JavaScript and the canvas element to hide information inside images. I've long been fascinated by steganography and this article and demonstration makes it even more accessible. If you can't be bothered to read the article, it describes a method of using the HTML5 File API and the canvas element to embed a message in images.","2012-07-24",[1151],"steganography","/tfarrel/2012-07/img/white.png","content:tfarrel:2012-07:index.md",{"user":1155,"name":1156},"tfarrel","Troy Farrel",1780330267101]