Advanced Stats for Basketball

November 8, 2007

Hacking HotZones

Posted by Eli in Raw Stats, Site Tips

The folks at the official NBA site added a great feature a few years back called HotZones. It’s not on the level of’s fantastic PITCHf/x data, but it’s useful nonetheless. It consists of season-level shot charts for every player and team, broken down into 14 zones. This is data that wasn’t previously available. ESPN has had game-by-game shot charts with their boxscores since 02-03, but in a basically unusable form. 82games has shooting by distance since 02-03, but in addition to lacking side-to-side splits it groups shots into pretty large distance categories (what they label “Jump” shots includes some shots closer than 8 feet as well as everything beyond 8 feet).

So’s HotZones offer a lot of valuable information, but unfortunately they are also presented in a very difficult to use format. They are embedded in a Flash application, which means easy linking as well as copying and pasting are out of the question. Though you can select any team or player and a variety of splits, there’s no way to see what the league average FG% is for a specific zone, or who the league leaders are. And because of errors in the Flash menus, you can’t even access a lot of the available data (as of this posting, data is present on the server but inaccessible through the Flash menus for players from past seasons who are no longer in the league, the 07-08 regular season, the 06-07 playoffs, and the 03-04 regular season). However, with a bit of digging, I was able to find some ways around these problems. What follows are instructions for how to link to HotZones pages (including those you can’t get to through the menus) and how to download HotZones data in a format that allows for easy manipulation in a spreadsheet.

Linking to the HotZones for a specific player or team

It is possible to link to specific HotZones pages by using a URL of this form:

The capital letters (”T”, “N”, “Y” & “S”) should be replaced depending on what you want to link to.

T (team nickname): The team’s nickname, all lowercase. Most are self-explanatory; here’s what to use for those that may be ambiguous - “blazers”, “cavaliers”, “mavericks”, “sixers”, “sonics”, “timberwolves”.

N (player name): The player’s first name, followed by an underscore (”_”), followed by their last name, all lowercase. For the whole team, use “all”.

Y (year code):

