Write on the margins of the internet. Powered by the AT Protocol. margin.at
extension web atproto comments
at main 15 kB view raw
1import { useState, useRef, useEffect } from "react"; 2import { Copy, ExternalLink, Check } from "lucide-react"; 3import { BlueskyIcon } from "./Icons"; 4 5const BLUESKY_COLOR = "#1185fe"; 6 7const WitchskyIcon = () => ( 8 <svg fill="none" viewBox="0 0 512 512" width="18" height="18"> 9 <path 10 fill="#ee5346" 11 d="M374.473 57.7173C367.666 50.7995 357.119 49.1209 348.441 53.1659C347.173 53.7567 342.223 56.0864 334.796 59.8613C326.32 64.1696 314.568 70.3869 301.394 78.0596C275.444 93.1728 242.399 114.83 218.408 139.477C185.983 172.786 158.719 225.503 140.029 267.661C130.506 289.144 122.878 308.661 117.629 322.81C116.301 326.389 115.124 329.63 114.104 332.478C87.1783 336.42 64.534 341.641 47.5078 348.101C37.6493 351.84 28.3222 356.491 21.0573 362.538C13.8818 368.511 6.00003 378.262 6.00003 391.822C6.00014 403.222 11.8738 411.777 17.4566 417.235C23.0009 422.655 29.9593 426.793 36.871 430.062C50.8097 436.653 69.5275 441.988 90.8362 446.249C133.828 454.846 192.21 460 256.001 460C319.79 460 378.172 454.846 421.164 446.249C442.472 441.988 461.19 436.653 475.129 430.062C482.041 426.793 488.999 422.655 494.543 417.235C500.039 411.862 505.817 403.489 505.996 392.353L506 391.822L505.995 391.188C505.754 377.959 498.012 368.417 490.945 362.534C483.679 356.485 474.35 351.835 464.491 348.095C446.749 341.366 422.906 335.982 394.476 331.987C393.6 330.57 392.633 328.995 391.595 327.273C386.477 318.777 379.633 306.842 372.737 293.115C358.503 264.781 345.757 232.098 344.756 206.636C343.87 184.121 351.638 154.087 360.819 127.789C365.27 115.041 369.795 103.877 373.207 95.9072C374.909 91.9309 376.325 88.7712 377.302 86.6328C377.79 85.5645 378.167 84.7524 378.416 84.2224C378.54 83.9579 378.632 83.7635 378.69 83.643C378.718 83.5829 378.739 83.5411 378.75 83.5181C378.753 83.5108 378.756 83.5049 378.757 83.5015C382.909 74.8634 381.196 64.5488 374.473 57.7173Z" 12 /> 13 </svg> 14); 15 16const BlackskyIcon = () => ( 17 <svg viewBox="0 0 285 285" width="18" height="18"> 18 <path 19 fill="#f9faf9" 20 d="M148.846 144.562C148.846 159.75 161.158 172.062 176.346 172.062H207.012V185.865H176.346C161.158 185.865 148.846 198.177 148.846 213.365V243.045H136.029V213.365C136.029 198.177 123.717 185.865 108.529 185.865H77.8633V172.062H108.529C123.717 172.062 136.029 159.75 136.029 144.562V113.896H148.846V144.562Z" 21 /> 22 <path 23 fill="#f9faf9" 24 d="M170.946 31.8766C160.207 42.616 160.207 60.0281 170.946 70.7675L192.631 92.4516L182.871 102.212L161.186 80.5275C150.447 69.7881 133.035 69.7881 122.296 80.5275L101.309 101.514L92.2456 92.4509L113.232 71.4642C123.972 60.7248 123.972 43.3128 113.232 32.5733L91.5488 10.8899L101.309 1.12988L122.993 22.814C133.732 33.5533 151.144 33.5534 161.884 22.814L183.568 1.12988L192.631 10.1925L170.946 31.8766Z" 25 /> 26 <path 27 fill="#f9faf9" 28 d="M79.0525 75.3259C75.1216 89.9962 83.8276 105.076 98.498 109.006L128.119 116.943L124.547 130.275L94.9267 122.338C80.2564 118.407 65.1772 127.113 61.2463 141.784L53.5643 170.453L41.1837 167.136L48.8654 138.467C52.7963 123.797 44.0902 108.718 29.4199 104.787L-0.201172 96.8497L3.37124 83.5173L32.9923 91.4542C47.6626 95.3851 62.7419 86.679 66.6728 72.0088L74.6098 42.3877L86.9895 45.7048L79.0525 75.3259Z" 29 /> 30 <path 31 fill="#f9faf9" 32 d="M218.413 71.4229C222.344 86.093 237.423 94.7992 252.094 90.8683L281.715 82.9313L285.287 96.2628L255.666 104.2C240.995 108.131 232.29 123.21 236.22 137.88L243.902 166.55L231.522 169.867L223.841 141.198C219.91 126.528 204.831 117.822 190.16 121.753L160.539 129.69L156.967 116.357L186.588 108.42C201.258 104.49 209.964 89.4103 206.033 74.74L198.096 45.1189L210.476 41.8018L218.413 71.4229Z" 33 /> 34 </svg> 35); 36 37const CatskyIcon = () => ( 38 <svg fill="none" viewBox="0 0 67.733328 67.733329" width="18" height="18"> 39 <path 40 fill="#cba7f7" 41 d="m 7.4595521,49.230487 -1.826355,1.186314 -0.00581,0.0064 c -0.6050542,0.41651 -1.129182,0.831427 -1.5159445,1.197382 -0.193382,0.182977 -0.3509469,0.347606 -0.4862911,0.535791 -0.067671,0.0941 -0.1322972,0.188188 -0.1933507,0.352343 -0.061048,0.164157 -0.1411268,0.500074 0.025624,0.844456 l 0.099589,0.200339 c 0.1666616,0.344173 0.4472046,0.428734 0.5969419,0.447854 0.1497358,0.01912 0.2507411,0.0024 0.352923,-0.02039 0.204367,-0.04555 0.4017284,-0.126033 0.6313049,-0.234117 0.4549828,-0.214229 1.0166476,-0.545006 1.6155328,-0.956275 l 0.014617,-0.01049 2.0855152,-1.357536 C 8.3399261,50.711052 7.8735929,49.979321 7.4596148,49.230532 Z" 42 /> 43 <path 44 fill="#cba7f7" 45 d="m 60.225246,49.199041 c -0.421632,0.744138 -0.895843,1.47112 -1.418104,2.178115 l 2.170542,1.413443 c 0.598885,0.411268 1.160549,0.742047 1.615532,0.956276 0.229578,0.108104 0.426937,0.188564 0.631304,0.234116 0.102186,0.02278 0.2061,0.03951 0.355838,0.02039 0.148897,-0.01901 0.427619,-0.104957 0.594612,-0.444358 l 0.0029,-0.0035 0.09667,-0.20034 h 0.0029 c 0.166756,-0.34438 0.08667,-0.680303 0.02562,-0.844455 -0.06104,-0.164158 -0.125675,-0.258251 -0.193352,-0.352343 -0.135356,-0.188186 -0.293491,-0.352814 -0.486873,-0.535792 -0.386891,-0.366 -0.911016,-0.780916 -1.516073,-1.197426 l -0.0082,-0.007 z" 46 /> 47 <path 48 fill="#cba7f7" 49 d="m 62.374822,42.996075 c -0.123437,0.919418 -0.330922,1.827482 -0.614997,2.71973 h 2.864745 c 0.698786,0 1.328766,-0.04848 1.817036,-0.1351 0.244137,-0.04331 0.449793,-0.09051 0.645864,-0.172979 0.09803,-0.04122 0.194035,-0.08458 0.315651,-0.190439 0.121618,-0.105868 0.330211,-0.348705 0.330211,-0.746032 v -0.233536 c 0,-0.397326 -0.208544,-0.637282 -0.330211,-0.743122 -0.121662,-0.105838 -0.217613,-0.152159 -0.315651,-0.193351 -0.196079,-0.08238 -0.401748,-0.129732 -0.645864,-0.17296 -0.488229,-0.08645 -1.118333,-0.132208 -1.817036,-0.132208 z" 50 /> 51 <path 52 fill="#cba7f7" 53 d="m 3.1074004,42.996075 c -0.6987018,0 -1.3264778,0.04576 -1.8147079,0.132208 -0.2441143,0.04324 -0.44978339,0.09059 -0.64586203,0.17296 -0.0980369,0.04118 -0.19398758,0.08751 -0.31565316,0.193351 C 0.20951466,43.600432 0.0015501,43.84039 0.0015501,44.237717 v 0.233535 c 0,0.397326 0.20800926,0.640175 0.32962721,0.746034 0.12161784,0.105867 0.21761904,0.149206 0.31565316,0.190437 0.19606972,0.08246 0.40172683,0.129657 0.64586203,0.172979 0.4882704,0.08663 1.1159226,0.1351 1.8147079,0.1351 H 5.9517617 C 5.6756425,44.822849 5.4740706,43.914705 5.3542351,42.996072 Z" 54 /> 55 <path 56 fill="#cba7f7" 57 d="m 64.667084,33.5073 c -0.430203,0 -0.690808,0.160181 -1.103618,0.372726 -0.41281,0.212535 -0.895004,0.507161 -1.40529,0.858434 l -0.84038,0.578305 c 0.360074,0.820951 0.644317,1.675211 0.844456,2.560741 l 1.136813,-0.78214 c 0.605058,-0.41651 1.12918,-0.834919 1.515944,-1.200875 0.193382,-0.182976 0.350947,-0.347609 0.486291,-0.535795 0.06767,-0.0941 0.132313,-0.188185 0.193351,-0.352341 0.06104,-0.164157 0.141126,-0.497171 -0.02562,-0.841544 L 65.369444,33.96156 C 65.163418,33.537073 64.829889,33.5073 64.669999,33.5073 Z" 58 /> 59 <path 60 fill="#cba7f7" 61 d="m 3.0648864,33.5073 c -0.1600423,3.64e-4 -0.4969719,0.0355 -0.7000249,0.45426 l -0.099589,0.203251 c -0.16676,0.344375 -0.089013,0.677388 -0.027951,0.841544 0.061047,0.164157 0.1285982,0.258248 0.1962636,0.352341 0.1353547,0.188186 0.2899962,0.352819 0.4833782,0.535795 0.386764,0.365956 0.9138003,0.784365 1.518856,1.200875 l 1.1478766,0.78971 c 0.2068,-0.879769 0.5000939,-1.727856 0.8706646,-2.542104 v -5.81e-4 L 5.5761273,34.73846 C 5.065553,34.38699 4.5814871,34.09259 4.1685053,33.880026 3.7555236,33.667462 3.4962107,33.506322 3.0648893,33.5073 Z" 62 /> 63 <path 64 fill="#cba7f7" 65 d="m 34.206496,25.930929 c -7.358038,0 -14.087814,1.669555 -18.851571,4.452678 -4.763758,2.783122 -7.4049994,6.472247 -7.4049994,10.665932 0,4.229683 2.6374854,8.946766 7.2694834,12.60017 4.631996,3.653402 11.153152,6.176813 18.420538,6.176813 7.267388,0 13.908863,-2.52485 18.657979,-6.185354 4.749117,-3.660501 7.485285,-8.390746 7.485285,-12.591629 0,-4.236884 -2.494219,-7.904081 -7.079874,-10.67732 -4.585655,-2.773237 -11.1388,-4.44129 -18.496841,-4.44129 z" 66 /> 67 <path 68 fill="#cba7f7" 69 d="m 51.797573,6.1189692 c -0.02945,-7.175e-4 -0.05836,4.17e-5 -0.08736,5.831e-4 -0.143066,0.00254 -0.278681,0.00746 -0.419898,0.094338 -0.483586,0.2975835 -0.980437,0.9277726 -1.446058,1.5345809 -1.170891,1.5259255 -2.372514,3.8701448 -4.229269,7.0095668 -0.839492,1.419423 -2.308256,4.55051 -3.891486,8.089307 4.831393,0.745951 9.148869,2.222975 12.643546,4.336427 2.130458,1.288425 3.976812,2.848736 5.416167,4.643344 C 58.614334,27.483611 57.260351,22.206768 56.421696,19.015263 55.149066,14.172268 54.241403,10.340754 53.185389,8.0524745 52.815225,7.2503647 52.540611,6.4969378 52.052073,6.1836069 51.974407,6.1337905 51.885945,6.1211124 51.79757,6.1189646 Z" 70 /> 71 <path 72 fill="#cba7f7" 73 d="m 15.935563,6.1189692 c -0.08837,0.00223 -0.176832,0.014766 -0.254502,0.064642 -0.48854,0.3133308 -0.763154,1.0667562 -1.13332,1.8688677 -1.056011,2.2882791 -1.963673,6.1197931 -3.236303,10.9627891 -0.85539,3.255187 -2.247014,8.680054 -3.4314032,13.071013 1.5346704,-1.910372 3.5390122,-3.56005 5.8517882,-4.91124 3.456591,-2.019439 7.668347,-3.458497 12.320324,-4.231015 C 24.452511,19.365796 22.96466,16.190327 22.117564,14.758042 20.260808,11.61862 19.059771,9.2744012 17.888878,7.7484762 17.423256,7.1416679 16.926404,6.5114787 16.442819,6.2138951 16.301603,6.127059 16.165987,6.1222115 16.02292,6.1195569 c -0.02901,-5.429e-4 -0.0579,-0.0013 -0.08734,-5.847e-4 z" 74 /> 75 </svg> 76); 77 78const DeerIcon = () => ( 79 <svg fill="none" viewBox="0 0 512 512" width="18" height="18"> 80 <path 81 fill="#739f7c" 82 d="m 149.96484,186.56641 46.09766,152.95898 c 0,0 -6.30222,-9.61174 -15.60547,-17.47656 -8.87322,-7.50128 -28.4082,-4.04492 -28.4082,-4.04492 0,0 6.14721,39.88867 15.53125,44.39843 10.71251,5.1482 22.19726,0.16993 22.19726,0.16993 0,0 11.7613,-4.87282 22.82032,31.82421 5.26534,17.47196 15.33258,50.877 20.9707,69.58594 2.16717,7.1913 8.83789,7.25781 8.83789,7.25781 0,0 6.67072,-0.0665 8.83789,-7.25781 5.63812,-18.70894 15.70536,-52.11398 20.9707,-69.58594 11.05902,-36.69703 22.82032,-31.82421 22.82032,-31.82421 0,0 11.48475,4.97827 22.19726,-0.16993 9.38404,-4.50976 15.5332,-44.39843 15.5332,-44.39843 0,0 -19.53693,-3.45636 -28.41015,4.04492 -9.30325,7.86482 -15.60547,17.47656 -15.60547,17.47656 l 46.09766,-152.95898 -49.32618,83.84179 -20.34375,-31.1914 6.35547,54.96875 -23.1582,39.36132 c 0,0 -2.97595,5.06226 -5.94336,4.68946 -0.009,-0.001 -0.0169,0.003 -0.0254,0.01 -0.008,-0.007 -0.0167,-0.0109 -0.0254,-0.01 -2.96741,0.3728 -5.94336,-4.68946 -5.94336,-4.68946 l -23.1582,-39.36132 6.35547,-54.96875 -20.34375,31.1914 z" 83 transform="matrix(2.6921023,0,0,1.7145911,-396.58283,-308.01527)" 84 /> 85 </svg> 86); 87 88const BLUESKY_FORKS = [ 89 { 90 name: "Bluesky", 91 domain: "bsky.app", 92 Icon: () => <BlueskyIcon size={18} color={BLUESKY_COLOR} />, 93 }, 94 { name: "Witchsky", domain: "witchsky.app", Icon: WitchskyIcon }, 95 { name: "Blacksky", domain: "blacksky.community", Icon: BlackskyIcon }, 96 { name: "Catsky", domain: "catsky.social", Icon: CatskyIcon }, 97 { name: "Deer", domain: "deer.social", Icon: DeerIcon }, 98]; 99 100export default function ShareMenu({ uri, text, customUrl, handle, type }) { 101 const [isOpen, setIsOpen] = useState(false); 102 const [copied, setCopied] = useState(false); 103 const menuRef = useRef(null); 104 105 const getShareUrl = () => { 106 if (customUrl) return customUrl; 107 if (!uri) return ""; 108 109 const uriParts = uri.split("/"); 110 const rkey = uriParts[uriParts.length - 1]; 111 112 if (handle && type) { 113 return `${window.location.origin}/${handle}/${type.toLowerCase()}/${rkey}`; 114 } 115 116 const did = uriParts[2]; 117 return `${window.location.origin}/at/${did}/${rkey}`; 118 }; 119 120 const shareUrl = getShareUrl(); 121 122 useEffect(() => { 123 const handleClickOutside = (e) => { 124 if (menuRef.current && !menuRef.current.contains(e.target)) { 125 setIsOpen(false); 126 } 127 }; 128 129 const card = menuRef.current?.closest(".card"); 130 if (card) { 131 if (isOpen) { 132 card.style.zIndex = "50"; 133 } else { 134 card.style.zIndex = ""; 135 } 136 } 137 138 if (isOpen) { 139 document.addEventListener("mousedown", handleClickOutside); 140 } 141 return () => document.removeEventListener("mousedown", handleClickOutside); 142 }, [isOpen]); 143 144 const handleShareToFork = (domain) => { 145 const composeText = text 146 ? `${text.substring(0, 200)}...\n\n${shareUrl}` 147 : shareUrl; 148 const composeUrl = `https://${domain}/intent/compose?text=${encodeURIComponent(composeText)}`; 149 window.open(composeUrl, "_blank"); 150 setIsOpen(false); 151 }; 152 153 const handleCopy = async () => { 154 try { 155 await navigator.clipboard.writeText(shareUrl); 156 setCopied(true); 157 setTimeout(() => { 158 setCopied(false); 159 setIsOpen(false); 160 }, 1500); 161 } catch { 162 prompt("Copy this link:", shareUrl); 163 } 164 }; 165 166 const handleSystemShare = async () => { 167 if (navigator.share) { 168 try { 169 await navigator.share({ 170 title: "Margin Annotation", 171 text: text?.substring(0, 100), 172 url: shareUrl, 173 }); 174 } catch { 175 /* ignore */ 176 } 177 } 178 setIsOpen(false); 179 }; 180 181 return ( 182 <div className="share-menu-container" ref={menuRef}> 183 <button 184 className="annotation-action" 185 onClick={() => setIsOpen(!isOpen)} 186 title="Share" 187 > 188 <svg 189 width="18" 190 height="18" 191 viewBox="0 0 24 24" 192 fill="none" 193 stroke="currentColor" 194 strokeWidth="2" 195 strokeLinecap="round" 196 strokeLinejoin="round" 197 > 198 <circle cx="18" cy="5" r="3" /> 199 <circle cx="6" cy="12" r="3" /> 200 <circle cx="18" cy="19" r="3" /> 201 <line x1="8.59" y1="13.51" x2="15.42" y2="17.49" /> 202 <line x1="15.41" y1="6.51" x2="8.59" y2="10.49" /> 203 </svg> 204 </button> 205 206 {isOpen && ( 207 <div className="share-menu"> 208 <div className="share-menu-section"> 209 <div className="share-menu-label">Share to</div> 210 {BLUESKY_FORKS.map((fork) => ( 211 <button 212 key={fork.domain} 213 className="share-menu-item" 214 onClick={() => handleShareToFork(fork.domain)} 215 > 216 <span className="share-menu-icon"> 217 <fork.Icon /> 218 </span> 219 <span>{fork.name}</span> 220 </button> 221 ))} 222 </div> 223 <div className="share-menu-divider" /> 224 <button className="share-menu-item" onClick={handleCopy}> 225 {copied ? <Check size={16} /> : <Copy size={16} />} 226 <span>{copied ? "Copied!" : "Copy Link"}</span> 227 </button> 228 {navigator.share && ( 229 <button className="share-menu-item" onClick={handleSystemShare}> 230 <ExternalLink size={16} /> 231 <span>More...</span> 232 </button> 233 )} 234 </div> 235 )} 236 </div> 237 ); 238}