学习目标

  1. 掌握使用Tailwind CSS实现GitHub Commit历史页面的整体布局
  2. 学习如何构建时间线布局,展示提交历史记录
  3. 实现提交详情卡片,包括作者、时间、消息和变更统计数据
  4. 理解如何使用Tailwind创建互动元素,如提交操作菜单和分支标签
  5. 学习构建响应式设计,确保页面在不同设备上的良好表现

页面分析

GitHub Commit历史页面是开发者查看项目变更历史的重要界面,它主要展示了按时间顺序排列的所有提交记录。该页面包含以下主要区域:

  1. 顶部导航栏 - 与前两节相同,包含GitHub logo、搜索框和用户菜单
  2. 仓库信息头部 - 显示仓库名称和基本信息
  3. 仓库导航菜单 - 提供Code、Issues、PR等功能入口
  4. 提交历史过滤器 - 用于筛选和查看特定分支的提交
  5. 提交时间线 - 按时间顺序展示的提交记录列表
  6. 分页控制器 - 用于在多页提交历史中导航

在本节中,我们将重点关注提交历史过滤器、提交时间线和分页控制器的实现,因为这些部分是Commit历史页的核心特色。

实现步骤

第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 Commit历史页 - 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-commit-green': '#aff5b4',
  25. 'github-commit-green-bg': '#dafbe1',
  26. 'github-commit-red': '#ffd7d5',
  27. 'github-commit-red-bg': '#ffebe9',
  28. },
  29. },
  30. },
  31. }
  32. </script>
  33. </head>
  34. <body class="bg-white text-github-text">
  35. <!-- 页面内容将在这里构建 -->
  36. </body>
  37. </html>

我们添加了用于Commit历史页面的特定颜色,如提交添加和删除的绿色和红色背景。

第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-4">
  4. <div class="flex flex-col sm:flex-row sm:items-center justify-between">
  5. <!-- 左侧: 分支选择器和提交计数 -->
  6. <div class="flex items-center flex-wrap mb-4 sm:mb-0">
  7. <!-- 分支选择器 -->
  8. <div class="flex items-center mr-4">
  9. <button class="flex items-center text-sm bg-github-bg border border-github-border rounded-md px-3 py-1.5 hover:bg-gray-100">
  10. <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">
  11. <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 6H5.25A2.25 2.25 0 0 0 3 8.25v10.5A2.25 2.25 0 0 0 5.25 21h10.5A2.25 2.25 0 0 0 18 18.75V10.5m-10.5 6L21 3m0 0h-5.25M21 3v5.25" />
  12. </svg>
  13. main
  14. <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 ml-1">
  15. <path stroke-linecap="round" stroke-linejoin="round" d="m19.5 8.25-7.5 7.5-7.5-7.5" />
  16. </svg>
  17. </button>
  18. </div>
  19. <!-- 提交计数 -->
  20. <div class="text-sm text-github-text-secondary">
  21. <span class="font-semibold text-github-text">254</span> commits
  22. </div>
  23. </div>
  24. <!-- 右侧: 代码频率和其他选项 -->
  25. <div class="flex items-center">
  26. <!-- 代码频率按钮 -->
  27. <a href="#" class="flex items-center text-sm text-github-text-secondary hover:text-github-blue mr-4">
  28. <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">
  29. <path stroke-linecap="round" stroke-linejoin="round" d="M3 13.125C3 12.504 3.504 12 4.125 12h2.25c.621 0 1.125.504 1.125 1.125v6.75C7.5 20.496 6.996 21 6.375 21h-2.25A1.125 1.125 0 0 1 3 19.875v-6.75ZM9.75 8.625c0-.621.504-1.125 1.125-1.125h2.25c.621 0 1.125.504 1.125 1.125v11.25c0 .621-.504 1.125-1.125 1.125h-2.25a1.125 1.125 0 0 1-1.125-1.125V8.625ZM16.5 4.125c0-.621.504-1.125 1.125-1.125h2.25C20.496 3 21 3.504 21 4.125v15.75c0 .621-.504 1.125-1.125 1.125h-2.25a1.125 1.125 0 0 1-1.125-1.125V4.125Z" />
  30. </svg>
  31. Code frequency
  32. </a>
  33. <!-- 搜索按钮 -->
  34. <a href="#" class="flex items-center text-sm text-github-text-secondary hover:text-github-blue">
  35. <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">
  36. <path stroke-linecap="round" stroke-linejoin="round" d="m21 21-5.197-5.197m0 0A7.5 7.5 0 1 0 5.196 5.196a7.5 7.5 0 0 0 10.607 10.607Z" />
  37. </svg>
  38. Search commits
  39. </a>
  40. </div>
  41. </div>
  42. </div>
  43. </div>

