学习目标

  1. 掌握使用Tailwind CSS实现GitHub代码查看页的整体布局
  2. 学习如何构建代码查看界面,包括语法高亮和行号显示
  3. 实现文件头部信息区域,包括文件路径和操作按钮
  4. 理解如何使用Tailwind创建交互式代码行号和行高亮功能
  5. 学习实现响应式设计,确保页面在不同设备上的良好表现

页面分析

GitHub代码查看页是开发者浏览和查看仓库代码的核心界面,它包含以下主要区域:

  1. 顶部导航栏 - 与仓库详情页相同,包含GitHub logo、搜索框和用户菜单
  2. 仓库信息头部 - 显示仓库名称和基本信息
  3. 仓库导航菜单 - 提供Code、Issues、PR等功能入口
  4. 文件路径导航 - 显示当前文件的目录路径
  5. 文件操作栏 - 包含各种文件操作按钮(查看原始文件、编辑、删除等)
  6. 代码查看区 - 显示代码内容,包括行号和语法高亮
  7. 文件底部 - 可能包含提交信息或其他元数据

在本节中,我们将重点关注文件路径导航、文件操作栏以及代码查看区域的实现,因为这些部分是代码查看页面的核心特色。

实现步骤

第1步:准备HTML骨架与Tailwind配置