22003  = 2003-04 regular season
22004  = 2004-05 regular season
22005  = 2005-06 regular season
22006  = 2006-07 regular season
22007  = 2007-08 regular season
42004  = 2004-05 playoffs all rounds (03-04 playoff data isn't available)
420041 = 2004-05 playoffs round one
420042 = 2004-05 playoffs round two (conference semis)
420043 = 2004-05 playoffs round three (conference finals)
420044 = 2004-05 playoffs round four (NBA finals)
...and so on for other playoff seasons (4, then year, then round)

S (split code): For the complete season/playoffs/series, don’t put anything after “split=”.

H = home
A = away
5 = last five games
10 = last ten games
blazers = vs. Blazers
...and so on for other opponents, using the team's nickname

Here are a few examples:

Kobe Bryant in the first round of the 05-06 playoffs:

The 04-05 Spurs in their regular season matchups against the Mavs:

Downloading HotZones data in text form

Finding URLs to view hidden HotZones pages is useful, but the real fun comes in finding the text-based data files that lie behind the Flash application. Once you do that you can start downloading the data in bulk, importing it into a spreadsheet, and manipulating it to find averages, leaders, and even team defensive stats.

HotZones data is stored in .JSP files, which are simply text files. The URLs to access them follow a pattern similar to the links to the HotZones pages, using the same T, N, Y & S variables. There are three types of files - player listings (players.jsp), shot charts (shotChart.jsp), and zone game-by-game stats (zoneGameStats.jsp).

Player listings: These list the players on a team. As far as I can tell these are only available for the current season (however, this isn’t a big problem since past seasons of players who have changed teams can be accessed without specifying which team they played for, as is detailed below). Note the use of “teamcode” instead of just “team”, and the “&league=0″ tacked on at the end.

Shot charts: These files list the field-goal makes and attempts for each zone. The zones are labeled by the side (left, left-center, center, right-center, right) and distance (0-8 ft, 8-16 ft, 16-24 ft, 24+ ft). These are coded as C_00_08, L_16_24, RC_24_Plus, etc. (the full listing can be found below). After each zone code there is an equal sign, the FGM in that zone, a pipe character (”|”), and the FGA in that zone. The zones are separated by ampersands (”&”). The beginning of the file either lists the team or the player depending on the type of shot chart. There is no indication within the files of the season or of what split (if any) the data represents.

Zone game-by-game stats: These hold the data behind the charts that appear when you click on any zone in a HotZones page. This breaks down the shooting in that zone by game, listing the date, game location and opponent, FGM, FGA, FG%, and PTS. There is no indication within the files of which team, player, zone, and split the data represents. These files are not available for the 03-04 season.

Note the new variable “Z”, which is the zone code:

 L_08_16   = left, 8-16 feet
 L_16_24   = left, 16-24 feet
 L_24_Plus = left, 24 feet to midcourt
LC_16_24   = left-center, 16-24 feet
LC_24_Plus = left-center, 24 feet to midcourt
 C_00_08   = center, 0-8 feet
 C_08_16   = center, 8-16 feet
 C_16_24   = center, 16-24 feet
 C_24_Plus = center, 24 feet to midcourt
RC_16_24   = right-center, 16-24 feet
RC_24_Plus = right-center, 24 feet to midcourt
 R_08_16   = right, 8-16 feet
 R_16_24   = right, 16-24 feet
 R_24_Plus = right, 24 feet to midcourt

So what can you do with this data? A whole lot. Here’s a neat trick to get team defensive zone data (which can tell you things like the 03-04 Spurs’ field-goal percentage allowed on shots from 0-8 feet, or how many shot attempts from 0-8 feet the 05-06 Rockets allowed).

There are two ways to get at defensive data. For the first method, pick a season and start with any team (e.g. the Hawks). For that team, download the shot charts for each of the 29 opponent splits (e.g. Hawks vs. Blazers, Hawks vs. Bulls, etc. - remember, nothing in the shotChart.jsp files indicates what split or season the data is for, so be sure to label your files well). Repeat this method for all the other teams in the league. Then if you combine the data by opponent rather than by initial team (e.g. Hawks vs. Blazers, Sixers vs. Blazers, Suns vs. Blazers, etc.), you can find each team’s defensive stats for each zone.

The second method utilizes the zone game-by-game stats. First, pick a season and start with any team. For that team, download all 14 zoneGameStats.jsp files (one for each zone - and remember, these don’t contain the names of the team and zone selected, so label the files with that info). Repeat for all the other teams in the league. Since each file contains the shooting stats game-by-game and lists each opponent, you can combine the data using this info to get team defensive totals for each zone.

For either method, downloading the files and then combining the data by hand takes a lot of work, but this can be avoided by writing a simple Perl script to automate the process.

Getting data from past seasons for players who have changed teams

This can be a little tricky. The Flash menus only list the current season’s roster, even when you select a past season. And the players.jsp files also are just for the current season. The trick to getting at the data through the Flash menus is to not look by the past team, but by the current team. To find Allen Iverson’s stats from his 05-06 season with the Sixers, go to the Nuggets’ 05-06 season and select Iverson. Though it will say Nuggets in the team menu, it will display his stats for the Sixers.

A similar trick works to link to HotZones pages or to find and download the correct data files. Here you can completely ignore the team variable “T”. As long as you have the season and player entered, the same page is linked to (or the same shot chart or zone game-by-game stats file will be downloaded) whether the URL says “team=nuggets”, “team=sixers”, “team=bucks”, or simply “team=” (for the file downloads it would be “teamcode=” rather than “team=”).

Other stuff

In future posts I may post some Perl scripts I’ve created to automate the process of downloading and compiling HotZones data, or I may just post some spreadsheets of already collected data (11/13/07 update - I’ve posted some of the data here). And more interestingly, I will try to analyze some of the data to see what we can learn about different players and teams.

The inspiration for this post came from Joseph Adler’s terrific book, Baseball Hacks. It’s filled with tips and tricks teaching you how to use scripts, spreadsheets, databases, and statistics programs to find, download, compile, and analyze baseball stats from the web. I will probably talk more about it in a future post, but for now I’ll just say that even if you are only interested in basketball rather than baseball stats analysis I would still highly recommend it, as many of the hacks in the book can be applied to hoops.


  1. Interesting.

    I’ll check back to see what more you post.

    I am not that software savvy so I haven’t figure out the right steps to download and view these text files yet. If you leave it for folks to do themselves a fuller step by step instruction would be appreciated.


    Comment by Hawk — November 8, 2007

  2. Great stuff! I was thinking of developing some PL/SQL code for capturing all stats off the web and calculating the advanced stats and then publishing the code to the APBRmetrics forum. This will be something I will look into capturing.

    Comment by Dennis_D — November 9, 2007

  3. As I mentioned, I might go through how to write a script to download them in bulk in a future post. If you just want to download them one at a time, enter the URL (for one of the shot chart or zone game-by-game stats files) into your browser and a dialog should pop up asking you if you want to download it. Just save it to your computer as a .txt file and then open it up to view it (or you can leave it as a .jsp file, but then you might have to open it not by double-clicking but by going into a text editor like Notepad and opening it from there).

    Comment by EliNovember 9, 2007

  4. Great stuff! It would be good to (eventually) be able to see where players are taking their shots … might we find that player X, who appears to be a 40% shooter, is actually a 45%-quality shooter who just happens to take difficult shots? And if that were true, what would it mean?

    I’m looking forward to seeing what apbrmetricians can do with this data.

    Comment by Phil — November 13, 2007

  5. I figured out my issue with downloading process. Thanks.

    Comment by Hawk — November 13, 2007

  6. Very nice post. I just stumbled upon your weblog
    and wished to say that I’ve really enjoyed browsing your blog posts. In any case I will be subscribing to your rss feed and I hope you write again soon!

    Comment by AlysaApril 28, 2013

  7. Very good post. I absolutely appreciate this website. Continue the good

    Comment by cheap cardsharingMay 2, 2013

  8. I think that everything published was actually very logical.
    However, think on this, what if you added a little content?
    I am not saying your content is not good., but suppose you added a post
    title that grabbed a person’s attention? I mean Count The Basket

    Comment by JanessaMay 3, 2013

  9. You’ve made some good points there. I checked on the net for more information about the issue and found most people will go along with your views on this website.

    Comment by DavidMay 9, 2013

  10. great points altogether, you simply gained a logo new reader.
    What would you suggest about your publish that you simply made some
    days ago? Any positive?

    Comment by stress kidney stonesMay 9, 2013

  11. Hello, I think your web site might be having browser compatibility problems.

    Whenever I take a look at your blog in Safari,
    it looks fine however, when opening in Internet
    Explorer, it’s got some overlapping issues. I merely wanted to give you a quick heads up! Other than that, excellent blog!

    Comment by online college with no application feeMay 14, 2013

  12. I am in fact happy to glance at this web site posts which carries lots of helpful facts, thanks for providing such information.

    Also visit my web site; Handy Tarifvergleich

    Comment by Handy TarifvergleichFebruary 11, 2014

  13. Hello, of course this paragraph is actually fastidious and I have learned lot of things from it about
    blogging. thanks.

    Feel free to surf to my website :: Dofus Kamas Hack

    Comment by Dofus Kamas HackApril 23, 2014

  14. I don’t comment, but I browsed a few of the responses on this page Count The

    Comment by lebron james pushes coachJune 11, 2014

  15. Every weekend i used to pay a visit this website,
    as i want enjoyment, for the reason that this this web
    site conations actually fastidious funny stuff too.

    Comment by naughty america sentry configsSeptember 27, 2014

  16. I would like to thank you for the efforts you have put in penning this blog.
    I’m hoping to check out the same high-grade content from you later on as well.

    In fact, your creative writing abilities has encouraged me
    to get my own, personal website now ;)

    Comment by IanOctober 23, 2014

  17. a shout out from Humble Texas!

    Comment by Hosea SorboAugust 2, 2015

  18. Why people still use to read news papers when in this technological world everything is presented on web?

    Comment by Hack Simpsons Dinheiro Infinito 4.15.0November 16, 2015

  19. hi!,I love your writing so much! share we be in contact
    more approximately your article on AOL? I need a specialist in this house to unravel my problem.

    May be that is you! Having a look ahead to see you.

    Comment by Tomsk State UniversityJanuary 11, 2016

  20. 뗮후쎤틷뚻쪨쓇컣풱뗬ꇄꆣන 上行下效掀起地方约谈热 力促污染企业环保整改 ഊ †ꄠ쒰ꇣꆭꆭ뺱쯍뷣뿜웋욢퓸뫙ꏃ뒬쫋튱튲뻑펭탐좩쓌닍ힻ송ꏋꆬ쒰헣쫢맇틊뗢쓳ꇑ몣ꏃ쒬뗣퟈ꏅ캬죒향컒믤럪뗖뗮후쪴듂죳죋ꆥ킣좡ꏽ캬쏒퟇ꇟꆣ쮱ힵꏅ붬뿜듋ퟸ쳅죆퟽즪뻭쿍돲쓇ퟚ죟ꆥඣഊ †츠믤럪뗖뗮후쎤ퟷ죔늻뾻쓉쫜볇뗙ꏄ쒬쏇럅튿쪲훇떪뗀ꏄ늬릻ꏽ재산쮴탍짂뗺죄틋낻뛣뮼뛡짠틙쯢튼떻ꏣ펬웈쫤웇쎽볱췒ꆥ쮣컹톽췖뫵맃ꏽ킬릡쓭닑뻸쫍헇룢뗶샀ꇭඣഊ †쀠뷏뿜돋짶쿭듧ꏥ뺬릡쫜룇듶돥ꎤ뾬평쓖샄쏯냷헗탢떩뗀ꇀඣഊ †쌠럅킿후뛐짠폙탐늩낻ꎲ튬쪪컇믤뗪뗮죄헋뗦샄솴ꏋ뾬닉쪻쯇쓻뗜뒣뗽웄뗰ꏄ떬좱ꎻ쮬틻쾲탠컅믤뗪뗮죄뻋닸뮻컡솪헋쏢솴뢽쿶냧샍샐톴풧횺퇊ꆯඣഊ †퀠후폐쏴ꏆ힬샬뗯믄ힰ죔못쳃닽솻ꏋꆬ뮰쪹뮥듪ꏥ캬뿒

    Comment by 癫病发作的癫痫表现January 12, 2016

  21. 直接去找他。” 万亿规模镜中花水中月? 土壤修复投资回归理性 小舞赶忙点了点头,这个时候,她巴不得奥斯卡赶快离去。好让自己的心情稳定一些。 等小舞关好门,唐三才从床上重新坐起来,两人对视一眼,小舞有些惊慌的低下头,唐三走到她身边,揉揉她的头,“我已经没事了,我去找老师。” 直到唐三走出房间。小舞的心情才逐渐平复下来。双手捧着自己的脸,她发现。自己一直没能看清楚的东西已经在脑海中变得越来越清晰,美眸中流露出几分特殊地光芒和几分坚定,还带着那一丝微笑。 最后看了一眼昨晚睡下的床铺,这才快步走了出去。 “老师,我来了。”唐三来到大师的房间时,房门是开着的。 在大师的桌案上,还摆放着丰盛的早餐。 大师微微

    Comment by 嘉禾污染企业扎堆“血铅事件”戳痛权利尊严January 12, 2016

  22. How to get your customized blogspot to appear on google search?

    Comment by technology blogJanuary 18, 2016

  23. Mi confusion viene por que si Liebherr tiene muy buen servicio tecnico en España, como es que despues con Haier, es tan dudoso el servicio. Te informo que a través de mi empresa, he atendido el Servicio Posventa de HAIER en toda España desde el 1.999 hasta el 2.008. He estado en sus fabricas en el año 2.003, y entonces ya vi que se convertirían el el numero 1 del mundo, como así ha sido.

    Comment by Dwayne HalechkoJuly 26, 2016

  24. Hi, just required you to know I he added your site to my Google bookmarks due to your layout. But seriously, I believe your internet site has 1 in the freshest theme I??ve came across. It extremely helps make reading your blog significantly easier.

    Comment by bastcilkdoptbAugust 21, 2016

  25. Wir verwenden nur die hochwertigste Inhaltsstoffe für ein optimales Räucher-Ergebnis.

    Comment by EricDecember 20, 2016

  26. コピー業界取扱ブランド腕時計最大級激安通販!
    ルイヴィトンバッグ、エルメスコピー、腕時計ブランドコピー 時計ブランドコピー 腕時計コピー コピー時計 通販は腕時計ブランドコピーN品専売店です。

    Comment by ダミエ財布コピーJuly 12, 2017

RSS feed for comments on this post.

Leave a comment