学习目标

  • 分析GitHub仓库列表页的结构和组件组成
  • 使用Tailwind CSS实现仓库卡片组件
  • 构建过滤器和排序控件
  • 实现分页导航功能
  • 创建响应式布局,确保在所有设备上都有良好的用户体验

GitHub仓库列表页分析

GitHub的仓库列表页是用户查看、搜索和管理其所有代码仓库的中心。这个页面布局清晰,功能丰富,包含多种UI组件:

  1. 顶部导航栏(复用前面章节的组件)
  2. 子导航标签(区分”Repositories”、”Projects”等)
  3. 过滤和搜索控件
  4. 仓库卡片列表
  5. 分页控件

这个页面的设计专注于高效浏览和管理大量仓库,提供多种方式来过滤和排序内容。

页面骨架结构

首先,让我们创建页面的基本HTML结构:

  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>Your Repositories - GitHub</title>
  7. <script src="https://cdn.tailwindcss.com"></script>
  8. <script>
  9. tailwind.config = {
  10. theme: {
  11. extend: {
  12. colors: {
  13. "github-blue": "#0969da",
  14. "github-green": "#2da44e",
  15. "github-red": "#cf222e",
  16. "github-yellow": "#bf8700",
  17. "github-purple": "#8250df",
  18. "github-bg": "#f6f8fa",
  19. "github-border": "#d0d7de",
  20. "github-text-primary": "#24292f",
  21. "github-text-secondary": "#57606a",
  22. "github-header-bg": "#24292f",
  23. "github-header-text": "#ffffff",
  24. "github-btn-primary": "#2da44e",
  25. "github-btn-primary-hover": "#2c974b",
  26. },
  27. fontFamily: {
  28. "github": [
  29. "-apple-system",
  30. "BlinkMacSystemFont",
  31. "Segoe UI",
  32. "Noto Sans",
  33. "Helvetica",
  34. "Arial",
  35. "sans-serif",
  36. "Apple Color Emoji",
  37. "Segoe UI Emoji"
  38. ],
  39. },
  40. borderRadius: {
  41. "github-md": "6px",
  42. },
  43. boxShadow: {
  44. "github-sm": "0 1px 0 rgba(27, 31, 36, 0.04)",
  45. "github-btn": "0 1px 0 rgba(27, 31, 36, 0.04), inset 0 1px 0 rgba(255, 255, 255, 0.25)",
  46. }
  47. }
  48. }
  49. }
  50. </script>
  51. </head>
  52. <body class="bg-white font-github text-github-text-primary">
  53. <!-- GitHub Header (复用) -->
  54. <header class="bg-github-header-bg text-github-header-text py-3 px-4 flex items-center justify-between">
  55. <div class="flex items-center w-full">
  56. <!-- GitHub Logo -->
  57. <svg height="32" aria-hidden="true" viewBox="0 0 16 16" version="1.1" width="32" class="fill-current">
  58. <path d="M8 0c4.42 0 8 3.58 8 8a8.013 8.013 0 0 1-5.45 7.59c-.4.08-.55-.17-.55-.38 0-.27.01-1.13.01-2.2 0-.75-.25-1.23-.54-1.48 1.78-.2 3.65-.88 3.65-3.95 0-.88-.31-1.59-.82-2.15.08-.2.36-1.02-.08-2.12 0 0-.67-.22-2.2.82-.64-.18-1.32-.27-2-.27-.68 0-1.36.09-2 .27-1.53-1.03-2.2-.82-2.2-.82-.44 1.1-.16 1.92-.08 2.12-.51.56-.82 1.28-.82 2.15 0 3.06 1.86 3.75 3.64 3.95-.23.2-.44.55-.51 1.07-.46.21-1.61.55-2.33-.66-.15-.24-.6-.83-1.23-.82-.67.01-.27.38.01.53.34.19.73.9.82 1.13.16.45.68 1.31 2.69.94 0 .67.01 1.3.01 1.49 0 .21-.15.45-.55.38A7.995 7.995 0 0 1 0 8c0-4.42 3.58-8 8-8Z"></path>
  59. </svg>
  60. <!-- Search bar -->
  61. <div class="relative mx-4 hidden md:block w-64">
  62. <input type="text" placeholder="Search or jump to..."
  63. class="bg-github-header-bg border border-gray-600 rounded-md text-sm text-white placeholder-gray-400 w-full py-1 px-3">
  64. </div>
  65. <!-- Navigation links -->
  66. <nav class="hidden md:flex items-center space-x-4">
  67. <a href="#" class="text-white hover:text-gray-300">Pull requests</a>
  68. <a href="#" class="text-white hover:text-gray-300">Issues</a>
  69. <a href="#" class="text-white hover:text-gray-300">Marketplace</a>
  70. <a href="#" class="text-white hover:text-gray-300">Explore</a>
  71. </nav>
  72. <!-- Mobile menu button -->
  73. <button class="ml-auto md:hidden text-white">
  74. <svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="w-6 h-6">
  75. <path stroke-linecap="round" stroke-linejoin="round" d="M3.75 6.75h16.5M3.75 12h16.5m-16.5 5.25h16.5" />
  76. </svg>
  77. </button>
  78. <!-- User menu -->
  79. <div class="hidden md:flex items-center ml-auto">
  80. <!-- Notifications icon -->
  81. <a href="#" class="text-white mr-4">
  82. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor">
  83. <path d="M8 16a2 2 0 0 0 1.985-1.75c.017-.137-.097-.25-.235-.25h-3.5c-.138 0-.252.113-.235.25A2 2 0 0 0 8 16ZM3 5a5 5 0 0 1 10 0v2.947c0 .05.015.098.042.139l1.703 2.555A1.519 1.519 0 0 1 13.482 13H2.518a1.516 1.516 0 0 1-1.263-2.36l1.703-2.554A.255.255 0 0 0 3 7.947Zm5-3.5A3.5 3.5 0 0 0 4.5 5v2.947c0 .346-.102.683-.294.97l-1.703 2.556a.017.017 0 0 0-.003.01l.001.006c0 .002.002.004.004.006l.006.004.007.001h10.964l.007-.001.006-.004.004-.006.001-.007a.017.017 0 0 0-.003-.01l-1.703-2.554a1.745 1.745 0 0 1-.294-.97V5A3.5 3.5 0 0 0 8 1.5Z"></path>
  84. </svg>
  85. </a>
  86. <!-- User avatar dropdown -->
  87. <div class="relative">
  88. <button class="flex items-center">
  89. <img src="https://avatars.githubusercontent.com/u/123456789?v=4" alt="User avatar" class="w-5 h-5 rounded-full">
  90. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="ml-1 text-white">
  91. <path d="m4.427 7.427 3.396 3.396a.25.25 0 0 0 .354 0l3.396-3.396A.25.25 0 0 0 11.396 7H4.604a.25.25 0 0 0-.177.427Z"></path>
  92. </svg>
  93. </button>
  94. </div>
  95. </div>
  96. </div>
  97. </header>
  98. <!-- Main Content Area -->
  99. <main class="container mx-auto px-4 py-6 max-w-7xl">
  100. <!-- Content will go here -->
  101. </main>
  102. </body>
  103. </html>