这个过滤器部分包含分支选择器和提交计数,以及右侧的代码频率和搜索按钮。在移动设备上,它们会垂直堆叠以提供更好的可读性。

第3步:实现提交时间线

提交时间线是本页面的核心,它以时间线形式展示所有提交记录:

  1. <!-- 提交时间线 -->
  2. <div class="container mx-auto px-4 sm:px-6 lg:px-8 py-4">
  3. <div class="relative">
  4. <!-- 提交列表 -->
  5. <ul class="relative">
  6. <!-- 提交项1 -->
  7. <li class="relative pl-8 py-4 border-b border-github-border">
  8. <!-- 时间线竖线 -->
  9. <div class="absolute top-0 left-3 h-full w-0.5 bg-github-border"></div>
  10. <!-- 提交点 -->
  11. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-bg border border-github-border"></div>
  12. <!-- 提交内容 -->
  13. <div class="flex flex-col md:flex-row md:items-center justify-between">
  14. <!-- 左侧: 提交信息 -->
  15. <div class="mb-2 md:mb-0">
  16. <h3 class="text-base font-semibold mb-1">
  17. <a href="#" class="text-github-text hover:text-github-blue">Fix bug in API response handling and update documentation</a>
  18. </h3>
  19. <!-- 作者和时间 -->
  20. <div class="flex items-center text-sm text-github-text-secondary">
  21. <a href="#" class="font-medium text-github-text hover:text-github-blue">
  22. <img src="https://github.com/identicons/user1.png" alt="User avatar" class="inline-block w-5 h-5 rounded-full mr-1">
  23. username
  24. </a>
  25. <span class="mx-1">committed</span>
  26. <a href="#" class="hover:text-github-blue" title="March 15, 2023, 14:32 GMT+8">2 days ago</a>
  27. </div>
  28. </div>
  29. <!-- 右侧: 提交哈希和操作按钮 -->
  30. <div class="flex items-center">
  31. <!-- 提交哈希 -->
  32. <a href="#" class="font-mono text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:text-github-blue">
  33. a2b9c8d
  34. </a>
  35. <!-- 复制按钮 -->
  36. <button class="ml-2 text-github-text-secondary hover:text-github-text">
  37. <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">
  38. <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" />
  39. </svg>
  40. </button>
  41. <!-- 浏览代码按钮 -->
  42. <a href="#" class="ml-2 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-5 h-5">
  44. <path stroke-linecap="round" stroke-linejoin="round" d="m6.75 7.5 3 2.25-3 2.25m4.5 0h3m-9 8.25h13.5A2.25 2.25 0 0 0 21 18V6a2.25 2.25 0 0 0-2.25-2.25H5.25A2.25 2.25 0 0 0 3 6v12a2.25 2.25 0 0 0 2.25 2.25Z" />
  45. </svg>
  46. </a>
  47. </div>
  48. </div>
  49. <!-- 变更统计数据 -->
  50. <div class="mt-3 flex items-center text-xs">
  51. <span class="flex items-center px-2 py-1 rounded-md bg-github-commit-green-bg text-github-text">
  52. <span class="font-mono mr-1">+27</span>
  53. <span>additions</span>
  54. </span>
  55. <span class="ml-1 flex items-center px-2 py-1 rounded-md bg-github-commit-red-bg text-github-text">
  56. <span class="font-mono mr-1">-12</span>
  57. <span>deletions</span>
  58. </span>
  59. </div>
  60. </li>
  61. <!-- 提交项2 - 带有分支标签 -->
  62. <li class="relative pl-8 py-4 border-b border-github-border">
  63. <!-- 时间线竖线 -->
  64. <div class="absolute top-0 left-3 h-full w-0.5 bg-github-border"></div>
  65. <!-- 提交点 -->
  66. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-green border border-github-green"></div>
  67. <!-- 分支标签 -->
  68. <div class="absolute top-4 -left-1">
  69. <span class="inline-flex items-center bg-github-blue text-white text-xs px-2 py-0.5 rounded-full">
  70. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-3 h-3 mr-0.5">
  71. <path stroke-linecap="round" stroke-linejoin="round" d="M13.5 6H5.25A2.25 2.25 0 0 0 3 8.25v10.5A2.25 2.25 0 0 0 5.25 21h10.5A2.25 2.25 0 0 0 18 18.75V10.5m-10.5 6L21 3m0 0h-5.25M21 3v5.25" />
  72. </svg>
  73. origin/feature
  74. </span>
  75. </div>
  76. <!-- 提交内容 -->
  77. <div class="flex flex-col md:flex-row md:items-center justify-between">
  78. <!-- 左侧: 提交信息 -->
  79. <div class="mb-2 md:mb-0">
  80. <h3 class="text-base font-semibold mb-1">
  81. <a href="#" class="text-github-text hover:text-github-blue">Implement new search feature with improved performance</a>
  82. </h3>
  83. <!-- 作者和时间 -->
  84. <div class="flex items-center text-sm text-github-text-secondary">
  85. <a href="#" class="font-medium text-github-text hover:text-github-blue">
  86. <img src="https://github.com/identicons/user2.png" alt="User avatar" class="inline-block w-5 h-5 rounded-full mr-1">
  87. collaborator
  88. </a>
  89. <span class="mx-1">committed</span>
  90. <a href="#" class="hover:text-github-blue" title="March 12, 2023, 09:17 GMT+8">5 days ago</a>
  91. </div>
  92. </div>
  93. <!-- 右侧: 提交哈希和操作按钮 -->
  94. <div class="flex items-center">
  95. <!-- 提交哈希 -->
  96. <a href="#" class="font-mono text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:text-github-blue">
  97. f7e8d9c
  98. </a>
  99. <!-- 复制按钮 -->
  100. <button class="ml-2 text-github-text-secondary hover:text-github-text">
  101. <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">
  102. <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" />
  103. </svg>
  104. </button>
  105. <!-- 浏览代码按钮 -->
  106. <a href="#" class="ml-2 text-github-text-secondary hover:text-github-blue">
  107. <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">
  108. <path stroke-linecap="round" stroke-linejoin="round" d="m6.75 7.5 3 2.25-3 2.25m4.5 0h3m-9 8.25h13.5A2.25 2.25 0 0 0 21 18V6a2.25 2.25 0 0 0-2.25-2.25H5.25A2.25 2.25 0 0 0 3 6v12a2.25 2.25 0 0 0 2.25 2.25Z" />
  109. </svg>
  110. </a>
  111. </div>
  112. </div>
  113. <!-- 变更统计数据 -->
  114. <div class="mt-3 flex items-center text-xs">
  115. <span class="flex items-center px-2 py-1 rounded-md bg-github-commit-green-bg text-github-text">
  116. <span class="font-mono mr-1">+156</span>
  117. <span>additions</span>
  118. </span>
  119. <span class="ml-1 flex items-center px-2 py-1 rounded-md bg-github-commit-red-bg text-github-text">
  120. <span class="font-mono mr-1">-8</span>
  121. <span>deletions</span>
  122. </span>
  123. </div>
  124. </li>
  125. <!-- 提交项3 - 合并提交 -->
  126. <li class="relative pl-8 py-4 border-b border-github-border">
  127. <!-- 时间线竖线 -->
  128. <div class="absolute top-0 left-3 h-full w-0.5 bg-github-border"></div>
  129. <!-- 提交点 -->
  130. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-purple border border-github-purple"></div>
  131. <!-- 提交内容 -->
  132. <div class="flex flex-col md:flex-row md:items-center justify-between">
  133. <!-- 左侧: 提交信息 -->
  134. <div class="mb-2 md:mb-0">
  135. <h3 class="text-base font-semibold mb-1">
  136. <a href="#" class="text-github-text hover:text-github-blue">Merge pull request #42 from user/feature-branch</a>
  137. </h3>
  138. <!-- 作者和时间 -->
  139. <div class="flex items-center text-sm text-github-text-secondary">
  140. <a href="#" class="font-medium text-github-text hover:text-github-blue">
  141. <img src="https://github.com/identicons/user3.png" alt="User avatar" class="inline-block w-5 h-5 rounded-full mr-1">
  142. maintainer
  143. </a>
  144. <span class="mx-1">committed</span>
  145. <a href="#" class="hover:text-github-blue" title="March 10, 2023, 16:45 GMT+8">1 week ago</a>
  146. </div>
  147. </div>
  148. <!-- 右侧: 提交哈希和操作按钮 -->
  149. <div class="flex items-center">
  150. <!-- 提交哈希 -->
  151. <a href="#" class="font-mono text-xs bg-github-bg border border-github-border rounded-md px-2 py-1 hover:text-github-blue">
  152. 0m1n2p3
  153. </a>
  154. <!-- 复制按钮 -->
  155. <button class="ml-2 text-github-text-secondary hover:text-github-text">
  156. <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">
  157. <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" />
  158. </svg>
  159. </button>
  160. <!-- 浏览代码按钮 -->
  161. <a href="#" class="ml-2 text-github-text-secondary hover:text-github-blue">
  162. <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">
  163. <path stroke-linecap="round" stroke-linejoin="round" d="m6.75 7.5 3 2.25-3 2.25m4.5 0h3m-9 8.25h13.5A2.25 2.25 0 0 0 21 18V6a2.25 2.25 0 0 0-2.25-2.25H5.25A2.25 2.25 0 0 0 3 6v12a2.25 2.25 0 0 0 2.25 2.25Z" />
  164. </svg>
  165. </a>
  166. </div>
  167. </div>
  168. <!-- 查看PR链接 -->
  169. <div class="mt-3">
  170. <a href="#" class="text-sm text-github-blue hover:underline flex items-center">
  171. <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">
  172. <path stroke-linecap="round" stroke-linejoin="round" d="M7.5 21 3 16.5m0 0L7.5 12M3 16.5h13.5m0-13.5L21 7.5m0 0L16.5 12M21 7.5H7.5" />
  173. </svg>
  174. View pull request #42
  175. </a>
  176. </div>
  177. </li>
  178. </ul>
  179. </div>
  180. </div>

