Programming

Automatically set window titles in GNU screen

Although GNU screen is actually a window manager, I mainly use it to turn nano into a tabbed text editor. If you do something similar, you might be interested in the way that I automatically set each window’s title to the filename that I’m editing (and reset it back to a default value when I exit the editor).

Make two additions to your .bashrc:

nano () { echo -ne "\ek${1##*/}\e\\"; /usr/bin/nano "$1"; }

This modifies the nano command to set the window title to the filename that’s being edited, sans any leading directories, before opening the file with the real nano. (If your true nano isn’t in /usr/bin/, make the appropriate substitution.)

export PROMPT_COMMAND='echo -ne "\ekbash\e\\"'

This has the effect of resetting the title to “bash” after nano exits, because the prompt has to be printed. (Note that it will reset any custom titles you set manually. There’s probably a better way to reset the title when a program exits, but this has worked for me.)

These two changes allow me to effectively use screen as a tabbed terminal text editor and see at a glance which files I have open in which “tabs.”

Standard
AOL, Plugins, Programming, WordPress

Import your chat transcripts into WordPress

Problem: You have years’ worth of chat (AIM/MSN/Adium/IRC) transcripts saved on your computer, and you’d love to stroll down memory lane and reread some of them, but how? Manually open each one in your browser or chat program?? There’s got to be a better way!

Solution: Install IM-porter, the WordPress plugin that imports your instant message transcripts to your WordPress blog so you can read through conversations from ages past just as easily as you browse the archives of a blog.

After activating the plugin, visit Tools > Import > Chat Transcripts. Then, you can upload a single transcript or a ZIP of transcripts.

IM-porter currently supports transcripts from:

  • AOL Instant Messenger
  • MSN Messenger
  • Adium
  • Colloquy

but you can extend it to parse other formats too. (See the README.)

Imported posts are given the chat post format, and you have the option of tagging them with the participants’ usernames, making them public or private, and adding them to a category.

The IM-porter configuration screen.

After they’re imported, chats will look something like this:

A chat imported by IM-porter

The output may not be perfect (the output will not be perfect), but the original raw transcript is saved as post meta so the post can be updated if you want or fixed by a later version of IM-porter.

If you want to style your chats like AIM used to display them (like in the screenshot I included), you can use this CSS:

.post.format-chat .entry-content p {
	margin-bottom: 0 !important;
}

.post.format-chat .entry-content span {
	color: #00f;
	font-weight: bold;
}

.post.format-chat .entry-content span.participant-1 {
	color: #f00;
}

.post.format-chat .entry-content span time {
	font-size: x-small;
}

Standard
Facebook, Programming, WordPress

Export your Facebook posts to WordPress

I’m a big proponent of owning the data that you create. I use WordPress (of course) wherever I blog, and I use the Keyring Social Importers plugin to make backup copies of my Twitter updates and Foursquare checkins. And as of today, I am also syncing my Facebook updates back to a private WordPress blog using Keyring Social Importers.

Not familiar with Keyring Social Importers? That’s too bad, it’s amazing. Install it, and within minutes, you can be importing data from any one of a dozen sites to your blog. Remember all of that data you put into Myspace/Jaiku/Bebo/Pownce and how it disappeared when the site shut down? Wouldn’t it have been nice to be able to save a copy of all of that? That’s what Keyring Social Importers makes possible.

There’s no built-in Facebook importer in Keyring Social Importers, so I wrote one. You can use it to save copies of your Facebook photos, photo albums, and status updates to an easy-to-browse (and easy to share) WordPress blog.

To use the Facebook importer, install Keyring Social Importers, copy the keyring-importer-facebook.php file into wp-content/plugins/keyring-social-importers/importers/, and then visit Import > Facebook to get started. Let me know what you think.

Note: I’ve only tested the importer on my own account, so it’s possible that it won’t be perfect. All imported data is set to private, just to be safe. Patches and bug reports are welcome.

Standard
Software

Convert iChat transcripts to (useable) XML

iChat (a.k.a. Messages) doesn’t store its chat logs in a format that can be (easily) parsed outside of iChat, so if you want to use your chat data for anything, you’ll need to convert those logs to something better. (Anything would be better. Even AIM’s <FONT>-ridden HTML 3 is more useful than iChat’s binary plists.)

One suggestion I found that came close was to convert the binary plist to XML so that the XML could be parsed with the tool of your choosing:

$ plutil -convert xml1 example.chat

However, the XML generated by that command doesn’t include enough information to reliably determine which participant sent each message. (Or if it does, it doesn’t make it obvious enough that I would dedicate time to writing a parser.)

After trying every other promising script and program, here is the only thing that worked:

  1. Install Adium.
  2. Use Adium’s File > Import to import all of your iChat transcripts.
  3. After it finishes, look in ~/Library/Application Support/Adium 2.0/Users/Default/Logs/ and voila! Folders full of XML chat logs with data you can actually use. The file extension is .chatlog, but the content is sensible XML.

This process worked for me to convert 5,000 transcripts created by iChat between 2007 and 2013, and I used Adium 1.5.8.

Standard
Programming

Websites shouldn’t close at 10pm

HealthCare.gov is getting a lot of flack for being inaccessible and just generally bad, so I thought I’d check if Minnesota’s state insurance exchange website is any better. I learned this:

And once the website is up and running, it won’t offer 24/7 service. Instead, it will operate from 6 a.m. to 10 p.m. Monday through Saturday, officials said. There will be no Sunday hours.

MNsure spokesman John Reich said the website’s limited hours won’t be permanent.

“The initial hours are so that we can continually update and improve the user experience and add new features,” he said in an email.

(via TwinCities.com)

I can understand (but don’t agree with the idea of) having some planned downtime when you deploy major new changes (like, maybe an hour), but shutting down every night at 10? That’s especially bad for a website targeted towards people who might be working two jobs and don’t get a chance to visit the site until they get off work late at night. They’re out of luck.


Shameless plug: At Automattic, we continually update and improve the user experience and add new features to products like WordPress.com as often as fifty times per day without making the site unavailable 43% of the time.

(If that sounds good to you, come work with us.)

Standard
Life

Eggs and Toothpicks

In our high-school physics class (a dozen years ago), our fun year-end project was to construct a container out of toothpicks and Elmer’s glue that could prevent an egg from cracking when dropped from a height of about ten feet. The idea was to build toothpick crumple zones around the egg such that the egg’s deceleration would be slowed enough that it wouldn’t crack.

I took a different approach:

2293948192_a3a7ac62e2_o

That’s a parachute made entirely out of dried glue. Not only is the parachute made out of glue, the strings connecting it to the egg basket are made out of glue too. I only used twelve toothpicks for the basket, definitely setting the class maximum for glue and minimum for toothpicks.

So how did it perform?

2293304470_45628d4010_o

Not well. The parachute didn’t catch the air until about a foot from the ground, and the toothpick basket was so flimsy that the egg didn’t stand a chance. The glue strings held up though, so I think that if I had dropped it from thirty feet, it would have had a pretty good chance of landing safely.

If there needs to be a moral to this story, I guess it would be “Don’t put all your eggs in one basket if the basket is made out of twelve toothpicks and attached to a glue parachute, unless you’re going to drop the basket from a height greater than ten feet, in which case, you might be ok.”

Standard
Life, Woodworking

I built a Minneapolis table

Earlier this year, I had an idea to build a table that is an inverted 3-D model of a metropolitan downtown. Take a bunch of 1×2 boards, cut them to scale lengths of the buildings in downtown Minneapolis, glue them together, and you’d have something like this:

1

Invert it, and you can see the city a little more clearly:

2

Each piece of wood represents half a city block between Hennepin and Portland Avenues. (The three tall ones in the middle are the IDS, Wells Fargo Center, and the Capella Tower.)

To get started, I cut a bunch of 3/4″ x 2″ stock I had laying around to the correct lengths and laid them out. (They’re miscellaneous woods, but I think oak, birch, and maple are represented.) Unfortunately, after cutting all 234 pieces of wood, I realized that my miter saw’s blade wasn’t true — it was off by about 1º — so I had to re-trim the ends of all of the blocks with a different saw.

After making those cuts, I glued the adjacent blocks together. I only have about a dozen clamps, so this took about ten gluing sessions.

3

The next step was to drill a hole in each block pair that would accept a threaded insert.

4

The original plan was to just glue all of the pieces to each other, but it’s likely (a.k.a. probable) that the table would fall apart as the wood moved due to humidity. I decided to bolt each piece to the bottom of a separate piece of wood, allowing all of the wood pieces to grow and shrink without colliding with each other. The threaded inserts allowed me to disassemble and reassemble the table as many times as I needed without tearing out any screw holes.