实现用户信息栏和导航标签

接下来,让我们添加用户信息栏和导航标签,这是仓库列表页的顶部部分:

  1. <!-- Main Content Area -->
  2. <main class="container mx-auto px-4 py-6 max-w-7xl">
  3. <!-- User info bar (mobile only) -->
  4. <div class="flex items-center mb-6 md:hidden">
  5. <img src="https://avatars.githubusercontent.com/u/123456789?v=4" alt="User avatar" class="w-8 h-8 rounded-full mr-2">
  6. <span class="font-bold">johndoe</span>
  7. </div>
  8. <!-- User tabs navigation -->
  9. <div class="border-b border-github-border mb-6 overflow-x-auto">
  10. <nav class="flex flex-nowrap -mb-px">
  11. <a href="#" class="px-4 py-3 border-b-2 border-transparent hover:border-gray-300 text-github-text-secondary hover:text-github-text-primary whitespace-nowrap">
  12. Overview
  13. </a>
  14. <a href="#" class="px-4 py-3 border-b-2 border-github-blue text-github-blue font-semibold whitespace-nowrap">
  15. Repositories <span class="ml-1 py-0.5 px-1.5 text-xs rounded-full bg-gray-200">28</span>
  16. </a>
  17. <a href="#" class="px-4 py-3 border-b-2 border-transparent hover:border-gray-300 text-github-text-secondary hover:text-github-text-primary whitespace-nowrap">
  18. Projects <span class="ml-1 py-0.5 px-1.5 text-xs rounded-full bg-gray-200">4</span>
  19. </a>
  20. <a href="#" class="px-4 py-3 border-b-2 border-transparent hover:border-gray-300 text-github-text-secondary hover:text-github-text-primary whitespace-nowrap">
  21. Packages
  22. </a>
  23. <a href="#" class="px-4 py-3 border-b-2 border-transparent hover:border-gray-300 text-github-text-secondary hover:text-github-text-primary whitespace-nowrap">
  24. Stars <span class="ml-1 py-0.5 px-1.5 text-xs rounded-full bg-gray-200">324</span>
  25. </a>
  26. </nav>
  27. </div>
  28. </main>