提交时间线使用相对定位的列表项和绝对定位的竖线和节点实现。每个提交项包含提交信息、作者信息、提交哈希和变更统计数据。

我们展示了三种不同类型的提交:

  1. 常规提交 - 使用灰色节点
  2. 分支提交 - 使用绿色节点和分支标签
  3. 合并提交 - 使用紫色节点和PR链接

第4步:实现分页控制器

分页控制器用于浏览多页提交历史:

  1. <!-- 分页控制器 -->
  2. <div class="container mx-auto px-4 sm:px-6 lg:px-8 py-4">
  3. <nav class="flex justify-center">
  4. <ul class="flex items-center space-x-1">
  5. <!-- 上一页 -->
  6. <li>
  7. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-border rounded-md text-github-text-secondary hover:text-github-blue hover:border-github-blue">
  8. <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">
  9. <path stroke-linecap="round" stroke-linejoin="round" d="M15.75 19.5 8.25 12l7.5-7.5" />
  10. </svg>
  11. </a>
  12. </li>
  13. <!-- 页码 -->
  14. <li>
  15. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-blue rounded-md bg-github-blue text-white font-medium">
  16. 1
  17. </a>
  18. </li>
  19. <li>
  20. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-border rounded-md text-github-text hover:text-github-blue hover:border-github-blue">
  21. 2
  22. </a>
  23. </li>
  24. <li>
  25. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-border rounded-md text-github-text hover:text-github-blue hover:border-github-blue">
  26. 3
  27. </a>
  28. </li>
  29. <!-- 省略号 -->
  30. <li>
  31. <span class="flex items-center justify-center w-8 h-8 text-github-text-secondary">
  32. ...
  33. </span>
  34. </li>
  35. <!-- 最后一页 -->
  36. <li>
  37. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-border rounded-md text-github-text hover:text-github-blue hover:border-github-blue">
  38. 21
  39. </a>
  40. </li>
  41. <!-- 下一页 -->
  42. <li>
  43. <a href="#" class="flex items-center justify-center w-8 h-8 border border-github-border rounded-md text-github-text-secondary hover:text-github-blue hover:border-github-blue">
  44. <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">
  45. <path stroke-linecap="round" stroke-linejoin="round" d="m8.25 4.5 7.5 7.5-7.5 7.5" />
  46. </svg>
  47. </a>
  48. </li>
  49. </ul>
  50. </nav>
  51. </div>