Here’s the top with the bolt holes mapped out; for each bolt, I drilled three holes: 1 pilot hole, one countersink to hide the head, and one for the shank of the bolt to pass through. In the picture below, I’ve attached the legs, drilled all of the pilot holes, and started drilling the countersink holes. (The big board is a glue-up of five maple boards.)

5

Here’s a closeup of the pilot holes.

6

Drilling the countersink holes generated a lot of shavings.

7

Same step, but without the shavings.

8

Here was the first dry-fit. How much would you guess that this table weighs?

9

I procured my first belt sander in order to clean up the edges of all of those pieces. Here’s an example of a pair before: there’s some glue leakage, and the two pieces of wood aren’t exactly the same width, so they don’t line up perfectly. Also pictured: my toe.

10

And here’s the same piece after belt sanding. I love belt sanding.

11

The tabletop is attached to the base with dowels and glue. Here’s the underside of the tabletop:

12

After staining both top pieces, reattaching all of the “buildings” underneath, making sure they were all square to each other, gluing on the top, and adding eight coats of shellac, this is the finished product.

14

A comparison between an updated render and the final product. Looks like I got all of the buildings in the right place!

15

The view from underneath. I thought this looked neat.

16

And from the front.

17

From the first cut to the last shellac coat, it took me about 25 hours over a month of nights and weekends, and I’m very pleased with it — I really enjoy having a one-of-a-kind piece of furniture.

Standard
Open Source

Reminiscing about old code

I imported some long-forgotten CVS repositories into GitHub last night: anyInventory and Toby Web Mail (which contains the first commits I ever made to a version control system). They are nine years old and eight years old, respectively.

Those two projects were my introduction to participating in the Open Source ecosystem. It felt magical that strangers were emailing me, asking if they could help develop and translate these projects that I had previously been working on by myself, albeit in the open.


It’s painful to read some of the code, knowing how insecure it is. There aren’t any indications that I knew about XSS or database query paramaterization. That being said, the applications still work. I was able to install anyInventory on my Web server in about 90 seconds. Thank you, PHP backwards compatibility.


My commit messages were not so great back then:

“Various trivial changes”

The commit? 12 changed files with 227 additions and 195 deletions.


Here’s the Slashdot post that was the impetus for anyInventory. I left a comment suggesting that users could solve their inventory problems by just sending all of their uncatalogued junk to me, and I included my actual mailing address. Despite concerns from other users about a physical Slashdot effect, not one Slashdotter took me up on my offer. (Also note the then-requisite SCO comment at the end of the thread.)

Drew Hearle, who submitted that Slashdot post, ended up working on the anyInventory project with me for a while.

I credit the amount of time I spent on Slashdot for my stances on open source, Web standards, and pony pranks.


I wrote Toby Web Mail so that I wouldn’t have to use Outlook Express any more, and it eventually became the basis of my college senior-year independent study project, anyMail. I think, but cannot confirm, that at one point, there were Web hosts offering both Toby and anyInventory as one-click installs. The thought of this now terrifies me. (See the security section above.)

Standard
WordPress

What software is being conceived right now that, in ten years, will fundamentally alter the landscape of the Web? Who is hacking on a personal project that will eventually be used by millions of people?

Asked ten years ago, the answers to these questions would be WordPress and Matt Mullenweg & Mike Little. I’m not sure what the answer to the first question will be ten years from now, but I hope I’m a part of the second answer.

Decade

Aside
Browser Add-ons, Mozilla, Mozilla Firefox, RSS, RSS Ticker

Major RSS Ticker Update Coming: What You Need to Know

RSS Ticker has been an alternative to Web-based feed readers since 2006, displaying feed updates directly in users’ browsers. It hasn’t seen significant change in a while, but some of the internal Firefox code used by RSS Ticker has changed enough that in order for it to remain functional in Firefox 22, its entire architecture would have to change. That’s a lot of work.

RSS Ticker

I didn’t want to abandon RSS Ticker’s users (especially with the shutdown of Google Reader imminent, already leaving one less feed reading option), but I also couldn’t dedicate the time to completely rewrite the add-on and keep all of its features. So here’s what I’ve done:

RSS Ticker has been completely rewritten. This has given me the opportunity to use the knowledge I’ve gained in the last seven years of programming to improve the overall design of the ticker and to restructure the code to play nicely with the new Firefox APIs.

What hasn’t changed? RSS Ticker will still scroll your feeds in your browser. You can still choose to put it at the top or bottom of your Firefox window. You can still exclude specific feeds. You can mark as read, mark feeds as read, open in tabs, open all in tabs, etc. You can temporarily disable the ticker. You can change the ticker speed, smoothness, and direction. You can hide the ticker automatically when it’s empty.

What has changed? In order to continue supporting RSS Ticker, I’ve had to drop a number of features. Here are some things you can no longer do with RSS Ticker:

  • Manually refresh the feeds.
  • Specify how often the feeds should refresh.
  • Randomize the order of the ticker items
  • Limit the number of items per feed
  • Display items that have already been read
  • Show unread items in bold
  • Manually limit the width of ticker items
  • Customize the context menu

I know some of you liked and used these features. I’m sorry I had to remove them, but it was the choice between removing them or abandoning the add-on altogether.

A few features haven’t been removed, but they have been changed (a.k.a. “improved”):

1. All of the remaining options (six of them, down from a total of 37) are displayed inline in RSS Ticker’s section of the Add-ons Manager.

RSS Ticker Options

2. If you want to temporarily disable the ticker, just uncheck it in the View > Toolbars menu.

Disable RSS Ticker

3. To remove a feed from the ticker (but not from your bookmarks), right-click on it in the Bookmarks Manager and uncheck “Show in RSS Ticker.”

Show a feed in RSS Ticker

This new version will be available in a couple of weeks after some more testing, but if you’d like to test it early, leave your e-mail address in a comment or ping me at chris@efinke.com and I’ll send you a copy.

Standard
Browser Add-ons, Mozilla, Mozilla Firefox, OPML Support, Programming

OPML Support updated for Firefox 20

I’ve just published an update to my OPML Support Firefox extension for the first time in three years. The extension previously added an OPML button to the toolbar in the Bookmarks Manager, but as of Firefox 20, the button disappears because of a change to the way that the Bookmarks Manager’s toolbar is assembled. Version 3 of OPML Support moves the Import OPML and Export OPML options into the existing Backup/Import button’s menu.

opml-menu-mac

Thanks to the OPML Support users who alerted me to the problem via email and in the comments here, since I don’t often have occasion to check whether my buttons are disappearing.

Standard
Automattic, Programming

The Plight of the Self-Taught Programmer

At each company-wide Automattic meetup, every employee is required to give a flash talk on a topic of his or her choosing. At our meetup last September, I gave a short overview of some of the programming terms I don’t pronounce correctly in my head, since I initially learned them on my own and took a guess at their pronunciation. Here are the slides from that fateful day.

The Plight of the Self-Taught Programmer: Chris Finke

a.k.a. Is it pronounced GIF or GIF?

GIF: A picture of a JIF peanut butter jar.

I say “GIF”, but apparently lots of other people pronounce it “GIF”.

animated-jif

Get it?

SQL: My Sequel 2. This time, it's personal.

I learned it as Ess-Cue-Ell and was very confused the first time someone asked me if I had worked with my sequel. “My sequel to what?”

/etc or Etsy?

Still not sure about this one. Thankfully, I never have to say it out loud.

C#: C Sharp or C Pound?

I remember reading a newsgroup post where someone was ridiculed for calling it “C Pound.” That someone was not me, but it could have been.

GUI: A hand covered in something gooey.

Gee-You-Eye is not as fun as the correct gooey pronunciation.

Apple's Operating System: Mackossex.

My college roommate was a diehard Apple fan and especially loved their operating system, “Mackossex” (Mac OS X).

XPI: Zippy.

Firefox extensions use a file format called XPI, pronounced “zippy”, not “ex-pee-eye.”

LaTeX is not the same as latex.

For years, I thought that LaTeX was spoken the same as latex until I heard a professor mention that he had used “layteck” to typeset his papers. In my senior year of college. Dodged a bullet there.

Egill: Not a chance.

I have a coworker from Iceland named Egill. It’s not “Ee-gill”, “egg-ill”, or “Ay-gill”. It doesn’t even have an L-sound if you say it properly, and I’ve been informed that because I’m not from Iceland, I probably can’t physically pronounce it.

