search.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. UI/UX Pro Max Search - BM25 search engine for UI/UX style guides
  5. Usage: python search.py "<query>" [--domain <domain>] [--stack <stack>] [--max-results 3]
  6. python search.py "<query>" --design-system [-p "Project Name"]
  7. python search.py "<query>" --design-system --persist [-p "Project Name"] [--page "dashboard"]
  8. Domains: style, prompt, color, chart, landing, product, ux, typography
  9. Stacks: html-tailwind, react, nextjs
  10. Persistence (Master + Overrides pattern):
  11. --persist Save design system to design-system/MASTER.md
  12. --page Also create a page-specific override file in design-system/pages/
  13. """
  14. import argparse
  15. import sys
  16. import io
  17. from core import CSV_CONFIG, AVAILABLE_STACKS, MAX_RESULTS, search, search_stack
  18. from design_system import generate_design_system, persist_design_system
  19. # Force UTF-8 for stdout/stderr to handle emojis on Windows (cp1252 default)
  20. if sys.stdout.encoding and sys.stdout.encoding.lower() != 'utf-8':
  21. sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
  22. if sys.stderr.encoding and sys.stderr.encoding.lower() != 'utf-8':
  23. sys.stderr = io.TextIOWrapper(sys.stderr.buffer, encoding='utf-8')
  24. def format_output(result):
  25. """Format results for Claude consumption (token-optimized)"""
  26. if "error" in result:
  27. return f"Error: {result['error']}"
  28. output = []
  29. if result.get("stack"):
  30. output.append(f"## UI Pro Max Stack Guidelines")
  31. output.append(f"**Stack:** {result['stack']} | **Query:** {result['query']}")
  32. else:
  33. output.append(f"## UI Pro Max Search Results")
  34. output.append(f"**Domain:** {result['domain']} | **Query:** {result['query']}")
  35. output.append(f"**Source:** {result['file']} | **Found:** {result['count']} results\n")
  36. for i, row in enumerate(result['results'], 1):
  37. output.append(f"### Result {i}")
  38. for key, value in row.items():
  39. value_str = str(value)
  40. if len(value_str) > 300:
  41. value_str = value_str[:300] + "..."
  42. output.append(f"- **{key}:** {value_str}")
  43. output.append("")
  44. return "\n".join(output)
  45. if __name__ == "__main__":
  46. parser = argparse.ArgumentParser(description="UI Pro Max Search")
  47. parser.add_argument("query", help="Search query")
  48. parser.add_argument("--domain", "-d", choices=list(CSV_CONFIG.keys()), help="Search domain")
  49. parser.add_argument("--stack", "-s", choices=AVAILABLE_STACKS, help="Stack-specific search (html-tailwind, react, nextjs)")
  50. parser.add_argument("--max-results", "-n", type=int, default=MAX_RESULTS, help="Max results (default: 3)")
  51. parser.add_argument("--json", action="store_true", help="Output as JSON")
  52. # Design system generation
  53. parser.add_argument("--design-system", "-ds", action="store_true", help="Generate complete design system recommendation")
  54. parser.add_argument("--project-name", "-p", type=str, default=None, help="Project name for design system output")
  55. parser.add_argument("--format", "-f", choices=["ascii", "markdown"], default="ascii", help="Output format for design system")
  56. # Persistence (Master + Overrides pattern)
  57. parser.add_argument("--persist", action="store_true", help="Save design system to design-system/MASTER.md (creates hierarchical structure)")
  58. parser.add_argument("--page", type=str, default=None, help="Create page-specific override file in design-system/pages/")
  59. parser.add_argument("--output-dir", "-o", type=str, default=None, help="Output directory for persisted files (default: current directory)")
  60. args = parser.parse_args()
  61. # Design system takes priority
  62. if args.design_system:
  63. result = generate_design_system(
  64. args.query,
  65. args.project_name,
  66. args.format,
  67. persist=args.persist,
  68. page=args.page,
  69. output_dir=args.output_dir
  70. )
  71. print(result)
  72. # Print persistence confirmation
  73. if args.persist:
  74. project_slug = args.project_name.lower().replace(' ', '-') if args.project_name else "default"
  75. print("\n" + "=" * 60)
  76. print(f"✅ Design system persisted to design-system/{project_slug}/")
  77. print(f" 📄 design-system/{project_slug}/MASTER.md (Global Source of Truth)")
  78. if args.page:
  79. page_filename = args.page.lower().replace(' ', '-')
  80. print(f" 📄 design-system/{project_slug}/pages/{page_filename}.md (Page Overrides)")
  81. print("")
  82. print(f"📖 Usage: When building a page, check design-system/{project_slug}/pages/[page].md first.")
  83. print(f" If exists, its rules override MASTER.md. Otherwise, use MASTER.md.")
  84. print("=" * 60)
  85. # Stack search
  86. elif args.stack:
  87. result = search_stack(args.query, args.stack, args.max_results)
  88. if args.json:
  89. import json
  90. print(json.dumps(result, indent=2, ensure_ascii=False))
  91. else:
  92. print(format_output(result))
  93. # Domain search
  94. else:
  95. result = search(args.query, args.domain, args.max_results)
  96. if args.json:
  97. import json
  98. print(json.dumps(result, indent=2, ensure_ascii=False))
  99. else:
  100. print(format_output(result))