@bbearai/react 0.3.0 → 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -790,7 +790,10 @@ function TestDetailScreen({ testId, nav }) {
790
790
  if (!client || !displayedAssignment || isSubmitting) return;
791
791
  setIsSubmitting(true);
792
792
  try {
793
- await client.updateAssignmentStatus(displayedAssignment.id, "failed");
793
+ const result = await client.updateAssignmentStatus(displayedAssignment.id, "failed");
794
+ if (!result.success) {
795
+ console.error("BugBear: Failed to mark assignment as failed", result.error);
796
+ }
794
797
  await refreshAssignments();
795
798
  nav.replace({
796
799
  name: "REPORT",
@@ -1540,7 +1543,16 @@ var import_jsx_runtime4 = require("react/jsx-runtime");
1540
1543
  function TestListScreen({ nav }) {
1541
1544
  const { assignments, currentAssignment, refreshAssignments } = useBugBear();
1542
1545
  const [filter, setFilter] = (0, import_react5.useState)("all");
1546
+ const [roleFilter, setRoleFilter] = (0, import_react5.useState)(null);
1543
1547
  const [collapsedFolders, setCollapsedFolders] = (0, import_react5.useState)(/* @__PURE__ */ new Set());
1548
+ const availableRoles = (0, import_react5.useMemo)(() => {
1549
+ const roleMap = /* @__PURE__ */ new Map();
1550
+ for (const a of assignments) {
1551
+ if (a.testCase.role) roleMap.set(a.testCase.role.id, a.testCase.role);
1552
+ }
1553
+ return Array.from(roleMap.values());
1554
+ }, [assignments]);
1555
+ const selectedRole = availableRoles.find((r) => r.id === roleFilter);
1544
1556
  const groupedAssignments = (0, import_react5.useMemo)(() => {
1545
1557
  const groups = /* @__PURE__ */ new Map();
1546
1558
  for (const assignment of assignments) {
@@ -1594,6 +1606,7 @@ function TestListScreen({ nav }) {
1594
1606
  });
1595
1607
  }, []);
1596
1608
  const filterAssignment = (a) => {
1609
+ if (roleFilter && a.testCase.role?.id !== roleFilter) return false;
1597
1610
  if (filter === "pending") return a.status === "pending" || a.status === "in_progress";
1598
1611
  if (filter === "completed") return a.status === "passed" || a.status === "failed";
1599
1612
  return true;
@@ -1626,7 +1639,7 @@ function TestListScreen({ nav }) {
1626
1639
  }
1627
1640
  };
1628
1641
  return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { children: [
1629
- /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { display: "flex", gap: 8, marginBottom: 16 }, children: filters.map((f) => /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1642
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("div", { style: { display: "flex", gap: 8, marginBottom: availableRoles.length >= 2 ? 8 : 16 }, children: filters.map((f) => /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1630
1643
  "button",
1631
1644
  {
1632
1645
  type: "button",
@@ -1653,6 +1666,79 @@ function TestListScreen({ nav }) {
1653
1666
  },
1654
1667
  f.key
1655
1668
  )) }),
1669
+ availableRoles.length >= 2 && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { marginBottom: 12 }, children: [
1670
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: { display: "flex", gap: 6, flexWrap: "wrap", marginBottom: selectedRole?.loginHint ? 8 : 0 }, children: [
1671
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
1672
+ "button",
1673
+ {
1674
+ type: "button",
1675
+ onClick: () => setRoleFilter(null),
1676
+ style: {
1677
+ paddingLeft: 10,
1678
+ paddingRight: 10,
1679
+ paddingTop: 4,
1680
+ paddingBottom: 4,
1681
+ borderRadius: 6,
1682
+ backgroundColor: !roleFilter ? colors.card : "transparent",
1683
+ border: !roleFilter ? `1px solid ${colors.border}` : "1px solid transparent",
1684
+ cursor: "pointer",
1685
+ fontSize: 11,
1686
+ color: !roleFilter ? colors.textPrimary : colors.textMuted,
1687
+ fontWeight: !roleFilter ? 600 : 400
1688
+ },
1689
+ children: "All Roles"
1690
+ }
1691
+ ),
1692
+ availableRoles.map((role) => {
1693
+ const isActive = roleFilter === role.id;
1694
+ return /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(
1695
+ "button",
1696
+ {
1697
+ type: "button",
1698
+ onClick: () => setRoleFilter(isActive ? null : role.id),
1699
+ style: {
1700
+ display: "flex",
1701
+ alignItems: "center",
1702
+ gap: 5,
1703
+ paddingLeft: 10,
1704
+ paddingRight: 10,
1705
+ paddingTop: 4,
1706
+ paddingBottom: 4,
1707
+ borderRadius: 6,
1708
+ backgroundColor: isActive ? role.color + "20" : "transparent",
1709
+ border: isActive ? `1px solid ${role.color}60` : "1px solid transparent",
1710
+ cursor: "pointer",
1711
+ fontSize: 11,
1712
+ color: isActive ? role.color : colors.textMuted,
1713
+ fontWeight: isActive ? 600 : 400
1714
+ },
1715
+ children: [
1716
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: {
1717
+ width: 6,
1718
+ height: 6,
1719
+ borderRadius: 3,
1720
+ backgroundColor: role.color,
1721
+ flexShrink: 0
1722
+ } }),
1723
+ role.name
1724
+ ]
1725
+ },
1726
+ role.id
1727
+ );
1728
+ })
1729
+ ] }),
1730
+ selectedRole?.loginHint && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("div", { style: {
1731
+ fontSize: 11,
1732
+ color: colors.textSecondary,
1733
+ backgroundColor: selectedRole.color + "10",
1734
+ border: `1px solid ${selectedRole.color}30`,
1735
+ borderRadius: 6,
1736
+ padding: "6px 10px"
1737
+ }, children: [
1738
+ "Log in as: ",
1739
+ selectedRole.loginHint
1740
+ ] })
1741
+ ] }),
1656
1742
  groupedAssignments.map((folder) => {
1657
1743
  const folderId = folder.group?.id || "ungrouped";
1658
1744
  const isCollapsed = collapsedFolders.has(folderId);
@@ -1825,7 +1911,26 @@ function TestListScreen({ nav }) {
1825
1911
  },
1826
1912
  children: assignment.testCase.priority
1827
1913
  }
1828
- )
1914
+ ),
1915
+ assignment.testCase.role && /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)(import_jsx_runtime4.Fragment, { children: [
1916
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: { fontSize: 11, color: colors.textDim }, children: "\xB7" }),
1917
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsxs)("span", { style: {
1918
+ display: "inline-flex",
1919
+ alignItems: "center",
1920
+ gap: 3,
1921
+ fontSize: 10,
1922
+ color: assignment.testCase.role.color,
1923
+ fontWeight: 500
1924
+ }, children: [
1925
+ /* @__PURE__ */ (0, import_jsx_runtime4.jsx)("span", { style: {
1926
+ width: 5,
1927
+ height: 5,
1928
+ borderRadius: 3,
1929
+ backgroundColor: assignment.testCase.role.color
1930
+ } }),
1931
+ assignment.testCase.role.name
1932
+ ] })
1933
+ ] })
1829
1934
  ] })