Finke = Fing-key. Rhymes with inky. And slinky.

As a public service, here’s a pronunciation guide to my name.

Standard
Life

I was talking with my wife tonight about the shutdown of Google Reader; she uses it heavily and relies on a number of features that aren’t all found in the alternatives. She was worried that on July 1st, she’ll be forced to settle for something that doesn’t meet all of her needs. My reply? “The worst case scenario is that you’ll have to tell me what you want, and I’ll build it for you.”

Being a programmer in the early twenty-first century is like being a blacksmith in the early twentieth century. No matter what is available to the public at large, you have the advantage of being able to build your own tools.

In which I forge a simile

Aside
JavaScript, Typo.js

A practical application of Typo.js

I released Typo.js about two years ago as a pure JavaScript implementation of the Hunspell spellchecker. I’ve been using it in Comment Snob for Chrome, but I haven’t seen many other projects using it. (JavaScript spellchecking is a very narrow niche, to be fair.)

A few days ago, however, I was made aware of a new project using Typo.js: NoTex. It’s a browser-based reStructuredText editor.

NoTex screenshot

The author, Hasan Karahan, has used Typo.js to support 87 (!) different dictionaries. I’m happy to report that spellchecking in the app is smooth and indistinguishable from the native browser spellchecker.

You can follow NoTex’s development on GitHub.

Standard
Life

Welcome to Twenty Thirteen

No, not the year, the theme! Twenty Thirteen will be the default theme in WordPress 3.6, but in order to help test it ahead of time, I’ve already enabled the trunk version of it here on my blog. I also took a few minutes to customize one of the default headers so as to give it a little personal flair; see if you can spot the super-subtle changes I made!

For posterity, and for anyone reading this via a feed reader, here’s how my site looks right now:

My blog with Twenty Thirteen enabled and a customized version of the default header.

It’s a beautiful (and more importantly, readable) theme designed by Joen Asmussen.

If you want to follow Twenty Thirteen’s development, subscribe to the Twenty Thirteen tag on the Make WordPress Core blog.

Standard
Programming

How to stream your ripped DVDs to TiVo with metadata and commentaries

pyTivo is a tool that allows you to stream video files from your computer to your TiVo. It has served me well for watching home movies, but when I decided to rip my DVD collection and watch movies via TiVo instead of using the physical discs, I found two problems:

  1. No metadata. All I see on TiVo is the title (which is just the filename of the video). I’d like to see the year the movie came out, the MPAA rating, the description, etc. just like I do for movies I recorded off of TV.
  2. No commentary tracks. The best part of owning a physical DVD is access to the commentary track(s); unfortunately, pyTivo isn’t able to stream the video in such a way that alternate audio can be selected via TiVo’s built-in audio selection, so the default leaves you with just the first audio track.

I’ve written two scripts to fix these problems, and they’re available at GitHub.

pytivo_metadata.py

pyTivo uses metadata files to supply information to the TiVo about each video file, like the description or the year the movie came out. pytivo_metadata.py generates metadata files for all videos in a given directory (and subdirectories) so you don’t have to write them yourself.

Here’s an example of a filename from a ripped DVD:

/path/to/movies/The Family Man.m4v

This is the movie “The Family Man,” starring Nicolas Cage. Without any metadata, this is what the movie looks like on TiVo:

IMG_6717

Running pytivo_metadata.py /path/to/movies/ will query The Open Movie Database API for information on all videos in that directory that don’t have metadata files; here’s the metadata that pytivo_metadata.py will save to The Family Man.m4v.txt:

title : The Family Man
vProgramGenre : Comedy
vProgramGenre : Drama
vProgramGenre : Romance
isEpisode : false
year : 2000
movieYear : 2000
description : A fast-lane investment broker, offered the opportunity to see how the other half lives, wakes up to find that his sports car and girlfriend have become a mini-van and wife.
mpaaRating : P3
vActor : Nicolas Cage
vActor : Téa Leoni
vActor : Don Cheadle
vActor : Jeremy Piven
vDirector : Brett Ratner
vWriter : David Diamond
vWriter : David Weissman

pyTivo will automatically use this file to supply the metadata for The Family Man.m4v to TiVo.

IMG_6718

(See the pyTivo wiki for more info on the metadata file format.)

No guarantees are made regarding the correct movie being returned from the API, so maybe do a cursory check of that after running the script. Or don’t. Whatever.