添加过滤和搜索控件

GitHub的仓库列表页面具有强大的过滤和搜索功能。让我们实现这些控件:

  1. <!-- After user tabs navigation -->
  2. <!-- Filter and search controls -->
  3. <div class="mb-4 flex flex-col sm:flex-row gap-3">
  4. <!-- Search repositories -->
  5. <div class="relative flex-grow order-2 sm:order-1">
  6. <input type="text" placeholder="Find a repository..."
  7. class="block w-full px-3 py-1.5 text-sm border border-github-border rounded-github-md focus:outline-none focus:border-github-blue focus:ring-1 focus:ring-github-blue">
  8. </div>
  9. <!-- Filter buttons -->
  10. <div class="flex flex-wrap gap-2 order-1 sm:order-2">
  11. <!-- Type dropdown -->
  12. <div class="relative">
  13. <button class="flex items-center px-3 py-1.5 text-sm font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  14. Type
  15. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="ml-1">
  16. <path d="m4.427 7.427 3.396 3.396a.25.25 0 0 0 .354 0l3.396-3.396A.25.25 0 0 0 11.396 7H4.604a.25.25 0 0 0-.177.427Z"></path>
  17. </svg>
  18. </button>
  19. </div>
  20. <!-- Language dropdown -->
  21. <div class="relative">
  22. <button class="flex items-center px-3 py-1.5 text-sm font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  23. Language
  24. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="ml-1">
  25. <path d="m4.427 7.427 3.396 3.396a.25.25 0 0 0 .354 0l3.396-3.396A.25.25 0 0 0 11.396 7H4.604a.25.25 0 0 0-.177.427Z"></path>
  26. </svg>
  27. </button>
  28. </div>
  29. <!-- Sort dropdown -->
  30. <div class="relative">
  31. <button class="flex items-center px-3 py-1.5 text-sm font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  32. Sort
  33. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="ml-1">
  34. <path d="m4.427 7.427 3.396 3.396a.25.25 0 0 0 .354 0l3.396-3.396A.25.25 0 0 0 11.396 7H4.604a.25.25 0 0 0-.177.427Z"></path>
  35. </svg>
  36. </button>
  37. </div>
  38. <!-- New repository button -->
  39. <a href="#" class="flex items-center px-3 py-1.5 text-sm font-medium rounded-github-md bg-github-btn-primary text-white hover:bg-github-btn-primary-hover border border-github-border border-opacity-20">
  40. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  41. <path d="M2 2.5A2.5 2.5 0 0 1 4.5 0h8.75a.75.75 0 0 1 .75.75v12.5a.75.75 0 0 1-.75.75h-2.5a.75.75 0 0 1 0-1.5h1.75v-2h-8a1 1 0 0 0-.714 1.7.75.75 0 1 1-1.072 1.05A2.495 2.495 0 0 1 2 11.5Zm10.5-1h-8a1 1 0 0 0-1 1v6.708A2.486 2.486 0 0 1 4.5 9h8ZM5 12.25a.25.25 0 0 1 .25-.25h3.5a.25.25 0 0 1 .25.25v.5a.25.25 0 0 1-.25.25h-3.5a.25.25 0 0 1-.25-.25Z"></path>
  42. </svg>
  43. New
  44. </a>
  45. </div>
  46. </div>

