AtCoderACPercentage

自分と同じくらいのレートの人々の何%がその問題を解けているかを表示する。

Verze ze dne 31. 07. 2019. Zobrazit nejnovější verzi.

  1. // ==UserScript==
  2. // @name AtCoderACPercentage
  3. // @namespace https://github.com/null-null-programming
  4. // @version 0.3
  5. // @description 自分と同じくらいのレートの人々の何%がその問題を解けているかを表示する。
  6. // @author null_null
  7. // @license MIT
  8. // @match https://atcoder.jp/contests/*/standings*
  9. // @require https://ajax.googleapis.com/ajax/libs/jquery/3.3.1/jquery.min.js
  10. // ==/UserScript==
  11.  
  12. (async function () {
  13. //jsonのときは何もしない
  14. if (window.location.href.endsWith("json")) return;
  15. //参加者自身のusername
  16. const userScreenName = getUserScreenName();
  17. //参加者自身のRating
  18. const userRating = await getUserRating(userScreenName);
  19. //コンテスト情報
  20. const contestData = await getContestStandingsData();
  21.  
  22. //問題名のリスト
  23. const problemNames = contestData.TaskInfo.map(task => task.TaskScreenName);
  24. //問題名の記号(AとかBとか)配列 <-これを作っておかないとF1 F2 などが来たときにバグる
  25. const Assignment = contestData.TaskInfo.map(task => task.Assignment);
  26.  
  27.  
  28. //コンテスト情報を辞書型に直す userScreenName->Data
  29. const contestResultData = {};
  30. //参加回数が15回以上のコンテスト参加者のuserScreenNameリスト
  31. let contestUserName = [];
  32.  
  33. contestData.StandingsData.forEach(res => {
  34. //辞書型に変換
  35. contestResultData[res.UserScreenName] = res;
  36.  
  37. //コンテスト参加回数10回未満、自分自身、未提出者は除いてリストに入れる
  38. if (res.Competitions >= 10 && res.UserScreenName !== userScreenName && res.TotalResult.Count > 0)
  39. contestUserName.push(res.UserScreenName);
  40. });
  41.  
  42. //TODO:評価関数の洗練
  43. //自身のレートとの絶対値の差が小さい順に並び替え。
  44. contestUserName.sort(function (x, y) {
  45. return Math.abs(userRating - contestResultData[x].Rating) - Math.abs(userRating - contestResultData[y].Rating);
  46. });
  47.  
  48. //TODO:選抜者人数の見直し
  49. //選抜者人数の10パーセントを選抜者人数とする。
  50. const USER_NAM = contestUserName.length * 0.1;
  51.  
  52. //自身のレートに近いUSER_NAM人の参加者を選抜
  53. contestUserName = contestUserName.slice(0, USER_NAM);
  54.  
  55. //何人が解けたかを問題ごとに集計
  56. let solvedPercentage = problemNames.map(problemName => {
  57. let sum = 0;
  58.  
  59. //各ユーザーごとに集計
  60. contestUserName.forEach(userName => {
  61. if ((contestResultData[userName].TaskResults[problemName] || -1).Score > 0) sum++;
  62. });
  63.  
  64. //小数第1位までパーセントを表示
  65. return Math.round(sum * 10 * 100 / USER_NAM) / 10;
  66. });
  67.  
  68. //結果を表示するテーブルを作成する。
  69.  
  70. //行を追加
  71. let table = document.getElementById('standings-tbody');
  72. let row = table.insertRow(-1);
  73.  
  74. //列を追加
  75. let cells = [];
  76.  
  77. for (let i = 0; i < problemNames.length + 1; i++) {
  78. cells[i] = row.insertCell(i);
  79.  
  80. if (i === 0) {
  81. //行の左端 題名を書き込む
  82. cells[i].innerText = 'AC Percentage';
  83. cells[i].style.color = '#00AA3E';
  84. cells[i].colSpan = '3';
  85. } else {
  86. //問題欄 計算結果と問題名アルファベットを書き込む
  87. cells[i].innerText = Assignment[i - 1] + ':' + solvedPercentage[i - 1] + '%';
  88. cells[i].style.color = '#888888';
  89. }
  90. cells[i].style.fontSize = '80%';
  91. }
  92.  
  93.  
  94. })();
  95.  
  96.  
  97. //参加しているコンテスト名を取得する。
  98. function getContestName() {
  99. let contestURL = location.href;
  100. let contestArray = contestURL.split('/');
  101. return contestArray[contestArray.length - 2];
  102. }
  103.  
  104. //コンテストデータを取得する。
  105. async function getContestStandingsData() {
  106. return await $.ajax(`https://atcoder.jp${window.location.pathname}/json`);
  107. }
  108.  
  109. //参加者自身のusernameを取得する。
  110. function getUserScreenName() {
  111. let userScreenName = document.querySelector("#navbar-collapse > ul.nav.navbar-nav.navbar-right > li:nth-child(2) > a").textContent.split(' ');
  112. return userScreenName[1];
  113. }
  114.  
  115. async function getUserRating(userScreenName) {
  116. let parser = new DOMParser();
  117. let archiveDom = parser.parseFromString((await $.get('https://atcoder.jp/users/' + userScreenName)), "text/html");
  118. let userRating = archiveDom.querySelector("#main-container > div.row > div.col-sm-9 > table > tbody > tr:nth-child(2) > td > span");
  119. return Number(userRating.innerText);
  120. }