分页控制器使用均匀的方形按钮实现,包括上一页、页码和下一页按钮。当前页码使用蓝色背景和白色文本高亮显示。

关键样式技术点解析

1. 时间线样式实现

GitHub的提交历史使用时间线布局,通过竖线连接各个提交节点,我们使用以下技术实现:

  1. <li class="relative pl-8 py-4 border-b border-github-border">
  2. <!-- 时间线竖线 -->
  3. <div class="absolute top-0 left-3 h-full w-0.5 bg-github-border"></div>
  4. <!-- 提交点 -->
  5. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-bg border border-github-border"></div>
  6. <!-- 提交内容 -->
  7. <!-- ... -->
  8. </li>

关键样式类解析:

  • relative: 为列表项设置相对定位,作为竖线和节点的定位参考
  • pl-8: 左侧填充为8个单位,为时间线腾出空间
  • py-4: 垂直方向填充为4个单位,创建提交项之间的间距
  • absolute top-0 left-3 h-full w-0.5: 绝对定位的竖线,高度为100%,宽度为0.5个单位
  • absolute top-6 left-2 w-3 h-3 rounded-full: 绝对定位的圆形节点
  • border-b border-github-border: 底部边框分隔不同的提交项

2. 提交类型视觉区分

GitHub通过不同的颜色区分不同类型的提交:

  1. <!-- 常规提交 -->
  2. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-bg border border-github-border"></div>
  3. <!-- 分支提交 -->
  4. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-green border border-github-green"></div>
  5. <!-- 合并提交 -->
  6. <div class="absolute top-6 left-2 w-3 h-3 rounded-full bg-github-purple border border-github-purple"></div>