实现仓库卡片列表

现在,让我们实现仓库卡片列表,这是页面的核心部分:

  1. <!-- After filter and search controls -->
  2. <!-- Repository list -->
  3. <div class="border-t border-github-border divide-y divide-github-border">
  4. <!-- Repository item 1 -->
  5. <div class="py-6">
  6. <div class="flex justify-between items-start">
  7. <div>
  8. <div class="flex items-baseline">
  9. <h3 class="text-xl font-semibold mr-2">
  10. <a href="#" class="text-github-blue hover:underline">awesome-project</a>
  11. </h3>
  12. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  13. </div>
  14. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  15. A collection of awesome resources for developers
  16. </p>
  17. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  18. <span class="flex items-center">
  19. <span class="w-3 h-3 rounded-full bg-yellow-500 mr-1"></span>
  20. JavaScript
  21. </span>
  22. <span class="flex items-center">
  23. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  24. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  25. </svg>
  26. 128
  27. </span>
  28. <span class="flex items-center">
  29. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  30. <path d="M5 5.372v.878c0 .414.336.75.75.75h4.5a.75.75 0 0 0 .75-.75v-.878a2.25 2.25 0 1 0-6 0Zm1.5 0a.75.75 0 1 1 1.5 0 .75.75 0 0 1-1.5 0Zm6.75.75a.75.75 0 0 0 0-1.5h-3a.75.75 0 0 0 0 1.5h3ZM4.5 15a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 13.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 12a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 10.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75Z"></path>
  31. </svg>
  32. 42
  33. </span>
  34. <span>Updated 2 days ago</span>
  35. </div>
  36. </div>
  37. <div>
  38. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  39. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  40. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  41. </svg>
  42. Star
  43. </button>
  44. </div>
  45. </div>
  46. </div>
  47. <!-- Repository item 2 -->
  48. <div class="py-6">
  49. <div class="flex justify-between items-start">
  50. <div>
  51. <div class="flex items-baseline">
  52. <h3 class="text-xl font-semibold mr-2">
  53. <a href="#" class="text-github-blue hover:underline">react-components</a>
  54. </h3>
  55. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  56. </div>
  57. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  58. Collection of reusable React components with TypeScript support
  59. </p>
  60. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  61. <span class="flex items-center">
  62. <span class="w-3 h-3 rounded-full bg-blue-500 mr-1"></span>
  63. TypeScript
  64. </span>
  65. <span class="flex items-center">
  66. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  67. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  68. </svg>
  69. 95
  70. </span>
  71. <span class="flex items-center">
  72. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  73. <path d="M5 5.372v.878c0 .414.336.75.75.75h4.5a.75.75 0 0 0 .75-.75v-.878a2.25 2.25 0 1 0-6 0Zm1.5 0a.75.75 0 1 1 1.5 0 .75.75 0 0 1-1.5 0Zm6.75.75a.75.75 0 0 0 0-1.5h-3a.75.75 0 0 0 0 1.5h3ZM4.5 15a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 13.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 12a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 10.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75Z"></path>
  74. </svg>
  75. 37
  76. </span>
  77. <span>Updated last week</span>
  78. </div>
  79. </div>
  80. <div>
  81. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  82. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  83. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  84. </svg>
  85. Star
  86. </button>
  87. </div>
  88. </div>
  89. </div>
  90. <!-- Repository item 3 (with pinned indicator) -->
  91. <div class="py-6">
  92. <div class="flex justify-between items-start">
  93. <div>
  94. <div class="flex items-baseline">
  95. <h3 class="text-xl font-semibold mr-2">
  96. <a href="#" class="text-github-blue hover:underline">tailwind-github-clone</a>
  97. </h3>
  98. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  99. <span class="ml-2 inline-flex items-center text-github-text-secondary">
  100. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  101. <path d="M7.775 3.275a.75.75 0 0 0 1.06 1.06l1.25-1.25a2 2 0 1 1 2.83 2.83l-2.5 2.5a2 2 0 0 1-2.83 0 .75.75 0 0 0-1.06 1.06 3.5 3.5 0 0 0 4.95 0l2.5-2.5a3.5 3.5 0 0 0-4.95-4.95l-1.25 1.25Zm-4.69 9.64a2 2 0 0 1 0-2.83l2.5-2.5a2 2 0 0 1 2.83 0 .75.75 0 0 0 1.06-1.06 3.5 3.5 0 0 0-4.95 0l-2.5 2.5a3.5 3.5 0 0 0 4.95 4.95l1.25-1.25a.75.75 0 0 0-1.06-1.06l-1.25 1.25a2 2 0 0 1-2.83 0Z"></path>
  102. </svg>
  103. Pinned
  104. </span>
  105. </div>
  106. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  107. A GitHub UI clone built with Tailwind CSS
  108. </p>
  109. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  110. <span class="flex items-center">
  111. <span class="w-3 h-3 rounded-full bg-blue-300 mr-1"></span>
  112. CSS
  113. </span>
  114. <span class="flex items-center">
  115. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  116. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  117. </svg>
  118. 54
  119. </span>
  120. <span class="flex items-center">
  121. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  122. <path d="M5 5.372v.878c0 .414.336.75.75.75h4.5a.75.75 0 0 0 .75-.75v-.878a2.25 2.25 0 1 0-6 0Zm1.5 0a.75.75 0 1 1 1.5 0 .75.75 0 0 1-1.5 0Zm6.75.75a.75.75 0 0 0 0-1.5h-3a.75.75 0 0 0 0 1.5h3ZM4.5 15a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 13.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 12a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 10.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75Z"></path>
  123. </svg>
  124. 15
  125. </span>
  126. <span>Updated 15 days ago</span>
  127. </div>
  128. </div>
  129. <div>
  130. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  131. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  132. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  133. </svg>
  134. Star
  135. </button>
  136. </div>
  137. </div>
  138. </div>
  139. <!-- Repository item 4 (with "Private" and "Archived" indicators) -->
  140. <div class="py-6">
  141. <div class="flex justify-between items-start">
  142. <div>
  143. <div class="flex items-baseline flex-wrap gap-2">
  144. <h3 class="text-xl font-semibold">
  145. <a href="#" class="text-github-blue hover:underline">legacy-project</a>
  146. </h3>
  147. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border bg-yellow-50">Private</span>
  148. <span class="text-xs px-2 py-0.5 rounded-full text-github-text-secondary bg-gray-100">Archived</span>
  149. </div>
  150. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  151. Legacy project from 2020, archived for reference
  152. </p>
  153. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  154. <span class="flex items-center">
  155. <span class="w-3 h-3 rounded-full bg-purple-500 mr-1"></span>
  156. PHP
  157. </span>
  158. <span class="flex items-center">
  159. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  160. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  161. </svg>
  162. 8
  163. </span>
  164. <span class="flex items-center">
  165. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  166. <path d="M5 5.372v.878c0 .414.336.75.75.75h4.5a.75.75 0 0 0 .75-.75v-.878a2.25 2.25 0 1 0-6 0Zm1.5 0a.75.75 0 1 1 1.5 0 .75.75 0 0 1-1.5 0Zm6.75.75a.75.75 0 0 0 0-1.5h-3a.75.75 0 0 0 0 1.5h3ZM4.5 15a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 13.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 12a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 10.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75Z"></path>
  167. </svg>
  168. 2
  169. </span>
  170. <span>Archived on Dec 5, 2023</span>
  171. </div>
  172. </div>
  173. <div>
  174. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  175. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  176. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  177. </svg>
  178. Star
  179. </button>
  180. </div>
  181. </div>
  182. </div>
  183. <!-- More repository items (5) -->
  184. <div class="py-6">
  185. <div class="flex justify-between items-start">
  186. <div>
  187. <div class="flex items-baseline">
  188. <h3 class="text-xl font-semibold mr-2">
  189. <a href="#" class="text-github-blue hover:underline">node-api-starter</a>
  190. </h3>
  191. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  192. </div>
  193. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  194. A starter template for Node.js APIs with Express and MongoDB
  195. </p>
  196. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  197. <span class="flex items-center">
  198. <span class="w-3 h-3 rounded-full bg-green-700 mr-1"></span>
  199. JavaScript
  200. </span>
  201. <span class="flex items-center">
  202. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  203. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  204. </svg>
  205. 72
  206. </span>
  207. <span class="flex items-center">
  208. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  209. <path d="M5 5.372v.878c0 .414.336.75.75.75h4.5a.75.75 0 0 0 .75-.75v-.878a2.25 2.25 0 1 0-6 0Zm1.5 0a.75.75 0 1 1 1.5 0 .75.75 0 0 1-1.5 0Zm6.75.75a.75.75 0 0 0 0-1.5h-3a.75.75 0 0 0 0 1.5h3ZM4.5 15a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 13.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 12a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75ZM4.5 10.5a.75.75 0 0 0 .75-.75v-.5a.75.75 0 0 0-1.5 0v.5c0 .414.336.75.75.75Z"></path>
  210. </svg>
  211. 24
  212. </span>
  213. <span>Updated 1 month ago</span>
  214. </div>
  215. </div>
  216. <div>
  217. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  218. <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" width="16" height="16" fill="currentColor" class="mr-1">
  219. <path d="M8 .25a.75.75 0 0 1 .673.418l1.882 3.815 4.21.612a.75.75 0 0 1 .416 1.279l-3.046 2.97.719 4.192a.751.751 0 0 1-1.088.791L8 12.347l-3.766 1.98a.75.75 0 0 1-1.088-.79l.72-4.194L.818 6.374a.75.75 0 0 1 .416-1.28l4.21-.611L7.327.668A.75.75 0 0 1 8 .25Z"></path>
  220. </svg>
  221. Star
  222. </button>
  223. </div>
  224. </div>
  225. </div>
  226. </div>