pytivo_commentary.py

If you rip a DVD and include multiple audio tracks in the file, you can stream each of those tracks via pyTivo by using pytivo_commentary.py.

If you’re using Handbrake, go to the Audio tab and add each commentary track as an additional entry:

Screen Shot 2013-01-21 at 8.29.54 PM

Then, add a note to the filename to let pytivo_commentary.py know that there’s a commentary track available:

The Family Man (Commentary).m4v

If the movie has more than one commentary, name it like this:

The Family Man (Commentary x 3).m4v

(Yes, The Family Man, starring Nicolas Cage, does have three commentary tracks; one of them is completely dedicated to the music score. What a country!)

pytivo_commentary.py will then generate multiple copies of the video and metadata file (one for each commentary) and add “Commentary #1” (or #2 or #3) to each title. The additional video files are only hardlinks to the original, so no additional disk space is needed.

Screen Shot 2013-01-21 at 8.58.59 PM

Then, add this line to the [Server] section of your pyTivo.conf (and restart the pyTivo service):

audio_lang=commentary

This causes pyTivo to choose the first audio track it can find that includes the label commentary. In the metadata file for each commentary track, you’ll see a line like this:

Override_mapAudio 0.2 : commentary

This tells pyTivo to explicitly use the label commentary for the second audio track instead of whatever the label was which means that when you stream this video, pyTivo will send the commentary audio track instead of the original audio. Because the metadata file for the non-commentary version doesn’t have this override, pyTivo will send the first audio track.

Conclusion

There you have it; now you can rip your DVDs to a hard drive, stream them to your TiVo, listen to the commentaries, and never have to worry about scratches, extra remotes, or mandatory previews before your movie.

These scripts were tested with the wmcbrine version of pyTivo (specifically, the version pulled as of this commit).

The scripts and their required module (pytivo_utilities.py) are available on GitHub in the pyTivo Utilities repository.

Standard
Automattic, Blogging, Life, WordPress

Work with me here.

In 2012, we committed and deployed changes to WordPress.com, on average, every 26 minutes. Why so long between deploys? Because we don’t have enough people working here! Help change that by coming to work with us. We’re looking to hire around 60 people this year; if my math is correct, that’s a lot!

Do you like the Internet? Working from home (or coffee shops or co-working spaces or libraries or your friend’s house or a grassy area with Wi-Fi access)? Making a difference for millions of users? Being part of a profitable company? Retina displays? Work with us.

Do any of these places sound like interesting locations for week-long team meetups? Hawaii, New Zealand, Mexico, Spain, New York City, Miami, San Francisco, Iceland, Boston, Las Vegas, San Diego, Italy, Berlin, Amsterdam, Montreal, Portugal, New Orleans, Chicago, Winnipeg, or Montevideo, Uruguay? Work with us. (And that’s just a selection of 2012 meetup locations.)

Are you a backend developer? Work with us. Mobile developer? Work with us. Customer service and support? Work with us. Designer? Work with us. Front-end developer? Work with us. Wordsmith? Work with us. Sysadmin? Work with us.

Do you dislike cubicles? Dress codes? Commuting? Working 9-5? Work with us.

Working at Automattic is the best job I’ve ever had, and I’ve even worked at AOL, so you know that it must be good. Won’t you come work with me and treat yourself to the best job you’ve ever had?

Standard
Apple, Bug Evangelism, iPhone

iPhone 5 shuts off when opening Camera

I’ve experienced a strange iPhone 5 or iOS 6.0.2 bug three times in the last week. Here are the steps to reproduce:

  1. Use phone until battery level is at the most 40%.
  2. In in an environment with a temperature below 50º Fahrenheit, play music through the phone speakers for at least 30 minutes.
  3. Confirm that the battery level is above 15%.
  4. Open Camera app, with or without stopping the music playback.
  5. Phone immediately shuts off as if the battery was dead.
  6. Attempt to turn phone on; screen shows “no battery” symbol.
  7. Plug phone in to a power source. The phone will turn back on after a few seconds, showing at least 15% battery power.

These specific steps are the result of me using my phone as a stereo while I work on projects in my garage in Minnesota in the winter.

I’ve reported the problem to Apple; has anyone else experienced this? I don’t know if the temperature, music playback, or actual battery level are relevant, but it’s definitely due to opening Camera.

Standard