首先创建一个基础HTML文件,并引入Tailwind CSS和必要的配置:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>GitHub 代码查看页 - Tailwind CSS 复刻</title>
  7. <!-- 引入Tailwind CSS -->
  8. <script src="https://cdn.tailwindcss.com"></script>
  9. <!-- 配置Tailwind为GitHub风格 -->
  10. <script>
  11. tailwind.config = {
  12. theme: {
  13. extend: {
  14. colors: {
  15. 'github-blue': '#0969da',
  16. 'github-green': '#2da44e',
  17. 'github-red': '#cf222e',
  18. 'github-yellow': '#bf8700',
  19. 'github-purple': '#8250df',
  20. 'github-bg': '#f6f8fa',
  21. 'github-border': '#d0d7de',
  22. 'github-text': '#24292f',
  23. 'github-text-secondary': '#57606a',
  24. 'github-line-selected': '#fff8c5',
  25. 'github-line-highlight': '#fffbdd',
  26. 'github-diff-add': '#e6ffec',
  27. 'github-diff-delete': '#ffebe9',
  28. },
  29. spacing: {
  30. '4.5': '1.125rem',
  31. }
  32. },
  33. },
  34. }
  35. </script>
  36. <!-- 添加Prism.js用于代码高亮 -->
  37. <link href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/themes/prism.min.css" rel="stylesheet" />
  38. <script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/prism.min.js"></script>
  39. <script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.25.0/components/prism-javascript.min.js"></script>
  40. <style>
  41. /* 自定义样式以匹配GitHub的代码高亮 */
  42. .token.comment { color: #6e7781; }
  43. .token.keyword { color: #cf222e; }
  44. .token.string { color: #0a3069; }
  45. .token.function { color: #8250df; }
  46. .token.number { color: #0550ae; }
  47. .token.operator { color: #24292f; }
  48. .token.class-name { color: #953800; }
  49. pre[class*="language-"] {
  50. margin: 0;
  51. padding: 0;
  52. background: transparent;
  53. }
  54. code[class*="language-"] {
  55. background: transparent;
  56. padding: 0;
  57. border-radius: 0;
  58. white-space: pre;
  59. word-break: normal;
  60. font-family: ui-monospace, SFMono-Regular, SF Mono, Menlo, Consolas, Liberation Mono, monospace;
  61. }
  62. /* 隐藏Prism行号,我们将自己实现 */
  63. .line-numbers .line-numbers-rows {
  64. display: none;
  65. }
  66. </style>
  67. </head>
  68. <body class="bg-white text-github-text">
  69. <!-- 页面内容将在这里构建 -->
  70. </body>
  71. </html>

我们增加了与代码高亮相关的颜色配置,并引入了Prism.js库来帮助我们实现语法高亮。同时添加了自定义CSS来匹配GitHub的代码高亮风格。

注意:通常在实际项目中,我们会通过配置Tailwind来实现自定义样式,但为了简化演示,这里直接使用了内联样式。

第2步:实现文件路径导航

文件路径导航显示当前文件在仓库中的位置:

  1. <!-- 文件路径导航 -->
  2. <div class="bg-white border-b border-github-border">
  3. <div class="container mx-auto px-4 sm:px-6 lg:px-8 py-3">
  4. <div class="flex flex-wrap items-center text-sm">
  5. <div class="flex items-center">
  6. <a href="#" class="text-github-blue hover:underline">repository-name</a>
  7. <span class="mx-1 text-github-text-secondary">/</span>
  8. </div>
  9. <!-- 中间目录 -->
  10. <div class="flex items-center">
  11. <a href="#" class="text-github-blue hover:underline">src</a>
  12. <span class="mx-1 text-github-text-secondary">/</span>
  13. </div>
  14. <div class="flex items-center">
  15. <a href="#" class="text-github-blue hover:underline">components</a>
  16. <span class="mx-1 text-github-text-secondary">/</span>
  17. </div>
  18. <!-- 当前文件 -->
  19. <div class="flex items-center font-semibold">
  20. <span>Button.js</span>
  21. </div>
  22. </div>
  23. </div>
  24. </div>

这个路径导航设计得非常简洁直观,使用斜杠分隔每个目录层级,并将当前文件加粗显示。

第3步:实现文件操作栏

文件操作栏包含各种文件操作按钮和文件信息:

  1. <!-- 文件操作栏 -->
  2. <div class="bg-github-bg border-b border-github-border">
  3. <div class="container mx-auto px-4 sm:px-6 lg:px-8 py-2">
  4. <div class="flex flex-col lg:flex-row lg:items-center justify-between">
  5. <!-- 左侧:文件信息和操作按钮 -->
  6. <div class="flex items-center flex-wrap mb-2 lg:mb-0">
  7. <!-- 行数和大小信息 -->
  8. <div class="flex items-center text-sm text-github-text-secondary mr-4">
  9. <span>78 lines (64 sloc)</span>
  10. <span class="mx-1">·</span>
  11. <span>2.3 KB</span>
  12. </div>
  13. <!-- 操作按钮组 -->
  14. <div class="flex items-center space-x-2">
  15. <!-- 编辑按钮 -->
  16. <button class="flex items-center text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:bg-gray-100">
  17. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  18. <path stroke-linecap="round" stroke-linejoin="round" d="m16.862 4.487 1.687-1.688a1.875 1.875 0 1 1 2.652 2.652L10.582 16.07a4.5 4.5 0 0 1-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 0 1 1.13-1.897l8.932-8.931Zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0 1 15.75 21H5.25A2.25 2.25 0 0 1 3 18.75V8.25A2.25 2.25 0 0 1 5.25 6H10" />
  19. </svg>
  20. Edit
  21. </button>
  22. <!-- 原始文件按钮 -->
  23. <button class="flex items-center text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:bg-gray-100">
  24. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  25. <path stroke-linecap="round" stroke-linejoin="round" d="M19.5 14.25v-2.625a3.375 3.375 0 0 0-3.375-3.375h-1.5A1.125 1.125 0 0 1 13.5 7.125v-1.5a3.375 3.375 0 0 0-3.375-3.375H8.25m0 12.75h7.5m-7.5 3H12M10.5 2.25H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 0 0-9-9Z" />
  26. </svg>
  27. Raw
  28. </button>
  29. <!-- 复制按钮 -->
  30. <button class="flex items-center text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:bg-gray-100">
  31. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  32. <path stroke-linecap="round" stroke-linejoin="round" d="M15.75 17.25v3.375c0 .621-.504 1.125-1.125 1.125h-9.75a1.125 1.125 0 0 1-1.125-1.125V7.875c0-.621.504-1.125 1.125-1.125H6.75a9.06 9.06 0 0 1 1.5.124m7.5 10.376h3.375c.621 0 1.125-.504 1.125-1.125V11.25c0-4.46-3.243-8.161-7.5-8.876a9.06 9.06 0 0 0-1.5-.124H9.375c-.621 0-1.125.504-1.125 1.125v3.5m7.5 10.375H9.375a1.125 1.125 0 0 1-1.125-1.125v-9.25m12 6.625v-1.875a3.375 3.375 0 0 0-3.375-3.375h-1.5a1.125 1.125 0 0 1-1.125-1.125v-1.5a3.375 3.375 0 0 0-3.375-3.375H9.75" />
  33. </svg>
  34. Copy
  35. </button>
  36. </div>
  37. </div>
  38. <!-- 右侧:Git信息和更多操作 -->
  39. <div class="flex items-center flex-wrap">
  40. <!-- Git信息 -->
  41. <div class="flex items-center text-xs mr-4">
  42. <a href="#" class="flex items-center text-github-text-secondary hover:text-github-blue">
  43. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  44. <path stroke-linecap="round" stroke-linejoin="round" d="M3 16.5v2.25A2.25 2.25 0 0 0 5.25 21h13.5A2.25 2.25 0 0 0 21 18.75V16.5M16.5 12 12 16.5m0 0L7.5 12m4.5 4.5V3" />
  45. </svg>
  46. <span>Latest commit</span>
  47. <span class="ml-1 font-mono text-github-blue">a2b9c8d</span>
  48. <span class="ml-1">on</span>
  49. <span class="ml-1 text-github-blue">Mar 15, 2023</span>
  50. </a>
  51. </div>
  52. <!-- 历史按钮 -->
  53. <a href="#" class="flex items-center text-xs text-github-text-secondary hover:text-github-blue mr-4">
  54. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  55. <path stroke-linecap="round" stroke-linejoin="round" d="M12 6v6h4.5m4.5 0a9 9 0 1 1-18 0 9 9 0 0 1 18 0Z" />
  56. </svg>
  57. <span>History</span>
  58. </a>
  59. <!-- 责任按钮 -->
  60. <a href="#" class="flex items-center text-xs text-github-text-secondary hover:text-github-blue">
  61. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-4 h-4 mr-1">
  62. <path stroke-linecap="round" stroke-linejoin="round" d="M15 19.128a9.38 9.38 0 0 0 2.625.372 9.337 9.337 0 0 0 4.121-.952 4.125 4.125 0 0 0-7.533-2.493M15 19.128v-.003c0-1.113-.285-2.16-.786-3.07M15 19.128v.106A12.318 12.318 0 0 1 8.624 21c-2.331 0-4.512-.645-6.374-1.766l-.001-.109a6.375 6.375 0 0 1 11.964-3.07M12 6.375a3.375 3.375 0 1 1-6.75 0 3.375 3.375 0 0 1 6.75 0Zm8.25 2.25a2.625 2.625 0 1 1-5.25 0 2.625 2.625 0 0 1 5.25 0Z" />
  63. </svg>
  64. <span>Blame</span>
  65. </a>
  66. </div>
  67. </div>
  68. </div>
  69. </div>

文件操作栏分为左右两部分:左侧显示文件基本信息和常用操作按钮,右侧显示Git相关信息和其他操作链接。在移动设备上,它们会垂直堆叠以提供更好的可读性。

第4步:实现代码查看区

代码查看区是本页面的核心部分,包括行号和语法高亮的代码内容:

  1. <!-- 代码查看区 -->
  2. <div class="container mx-auto px-4 sm:px-6 lg:px-8 py-4">
  3. <div class="border border-github-border rounded-md overflow-hidden">
  4. <!-- 文件语言栏 -->
  5. <div class="bg-github-bg border-b border-github-border px-4 py-2 flex justify-between items-center">
  6. <div class="text-sm font-medium">
  7. JavaScript
  8. </div>
  9. <div>
  10. <button class="text-github-text-secondary hover:text-github-text">
  11. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-5 h-5">
  12. <path stroke-linecap="round" stroke-linejoin="round" d="M3.75 3.75v4.5m0-4.5h4.5m-4.5 0L9 9M3.75 20.25v-4.5m0 4.5h4.5m-4.5 0L9 15M20.25 3.75h-4.5m4.5 0v4.5m0-4.5L15 9m5.25 11.25h-4.5m4.5 0v-4.5m0 4.5L15 15" />
  13. </svg>
  14. </button>
  15. </div>
  16. </div>
  17. <!-- 代码内容 -->
  18. <div class="overflow-x-auto">
  19. <table class="w-full border-collapse">
  20. <tbody>
  21. <!-- 代码行 -->
  22. <tr>
  23. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L1">
  24. <a href="#L1" class="block py-1">1</a>
  25. </td>
  26. <td class="py-0 pl-4 pr-4 whitespace-pre">
  27. <div class="py-1">
  28. <code class="language-javascript">import React from 'react';</code>
  29. </div>
  30. </td>
  31. </tr>
  32. <tr>
  33. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L2">
  34. <a href="#L2" class="block py-1">2</a>
  35. </td>
  36. <td class="py-0 pl-4 pr-4 whitespace-pre">
  37. <div class="py-1">
  38. <code class="language-javascript">import PropTypes from 'prop-types';</code>
  39. </div>
  40. </td>
  41. </tr>
  42. <tr>
  43. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L3">
  44. <a href="#L3" class="block py-1">3</a>
  45. </td>
  46. <td class="py-0 pl-4 pr-4 whitespace-pre">
  47. <div class="py-1">
  48. <code class="language-javascript"></code>
  49. </div>
  50. </td>
  51. </tr>
  52. <tr class="bg-github-line-highlight">
  53. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-github-line-highlight border-r border-github-border" id="L4">
  54. <a href="#L4" class="block py-1">4</a>
  55. </td>
  56. <td class="py-0 pl-4 pr-4 whitespace-pre">
  57. <div class="py-1">
  58. <code class="language-javascript">/**</code>
  59. </div>
  60. </td>
  61. </tr>
  62. <tr>
  63. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L5">
  64. <a href="#L5" class="block py-1">5</a>
  65. </td>
  66. <td class="py-0 pl-4 pr-4 whitespace-pre">
  67. <div class="py-1">
  68. <code class="language-javascript"> * Button component for primary actions in forms and dialogs.</code>
  69. </div>
  70. </td>
  71. </tr>
  72. <tr>
  73. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L6">
  74. <a href="#L6" class="block py-1">6</a>
  75. </td>
  76. <td class="py-0 pl-4 pr-4 whitespace-pre">
  77. <div class="py-1">
  78. <code class="language-javascript"> */</code>
  79. </div>
  80. </td>
  81. </tr>
  82. <tr>
  83. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L7">
  84. <a href="#L7" class="block py-1">7</a>
  85. </td>
  86. <td class="py-0 pl-4 pr-4 whitespace-pre">
  87. <div class="py-1">
  88. <code class="language-javascript">const Button = ({ children, variant, size, disabled, onClick }) => {</code>
  89. </div>
  90. </td>
  91. </tr>
  92. <tr>
  93. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L8">
  94. <a href="#L8" class="block py-1">8</a>
  95. </td>
  96. <td class="py-0 pl-4 pr-4 whitespace-pre">
  97. <div class="py-1">
  98. <code class="language-javascript"> // Base classes always applied</code>
  99. </div>
  100. </td>
  101. </tr>
  102. <tr>
  103. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L9">
  104. <a href="#L9" class="block py-1">9</a>
  105. </td>
  106. <td class="py-0 pl-4 pr-4 whitespace-pre">
  107. <div class="py-1">
  108. <code class="language-javascript"> const baseClasses = 'inline-flex items-center justify-center rounded-md font-medium focus:outline-none focus:ring-2 focus:ring-offset-2';</code>
  109. </div>
  110. </td>
  111. </tr>
  112. <tr>
  113. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L10">
  114. <a href="#L10" class="block py-1">10</a>
  115. </td>
  116. <td class="py-0 pl-4 pr-4 whitespace-pre">
  117. <div class="py-1">
  118. <code class="language-javascript"></code>
  119. </div>
  120. </td>
  121. </tr>
  122. <tr>
  123. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L11">
  124. <a href="#L11" class="block py-1">11</a>
  125. </td>
  126. <td class="py-0 pl-4 pr-4 whitespace-pre">
  127. <div class="py-1">
  128. <code class="language-javascript"> // Variant specific classes</code>
  129. </div>
  130. </td>
  131. </tr>
  132. <tr>
  133. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L12">
  134. <a href="#L12" class="block py-1">12</a>
  135. </td>
  136. <td class="py-0 pl-4 pr-4 whitespace-pre">
  137. <div class="py-1">
  138. <code class="language-javascript"> const variantClasses = {</code>
  139. </div>
  140. </td>
  141. </tr>
  142. <tr>
  143. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L13">
  144. <a href="#L13" class="block py-1">13</a>
  145. </td>
  146. <td class="py-0 pl-4 pr-4 whitespace-pre">
  147. <div class="py-1">
  148. <code class="language-javascript"> primary: 'bg-blue-600 text-white hover:bg-blue-700 focus:ring-blue-500',</code>
  149. </div>
  150. </td>
  151. </tr>
  152. <tr>
  153. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L14">
  154. <a href="#L14" class="block py-1">14</a>
  155. </td>
  156. <td class="py-0 pl-4 pr-4 whitespace-pre">
  157. <div class="py-1">
  158. <code class="language-javascript"> secondary: 'bg-gray-200 text-gray-900 hover:bg-gray-300 focus:ring-gray-500',</code>
  159. </div>
  160. </td>
  161. </tr>
  162. <tr>
  163. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L15">
  164. <a href="#L15" class="block py-1">15</a>
  165. </td>
  166. <td class="py-0 pl-4 pr-4 whitespace-pre">
  167. <div class="py-1">
  168. <code class="language-javascript"> danger: 'bg-red-600 text-white hover:bg-red-700 focus:ring-red-500',</code>
  169. </div>
  170. </td>
  171. </tr>
  172. <!-- 更多代码行... -->
  173. </tbody>
  174. </table>
  175. </div>
  176. </div>
  177. </div>

代码查看区使用了表格布局,左侧是行号,右侧是代码内容。表格的每一行代表代码的一行,并使用Prism.js进行语法高亮。

为了展示行高亮功能,我们对第4行添加了高亮背景色。在实际应用中,通常会根据URL中的锚点来确定哪一行需要高亮显示。

第5步:添加更多代码行和文件底部信息

为了更完整地展示代码文件,我们添加更多代码行,并在底部添加文件信息:

  1. <!-- 更多代码行... -->
  2. <tr>
  3. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L16">
  4. <a href="#L16" class="block py-1">16</a>
  5. </td>
  6. <td class="py-0 pl-4 pr-4 whitespace-pre">
  7. <div class="py-1">
  8. <code class="language-javascript"> };</code>
  9. </div>
  10. </td>
  11. </tr>
  12. <tr>
  13. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L17">
  14. <a href="#L17" class="block py-1">17</a>
  15. </td>
  16. <td class="py-0 pl-4 pr-4 whitespace-pre">
  17. <div class="py-1">
  18. <code class="language-javascript"></code>
  19. </div>
  20. </td>
  21. </tr>
  22. <tr>
  23. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L16">
  24. <a href="#L16" class="block py-1">16</a>
  25. </td>
  26. <td class="py-0 pl-4 pr-4 whitespace-pre">
  27. <div class="py-1">
  28. <code class="language-javascript"> };</code>
  29. </div>
  30. </td>
  31. </tr>
  32. <tr>
  33. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L17">
  34. <a href="#L17" class="block py-1">17</a>
  35. </td>
  36. <td class="py-0 pl-4 pr-4 whitespace-pre">
  37. <div class="py-1">
  38. <code class="language-javascript"></code>
  39. </div>
  40. </td>
  41. </tr>
  42. <tr>
  43. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L18">
  44. <a href="#L18" class="block py-1">18</a>
  45. </td>
  46. <td class="py-0 pl-4 pr-4 whitespace-pre">
  47. <div class="py-1">
  48. <code class="language-javascript"> // Size specific classes</code>
  49. </div>
  50. </td>
  51. </tr>
  52. <tr>
  53. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L19">
  54. <a href="#L19" class="block py-1">19</a>
  55. </td>
  56. <td class="py-0 pl-4 pr-4 whitespace-pre">
  57. <div class="py-1">
  58. <code class="language-javascript"> const sizeClasses = {</code>
  59. </div>
  60. </td>
  61. </tr>
  62. <tr>
  63. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L20">
  64. <a href="#L20" class="block py-1">20</a>
  65. </td>
  66. <td class="py-0 pl-4 pr-4 whitespace-pre">
  67. <div class="py-1">
  68. <code class="language-javascript"> small: 'px-3 py-1.5 text-sm',</code>
  69. </div>
  70. </td>
  71. </tr>
  72. <tr>
  73. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L21">
  74. <a href="#L21" class="block py-1">21</a>
  75. </td>
  76. <td class="py-0 pl-4 pr-4 whitespace-pre">
  77. <div class="py-1">
  78. <code class="language-javascript"> medium: 'px-4 py-2 text-base',</code>
  79. </div>
  80. </td>
  81. </tr>
  82. <tr>
  83. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L22">
  84. <a href="#L22" class="block py-1">22</a>
  85. </td>
  86. <td class="py-0 pl-4 pr-4 whitespace-pre">
  87. <div class="py-1">
  88. <code class="language-javascript"> large: 'px-5 py-2.5 text-lg',</code>
  89. </div>
  90. </td>
  91. </tr>
  92. <tr>
  93. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L23">
  94. <a href="#L23" class="block py-1">23</a>
  95. </td>
  96. <td class="py-0 pl-4 pr-4 whitespace-pre">
  97. <div class="py-1">
  98. <code class="language-javascript"> };</code>
  99. </div>
  100. </td>
  101. </tr>
  102. <tr>
  103. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L24">
  104. <a href="#L24" class="block py-1">24</a>
  105. </td>
  106. <td class="py-0 pl-4 pr-4 whitespace-pre">
  107. <div class="py-1">
  108. <code class="language-javascript"></code>
  109. </div>
  110. </td>
  111. </tr>
  112. <tr>
  113. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L25">
  114. <a href="#L25" class="block py-1">25</a>
  115. </td>
  116. <td class="py-0 pl-4 pr-4 whitespace-pre">
  117. <div class="py-1">
  118. <code class="language-javascript"> // Combine all classes</code>
  119. </div>
  120. </td>
  121. </tr>
  122. <tr>
  123. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L26">
  124. <a href="#L26" class="block py-1">26</a>
  125. </td>
  126. <td class="py-0 pl-4 pr-4 whitespace-pre">
  127. <div class="py-1">
  128. <code class="language-javascript"> const classes = [</code>
  129. </div>
  130. </td>
  131. </tr>
  132. <tr>
  133. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L27">
  134. <a href="#L27" class="block py-1">27</a>
  135. </td>
  136. <td class="py-0 pl-4 pr-4 whitespace-pre">
  137. <div class="py-1">
  138. <code class="language-javascript"> baseClasses,</code>
  139. </div>
  140. </td>
  141. </tr>
  142. <tr>
  143. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L28">
  144. <a href="#L28" class="block py-1">28</a>
  145. </td>
  146. <td class="py-0 pl-4 pr-4 whitespace-pre">
  147. <div class="py-1">
  148. <code class="language-javascript"> variantClasses[variant],</code>
  149. </div>
  150. </td>
  151. </tr>
  152. <tr>
  153. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L29">
  154. <a href="#L29" class="block py-1">29</a>
  155. </td>
  156. <td class="py-0 pl-4 pr-4 whitespace-pre">
  157. <div class="py-1">
  158. <code class="language-javascript"> sizeClasses[size],</code>
  159. </div>
  160. </td>
  161. </tr>
  162. <tr>
  163. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L30">
  164. <a href="#L30" class="block py-1">30</a>
  165. </td>
  166. <td class="py-0 pl-4 pr-4 whitespace-pre">
  167. <div class="py-1">
  168. <code class="language-javascript"> disabled ? 'opacity-50 cursor-not-allowed' : 'cursor-pointer',</code>
  169. </div>
  170. </td>
  171. </tr>
  172. <tr>
  173. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L31">
  174. <a href="#L31" class="block py-1">31</a>
  175. </td>
  176. <td class="py-0 pl-4 pr-4 whitespace-pre">
  177. <div class="py-1">
  178. <code class="language-javascript"> ].join(' ');</code>
  179. </div>
  180. </td>
  181. </tr>
  182. <tr>
  183. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L32">
  184. <a href="#L32" class="block py-1">32</a>
  185. </td>
  186. <td class="py-0 pl-4 pr-4 whitespace-pre">
  187. <div class="py-1">
  188. <code class="language-javascript"></code>
  189. </div>
  190. </td>
  191. </tr>
  192. <tr>
  193. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L33">
  194. <a href="#L33" class="block py-1">33</a>
  195. </td>
  196. <td class="py-0 pl-4 pr-4 whitespace-pre">
  197. <div class="py-1">
  198. <code class="language-javascript"> return (</code>
  199. </div>
  200. </td>
  201. </tr>
  202. <tr>
  203. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L34">
  204. <a href="#L34" class="block py-1">34</a>
  205. </td>
  206. <td class="py-0 pl-4 pr-4 whitespace-pre">
  207. <div class="py-1">
  208. <code class="language-javascript"> <button</code>
  209. </div>
  210. </td>
  211. </tr>
  212. <tr>
  213. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L35">
  214. <a href="#L35" class="block py-1">35</a>
  215. </td>
  216. <td class="py-0 pl-4 pr-4 whitespace-pre">
  217. <div class="py-1">
  218. <code class="language-javascript"> className={classes}</code>
  219. </div>
  220. </td>
  221. </tr>
  222. <tr>
  223. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L36">
  224. <a href="#L36" class="block py-1">36</a>
  225. </td>
  226. <td class="py-0 pl-4 pr-4 whitespace-pre">
  227. <div class="py-1">
  228. <code class="language-javascript"> disabled={disabled}</code>
  229. </div>
  230. </td>
  231. </tr>
  232. <tr>
  233. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L37">
  234. <a href="#L37" class="block py-1">37</a>
  235. </td>
  236. <td class="py-0 pl-4 pr-4 whitespace-pre">
  237. <div class="py-1">
  238. <code class="language-javascript"> onClick={onClick}</code>
  239. </div>
  240. </td>
  241. </tr>
  242. <tr>
  243. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L38">
  244. <a href="#L38" class="block py-1">38</a>
  245. </td>
  246. <td class="py-0 pl-4 pr-4 whitespace-pre">
  247. <div class="py-1">
  248. <code class="language-javascript"> ></code>
  249. </div>
  250. </td>
  251. </tr>
  252. <tr>
  253. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L39">
  254. <a href="#L39" class="block py-1">39</a>
  255. </td>
  256. <td class="py-0 pl-4 pr-4 whitespace-pre">
  257. <div class="py-1">
  258. <code class="language-javascript"> {children}</code>
  259. </div>
  260. </td>
  261. </tr>
  262. <tr>
  263. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L40">
  264. <a href="#L40" class="block py-1">40</a>
  265. </td>
  266. <td class="py-0 pl-4 pr-4 whitespace-pre">
  267. <div class="py-1">
  268. <code class="language-javascript"> </button></code>
  269. </div>
  270. </td>
  271. </tr>
  272. <tr>
  273. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L41">
  274. <a href="#L41" class="block py-1">41</a>
  275. </td>
  276. <td class="py-0 pl-4 pr-4 whitespace-pre">
  277. <div class="py-1">
  278. <code class="language-javascript"> );</code>
  279. </div>
  280. </td>
  281. </tr>
  282. <tr>
  283. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L42">
  284. <a href="#L42" class="block py-1">42</a>
  285. </td>
  286. <td class="py-0 pl-4 pr-4 whitespace-pre">
  287. <div class="py-1">
  288. <code class="language-javascript">};</code>
  289. </div>
  290. </td>
  291. </tr>
  292. <tr>
  293. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L43">
  294. <a href="#L43" class="block py-1">43</a>
  295. </td>
  296. <td class="py-0 pl-4 pr-4 whitespace-pre">
  297. <div class="py-1">
  298. <code class="language-javascript"></code>
  299. </div>
  300. </td>
  301. </tr>
  302. <tr>
  303. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L44">
  304. <a href="#L44" class="block py-1">44</a>
  305. </td>
  306. <td class="py-0 pl-4 pr-4 whitespace-pre">
  307. <div class="py-1">
  308. <code class="language-javascript">// Default props</code>
  309. </div>
  310. </td>
  311. </tr>
  312. <tr>
  313. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L45">
  314. <a href="#L45" class="block py-1">45</a>
  315. </td>
  316. <td class="py-0 pl-4 pr-4 whitespace-pre">
  317. <div class="py-1">
  318. <code class="language-javascript">Button.defaultProps = {</code>
  319. </div>
  320. </td>
  321. </tr>
  322. <tr>
  323. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L46">
  324. <a href="#L46" class="block py-1">46</a>
  325. </td>
  326. <td class="py-0 pl-4 pr-4 whitespace-pre">
  327. <div class="py-1">
  328. <code class="language-javascript"> variant: 'primary',</code>
  329. </div>
  330. </td>
  331. </tr>
  332. <tr>
  333. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L47">
  334. <a href="#L47" class="block py-1">47</a>
  335. </td>
  336. <td class="py-0 pl-4 pr-4 whitespace-pre">
  337. <div class="py-1">
  338. <code class="language-javascript"> size: 'medium',</code>
  339. </div>
  340. </td>
  341. </tr>
  342. <tr>
  343. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L48">
  344. <a href="#L48" class="block py-1">48</a>
  345. </td>
  346. <td class="py-0 pl-4 pr-4 whitespace-pre">
  347. <div class="py-1">
  348. <code class="language-javascript"> disabled: false,</code>
  349. </div>
  350. </td>
  351. </tr>
  352. <tr>
  353. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L49">
  354. <a href="#L49" class="block py-1">49</a>
  355. </td>
  356. <td class="py-0 pl-4 pr-4 whitespace-pre">
  357. <div class="py-1">
  358. <code class="language-javascript"> onClick: () => {},</code>
  359. </div>
  360. </td>
  361. </tr>
  362. <tr>
  363. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L50">
  364. <a href="#L50" class="block py-1">50</a>
  365. </td>
  366. <td class="py-0 pl-4 pr-4 whitespace-pre">
  367. <div class="py-1">
  368. <code class="language-javascript">};</code>
  369. </div>
  370. </td>
  371. </tr>
  372. <tr>
  373. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L51">
  374. <a href="#L51" class="block py-1">51</a>
  375. </td>
  376. <td class="py-0 pl-4 pr-4 whitespace-pre">
  377. <div class="py-1">
  378. <code class="language-javascript"></code>
  379. </div>
  380. </td>
  381. </tr>
  382. <tr>
  383. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L52">
  384. <a href="#L52" class="block py-1">52</a>
  385. </td>
  386. <td class="py-0 pl-4 pr-4 whitespace-pre">
  387. <div class="py-1">
  388. <code class="language-javascript">// PropTypes</code>
  389. </div>
  390. </td>
  391. </tr>
  392. <tr>
  393. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L53">
  394. <a href="#L53" class="block py-1">53</a>
  395. </td>
  396. <td class="py-0 pl-4 pr-4 whitespace-pre">
  397. <div class="py-1">
  398. <code class="language-javascript">Button.propTypes = {</code>
  399. </div>
  400. </td>
  401. </tr>
  402. <tr>
  403. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L54">
  404. <a href="#L54" class="block py-1">54</a>
  405. </td>
  406. <td class="py-0 pl-4 pr-4 whitespace-pre">
  407. <div class="py-1">
  408. <code class="language-javascript"> children: PropTypes.node.isRequired,</code>
  409. </div>
  410. </td>
  411. </tr>
  412. <tr>
  413. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L55">
  414. <a href="#L55" class="block py-1">55</a>
  415. </td>
  416. <td class="py-0 pl-4 pr-4 whitespace-pre">
  417. <div class="py-1">
  418. <code class="language-javascript"> variant: PropTypes.oneOf(['primary', 'secondary', 'danger']),</code>
  419. </div>
  420. </td>
  421. </tr>
  422. <tr>
  423. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L56">
  424. <a href="#L56" class="block py-1">56</a>
  425. </td>
  426. <td class="py-0 pl-4 pr-4 whitespace-pre">
  427. <div class="py-1">
  428. <code class="language-javascript"> size: PropTypes.oneOf(['small', 'medium', 'large']),</code>
  429. </div>
  430. </td>
  431. </tr>
  432. <tr>
  433. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L57">
  434. <a href="#L57" class="block py-1">57</a>
  435. </td>
  436. <td class="py-0 pl-4 pr-4 whitespace-pre">
  437. <div class="py-1">
  438. <code class="language-javascript"> disabled: PropTypes.bool,</code>
  439. </div>
  440. </td>
  441. </tr>
  442. <tr>
  443. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L58">
  444. <a href="#L58" class="block py-1">58</a>
  445. </td>
  446. <td class="py-0 pl-4 pr-4 whitespace-pre">
  447. <div class="py-1">
  448. <code class="language-javascript"> onClick: PropTypes.func,</code>
  449. </div>
  450. </td>
  451. </tr>
  452. <tr>
  453. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L59">
  454. <a href="#L59" class="block py-1">59</a>
  455. </td>
  456. <td class="py-0 pl-4 pr-4 whitespace-pre">
  457. <div class="py-1">
  458. <code class="language-javascript">};</code>
  459. </div>
  460. </td>
  461. </tr>
  462. <tr>
  463. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L60">
  464. <a href="#L60" class="block py-1">60</a>
  465. </td>
  466. <td class="py-0 pl-4 pr-4 whitespace-pre">
  467. <div class="py-1">
  468. <code class="language-javascript"></code>
  469. </div>
  470. </td>
  471. </tr>
  472. <tr>
  473. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L61">
  474. <a href="#L61" class="block py-1">61</a>
  475. </td>
  476. <td class="py-0 pl-4 pr-4 whitespace-pre">
  477. <div class="py-1">
  478. <code class="language-javascript">export default Button;</code>
  479. </div>
  480. </td>
  481. </tr>

第6步:实现文件底部信息

最后,添加文件底部信息:

  1. <!-- 文件底部信息 -->
  2. <div class="border-t border-github-border py-2 px-4 text-center text-xs text-github-text-secondary">
  3. <p>You can't perform that action at this time.</p>
  4. <p>You signed in with another tab or window. Reload to refresh your session.</p>
  5. </div>

这个底部信息通常会出现在一些页面状态下,比如用户会话过期或没有权限执行某些操作时。

关键样式技术点解析

1. 代码行与行号的实现

GitHub代码查看页面最核心的部分是代码行与行号的实现,我们使用表格布局来确保行号和代码内容能够保持对齐:

  1. <table class="w-full border-collapse">
  2. <tbody>
  3. <tr>
  4. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-white border-r border-github-border" id="L1">
  5. <a href="#L1" class="block py-1">1</a>
  6. </td>
  7. <td class="py-0 pl-4 pr-4 whitespace-pre">
  8. <div class="py-1">
  9. <code class="language-javascript">import React from 'react';</code>
  10. </div>
  11. </td>
  12. </tr>
  13. </tbody>
  14. </table>

关键样式类解析:

  • border-collapse: 确保表格边框合并,创建一个连续的视觉效果
  • py-0 pr-2 pl-4: 精确控制每个单元格的内边距
  • text-right select-none: 使行号右对齐且不可选
  • text-github-text-secondary text-xs: 行号使用次要文本颜色和小号字体
  • w-10: 给行号列一个固定宽度
  • border-r border-github-border: 添加右边框分隔行号和代码
  • whitespace-pre: 保留代码中的空格和换行
  • py-1: 为代码行内容提供适当的垂直间距

2. 行高亮实现

GitHub代码查看页支持行高亮功能,当用户点击行号或通过URL锚点导航时:

  1. <tr class="bg-github-line-highlight">
  2. <td class="py-0 pr-2 pl-4 text-right select-none text-github-text-secondary text-xs w-10 bg-github-line-highlight border-r border-github-border" id="L4">
  3. <a href="#L4" class="block py-1">4</a>
  4. </td>
  5. <td class="py-0 pl-4 pr-4 whitespace-pre">
  6. <div class="py-1">
  7. <code class="language-javascript">/**</code>
  8. </div>
  9. </td>
  10. </tr>

关键样式类:

  • bg-github-line-highlight: 为整行添加高亮背景色
  • 同样的颜色应用于行号单元格 bg-github-line-highlight,保持视觉一致性

3. 语法高亮集成

我们使用Prism.js实现代码语法高亮,并自定义样式以匹配GitHub的风格:

  1. /* 自定义样式以匹配GitHub的代码高亮 */
  2. .token.comment { color: #6e7781; }
  3. .token.keyword { color: #cf222e; }
  4. .token.string { color: #0a3069; }
  5. .token.function { color: #8250df; }
  6. .token.number { color: #0550ae; }
  7. .token.operator { color: #24292f; }
  8. .token.class-name { color: #953800; }

每种代码元素(注释、关键字、字符串等)都有特定的颜色,精确匹配GitHub的代码高亮风格。

4. 响应式导航与操作按钮

文件操作栏在不同屏幕尺寸下有不同的布局:

  1. <div class="flex flex-col lg:flex-row lg:items-center justify-between">
  2. <!-- 左侧内容 -->
  3. <div class="flex items-center flex-wrap mb-2 lg:mb-0">
  4. <!-- 内容 -->
  5. </div>
  6. <!-- 右侧内容 -->
  7. <div class="flex items-center flex-wrap">
  8. <!-- 内容 -->
  9. </div>
  10. </div>

关键样式类:

  • flex flex-col lg:flex-row: 在移动设备上垂直堆叠,在大屏幕上水平排列
  • flex-wrap: 允许按钮在空间不足时自动换行
  • mb-2 lg:mb-0: 在移动设备上添加底部边距,在大屏幕上移除

5. 按钮组与链接样式

GitHub代码查看页面有多种按钮和链接样式:

  1. <button class="flex items-center text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:bg-gray-100">
  2. <svg class="w-4 h-4 mr-1"><!-- ... --></svg>
  3. Edit
  4. </button>
  5. <a href="#" class="flex items-center text-xs text-github-text-secondary hover:text-github-blue">
  6. <svg class="w-4 h-4 mr-1"><!-- ... --></svg>
  7. <span>History</span>
  8. </a>

按钮样式:

  • 浅灰背景 (bg-github-bg)
  • 细边框 (border border-github-border)
  • 圆角 (rounded-md)
  • 小字体 (text-xs)
  • 图标+文本组合 (flex items-center)
  • 悬停效果 (hover:bg-gray-100)

链接样式:

  • 次要文本颜色 (text-github-text-secondary)
  • 悬停时变为蓝色 (hover:text-github-blue)
  • 无背景和边框
  • 同样的图标+文本布局

常见问题与解决方案

  1. 问题: 如何保持代码格式和空格不变
    解决方案: 使用 whitespace-pre 类保留代码中的所有空格和换行,而不是默认的文本换行行为。

  2. 问题: 行号与代码内容对齐
    解决方案: 使用表格布局确保行号与代码内容垂直对齐,即使在代码内容高度变化时也能保持一致。

  3. 问题: 如何实现行高亮功能
    解决方案: 为整个表格行添加背景色类 bg-github-line-highlight,并确保行号单元格也使用相同的背景色。

  4. 问题: 代码超出视口宽度时的处理
    解决方案: 使用 overflow-x-auto 类允许代码区域水平滚动,而不破坏页面整体布局。

  5. 问题: 语法高亮与GitHub样式不匹配
    解决方案: 使用自定义CSS覆盖Prism.js的默认样式,精确匹配GitHub的代码高亮颜色。

扩展与练习建议

练习1:实现代码行选择功能

扩展代码,使用JavaScript实现多行选择功能,当用户点击行号并按住Shift键点击另一行时,高亮显示所选范围内的所有行。

练习2:添加代码注释功能

在特定代码行右侧添加注释图标按钮,点击后弹出注释输入框。

练习3:实现代码折叠功能

为代码添加折叠功能,可以折叠/展开特定的代码块(如函数、类定义等)。

练习4:实现”复制到剪贴板”功能

使用JavaScript实现”复制”按钮的功能,点击后将代码复制到剪贴板,并显示成功提示。

练习5:添加行内代码建议功能

实现类似GitHub Copilot的简单版本,在特定代码行下方显示建议的下一行代码。

总结

在本节中,我们成功实现了GitHub代码查看页的核心界面元素,重点关注代码显示、语法高亮和行号实现。关键实现点包括:

  1. 使用表格布局实现代码行与行号的精确对齐
  2. 通过Prism.js实现语法高亮,并自定义样式匹配GitHub的风格
  3. 实现行高亮功能,支持通过URL锚点定位到特定代码行
  4. 创建响应式文件操作栏,包含编辑、原始查看和复制等功能
  5. 精确还原GitHub的交互元素,如按钮、链接和图标组合

通过这些实现,我们不仅展示了Tailwind CSS在复杂UI组件中的应用,也深入理解了GitHub代码查看界面的设计原则。代码查看页面的设计重点在于提供清晰的代码可读性、便捷的导航和实用的代码操作功能。

在下一节中,我们将探讨Commit历史页面的实现,学习如何创建时间线和变更记录视图。