添加分页导航组件

最后,我们需要添加分页导航组件来浏览更多的仓库:

  1. <!-- After repository list -->
  2. <!-- Pagination -->
  3. <div class="mt-6 flex justify-center">
  4. <nav class="inline-flex items-center">
  5. <a href="#" class="px-3 py-1 rounded-l-md border border-github-border bg-white text-github-text-primary hover:bg-github-bg disabled:opacity-50 disabled:pointer-events-none" disabled>
  6. Previous
  7. </a>
  8. <a href="#" class="px-3 py-1 border-t border-b border-github-border bg-github-blue text-white font-medium">
  9. 1
  10. </a>
  11. <a href="#" class="px-3 py-1 border-t border-b border-github-border bg-white text-github-text-primary hover:bg-github-bg">
  12. 2
  13. </a>
  14. <a href="#" class="px-3 py-1 border-t border-b border-github-border bg-white text-github-text-primary hover:bg-github-bg">
  15. 3
  16. </a>
  17. <span class="px-3 py-1 border-t border-b border-github-border bg-white text-github-text-secondary">
  18. ...
  19. </span>
  20. <a href="#" class="px-3 py-1 border-t border-b border-github-border bg-white text-github-text-primary hover:bg-github-bg">
  21. 8
  22. </a>
  23. <a href="#" class="px-3 py-1 rounded-r-md border border-github-border bg-white text-github-text-primary hover:bg-github-bg">
  24. Next
  25. </a>
  26. </nav>
  27. </div>