1830
1935
  ] }),
1831
1936
  /* @__PURE__ */ (0, import_jsx_runtime4.jsx)(
@@ -2385,6 +2490,7 @@ function ReportScreen({ nav, prefill }) {
2385
2490
  const observedRoute = (0, import_react8.useRef)(
2386
2491
  typeof window !== "undefined" ? window.location.pathname : "unknown"
2387
2492
  );
2493
+ const isRetestFailure = prefill?.type === "test_fail";
2388
2494
  const isBugType = reportType === "bug" || reportType === "test_fail";
2389
2495
  const handleSubmit = async () => {
2390
2496
  if (!client || !description.trim() || images.isUploading) return;
@@ -2433,7 +2539,68 @@ function ReportScreen({ nav, prefill }) {
2433
2539
  { sev: "medium", color: "#eab308" },
2434
2540
  { sev: "low", color: "#6b7280" }
2435
2541
  ];
2436
- return /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)("div", { children: [
2542
+ return /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { children: isRetestFailure ? /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)(import_jsx_runtime8.Fragment, { children: [
2543
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)("div", { style: styles2.retestBanner, children: [
2544
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("span", { style: { fontSize: 16 }, children: "\u{1F504}" }),
2545
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)("div", { children: [
2546
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.retestTitle, children: "Bug Still Present" }),
2547
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.retestSubtitle, children: "The fix did not resolve this issue" })
2548
+ ] })
2549
+ ] }),
2550
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)("div", { style: styles2.section, children: [
2551
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.label, children: "Severity" }),
2552
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.severityRow, children: severityOptions.map(({ sev, color }) => /* @__PURE__ */ (0, import_jsx_runtime8.jsx)(
2553
+ "button",
2554
+ {
2555
+ onClick: () => setSeverity(sev),
2556
+ style: {
2557
+ ...styles2.sevButton,
2558
+ ...severity === sev ? { backgroundColor: `${color}30`, borderColor: color } : {}
2559
+ },
2560
+ children: /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("span", { style: { ...styles2.sevText, ...severity === sev ? { color } : {} }, children: sev })
2561
+ },
2562
+ sev
2563
+ )) })
2564
+ ] }),
2565
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)("div", { style: styles2.section, children: [
2566
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.label, children: "What went wrong?" }),
2567
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)(
2568
+ "textarea",
2569
+ {
2570
+ style: styles2.descInput,
2571
+ value: description,
2572
+ onChange: (e) => setDescription(e.target.value),
2573
+ placeholder: "Describe what you observed. What still doesn't work?",
2574
+ rows: 4
2575
+ }
2576
+ )
2577
+ ] }),
2578
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)(
2579
+ ImagePickerButtons,
2580
+ {
2581
+ images: images.images,
2582
+ maxImages: 5,
2583
+ onPickGallery: images.pickFromGallery,
2584
+ onPickCamera: images.pickFromCamera,
2585
+ onRemove: images.removeImage,
2586
+ label: "Attachments (optional)"
2587
+ }
2588
+ ),
2589
+ error && /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.errorBanner, children: error }),
2590
+ /* @__PURE__ */ (0, import_jsx_runtime8.jsx)(
2591
+ "button",
2592
+ {
2593
+ style: {
2594
+ ...styles2.submitButton,
2595
+ ...styles2.submitButtonRetest,
2596
+ ...!description.trim() || submitting || images.isUploading ? styles2.submitButtonDisabled : {}
2597
+ },
2598
+ onClick: handleSubmit,
2599
+ disabled: !description.trim() || submitting || images.isUploading,
2600
+ children: images.isUploading ? "Uploading images..." : submitting ? "Submitting..." : error ? "Retry" : "Submit Failed Retest"
2601
+ }
2602
+ )
2603
+ ] }) : /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)(import_jsx_runtime8.Fragment, { children: [
2437
2604
  /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.label, children: "What are you reporting?" }),
2438
2605
  /* @__PURE__ */ (0, import_jsx_runtime8.jsx)("div", { style: styles2.typeRow, children: typeOptions.map(({ type, label, icon }) => /* @__PURE__ */ (0, import_jsx_runtime8.jsxs)(
2439
2606
  "button",
@@ -2540,7 +2707,7 @@ function ReportScreen({ nav, prefill }) {
2540
2707
  children: images.isUploading ? "Uploading images..." : submitting ? "Submitting..." : error ? "Retry" : "Submit Report"
2541
2708
  }
2542
2709
  )
2543
- ] });
2710
+ ] }) });
2544
2711
  }