我们使用不同的背景和边框颜色来区分提交类型:

  • 常规提交:灰色背景和边框
  • 分支提交:绿色背景和边框
  • 合并提交:紫色背景和边框

3. 变更统计数据的视觉表示

提交的代码变更统计(添加和删除)使用颜色编码显示:

  1. <div class="mt-3 flex items-center text-xs">
  2. <span class="flex items-center px-2 py-1 rounded-md bg-github-commit-green-bg text-github-text">
  3. <span class="font-mono mr-1">+27</span>
  4. <span>additions</span>
  5. </span>
  6. <span class="ml-1 flex items-center px-2 py-1 rounded-md bg-github-commit-red-bg text-github-text">
  7. <span class="font-mono mr-1">-12</span>
  8. <span>deletions</span>
  9. </span>
  10. </div>

关键样式:

  • 添加:使用 bg-github-commit-green-bg(浅绿色背景)
  • 删除:使用 bg-github-commit-red-bg(浅红色背景)
  • font-mono:使用等宽字体显示数字,提高可读性和对齐性
  • rounded-md:圆角边框提高视觉效果
  • text-xs:小号字体保持紧凑

4. 分支标签实现

分支标签使用绝对定位显示在时间线的左侧:

  1. <!-- 分支标签 -->
  2. <div class="absolute top-4 -left-1">
  3. <span class="inline-flex items-center bg-github-blue text-white text-xs px-2 py-0.5 rounded-full">
  4. <svg class="w-3 h-3 mr-0.5"><!-- ... --></svg>
  5. origin/feature
  6. </span>
  7. </div>