响应式设计考虑

GitHub的仓库列表页在移动设备上有一些特殊的考虑。我们已经在示例中实现了一些关键的响应式特性:

  1. 移动优先的过滤器布局

    • 在小屏幕上,过滤器控件堆叠显示,搜索框在下方
    • 在较大屏幕上,它们变为水平排列,搜索框在左侧
  2. 导航变换

    • 在移动设备上,主导航栏简化为汉堡菜单
    • 用户信息显示在页面顶部

关键样式技术点

1. 仓库卡片模式

GitHub的仓库卡片设计是该页面的核心元素。每个卡片包含仓库名称、描述、统计信息和操作按钮,采用清晰的层次结构:

  1. <div class="py-6">
  2. <div class="flex justify-between items-start">
  3. <!-- 左侧:仓库信息 -->
  4. <div>
  5. <!-- 标题行 -->
  6. <div class="flex items-baseline">
  7. <h3 class="text-xl font-semibold mr-2">
  8. <a href="#" class="text-github-blue hover:underline">repository-name</a>
  9. </h3>
  10. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  11. </div>
  12. <!-- 描述 -->
  13. <p class="text-sm text-github-text-secondary mt-1 mb-3">
  14. Repository description goes here
  15. </p>
  16. <!-- 元数据行 -->
  17. <div class="flex flex-wrap items-center text-xs text-github-text-secondary gap-x-4 gap-y-2">
  18. <!-- 各种统计信息... -->
  19. </div>
  20. </div>
  21. <!-- 右侧:动作按钮 -->
  22. <div>
  23. <button class="flex items-center px-3 py-1 text-xs font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  24. <!-- 按钮内容 -->
  25. </button>
  26. </div>
  27. </div>
  28. </div>