2545
2712
  var styles2 = {
2546
2713
  label: {
@@ -2663,6 +2830,31 @@ var styles2 = {
2663
2830
  submitButtonDisabled: {
2664
2831
  opacity: 0.5,
2665
2832
  cursor: "not-allowed"
2833
+ },
2834
+ submitButtonRetest: {
2835
+ backgroundColor: "#b45309"
2836
+ },
2837
+ retestBanner: {
2838
+ display: "flex",
2839
+ flexDirection: "row",
2840
+ alignItems: "center",
2841
+ gap: 10,
2842
+ backgroundColor: "#422006",
2843
+ border: "1px solid #854d0e",
2844
+ borderRadius: 10,
2845
+ padding: "12px 14px",
2846
+ marginBottom: 20
2847
+ },
2848
+ retestTitle: {
2849
+ fontSize: 15,
2850
+ fontWeight: 600,
2851
+ color: "#fbbf24",
2852
+ lineHeight: "20px"
2853
+ },
2854
+ retestSubtitle: {
2855
+ fontSize: 12,
2856
+ color: "#d97706",
2857
+ lineHeight: "16px"
2666
2858
  }
2667
2859
  };
2668
2860
 
package/dist/index.mjs CHANGED
@@ -761,7 +761,10 @@ function TestDetailScreen({ testId, nav }) {
761
761
  if (!client || !displayedAssignment || isSubmitting) return;
762
762
  setIsSubmitting(true);
763
763
  try {
764
- await client.updateAssignmentStatus(displayedAssignment.id, "failed");
764
+ const result = await client.updateAssignmentStatus(displayedAssignment.id, "failed");
765
+ if (!result.success) {
766
+ console.error("BugBear: Failed to mark assignment as failed", result.error);
767
+ }
765
768
  await refreshAssignments();
766
769
  nav.replace({
767
770
  name: "REPORT",
@@ -1511,7 +1514,16 @@ import { Fragment as Fragment2, jsx as jsx4, jsxs as jsxs3 } from "react/jsx-run
1511
1514
  function TestListScreen({ nav }) {
1512
1515
  const { assignments, currentAssignment, refreshAssignments } = useBugBear();
1513
1516
  const [filter, setFilter] = useState3("all");
1517
+ const [roleFilter, setRoleFilter] = useState3(null);
1514
1518
  const [collapsedFolders, setCollapsedFolders] = useState3(/* @__PURE__ */ new Set());
1519
+ const availableRoles = useMemo(() => {
1520
+ const roleMap = /* @__PURE__ */ new Map();
1521
+ for (const a of assignments) {
1522
+ if (a.testCase.role) roleMap.set(a.testCase.role.id, a.testCase.role);
1523
+ }
1524
+ return Array.from(roleMap.values());
1525
+ }, [assignments]);
1526
+ const selectedRole = availableRoles.find((r) => r.id === roleFilter);
1515
1527
  const groupedAssignments = useMemo(() => {
1516
1528
  const groups = /* @__PURE__ */ new Map();
1517
1529
  for (const assignment of assignments) {
@@ -1565,6 +1577,7 @@ function TestListScreen({ nav }) {
1565
1577
  });
1566
1578
  }, []);
1567
1579
  const filterAssignment = (a) => {
1580
+ if (roleFilter && a.testCase.role?.id !== roleFilter) return false;
1568
1581
  if (filter === "pending") return a.status === "pending" || a.status === "in_progress";
1569
1582
  if (filter === "completed") return a.status === "passed" || a.status === "failed";
1570
1583
  return true;
@@ -1597,7 +1610,7 @@ function TestListScreen({ nav }) {
1597
1610
  }
1598
1611
  };
1599
1612
  return /* @__PURE__ */ jsxs3("div", { children: [
1600
- /* @__PURE__ */ jsx4("div", { style: { display: "flex", gap: 8, marginBottom: 16 }, children: filters.map((f) => /* @__PURE__ */ jsxs3(
1613
+ /* @__PURE__ */ jsx4("div", { style: { display: "flex", gap: 8, marginBottom: availableRoles.length >= 2 ? 8 : 16 }, children: filters.map((f) => /* @__PURE__ */ jsxs3(
1601
1614
  "button",
1602
1615
  {
1603
1616
  type: "button",
@@ -1624,6 +1637,79 @@ function TestListScreen({ nav }) {
1624
1637
  },
1625
1638
  f.key
1626
1639
  )) }),
1640
+ availableRoles.length >= 2 && /* @__PURE__ */ jsxs3("div", { style: { marginBottom: 12 }, children: [
1641
+ /* @__PURE__ */ jsxs3("div", { style: { display: "flex", gap: 6, flexWrap: "wrap", marginBottom: selectedRole?.loginHint ? 8 : 0 }, children: [
1642
+ /* @__PURE__ */ jsx4(
1643
+ "button",
1644
+ {
1645
+ type: "button",
1646
+ onClick: () => setRoleFilter(null),
1647
+ style: {
1648
+ paddingLeft: 10,
1649
+ paddingRight: 10,
1650
+ paddingTop: 4,
1651
+ paddingBottom: 4,
1652
+ borderRadius: 6,
1653
+ backgroundColor: !roleFilter ? colors.card : "transparent",
1654
+ border: !roleFilter ? `1px solid ${colors.border}` : "1px solid transparent",
1655
+ cursor: "pointer",
1656
+ fontSize: 11,
1657
+ color: !roleFilter ? colors.textPrimary : colors.textMuted,
1658
+ fontWeight: !roleFilter ? 600 : 400
1659
+ },
1660
+ children: "All Roles"
1661
+ }
1662
+ ),
1663
+ availableRoles.map((role) => {
1664
+ const isActive = roleFilter === role.id;
1665
+ return /* @__PURE__ */ jsxs3(
1666
+ "button",
1667
+ {
1668
+ type: "button",
1669
+ onClick: () => setRoleFilter(isActive ? null : role.id),
1670
+ style: {
1671
+ display: "flex",
1672
+ alignItems: "center",
1673
+ gap: 5,
1674
+ paddingLeft: 10,
1675
+ paddingRight: 10,
1676
+ paddingTop: 4,
1677
+ paddingBottom: 4,
1678
+ borderRadius: 6,
1679
+ backgroundColor: isActive ? role.color + "20" : "transparent",
1680
+ border: isActive ? `1px solid ${role.color}60` : "1px solid transparent",
1681
+ cursor: "pointer",
1682
+ fontSize: 11,
1683
+ color: isActive ? role.color : colors.textMuted,
1684
+ fontWeight: isActive ? 600 : 400
1685
+ },
1686
+ children: [
1687
+ /* @__PURE__ */ jsx4("span", { style: {
1688
+ width: 6,
1689
+ height: 6,
1690
+ borderRadius: 3,
1691
+ backgroundColor: role.color,
1692
+ flexShrink: 0
1693
+ } }),
1694
+ role.name
1695
+ ]
1696
+ },
1697
+ role.id
1698
+ );
1699
+ })
1700
+ ] }),
1701
+ selectedRole?.loginHint && /* @__PURE__ */ jsxs3("div", { style: {
1702
+ fontSize: 11,
1703
+ color: colors.textSecondary,
1704
+ backgroundColor: selectedRole.color + "10",
1705
+ border: `1px solid ${selectedRole.color}30`,
1706
+ borderRadius: 6,
1707
+ padding: "6px 10px"
1708
+ }, children: [
1709
+ "Log in as: ",
1710
+ selectedRole.loginHint
1711
+ ] })
1712
+ ] }),
1627
1713
  groupedAssignments.map((folder) => {
1628
1714
  const folderId = folder.group?.id || "ungrouped";
1629
1715
  const isCollapsed = collapsedFolders.has(folderId);
@@ -1796,7 +1882,26 @@ function TestListScreen({ nav }) {
1796
1882
  },
1797
1883
  children: assignment.testCase.priority
1798
1884
  }
1799
- )
1885
+ ),
1886
+ assignment.testCase.role && /* @__PURE__ */ jsxs3(Fragment2, { children: [
1887
+ /* @__PURE__ */ jsx4("span", { style: { fontSize: 11, color: colors.textDim }, children: "\xB7" }),
1888
+ /* @__PURE__ */ jsxs3("span", { style: {
1889
+ display: "inline-flex",
1890
+ alignItems: "center",
1891
+ gap: 3,
1892
+ fontSize: 10,
1893
+ color: assignment.testCase.role.color,
1894
+ fontWeight: 500
1895
+ }, children: [
1896
+ /* @__PURE__ */ jsx4("span", { style: {
1897
+ width: 5,
1898
+ height: 5,
1899
+ borderRadius: 3,
1900
+ backgroundColor: assignment.testCase.role.color
1901
+ } }),
1902
+ assignment.testCase.role.name
1903
+ ] })
1904
+ ] })
1800
1905
  ] })
1801
1906
  ] }),
1802
1907
  /* @__PURE__ */ jsx4(
@@ -2343,7 +2448,7 @@ var styles = {
2343
2448
 
2344
2449
  // src/widget/screens/ReportScreen.tsx
2345
2450
  import { useState as useState6, useRef as useRef2 } from "react";
2346
- import { jsx as jsx8, jsxs as jsxs7 } from "react/jsx-runtime";
2451
+ import { Fragment as Fragment3, jsx as jsx8, jsxs as jsxs7 } from "react/jsx-runtime";
2347
2452
  function ReportScreen({ nav, prefill }) {
2348
2453
  const { client, refreshAssignments, uploadImage } = useBugBear();
2349
2454
  const images = useImageAttachments(uploadImage, 5, "screenshots");
@@ -2356,6 +2461,7 @@ function ReportScreen({ nav, prefill }) {
2356
2461
  const observedRoute = useRef2(
2357
2462
  typeof window !== "undefined" ? window.location.pathname : "unknown"
2358
2463
  );
2464
+ const isRetestFailure = prefill?.type === "test_fail";
2359
2465
  const isBugType = reportType === "bug" || reportType === "test_fail";
2360
2466
  const handleSubmit = async () => {
2361
2467
  if (!client || !description.trim() || images.isUploading) return;
@@ -2404,7 +2510,68 @@ function ReportScreen({ nav, prefill }) {
2404
2510
  { sev: "medium", color: "#eab308" },
2405
2511
  { sev: "low", color: "#6b7280" }
2406
2512
  ];
2407
- return /* @__PURE__ */ jsxs7("div", { children: [
2513
+ return /* @__PURE__ */ jsx8("div", { children: isRetestFailure ? /* @__PURE__ */ jsxs7(Fragment3, { children: [
2514
+ /* @__PURE__ */ jsxs7("div", { style: styles2.retestBanner, children: [
2515
+ /* @__PURE__ */ jsx8("span", { style: { fontSize: 16 }, children: "\u{1F504}" }),
2516
+ /* @__PURE__ */ jsxs7("div", { children: [
2517
+ /* @__PURE__ */ jsx8("div", { style: styles2.retestTitle, children: "Bug Still Present" }),
2518
+ /* @__PURE__ */ jsx8("div", { style: styles2.retestSubtitle, children: "The fix did not resolve this issue" })
2519
+ ] })
2520
+ ] }),
2521
+ /* @__PURE__ */ jsxs7("div", { style: styles2.section, children: [
2522
+ /* @__PURE__ */ jsx8("div", { style: styles2.label, children: "Severity" }),
2523
+ /* @__PURE__ */ jsx8("div", { style: styles2.severityRow, children: severityOptions.map(({ sev, color }) => /* @__PURE__ */ jsx8(
2524
+ "button",
2525
+ {
2526
+ onClick: () => setSeverity(sev),
2527
+ style: {
2528
+ ...styles2.sevButton,
2529
+ ...severity === sev ? { backgroundColor: `${color}30`, borderColor: color } : {}
2530
+ },
2531
+ children: /* @__PURE__ */ jsx8("span", { style: { ...styles2.sevText, ...severity === sev ? { color } : {} }, children: sev })
2532
+ },
2533
+ sev
2534
+ )) })
2535
+ ] }),
2536
+ /* @__PURE__ */ jsxs7("div", { style: styles2.section, children: [
2537
+ /* @__PURE__ */ jsx8("div", { style: styles2.label, children: "What went wrong?" }),
2538
+ /* @__PURE__ */ jsx8(
2539
+ "textarea",
2540
+ {
2541
+ style: styles2.descInput,
2542
+ value: description,
2543
+ onChange: (e) => setDescription(e.target.value),
2544
+ placeholder: "Describe what you observed. What still doesn't work?",
2545
+ rows: 4
2546
+ }
2547
+ )
2548
+ ] }),
2549
+ /* @__PURE__ */ jsx8(
2550
+ ImagePickerButtons,
2551
+ {
2552
+ images: images.images,
2553
+ maxImages: 5,
2554
+ onPickGallery: images.pickFromGallery,
2555
+ onPickCamera: images.pickFromCamera,
2556
+ onRemove: images.removeImage,
2557
+ label: "Attachments (optional)"
2558
+ }
2559
+ ),
2560
+ error && /* @__PURE__ */ jsx8("div", { style: styles2.errorBanner, children: error }),
2561
+ /* @__PURE__ */ jsx8(
2562
+ "button",
2563
+ {
2564
+ style: {
2565
+ ...styles2.submitButton,
2566
+ ...styles2.submitButtonRetest,
2567
+ ...!description.trim() || submitting || images.isUploading ? styles2.submitButtonDisabled : {}
2568
+ },
2569
+ onClick: handleSubmit,
2570
+ disabled: !description.trim() || submitting || images.isUploading,
2571
+ children: images.isUploading ? "Uploading images..." : submitting ? "Submitting..." : error ? "Retry" : "Submit Failed Retest"
2572
+ }
2573
+ )
2574
+ ] }) : /* @__PURE__ */ jsxs7(Fragment3, { children: [
2408
2575
  /* @__PURE__ */ jsx8("div", { style: styles2.label, children: "What are you reporting?" }),
2409
2576
  /* @__PURE__ */ jsx8("div", { style: styles2.typeRow, children: typeOptions.map(({ type, label, icon }) => /* @__PURE__ */ jsxs7(
2410
2577
  "button",
@@ -2511,7 +2678,7 @@ function ReportScreen({ nav, prefill }) {
2511
2678
  children: images.isUploading ? "Uploading images..." : submitting ? "Submitting..." : error ? "Retry" : "Submit Report"
2512
2679
  }
2513
2680
  )
2514
- ] });
2681
+ ] }) });
2515
2682
  }
2516
2683
  var styles2 = {
2517
2684
  label: {
@@ -2634,6 +2801,31 @@ var styles2 = {
2634
2801
  submitButtonDisabled: {
2635
2802
  opacity: 0.5,
2636
2803
  cursor: "not-allowed"
2804
+ },
2805
+ submitButtonRetest: {
2806
+ backgroundColor: "#b45309"
2807
+ },
2808
+ retestBanner: {
2809
+ display: "flex",
2810
+ flexDirection: "row",
2811
+ alignItems: "center",
2812
+ gap: 10,
2813
+ backgroundColor: "#422006",
2814
+ border: "1px solid #854d0e",
2815
+ borderRadius: 10,
2816
+ padding: "12px 14px",
2817
+ marginBottom: 20
2818
+ },
2819
+ retestTitle: {
2820
+ fontSize: 15,
2821
+ fontWeight: 600,
2822
+ color: "#fbbf24",
2823
+ lineHeight: "20px"
2824
+ },
2825
+ retestSubtitle: {
2826
+ fontSize: 12,
2827
+ color: "#d97706",
2828
+ lineHeight: "16px"
2637
2829
  }
2638
2830
  };
2639
2831
 
@@ -3823,7 +4015,7 @@ var styles4 = {
3823
4015
  var BUGBEAR_LOGO_BASE64 = "";
3824
4016
 
3825
4017
  // src/BugBearPanel.tsx
3826
- import { Fragment as Fragment3, jsx as jsx14, jsxs as jsxs13 } from "react/jsx-runtime";
4018
+ import { Fragment as Fragment4, jsx as jsx14, jsxs as jsxs13 } from "react/jsx-runtime";
3827
4019
  function BugBearIcon({ size = 24 }) {
3828
4020
  return /* @__PURE__ */ jsx14(
3829
4021
  "img",
@@ -4106,7 +4298,7 @@ function BugBearPanel({
4106
4298
  },
4107
4299
  children: "\u2190 Back"
4108
4300
  }
4109
- ) : /* @__PURE__ */ jsxs13(Fragment3, { children: [
4301
+ ) : /* @__PURE__ */ jsxs13(Fragment4, { children: [
4110
4302
  /* @__PURE__ */ jsx14(BugBearIcon, { size: 28 }),
4111
4303
  /* @__PURE__ */ jsxs13("div", { children: [
4112
4304
  /* @__PURE__ */ jsxs13("div", { style: { display: "flex", alignItems: "center", gap: 8 }, children: [
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@bbearai/react",
3
- "version": "0.3.0",
3
+ "version": "0.4.0",
4
4
  "description": "BugBear React components for web apps",
5
5
  "main": "./dist/index.js",
6
6
  "module": "./dist/index.mjs",
@@ -41,7 +41,7 @@
41
41
  "prepublishOnly": "npm run build"
42
42
  },
43
43
  "dependencies": {
44
- "@bbearai/core": "^0.3.0"
44
+ "@bbearai/core": "^0.4.0"
45
45
  },
46
46
  "peerDependencies": {
47
47
  "react": "^18.0.0 || ^19.0.0",