关键样式:

  • absolute top-4 -left-1:将标签定位在时间线左侧
  • bg-github-blue text-white:蓝色背景和白色文本,高对比度提高可读性
  • rounded-full:完全圆角形成胶囊状标签
  • inline-flex items-center:水平布局图标和文本

5. 响应式布局实现

提交信息在不同屏幕尺寸上有不同的布局:

  1. <div class="flex flex-col md:flex-row md:items-center justify-between">
  2. <!-- 左侧: 提交信息 -->
  3. <div class="mb-2 md:mb-0">
  4. <!-- 内容 -->
  5. </div>
  6. <!-- 右侧: 提交哈希和操作按钮 -->
  7. <div class="flex items-center">
  8. <!-- 内容 -->
  9. </div>
  10. </div>

响应式布局关键点:

  • flex flex-col md:flex-row:在移动设备上垂直堆叠,在中等屏幕以上水平排列
  • mb-2 md:mb-0:在移动设备上添加底部边距,在中等屏幕以上移除
  • justify-between:在水平排列时,将两组元素分别推向左右两侧

常见问题与解决方案

  1. 问题: 时间线在不同高度提交项中的连续性
    解决方案: 使用绝对定位和h-full确保时间线竖线贯穿整个提交项,无论内容高度如何变化。

  2. 问题: 提交信息在小屏幕上的排版
    解决方案: 使用响应式类flex-col md:flex-row在小屏幕上垂直堆叠提交信息和操作按钮。

  3. 问题: 分支标签与时间线的重叠
    解决方案: 使用负边距-left-1将分支标签定位在时间线左侧,确保不重叠。

  4. 问题: 提交节点与时间线的对齐
    解决方案: 精确调整topleft值,确保节点在竖线上居中对齐。

  5. 问题: 分页按钮在不同数量页码下的显示
    解决方案: 使用省略号表示中间页码,并始终显示第一页和最后一页,保持界面简洁。

扩展与练习建议

练习1:实现提交详情展开功能

扩展代码,实现点击提交后展开显示详细的变更内容,包括文件列表和代码差异。

练习2:添加提交过滤功能

实现一个过滤器,允许用户按作者、时间范围或关键词过滤提交记录。

练习3:实现提交图表可视化

使用SVG或Canvas创建一个提交历史图表,显示分支和合并关系。

练习4:添加提交比较功能

实现选择两个提交并比较它们之间差异的功能。

练习5:实现提交搜索功能

实现搜索框,允许用户搜索特定的提交信息、作者或文件变更。

总结

在本节中,我们成功实现了GitHub Commit历史页面的核心界面元素,重点关注提交时间线和分页控制器的实现。关键实现点包括:

  1. 使用相对和绝对定位实现垂直时间线和提交节点
  2. 通过不同颜色区分常规提交、分支提交和合并提交
  3. 使用视觉编码(绿色和红色背景)直观展示代码添加和删除
  4. 实现分支标签和PR链接,提供额外的上下文信息
  5. 创建响应式布局,确保在不同设备上有良好的用户体验

通过这些实现,我们不仅展示了Tailwind CSS在复杂时间线布局中的应用,也深入理解了GitHub提交历史页面的设计原则。提交历史页面的设计重点在于清晰展示版本变更的时间序列,并提供丰富的上下文信息和操作选项。

在下一节中,我们将探讨议题(Issues)列表页面的实现,学习如何创建状态标记和过滤器系统。