这种结构使用了Flexbox来创建左右两侧的布局,同时在较小屏幕上仍然能够保持良好的阅读体验。

2. 分隔线和边界

GitHub使用细微的边框和分隔线来区分内容区域,我们使用以下Tailwind类实现:

  1. <div class="border-t border-github-border divide-y divide-github-border">
  2. <!-- 仓库项目 -->
  3. </div>

这种方法为每个仓库项目添加了上边框,并在列表的顶部添加了一个边框,创建了清晰的视觉分隔。

3. 标签和指示器

GitHub使用各种标签和状态指示器来显示仓库状态(公开/私有、已归档、已固定等):

  1. <!-- 公开仓库标签 -->
  2. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  3. <!-- 私有仓库标签 -->
  4. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border bg-yellow-50">Private</span>
  5. <!-- 已归档标签 -->
  6. <span class="text-xs px-2 py-0.5 rounded-full text-github-text-secondary bg-gray-100">Archived</span>
  7. <!-- 固定指示器 -->
  8. <span class="ml-2 inline-flex items-center text-github-text-secondary">
  9. <svg><!-- ... --></svg>
  10. Pinned
  11. </span>

这种一致的设计语言帮助用户快速识别仓库的状态和特性。

4. 语言颜色点

GitHub为不同的编程语言使用彩色圆点,这是一个独特的视觉元素:

  1. <span class="flex items-center">
  2. <span class="w-3 h-3 rounded-full bg-blue-500 mr-1"></span>
  3. TypeScript
  4. </span>

我们使用不同的背景色来模拟GitHub的语言颜色系统。

5. 过滤器和排序控件

过滤器和排序控件使用一致的按钮样式,具有轻微的背景色和边框:

  1. <button class="flex items-center px-3 py-1.5 text-sm font-medium border border-github-border rounded-github-md bg-github-bg hover:bg-gray-100">
  2. Type
  3. <svg><!-- 下拉箭头图标 --></svg>
  4. </button>

这些按钮在悬停时有轻微的背景色变化,提供了良好的交互反馈。

常见问题与解决方案

问题1:移动设备上过滤控件的排列

解决方案:使用Flexbox的order属性和flex-col/flex-row条件类来根据屏幕尺寸改变元素顺序:

  1. <div class="mb-4 flex flex-col sm:flex-row gap-3">
  2. <!-- 搜索框 - 在移动设备上在下方,在桌面上在左侧 -->
  3. <div class="relative flex-grow order-2 sm:order-1">
  4. <!-- 搜索框内容 -->
  5. </div>
  6. <!-- 过滤器按钮 - 在移动设备上在上方,在桌面上在右侧 -->
  7. <div class="flex flex-wrap gap-2 order-1 sm:order-2">
  8. <!-- 过滤器按钮内容 -->
  9. </div>
  10. </div>

问题2:卡片内容在小屏幕上的溢出

解决方案:使用Flexbox换行和适当的间距来确保内容在小屏幕上能够正确换行:

  1. <div class="flex flex-wrap items-baseline gap-2">
  2. <h3 class="text-xl font-semibold">
  3. <a href="#" class="text-github-blue hover:underline">repository-name</a>
  4. </h3>
  5. <span class="text-xs px-2 py-0.5 rounded-full border border-github-border">Public</span>
  6. <span class="text-xs px-2 py-0.5 rounded-full text-github-text-secondary bg-gray-100">Archived</span>
  7. </div>

flex-wrapgap-2 类确保元素在需要时会换行,并保持适当的间距。

问题3:分页控件在小屏幕上的显示

解决方案:使居中布局和横向滚动来确保分页控件在小屏幕上可访问:

  1. <div class="mt-6 flex justify-center overflow-x-auto">
  2. <nav class="inline-flex items-center">
  3. <!-- 分页控件内容 -->
  4. </nav>
  5. </div>

overflow-x-auto 类允许用户在必要时水平滚动以查看所有分页控件。

扩展与练习建议

  1. 实现下拉过滤器:为”Type”、”Language”和”Sort”按钮创建下拉菜单,显示相应的过滤选项。

  2. 添加批量选择功能:实现GitHub的批量操作功能,允许用户选择多个仓库并执行批量操作(如归档、转移等)。

  3. 创建空状态页面:设计一个”无仓库”的空状态页面,当用户没有仓库或过滤条件没有匹配结果时显示。

  4. 添加筛选结果标签:实现GitHub的筛选标签功能,显示当前应用的过滤器,并允许用户点击移除它们。

  5. 集成暗模式:为仓库列表页添加暗模式支持,确保所有元素在暗背景上有适当的对比度。

总结

在本节中,我们成功实现了GitHub的仓库列表页面,包括仓库卡片、过滤控件和分页导航。这个页面展示了如何使用Tailwind CSS创建复杂的列表布局,包含多种状态指示器和交互元素。

关键技术点包括:

  • 使用Flexbox创建灵活的卡片布局
  • 采用分隔线和边框创建清晰的视觉层次
  • 实现多种类型的状态标签和指示器
  • 创建响应式过滤器和搜索控件
  • 设计分页导航组件

GitHub的仓库列表页面是一个信息密集型界面,通过精心的设计和布局,使用户能够轻松浏览和管理大量仓库。通过复刻这个页面,我们学习了如何在保持视觉清晰度的同时呈现丰富的信息和功能。

在下一节中,我们将探索GitHub的搜索结果页面,学习如何实现更复杂的过滤器和搜索结果显示。