Brak opisu

DashboardView.swift 169KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683
  1. //
  2. // DashboardView.swift
  3. // App for Indeed
  4. //
  5. import Cocoa
  6. import QuartzCore
  7. import Security
  8. private enum JobListingCardContext {
  9. case homeSearchResults
  10. case savedJobsPage
  11. }
  12. private enum PremiumSheetLayout {
  13. /// Grow the sheet past the host content rect on each side to hide compositing hairlines.
  14. static let overscanPerEdge: CGFloat = 2
  15. /// Additional growth on the top edge only (pt).
  16. static let overscanExtraTop: CGFloat = 0.5
  17. }
  18. final class DashboardView: NSView, NSTextFieldDelegate, NSSharingServicePickerDelegate, NSSharingServiceDelegate {
  19. /// Indeed.com-inspired neutrals and brand blue (white surfaces, `#2557a7` accent, `#2d2d2d` / `#767676` text, `#d4d2d0` borders).
  20. private enum Theme {
  21. static let brandBlue = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 1)
  22. static let pageBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  23. static let chromeBackground = NSColor(srgbRed: 247 / 255, green: 247 / 255, blue: 247 / 255, alpha: 1)
  24. static let sidebarBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  25. static let mainHostBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  26. /// Welcome hero (matches reference: `#0052CC` heading, `#334155` subline, `#EFF6FF` icon well).
  27. static let welcomeHeroHeadingBlue = NSColor(srgbRed: 0, green: 82 / 255, blue: 204 / 255, alpha: 1)
  28. static let welcomeHeroSubtitleText = NSColor(srgbRed: 51 / 255, green: 65 / 255, blue: 85 / 255, alpha: 1)
  29. static let welcomeHeroIconWell = NSColor(srgbRed: 239 / 255, green: 246 / 255, blue: 255 / 255, alpha: 1)
  30. /// Light decorative strokes / sparkles behind the welcome hero.
  31. static let welcomeHeroWaveTint = NSColor(srgbRed: 186 / 255, green: 210 / 255, blue: 253 / 255, alpha: 1)
  32. /// Subtitle on the welcome hero: dark neutral gray to match the reference layout.
  33. static let welcomeSubtitleText = NSColor(srgbRed: 64 / 255, green: 64 / 255, blue: 64 / 255, alpha: 1)
  34. static let selectionFill = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 0.12)
  35. static let cardBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  36. static let toggleBackground = NSColor(srgbRed: 232 / 255, green: 232 / 255, blue: 232 / 255, alpha: 1)
  37. static let primaryText = NSColor(srgbRed: 45 / 255, green: 45 / 255, blue: 45 / 255, alpha: 1)
  38. static let secondaryText = NSColor(srgbRed: 118 / 255, green: 118 / 255, blue: 118 / 255, alpha: 1)
  39. static let tertiaryText = NSColor(srgbRed: 118 / 255, green: 118 / 255, blue: 118 / 255, alpha: 1)
  40. static let border = NSColor(srgbRed: 212 / 255, green: 210 / 255, blue: 208 / 255, alpha: 1)
  41. static let featureIconWell = NSColor(srgbRed: 220 / 255, green: 235 / 255, blue: 252 / 255, alpha: 1)
  42. /// Job search bar outer stroke (soft blue-gray, pill field in the reference UI).
  43. static let searchBarBorder = NSColor(srgbRed: 180 / 255, green: 200 / 255, blue: 228 / 255, alpha: 1)
  44. /// Search bar border on hover (brand-tinted, matches focus affordance elsewhere).
  45. static let searchBarBorderHover = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 0.55)
  46. static let proCardFill = NSColor(srgbRed: 239 / 255, green: 244 / 255, blue: 252 / 255, alpha: 1)
  47. static let proCardBorder = NSColor(srgbRed: 212 / 255, green: 210 / 255, blue: 208 / 255, alpha: 1)
  48. static let proAccent = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 1)
  49. static let proCTABackground = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 1)
  50. static let proCTAText = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  51. /// Hover states: darker brand blue, stronger tints, and subtle neutral fills used across CTAs, toggles, and the sidebar.
  52. static let brandBlueHover = NSColor(srgbRed: 28 / 255, green: 70 / 255, blue: 140 / 255, alpha: 1)
  53. static let selectionFillHover = NSColor(srgbRed: 37 / 255, green: 87 / 255, blue: 167 / 255, alpha: 0.2)
  54. static let neutralHoverFill = NSColor(srgbRed: 240 / 255, green: 240 / 255, blue: 240 / 255, alpha: 1)
  55. static let sidebarRowHoverFill = NSColor(srgbRed: 0, green: 0, blue: 0, alpha: 0.04)
  56. static let settingsPageBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  57. static let settingsGroupBackground = NSColor(srgbRed: 1, green: 1, blue: 1, alpha: 1)
  58. static let settingsIconBackground = NSColor(srgbRed: 239 / 255, green: 244 / 255, blue: 252 / 255, alpha: 1)
  59. static let settingsDivider = NSColor(srgbRed: 228 / 255, green: 228 / 255, blue: 228 / 255, alpha: 1)
  60. }
  61. /// Multiline `NSTextField` often ignores `alignment` for wrapped runs; explicit paragraph alignment matches the title.
  62. private static func jobListingDescriptionAttributedString(_ plain: String) -> NSAttributedString {
  63. let paragraph = NSMutableParagraphStyle()
  64. paragraph.alignment = .left
  65. paragraph.lineBreakMode = .byWordWrapping
  66. paragraph.baseWritingDirection = .leftToRight
  67. let font = NSFont.systemFont(ofSize: 13, weight: .regular)
  68. return NSAttributedString(string: plain, attributes: [
  69. .font: font,
  70. .foregroundColor: Theme.secondaryText,
  71. .paragraphStyle: paragraph
  72. ])
  73. }
  74. /// Horizontal row for sidebar + main; plain view + constraints keep both panels top/bottom aligned (stack view height alignment was inconsistent).
  75. private let panelsRow = NSView()
  76. private let chromeContainer = NSView()
  77. private let sidebar = NSStackView()
  78. private let mainHost = NSView()
  79. private let mainOverlay = NSStackView()
  80. private let greetingLabel = NSTextField(labelWithString: "")
  81. private let subtitleLabel = NSTextField(labelWithString: "")
  82. private let searchBarShadowHost = NSView()
  83. private let searchCard = HoverableView()
  84. private let jobSearchIcon = NSImageView()
  85. private let jobKeywordsField = NSTextField()
  86. private let findJobsButton = HoverableButton()
  87. private let findJobsCTAPill = HoverableView()
  88. private let sendIconView = NSImageView()
  89. private let sendLabel = NSTextField(labelWithString: "Send")
  90. private let sendContentStack = NSStackView()
  91. private let findJobsCTAHost = NSView()
  92. private let welcomeHeroHost = NSView()
  93. private let welcomeHeroBackgroundView = WelcomeHeroBackgroundView()
  94. private let welcomeLogoWell = NSView()
  95. private let welcomeLogoView = IndeedLogoView(displayHeight: 40, variant: .compact)
  96. private let featureCardsRow = NSStackView()
  97. private enum FeatureShortcut: Int { case role = 0, company = 1, skill = 2 }
  98. private let clearChatButton = NSButton(title: "Clear chat", target: nil, action: nil)
  99. private let chatScrollView = NSScrollView()
  100. private let chatDocumentView = JobListingsDocumentView()
  101. private let chatStack = NSStackView()
  102. /// Shown when a sidebar item other than Home is selected.
  103. private let nonHomeHost = NSView()
  104. /// Full-bleed Indeed apply / listing web view inside the main panel (same window as the dashboard).
  105. private let indeedJobBrowserHost = NSView()
  106. private let nonHomeGenericContainer = NSView()
  107. private let nonHomeTitleLabel = NSTextField(labelWithString: "")
  108. private let nonHomeSubtitleLabel = NSTextField(wrappingLabelWithString: "")
  109. private let savedJobsPageContainer = NSView()
  110. private let savedJobsPageTitleLabel = NSTextField(labelWithString: "Saved Jobs")
  111. private let savedJobsPageSubtitleLabel = NSTextField(wrappingLabelWithString: "")
  112. private let savedJobsScrollView = NSScrollView()
  113. private let savedJobsDocumentView = JobListingsDocumentView()
  114. private let savedJobsStack = NSStackView()
  115. private let settingsPageContainer = NSView()
  116. private weak var appearanceModeSegment: NSSegmentedControl?
  117. private let cvMakerPageContainer = NSView()
  118. private lazy var cvMakerPageView: CVMakerPageView = {
  119. CVMakerPageView()
  120. }()
  121. private let profilePageContainer = NSView()
  122. private lazy var profilesListPageView: ProfilesListPageView = {
  123. ProfilesListPageView()
  124. }()
  125. private lazy var myProfilePageView: MyProfilePageView = {
  126. MyProfilePageView()
  127. }()
  128. /// When true, `myProfilePageView` is visible instead of the profiles list.
  129. private var isProfileEditorPresented = false
  130. /// When true, the merged CV preview is visible instead of the profiles list or editor.
  131. private var isCVDocumentPreviewPresented = false
  132. /// Exact template chosen in CV Maker until the user leaves Profile or starts a new CV Maker hand-off (avoids re-resolving by id and picking a different row).
  133. private var pendingCVTemplate: CVTemplate?
  134. private let cvFilledPreviewPageView = CVFilledPreviewPageView()
  135. private var currentSidebarItems: [SidebarItem] = []
  136. private var selectedSidebarIndex: Int = 0
  137. /// When true, the **Indeed** sidebar row is highlighted instead of `selectedSidebarIndex`.
  138. private var isIndeedSidebarSelected = false
  139. /// All jobs that have been shown in the current chat session, oldest first. Used to deduplicate continuation searches so "show more" doesn't re-display the same listings.
  140. private var lastSearchResults: [JobListing] = []
  141. /// "Show more jobs" row under the latest assistant message that listed jobs; removed when a newer listing block replaces it.
  142. private var trailingLoadMoreJobsRow: NSView?
  143. private weak var trailingLoadMoreJobsButton: HoverableButton?
  144. /// Most recently saved jobs appear first; persisted across launches.
  145. private var savedJobOrder: [JobListing] = []
  146. private var chatMessages: [ChatMessage] = []
  147. private var isAwaitingResponse = false
  148. /// Shown under the latest user message while a job search request is in flight.
  149. private var chatThinkingRowHost: NSView?
  150. private let jobSearchService = OpenAIJobSearchService()
  151. private var premiumPlansWindowController: PremiumPlansWindowController?
  152. private var indeedJobBrowserViewController: IndeedJobBrowserViewController?
  153. private var isIndeedJobBrowserPresented = false
  154. private weak var sidebarUpgradeCard: NSView?
  155. private weak var sidebarUpgradeHeadline: NSTextField?
  156. private weak var sidebarUpgradeDescription: NSTextField?
  157. private weak var sidebarUpgradeButton: HoverableButton?
  158. private var subscriptionObserver: NSObjectProtocol?
  159. /// Retains the system share picker until the user picks a destination or dismisses the menu.
  160. private var appSharePicker: NSSharingServicePicker?
  161. /// Upper bound sent to the model per request (was fixed at 8 in the prompt). Clamped when calling the API.
  162. private static let jobsPerSearchDefault = 15
  163. private static let jobsPerSearchMin = 1
  164. private static let jobsPerSearchMaxCap = 25
  165. private static func clampedJobsPerRequest(_ requested: Int = jobsPerSearchDefault) -> Int {
  166. min(jobsPerSearchMaxCap, max(jobsPerSearchMin, requested))
  167. }
  168. override init(frame frameRect: NSRect) {
  169. super.init(frame: frameRect)
  170. setupLayout()
  171. }
  172. required init?(coder: NSCoder) {
  173. super.init(coder: coder)
  174. setupLayout()
  175. }
  176. deinit {
  177. if let subscriptionObserver {
  178. NotificationCenter.default.removeObserver(subscriptionObserver)
  179. }
  180. }
  181. override func viewDidMoveToWindow() {
  182. super.viewDidMoveToWindow()
  183. guard window != nil else { return }
  184. Task { @MainActor in
  185. await SubscriptionStore.shared.refreshEntitlements(deep: true)
  186. self.applyProSubscriptionToSidebar()
  187. }
  188. }
  189. override func layout() {
  190. super.layout()
  191. updateSearchBarShadowPath()
  192. updateFindJobsCTAShadowPath()
  193. updateJobListingDescriptionWidths()
  194. updateChatBubbleWidths()
  195. }
  196. func render(_ data: DashboardData) {
  197. dismissIndeedJobBrowserEmbedded()
  198. greetingLabel.stringValue = "Welcome"
  199. subtitleLabel.stringValue = data.subtitle
  200. currentSidebarItems = data.sidebarItems
  201. if selectedSidebarIndex >= currentSidebarItems.count {
  202. selectedSidebarIndex = max(0, currentSidebarItems.count - 1)
  203. }
  204. configureSidebar()
  205. savedJobOrder = Self.normalizedSavedJobs(SavedJobsStore.load())
  206. resetChatState()
  207. updateMainContentVisibility()
  208. }
  209. private func setupLayout() {
  210. wantsLayer = true
  211. // Match chrome so the outer margin (inset chrome container) is grey, not an extra white ring.
  212. layer?.backgroundColor = Theme.chromeBackground.cgColor
  213. panelsRow.translatesAutoresizingMaskIntoConstraints = false
  214. chromeContainer.translatesAutoresizingMaskIntoConstraints = false
  215. chromeContainer.wantsLayer = true
  216. chromeContainer.layer?.backgroundColor = Theme.chromeBackground.cgColor
  217. chromeContainer.layer?.cornerRadius = 18
  218. chromeContainer.layer?.masksToBounds = true
  219. addSubview(chromeContainer)
  220. chromeContainer.addSubview(panelsRow)
  221. sidebar.orientation = .vertical
  222. sidebar.spacing = 10
  223. sidebar.distribution = .fill
  224. sidebar.alignment = .leading
  225. sidebar.wantsLayer = true
  226. sidebar.layer?.backgroundColor = Theme.sidebarBackground.cgColor
  227. sidebar.layer?.cornerRadius = 16
  228. sidebar.edgeInsets = NSEdgeInsets(top: 20, left: 6, bottom: 6, right: 6)
  229. sidebar.translatesAutoresizingMaskIntoConstraints = false
  230. mainHost.translatesAutoresizingMaskIntoConstraints = false
  231. mainHost.wantsLayer = true
  232. mainHost.layer?.backgroundColor = Theme.mainHostBackground.cgColor
  233. mainHost.layer?.cornerRadius = 16
  234. mainHost.layer?.masksToBounds = true
  235. sidebar.setContentHuggingPriority(.required, for: .horizontal)
  236. mainHost.setContentHuggingPriority(.defaultLow, for: .horizontal)
  237. mainHost.addSubview(mainOverlay)
  238. configureNonHomePlaceholder()
  239. mainHost.addSubview(nonHomeHost)
  240. indeedJobBrowserHost.translatesAutoresizingMaskIntoConstraints = false
  241. indeedJobBrowserHost.wantsLayer = true
  242. indeedJobBrowserHost.layer?.backgroundColor = Theme.mainHostBackground.cgColor
  243. indeedJobBrowserHost.isHidden = true
  244. mainHost.addSubview(indeedJobBrowserHost)
  245. mainOverlay.orientation = .vertical
  246. mainOverlay.spacing = 0
  247. mainOverlay.alignment = .centerX
  248. mainOverlay.distribution = .fill
  249. mainOverlay.translatesAutoresizingMaskIntoConstraints = false
  250. mainOverlay.setContentHuggingPriority(.defaultLow, for: .vertical)
  251. greetingLabel.font = .systemFont(ofSize: 28, weight: .bold)
  252. greetingLabel.textColor = Theme.welcomeHeroHeadingBlue
  253. greetingLabel.alignment = .center
  254. greetingLabel.maximumNumberOfLines = 1
  255. subtitleLabel.font = .systemFont(ofSize: 14, weight: .regular)
  256. subtitleLabel.textColor = Theme.welcomeHeroSubtitleText
  257. subtitleLabel.alignment = .center
  258. subtitleLabel.maximumNumberOfLines = 2
  259. let topInset = NSView()
  260. topInset.translatesAutoresizingMaskIntoConstraints = false
  261. topInset.heightAnchor.constraint(equalToConstant: 12).isActive = true
  262. configureSearchBar()
  263. configureChatViews()
  264. welcomeHeroHost.translatesAutoresizingMaskIntoConstraints = false
  265. welcomeHeroBackgroundView.translatesAutoresizingMaskIntoConstraints = false
  266. welcomeHeroBackgroundView.waveTint = Theme.welcomeHeroWaveTint
  267. welcomeLogoWell.translatesAutoresizingMaskIntoConstraints = false
  268. welcomeLogoWell.wantsLayer = true
  269. welcomeLogoWell.layer?.backgroundColor = Theme.welcomeHeroIconWell.cgColor
  270. welcomeLogoWell.layer?.cornerRadius = 28
  271. if #available(macOS 11.0, *) {
  272. welcomeLogoWell.layer?.cornerCurve = .continuous
  273. }
  274. welcomeLogoView.translatesAutoresizingMaskIntoConstraints = false
  275. welcomeLogoWell.addSubview(welcomeLogoView)
  276. let welcomeHeroContent = NSStackView(views: [welcomeLogoWell, greetingLabel, subtitleLabel])
  277. welcomeHeroContent.orientation = .vertical
  278. welcomeHeroContent.spacing = 8
  279. welcomeHeroContent.alignment = .centerX
  280. welcomeHeroContent.translatesAutoresizingMaskIntoConstraints = false
  281. welcomeHeroHost.addSubview(welcomeHeroBackgroundView)
  282. welcomeHeroHost.addSubview(welcomeHeroContent)
  283. welcomeHeroHost.setContentHuggingPriority(.defaultHigh, for: .vertical)
  284. NSLayoutConstraint.activate([
  285. welcomeHeroBackgroundView.leadingAnchor.constraint(equalTo: welcomeHeroHost.leadingAnchor),
  286. welcomeHeroBackgroundView.trailingAnchor.constraint(equalTo: welcomeHeroHost.trailingAnchor),
  287. welcomeHeroBackgroundView.topAnchor.constraint(equalTo: welcomeHeroHost.topAnchor),
  288. welcomeHeroBackgroundView.bottomAnchor.constraint(equalTo: welcomeHeroHost.bottomAnchor),
  289. welcomeHeroContent.centerXAnchor.constraint(equalTo: welcomeHeroHost.centerXAnchor),
  290. welcomeHeroContent.leadingAnchor.constraint(greaterThanOrEqualTo: welcomeHeroHost.leadingAnchor, constant: 16),
  291. welcomeHeroContent.trailingAnchor.constraint(lessThanOrEqualTo: welcomeHeroHost.trailingAnchor, constant: -16),
  292. welcomeHeroContent.topAnchor.constraint(equalTo: welcomeHeroHost.topAnchor, constant: 4),
  293. welcomeHeroContent.bottomAnchor.constraint(equalTo: welcomeHeroHost.bottomAnchor, constant: -2),
  294. welcomeLogoWell.widthAnchor.constraint(equalToConstant: 56),
  295. welcomeLogoWell.heightAnchor.constraint(equalToConstant: 56),
  296. welcomeLogoView.centerXAnchor.constraint(equalTo: welcomeLogoWell.centerXAnchor),
  297. welcomeLogoView.centerYAnchor.constraint(equalTo: welcomeLogoWell.centerYAnchor)
  298. ])
  299. configureFeatureShortcutCards()
  300. featureCardsRow.setContentHuggingPriority(.defaultHigh, for: .vertical)
  301. let heroCardsSpacer = NSView()
  302. heroCardsSpacer.translatesAutoresizingMaskIntoConstraints = false
  303. heroCardsSpacer.heightAnchor.constraint(equalToConstant: 14).isActive = true
  304. let midSpacer = NSView()
  305. midSpacer.translatesAutoresizingMaskIntoConstraints = false
  306. midSpacer.heightAnchor.constraint(equalToConstant: 6).isActive = true
  307. let chatTopSpacer = NSView()
  308. chatTopSpacer.translatesAutoresizingMaskIntoConstraints = false
  309. chatTopSpacer.heightAnchor.constraint(equalToConstant: 8).isActive = true
  310. let chatBottomSpacer = NSView()
  311. chatBottomSpacer.translatesAutoresizingMaskIntoConstraints = false
  312. chatBottomSpacer.heightAnchor.constraint(equalToConstant: 8).isActive = true
  313. mainOverlay.addArrangedSubview(topInset)
  314. mainOverlay.addArrangedSubview(welcomeHeroHost)
  315. mainOverlay.addArrangedSubview(heroCardsSpacer)
  316. mainOverlay.addArrangedSubview(featureCardsRow)
  317. mainOverlay.addArrangedSubview(midSpacer)
  318. mainOverlay.addArrangedSubview(chatTopSpacer)
  319. let chatHeaderRow = NSStackView()
  320. chatHeaderRow.orientation = .horizontal
  321. chatHeaderRow.spacing = 8
  322. chatHeaderRow.alignment = .centerY
  323. chatHeaderRow.distribution = .fill
  324. chatHeaderRow.translatesAutoresizingMaskIntoConstraints = false
  325. let chatHeaderLeadingSpacer = NSView()
  326. chatHeaderLeadingSpacer.translatesAutoresizingMaskIntoConstraints = false
  327. chatHeaderLeadingSpacer.setContentHuggingPriority(.defaultLow, for: .horizontal)
  328. clearChatButton.translatesAutoresizingMaskIntoConstraints = false
  329. clearChatButton.bezelStyle = .rounded
  330. clearChatButton.font = .systemFont(ofSize: 12, weight: .medium)
  331. clearChatButton.target = self
  332. clearChatButton.action = #selector(didTapClearChat)
  333. clearChatButton.toolTip = "Remove all messages and start a new conversation"
  334. clearChatButton.setContentHuggingPriority(.required, for: .horizontal)
  335. chatHeaderRow.addArrangedSubview(chatHeaderLeadingSpacer)
  336. chatHeaderRow.addArrangedSubview(clearChatButton)
  337. mainOverlay.addArrangedSubview(chatHeaderRow)
  338. mainOverlay.addArrangedSubview(chatScrollView)
  339. mainOverlay.addArrangedSubview(chatBottomSpacer)
  340. mainOverlay.addArrangedSubview(searchBarShadowHost)
  341. panelsRow.addSubview(sidebar)
  342. panelsRow.addSubview(mainHost)
  343. NSLayoutConstraint.activate([
  344. chromeContainer.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 6),
  345. chromeContainer.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -6),
  346. chromeContainer.topAnchor.constraint(equalTo: topAnchor, constant: 6),
  347. chromeContainer.bottomAnchor.constraint(equalTo: bottomAnchor, constant: -6),
  348. panelsRow.leadingAnchor.constraint(equalTo: chromeContainer.leadingAnchor, constant: 6),
  349. panelsRow.trailingAnchor.constraint(equalTo: chromeContainer.trailingAnchor, constant: -8),
  350. panelsRow.topAnchor.constraint(equalTo: safeAreaLayoutGuide.topAnchor, constant: 2),
  351. panelsRow.bottomAnchor.constraint(equalTo: chromeContainer.bottomAnchor, constant: -8),
  352. sidebar.leadingAnchor.constraint(equalTo: panelsRow.leadingAnchor),
  353. sidebar.topAnchor.constraint(equalTo: panelsRow.topAnchor),
  354. sidebar.bottomAnchor.constraint(equalTo: panelsRow.bottomAnchor),
  355. sidebar.widthAnchor.constraint(equalToConstant: 218),
  356. mainHost.leadingAnchor.constraint(equalTo: sidebar.trailingAnchor, constant: 6),
  357. mainHost.trailingAnchor.constraint(equalTo: panelsRow.trailingAnchor),
  358. mainHost.topAnchor.constraint(equalTo: panelsRow.topAnchor),
  359. mainHost.bottomAnchor.constraint(equalTo: panelsRow.bottomAnchor),
  360. mainHost.widthAnchor.constraint(greaterThanOrEqualToConstant: 720),
  361. mainOverlay.leadingAnchor.constraint(equalTo: mainHost.leadingAnchor),
  362. mainOverlay.trailingAnchor.constraint(equalTo: mainHost.trailingAnchor),
  363. mainOverlay.topAnchor.constraint(equalTo: mainHost.topAnchor),
  364. mainOverlay.bottomAnchor.constraint(equalTo: mainHost.bottomAnchor, constant: -24),
  365. nonHomeHost.leadingAnchor.constraint(equalTo: mainHost.leadingAnchor),
  366. nonHomeHost.trailingAnchor.constraint(equalTo: mainHost.trailingAnchor),
  367. nonHomeHost.topAnchor.constraint(equalTo: mainHost.topAnchor),
  368. nonHomeHost.bottomAnchor.constraint(equalTo: mainHost.bottomAnchor, constant: -24),
  369. indeedJobBrowserHost.leadingAnchor.constraint(equalTo: mainHost.leadingAnchor),
  370. indeedJobBrowserHost.trailingAnchor.constraint(equalTo: mainHost.trailingAnchor),
  371. indeedJobBrowserHost.topAnchor.constraint(equalTo: mainHost.topAnchor),
  372. indeedJobBrowserHost.bottomAnchor.constraint(equalTo: mainHost.bottomAnchor, constant: -24),
  373. searchBarShadowHost.widthAnchor.constraint(equalTo: mainOverlay.widthAnchor, multiplier: 0.92),
  374. featureCardsRow.widthAnchor.constraint(equalTo: mainOverlay.widthAnchor, multiplier: 0.92),
  375. chatHeaderRow.widthAnchor.constraint(equalTo: mainOverlay.widthAnchor, multiplier: 0.92),
  376. chatScrollView.widthAnchor.constraint(equalTo: mainOverlay.widthAnchor, multiplier: 0.92),
  377. welcomeHeroHost.leadingAnchor.constraint(equalTo: mainOverlay.leadingAnchor),
  378. welcomeHeroHost.trailingAnchor.constraint(equalTo: mainOverlay.trailingAnchor),
  379. greetingLabel.leadingAnchor.constraint(equalTo: mainOverlay.leadingAnchor, constant: 16),
  380. greetingLabel.trailingAnchor.constraint(equalTo: mainOverlay.trailingAnchor, constant: -16),
  381. subtitleLabel.leadingAnchor.constraint(equalTo: greetingLabel.leadingAnchor),
  382. subtitleLabel.trailingAnchor.constraint(equalTo: greetingLabel.trailingAnchor),
  383. welcomeHeroContent.widthAnchor.constraint(equalTo: mainOverlay.widthAnchor, constant: -32)
  384. ])
  385. registerSubscriptionObserverOnce()
  386. }
  387. private func registerSubscriptionObserverOnce() {
  388. guard subscriptionObserver == nil else { return }
  389. subscriptionObserver = NotificationCenter.default.addObserver(
  390. forName: .subscriptionStatusDidChange,
  391. object: nil,
  392. queue: .main
  393. ) { [weak self] _ in
  394. self?.applyProSubscriptionToSidebar()
  395. }
  396. }
  397. private func applyProSubscriptionToSidebar() {
  398. let active = SubscriptionStore.shared.isProActive
  399. sidebarUpgradeCard?.isHidden = false
  400. guard let headline = sidebarUpgradeHeadline,
  401. let upgradeDescription = sidebarUpgradeDescription,
  402. let upgradeButton = sidebarUpgradeButton else { return }
  403. let descriptionWidth: CGFloat = 158
  404. if active {
  405. headline.stringValue = "You're on Pro"
  406. upgradeDescription.stringValue = "Manage billing, renewals, and plans in Premium."
  407. upgradeDescription.preferredMaxLayoutWidth = descriptionWidth
  408. upgradeButton.title = "Manage Subscription"
  409. } else {
  410. headline.stringValue = "Upgrade to Pro"
  411. upgradeDescription.stringValue = "Unlimited AI matches, smart alerts, and interview prep—all in one place."
  412. upgradeDescription.preferredMaxLayoutWidth = descriptionWidth
  413. upgradeButton.title = "Try Pro"
  414. }
  415. }
  416. /// Returns `false` and presents the paywall when the user does not have an active Pro subscription.
  417. @discardableResult
  418. private func ensureProAccess() -> Bool {
  419. guard SubscriptionStore.shared.isProActive else {
  420. presentPremiumPlansSheet()
  421. return false
  422. }
  423. return true
  424. }
  425. private func presentPremiumPlansSheet() {
  426. guard let hostWindow = window else { return }
  427. if premiumPlansWindowController == nil {
  428. premiumPlansWindowController = PremiumPlansWindowController()
  429. }
  430. guard let paywallWindow = premiumPlansWindowController?.window else { return }
  431. if hostWindow.attachedSheet === paywallWindow {
  432. return
  433. }
  434. paywallWindow.styleMask = [.borderless, .closable, .resizable]
  435. paywallWindow.isOpaque = true
  436. paywallWindow.backgroundColor = PremiumPlansWindowController.paywallSheetBackground
  437. let hostContentRect = hostWindow.contentRect(forFrameRect: hostWindow.frame)
  438. let overscan = PremiumSheetLayout.overscanPerEdge
  439. var expandedContentRect = hostContentRect.insetBy(dx: -overscan, dy: -overscan)
  440. expandedContentRect.size.height += PremiumSheetLayout.overscanExtraTop
  441. let paywallFrame = paywallWindow.frameRect(forContentRect: expandedContentRect)
  442. paywallWindow.setFrame(paywallFrame, display: false)
  443. let lockedSize = paywallWindow.frame.size
  444. paywallWindow.minSize = lockedSize
  445. paywallWindow.maxSize = lockedSize
  446. hostWindow.beginSheet(paywallWindow)
  447. }
  448. private func configureFeatureShortcutCards() {
  449. featureCardsRow.orientation = .horizontal
  450. featureCardsRow.spacing = 16
  451. featureCardsRow.distribution = .fillEqually
  452. featureCardsRow.alignment = .top
  453. featureCardsRow.translatesAutoresizingMaskIntoConstraints = false
  454. let specs: [(symbol: String, title: String, subtitle: String, action: Selector)] = [
  455. ("briefcase", "Role", "Explore similar or better job roles", #selector(didTapFeatureRole)),
  456. ("building.2", "Company", "Find opportunities at other companies", #selector(didTapFeatureCompany)),
  457. ("chevron.left.forwardslash.chevron.right", "Skill", "Match jobs that fit your skills", #selector(didTapFeatureSkill))
  458. ]
  459. for spec in specs {
  460. let card = FeatureShortcutCardView(
  461. symbolName: spec.symbol,
  462. title: spec.title,
  463. subtitle: spec.subtitle,
  464. target: self,
  465. action: spec.action
  466. )
  467. featureCardsRow.addArrangedSubview(card)
  468. }
  469. }
  470. private func configureChatViews() {
  471. chatDocumentView.translatesAutoresizingMaskIntoConstraints = false
  472. chatStack.orientation = .vertical
  473. chatStack.spacing = 18
  474. chatStack.alignment = .width
  475. chatStack.distribution = .fill
  476. chatStack.translatesAutoresizingMaskIntoConstraints = false
  477. chatDocumentView.addSubview(chatStack)
  478. NSLayoutConstraint.activate([
  479. chatStack.leadingAnchor.constraint(equalTo: chatDocumentView.leadingAnchor, constant: 4),
  480. chatStack.trailingAnchor.constraint(equalTo: chatDocumentView.trailingAnchor, constant: -4),
  481. chatStack.topAnchor.constraint(equalTo: chatDocumentView.topAnchor, constant: 8),
  482. chatStack.bottomAnchor.constraint(equalTo: chatDocumentView.bottomAnchor, constant: -8)
  483. ])
  484. chatScrollView.translatesAutoresizingMaskIntoConstraints = false
  485. chatScrollView.hasVerticalScroller = true
  486. chatScrollView.hasHorizontalScroller = false
  487. // Legacy reserves a dedicated track to the right of the clip view so the thumb never sits on top of cards/buttons.
  488. chatScrollView.scrollerStyle = .legacy
  489. chatScrollView.autohidesScrollers = true
  490. chatScrollView.drawsBackground = false
  491. chatScrollView.borderType = .noBorder
  492. chatScrollView.documentView = chatDocumentView
  493. chatScrollView.setContentHuggingPriority(.defaultLow, for: .vertical)
  494. chatScrollView.setContentCompressionResistancePriority(.defaultLow, for: .vertical)
  495. // Match Saved Jobs: pin document width to the clip view so cards and bubbles track window width instead of sticking to a narrow intrinsic width.
  496. NSLayoutConstraint.activate([
  497. chatDocumentView.topAnchor.constraint(equalTo: chatScrollView.contentView.topAnchor),
  498. chatDocumentView.leadingAnchor.constraint(equalTo: chatScrollView.contentView.leadingAnchor),
  499. chatDocumentView.widthAnchor.constraint(equalTo: chatScrollView.contentView.widthAnchor)
  500. ])
  501. }
  502. private var prefersReducedMotion: Bool {
  503. NSWorkspace.shared.accessibilityDisplayShouldReduceMotion
  504. }
  505. private func updateJobListingDescriptionWidths() {
  506. updateDescriptionColumnWidths(in: savedJobsStack, containerWidth: savedJobsDocumentView.bounds.width)
  507. walkChatJobStacks { stack in
  508. updateDescriptionColumnWidths(in: stack, containerWidth: stack.bounds.width)
  509. }
  510. }
  511. private func walkChatJobStacks(_ visitor: (ChatJobsStackView) -> Void) {
  512. func walk(_ view: NSView) {
  513. if let stack = view as? ChatJobsStackView {
  514. visitor(stack)
  515. }
  516. for sub in view.subviews { walk(sub) }
  517. }
  518. walk(chatStack)
  519. }
  520. /// Chat bubble text fields are wrapping labels whose `preferredMaxLayoutWidth` needs to track the available row width so long strings reflow correctly when the window resizes.
  521. private func updateChatBubbleWidths() {
  522. func walk(_ view: NSView) {
  523. if let label = view as? ChatBubbleLabel,
  524. let bubble = label.superview, bubble.bounds.width > 1 {
  525. let target = max(40, bubble.bounds.width - 28)
  526. if abs(label.preferredMaxLayoutWidth - target) > 0.5 {
  527. label.preferredMaxLayoutWidth = target
  528. label.invalidateIntrinsicContentSize()
  529. }
  530. }
  531. for sub in view.subviews { walk(sub) }
  532. }
  533. walk(chatStack)
  534. }
  535. private func updateDescriptionColumnWidths(in stack: NSStackView, containerWidth: CGFloat) {
  536. guard containerWidth > 1 else { return }
  537. // Matches `contentColumn` insets on the card (16 leading + 16 trailing). The description spans the full row below the buttons, so never subtract a “button strip” here — a too-narrow `preferredMaxLayoutWidth` inside a wider field makes wrapped `NSTextField` text lay out like a trailing-aligned band after resizes.
  538. let contentHorizontalInset: CGFloat = 32
  539. var didChange = false
  540. for card in stack.arrangedSubviews {
  541. guard let desc = card.viewWithTag(502) as? NSTextField else { continue }
  542. let cardWidth = card.bounds.width > 1 ? card.bounds.width : containerWidth
  543. let fallbackColumn = max(1, cardWidth - contentHorizontalInset)
  544. let columnWidth: CGFloat
  545. if desc.bounds.width > 1 {
  546. columnWidth = desc.bounds.width
  547. } else if let column = desc.superview, column.bounds.width > 1 {
  548. columnWidth = column.bounds.width
  549. } else {
  550. columnWidth = fallbackColumn
  551. }
  552. if abs(desc.preferredMaxLayoutWidth - columnWidth) > 0.5 {
  553. desc.preferredMaxLayoutWidth = columnWidth
  554. desc.invalidateIntrinsicContentSize()
  555. didChange = true
  556. }
  557. }
  558. if didChange {
  559. stack.needsLayout = true
  560. }
  561. }
  562. private static func normalizedSavedJobs(_ jobs: [JobListing]) -> [JobListing] {
  563. var seen = Set<JobListing>()
  564. var out: [JobListing] = []
  565. for job in jobs where seen.insert(job).inserted {
  566. out.append(job)
  567. }
  568. return out
  569. }
  570. private func isJobSaved(_ job: JobListing) -> Bool {
  571. savedJobOrder.contains(job)
  572. }
  573. private func persistSavedJobs() {
  574. SavedJobsStore.save(savedJobOrder)
  575. }
  576. private func applySavedState(_ saved: Bool, for job: JobListing) {
  577. if saved {
  578. savedJobOrder.removeAll { $0 == job }
  579. savedJobOrder.insert(job, at: 0)
  580. } else {
  581. savedJobOrder.removeAll { $0 == job }
  582. }
  583. persistSavedJobs()
  584. }
  585. private func jobListingHostSubtitle(_ job: JobListing) -> String {
  586. guard let raw = job.url, let url = URL(string: raw), let host = url.host?.lowercased() else {
  587. return "Indeed"
  588. }
  589. if host.hasPrefix("www.") {
  590. return String(host.dropFirst(4))
  591. }
  592. return host
  593. }
  594. private func jobListingCategorySymbol(for job: JobListing) -> String {
  595. let blob = (job.title + " " + job.description).lowercased()
  596. if blob.contains("machine learning") || blob.contains("deep learning") || blob.contains(" ml ") {
  597. return "brain.head.profile"
  598. }
  599. if blob.contains("audio") || blob.contains(" sound ") || blob.contains("dsp") {
  600. return "waveform"
  601. }
  602. if blob.contains("ios") || blob.contains("swift") || blob.contains("mobile") {
  603. return "iphone"
  604. }
  605. if blob.contains("design") || blob.contains(" ux") || blob.contains("figma") {
  606. return "paintpalette.fill"
  607. }
  608. if blob.contains("data ") || blob.contains("analytics") {
  609. return "chart.bar.fill"
  610. }
  611. if blob.contains("ai") || blob.contains("llm") || blob.contains("nlp") {
  612. return "cpu"
  613. }
  614. return "briefcase.fill"
  615. }
  616. private func makeJobListingCard(_ job: JobListing, context: JobListingCardContext) -> NSView {
  617. let card = NSView()
  618. card.translatesAutoresizingMaskIntoConstraints = false
  619. card.wantsLayer = true
  620. card.layer?.backgroundColor = Theme.cardBackground.cgColor
  621. card.layer?.cornerRadius = 14
  622. card.layer?.borderWidth = 1
  623. card.layer?.borderColor = Theme.border.cgColor
  624. card.layer?.masksToBounds = true
  625. let iconBox = NSView()
  626. iconBox.translatesAutoresizingMaskIntoConstraints = false
  627. iconBox.wantsLayer = true
  628. iconBox.layer?.backgroundColor = Theme.brandBlue.cgColor
  629. iconBox.layer?.cornerRadius = 12
  630. if #available(macOS 11.0, *) {
  631. iconBox.layer?.cornerCurve = .continuous
  632. }
  633. let categoryIcon = NSImageView()
  634. categoryIcon.translatesAutoresizingMaskIntoConstraints = false
  635. categoryIcon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 22, weight: .medium)
  636. categoryIcon.image = NSImage(systemSymbolName: jobListingCategorySymbol(for: job), accessibilityDescription: nil)
  637. categoryIcon.contentTintColor = .white
  638. iconBox.addSubview(categoryIcon)
  639. let titleField = NSTextField(labelWithString: job.title)
  640. titleField.font = .systemFont(ofSize: 16, weight: .semibold)
  641. titleField.textColor = Theme.brandBlue
  642. titleField.maximumNumberOfLines = 2
  643. titleField.lineBreakMode = .byWordWrapping
  644. titleField.alignment = .left
  645. titleField.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  646. titleField.translatesAutoresizingMaskIntoConstraints = false
  647. let buildingIcon = NSImageView()
  648. buildingIcon.translatesAutoresizingMaskIntoConstraints = false
  649. buildingIcon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 12, weight: .medium)
  650. buildingIcon.image = NSImage(systemSymbolName: "building.2.fill", accessibilityDescription: nil)
  651. buildingIcon.contentTintColor = Theme.welcomeSubtitleText
  652. let companyLabel = NSTextField(labelWithString: jobListingHostSubtitle(job))
  653. companyLabel.font = .systemFont(ofSize: 12, weight: .medium)
  654. companyLabel.textColor = Theme.welcomeSubtitleText
  655. companyLabel.maximumNumberOfLines = 1
  656. companyLabel.lineBreakMode = .byTruncatingTail
  657. companyLabel.translatesAutoresizingMaskIntoConstraints = false
  658. let companyRow = NSStackView(views: [buildingIcon, companyLabel])
  659. companyRow.orientation = .horizontal
  660. companyRow.spacing = 5
  661. companyRow.alignment = .centerY
  662. companyRow.translatesAutoresizingMaskIntoConstraints = false
  663. let descriptionField = NSTextField(wrappingLabelWithString: job.description)
  664. descriptionField.font = .systemFont(ofSize: 13, weight: .regular)
  665. descriptionField.textColor = Theme.secondaryText
  666. descriptionField.maximumNumberOfLines = 2
  667. descriptionField.lineBreakMode = .byWordWrapping
  668. descriptionField.alignment = .left
  669. descriptionField.baseWritingDirection = .leftToRight
  670. descriptionField.attributedStringValue = Self.jobListingDescriptionAttributedString(job.description)
  671. if let cell = descriptionField.cell as? NSTextFieldCell {
  672. cell.alignment = .left
  673. cell.wraps = true
  674. }
  675. descriptionField.setContentHuggingPriority(.defaultLow, for: .horizontal)
  676. descriptionField.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  677. descriptionField.tag = 502
  678. descriptionField.translatesAutoresizingMaskIntoConstraints = false
  679. let applyButton = JobPayloadButton(title: "Apply", target: self, action: #selector(didTapJobApply(_:)))
  680. applyButton.jobPayload = job
  681. applyButton.cardContext = context
  682. applyButton.isBordered = false
  683. applyButton.bezelStyle = .rounded
  684. applyButton.font = .systemFont(ofSize: 13, weight: .semibold)
  685. applyButton.wantsLayer = true
  686. applyButton.layer?.cornerRadius = 8
  687. applyButton.layer?.backgroundColor = Theme.brandBlue.cgColor
  688. applyButton.contentTintColor = Theme.proCTAText
  689. applyButton.focusRingType = .none
  690. applyButton.pointerCursor = true
  691. applyButton.hoverHandler = { [weak applyButton] hovering in
  692. applyButton?.layer?.backgroundColor = (hovering ? Theme.brandBlueHover : Theme.brandBlue).cgColor
  693. }
  694. applyButton.setContentHuggingPriority(.required, for: .horizontal)
  695. applyButton.setContentCompressionResistancePriority(.required, for: .horizontal)
  696. let savedOn = isJobSaved(job)
  697. let savedButton = SaveJobPayloadButton(title: savedOn ? "Saved" : "Save", target: self, action: #selector(didTapJobSaved(_:)))
  698. savedButton.jobPayload = job
  699. savedButton.cardContext = context
  700. savedButton.setButtonType(.toggle)
  701. savedButton.isBordered = false
  702. savedButton.bezelStyle = .rounded
  703. savedButton.font = .systemFont(ofSize: 13, weight: .semibold)
  704. savedButton.image = NSImage(systemSymbolName: savedOn ? "heart.fill" : "heart", accessibilityDescription: nil)
  705. savedButton.imagePosition = .imageLeading
  706. savedButton.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 11, weight: .semibold)
  707. savedButton.focusRingType = .none
  708. savedButton.state = savedOn ? .on : .off
  709. savedButton.pointerCursor = true
  710. savedButton.hoverHandler = { [weak self, weak savedButton] _ in
  711. guard let savedButton = savedButton else { return }
  712. self?.styleJobSavedButton(savedButton)
  713. }
  714. styleJobSavedButton(savedButton)
  715. savedButton.setContentHuggingPriority(.required, for: .horizontal)
  716. savedButton.setContentCompressionResistancePriority(.required, for: .horizontal)
  717. let dismissButton = JobPayloadButton()
  718. dismissButton.jobPayload = job
  719. dismissButton.cardContext = context
  720. dismissButton.image = NSImage(systemSymbolName: "xmark", accessibilityDescription: "Dismiss")
  721. dismissButton.imagePosition = .imageOnly
  722. dismissButton.imageScaling = .scaleProportionallyDown
  723. dismissButton.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 12, weight: .semibold)
  724. dismissButton.isBordered = false
  725. dismissButton.bezelStyle = .rounded
  726. dismissButton.contentTintColor = Theme.secondaryText
  727. dismissButton.target = self
  728. dismissButton.action = #selector(didTapJobDismiss(_:))
  729. dismissButton.toolTip = context == .savedJobsPage ? "Remove from saved" : "Dismiss"
  730. dismissButton.focusRingType = .none
  731. dismissButton.wantsLayer = true
  732. dismissButton.layer?.cornerRadius = 8
  733. dismissButton.layer?.backgroundColor = NSColor.clear.cgColor
  734. dismissButton.pointerCursor = true
  735. dismissButton.hoverHandler = { [weak dismissButton] hovering in
  736. dismissButton?.layer?.backgroundColor = (hovering ? Theme.neutralHoverFill : NSColor.clear).cgColor
  737. dismissButton?.contentTintColor = hovering ? Theme.primaryText : Theme.secondaryText
  738. }
  739. dismissButton.setContentHuggingPriority(.required, for: .horizontal)
  740. let buttonRow = NSStackView(views: [applyButton, savedButton, dismissButton])
  741. buttonRow.orientation = .horizontal
  742. buttonRow.spacing = 8
  743. buttonRow.alignment = .top
  744. buttonRow.translatesAutoresizingMaskIntoConstraints = false
  745. buttonRow.setContentHuggingPriority(.required, for: .horizontal)
  746. buttonRow.setContentCompressionResistancePriority(.required, for: .horizontal)
  747. buttonRow.setContentHuggingPriority(.required, for: .vertical)
  748. buttonRow.setContentCompressionResistancePriority(.required, for: .vertical)
  749. applyButton.setContentCompressionResistancePriority(.required, for: .horizontal)
  750. savedButton.setContentCompressionResistancePriority(.required, for: .horizontal)
  751. dismissButton.setContentCompressionResistancePriority(.required, for: .horizontal)
  752. let middleColumn = NSStackView(views: [titleField, companyRow, descriptionField])
  753. middleColumn.orientation = .vertical
  754. middleColumn.spacing = 5
  755. middleColumn.alignment = .leading
  756. middleColumn.translatesAutoresizingMaskIntoConstraints = false
  757. middleColumn.setContentHuggingPriority(.defaultLow, for: .horizontal)
  758. middleColumn.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  759. let contentRow = NSStackView(views: [iconBox, middleColumn])
  760. contentRow.orientation = .horizontal
  761. contentRow.spacing = 14
  762. contentRow.alignment = .top
  763. contentRow.distribution = .fill
  764. contentRow.translatesAutoresizingMaskIntoConstraints = false
  765. card.addSubview(contentRow)
  766. card.addSubview(buttonRow)
  767. let actionCornerInset: CGFloat = 8
  768. let contentToActionsGap: CGFloat = 12
  769. let bodyTrailingInset: CGFloat = 16
  770. NSLayoutConstraint.activate([
  771. contentRow.leadingAnchor.constraint(equalTo: card.leadingAnchor, constant: 16),
  772. contentRow.trailingAnchor.constraint(equalTo: card.trailingAnchor, constant: -bodyTrailingInset),
  773. contentRow.topAnchor.constraint(equalTo: card.topAnchor, constant: 14),
  774. contentRow.bottomAnchor.constraint(equalTo: card.bottomAnchor, constant: -14),
  775. buttonRow.topAnchor.constraint(equalTo: card.topAnchor, constant: actionCornerInset),
  776. buttonRow.trailingAnchor.constraint(equalTo: card.trailingAnchor, constant: -actionCornerInset),
  777. middleColumn.trailingAnchor.constraint(lessThanOrEqualTo: buttonRow.leadingAnchor, constant: -contentToActionsGap),
  778. iconBox.widthAnchor.constraint(equalToConstant: 58),
  779. iconBox.heightAnchor.constraint(equalToConstant: 58),
  780. categoryIcon.centerXAnchor.constraint(equalTo: iconBox.centerXAnchor),
  781. categoryIcon.centerYAnchor.constraint(equalTo: iconBox.centerYAnchor),
  782. buildingIcon.widthAnchor.constraint(equalToConstant: 14),
  783. buildingIcon.heightAnchor.constraint(equalToConstant: 14),
  784. applyButton.widthAnchor.constraint(greaterThanOrEqualToConstant: 76),
  785. applyButton.heightAnchor.constraint(equalToConstant: 32),
  786. savedButton.widthAnchor.constraint(greaterThanOrEqualToConstant: 84),
  787. savedButton.heightAnchor.constraint(equalToConstant: 32),
  788. dismissButton.widthAnchor.constraint(equalToConstant: 32),
  789. dismissButton.heightAnchor.constraint(equalToConstant: 32),
  790. descriptionField.widthAnchor.constraint(equalTo: middleColumn.widthAnchor)
  791. ])
  792. return card
  793. }
  794. private func styleJobSavedButton(_ button: NSButton) {
  795. button.wantsLayer = true
  796. button.layer?.cornerRadius = 10
  797. let hovering = (button as? HoverableButton)?.isHovering ?? false
  798. // Reference: white surface, soft blue outline, brand blue icon + label (no tinted fill on hover).
  799. button.layer?.backgroundColor = Theme.cardBackground.cgColor
  800. button.layer?.borderWidth = 1
  801. button.layer?.borderColor = (hovering ? Theme.searchBarBorderHover : Theme.searchBarBorder).cgColor
  802. button.contentTintColor = Theme.brandBlue
  803. }
  804. @objc private func didTapJobApply(_ sender: NSButton) {
  805. guard let job = (sender as? JobPayloadButton)?.jobPayload else { return }
  806. presentIndeedJobBrowser(url: Self.resolvedIndeedApplyURL(for: job))
  807. }
  808. /// Apply always loads a live Indeed page. Single-job URLs (`/viewjob`, `/pagead`, `/rc/clk`, …) are often expired or incorrectly synthesized by the model and show Indeed’s 404; we only trust **search** URLs whose path is `/jobs`. Otherwise we open a fresh `/jobs?q=<title>` (same regional host when the model provided one).
  809. private static func resolvedIndeedApplyURL(for job: JobListing) -> URL {
  810. let title = job.title.trimmingCharacters(in: .whitespacesAndNewlines)
  811. let raw = job.url?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
  812. var preferredHost: String?
  813. if !raw.isEmpty, let components = URLComponents(string: raw), let host = components.host {
  814. let lower = host.lowercased()
  815. if isIndeedApplyHost(lower) {
  816. preferredHost = host
  817. let pathLower = components.path.lowercased()
  818. let isJobsSearchPath = pathLower == "/jobs" || pathLower.hasSuffix("/jobs")
  819. if isJobsSearchPath, let url = components.url {
  820. return url
  821. }
  822. }
  823. }
  824. return indeedJobsSearchURL(title: title, preferredHost: preferredHost)
  825. }
  826. private static func indeedJobsSearchURL(title: String, preferredHost: String?) -> URL {
  827. let allowed = CharacterSet.urlQueryAllowed
  828. let q = title.addingPercentEncoding(withAllowedCharacters: allowed) ?? ""
  829. let host: String
  830. if let preferredHost, isIndeedApplyHost(preferredHost.lowercased()) {
  831. host = preferredHost
  832. } else {
  833. host = "www.indeed.com"
  834. }
  835. if let url = URL(string: "https://\(host)/jobs?q=\(q)") {
  836. return url
  837. }
  838. return URL(string: "https://www.indeed.com/jobs?q=\(q)")!
  839. }
  840. private static func isIndeedApplyHost(_ host: String) -> Bool {
  841. if host == "indeed.com" { return true }
  842. if host.hasPrefix("indeed.") { return true }
  843. return host.contains(".indeed.")
  844. }
  845. private static let indeedBrowseHomeURL = URL(string: "https://www.indeed.com/")!
  846. private static let externalIndeedAppURLSchemes = ["indeed://", "indeedjobs://"]
  847. private func selectIndeedSidebar() {
  848. isIndeedSidebarSelected = true
  849. if !isIndeedJobBrowserPresented {
  850. openIndeedFromSidebar()
  851. }
  852. configureSidebar()
  853. updateMainContentVisibility()
  854. }
  855. /// Opens the installed Indeed app when a handler is registered; otherwise loads Indeed in the embedded browser.
  856. private func openIndeedFromSidebar() {
  857. for scheme in Self.externalIndeedAppURLSchemes {
  858. guard let url = URL(string: scheme) else { continue }
  859. if NSWorkspace.shared.urlForApplication(toOpen: url) != nil {
  860. NSWorkspace.shared.open(url)
  861. return
  862. }
  863. }
  864. presentIndeedJobBrowser(url: Self.indeedBrowseHomeURL)
  865. }
  866. private func presentIndeedJobBrowser(url: URL) {
  867. guard let parentVC = hostingViewController else { return }
  868. if indeedJobBrowserViewController == nil {
  869. let vc = IndeedJobBrowserViewController()
  870. vc.onDismissEmbedded = { [weak self] in
  871. self?.dismissIndeedJobBrowserEmbedded()
  872. }
  873. vc.embed(in: indeedJobBrowserHost, parent: parentVC)
  874. indeedJobBrowserViewController = vc
  875. }
  876. indeedJobBrowserViewController?.loadPage(url)
  877. isIndeedJobBrowserPresented = true
  878. updateMainContentVisibility()
  879. }
  880. private func dismissIndeedJobBrowserEmbedded() {
  881. guard isIndeedJobBrowserPresented else { return }
  882. isIndeedJobBrowserPresented = false
  883. if isIndeedSidebarSelected {
  884. isIndeedSidebarSelected = false
  885. configureSidebar()
  886. }
  887. updateMainContentVisibility()
  888. }
  889. private var hostingViewController: NSViewController? {
  890. var responder: NSResponder? = self
  891. while let current = responder {
  892. if let viewController = current as? NSViewController {
  893. return viewController
  894. }
  895. responder = current.nextResponder
  896. }
  897. return nil
  898. }
  899. @objc private func didTapJobSaved(_ sender: NSButton) {
  900. guard let job = (sender as? JobPayloadButton)?.jobPayload else { return }
  901. let willSave = !isJobSaved(job)
  902. applySavedState(willSave, for: job)
  903. sender.state = willSave ? .on : .off
  904. sender.title = willSave ? "Saved" : "Save"
  905. sender.image = NSImage(systemSymbolName: willSave ? "heart.fill" : "heart", accessibilityDescription: nil)
  906. styleJobSavedButton(sender)
  907. if isSavedJobsSidebarIndex(selectedSidebarIndex) {
  908. reloadSavedJobsListings()
  909. }
  910. }
  911. @objc private func didTapJobDismiss(_ sender: NSButton) {
  912. guard let button = sender as? JobPayloadButton, let job = button.jobPayload else { return }
  913. switch button.cardContext {
  914. case .homeSearchResults:
  915. removeJobCardFromChat(originating: button, job: job)
  916. case .savedJobsPage:
  917. applySavedState(false, for: job)
  918. reloadSavedJobsListings()
  919. }
  920. }
  921. /// Walks up from a dismiss button until it finds the enclosing chat job stack, then removes only the card that owns the button. Other chat history (older searches, the assistant summary text) is untouched.
  922. private func removeJobCardFromChat(originating button: NSView, job: JobListing) {
  923. var node: NSView? = button
  924. var card: NSView?
  925. var stack: ChatJobsStackView?
  926. while let v = node {
  927. if let parent = v.superview as? ChatJobsStackView {
  928. card = v
  929. stack = parent
  930. break
  931. }
  932. node = v.superview
  933. }
  934. guard let card, let stack else { return }
  935. stack.removeArrangedSubview(card)
  936. card.removeFromSuperview()
  937. lastSearchResults.removeAll { $0 == job }
  938. }
  939. private func configureSearchBar() {
  940. let pillCorner: CGFloat = 27
  941. let barHeight: CGFloat = 54
  942. searchBarShadowHost.translatesAutoresizingMaskIntoConstraints = false
  943. searchBarShadowHost.wantsLayer = true
  944. searchBarShadowHost.layer?.masksToBounds = false
  945. searchBarShadowHost.layer?.shadowColor = NSColor.black.withAlphaComponent(0.18).cgColor
  946. searchBarShadowHost.layer?.shadowOffset = CGSize(width: 0, height: 2)
  947. searchBarShadowHost.layer?.shadowRadius = 10
  948. searchBarShadowHost.layer?.shadowOpacity = 1
  949. searchBarShadowHost.setContentHuggingPriority(.defaultHigh, for: .vertical)
  950. searchCard.translatesAutoresizingMaskIntoConstraints = false
  951. searchCard.wantsLayer = true
  952. searchCard.layer?.backgroundColor = Theme.cardBackground.cgColor
  953. searchCard.layer?.cornerRadius = pillCorner
  954. searchCard.layer?.borderWidth = 1
  955. searchCard.layer?.borderColor = Theme.searchBarBorder.cgColor
  956. searchCard.layer?.masksToBounds = true
  957. searchCard.hoverHandler = { [weak self] hovering in
  958. guard let self else { return }
  959. CATransaction.begin()
  960. CATransaction.setAnimationDuration(0.15)
  961. self.searchCard.layer?.backgroundColor = (hovering ? Theme.neutralHoverFill : Theme.cardBackground).cgColor
  962. self.searchCard.layer?.borderColor = (hovering ? Theme.searchBarBorderHover : Theme.searchBarBorder).cgColor
  963. self.searchBarShadowHost.layer?.shadowColor = NSColor.black.withAlphaComponent(hovering ? 0.24 : 0.18).cgColor
  964. self.searchBarShadowHost.layer?.shadowRadius = hovering ? 12 : 10
  965. CATransaction.commit()
  966. }
  967. searchBarShadowHost.addSubview(searchCard)
  968. func configureField(_ field: NSTextField, placeholder: String) {
  969. field.translatesAutoresizingMaskIntoConstraints = false
  970. field.isBordered = false
  971. field.drawsBackground = false
  972. field.focusRingType = .none
  973. field.font = .systemFont(ofSize: 14, weight: .regular)
  974. field.textColor = Theme.primaryText
  975. field.delegate = self
  976. field.placeholderAttributedString = NSAttributedString(
  977. string: placeholder,
  978. attributes: [
  979. .foregroundColor: Theme.secondaryText,
  980. .font: NSFont.systemFont(ofSize: 14, weight: .regular)
  981. ]
  982. )
  983. field.cell?.usesSingleLineMode = true
  984. field.cell?.wraps = false
  985. field.cell?.isScrollable = true
  986. field.target = self
  987. field.action = #selector(didSubmitSearch)
  988. }
  989. jobSearchIcon.translatesAutoresizingMaskIntoConstraints = false
  990. jobSearchIcon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 16, weight: .semibold)
  991. jobSearchIcon.image = NSImage(systemSymbolName: "sparkles", accessibilityDescription: "Ask AI")
  992. jobSearchIcon.contentTintColor = Theme.brandBlue
  993. configureField(jobKeywordsField, placeholder: "Ask for roles, skills, salary, or job descriptions...")
  994. let ctaHeight: CGFloat = 42
  995. let ctaCorner = ctaHeight / 2
  996. findJobsCTAHost.translatesAutoresizingMaskIntoConstraints = false
  997. findJobsCTAHost.wantsLayer = true
  998. findJobsCTAHost.layer?.masksToBounds = false
  999. findJobsCTAHost.layer?.shadowColor = NSColor.black.cgColor
  1000. findJobsCTAHost.layer?.shadowOpacity = 0.16
  1001. findJobsCTAHost.layer?.shadowOffset = CGSize(width: 0, height: 2)
  1002. findJobsCTAHost.layer?.shadowRadius = 6
  1003. let sendContentPadding: CGFloat = 16
  1004. findJobsCTAPill.translatesAutoresizingMaskIntoConstraints = false
  1005. findJobsCTAPill.wantsLayer = true
  1006. findJobsCTAPill.layer?.cornerRadius = ctaCorner
  1007. if #available(macOS 11.0, *) {
  1008. findJobsCTAPill.layer?.cornerCurve = .continuous
  1009. }
  1010. findJobsCTAPill.layer?.masksToBounds = true
  1011. findJobsCTAPill.layer?.backgroundColor = Theme.brandBlue.cgColor
  1012. sendIconView.translatesAutoresizingMaskIntoConstraints = false
  1013. sendIconView.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 11, weight: .semibold)
  1014. sendIconView.image = NSImage(systemSymbolName: "paperplane.fill", accessibilityDescription: nil)
  1015. sendIconView.contentTintColor = Theme.proCTAText
  1016. sendLabel.font = .systemFont(ofSize: 14, weight: .semibold)
  1017. sendLabel.textColor = Theme.proCTAText
  1018. sendLabel.alignment = .center
  1019. sendContentStack.orientation = .horizontal
  1020. sendContentStack.spacing = 6
  1021. sendContentStack.alignment = .centerY
  1022. sendContentStack.translatesAutoresizingMaskIntoConstraints = false
  1023. sendContentStack.addArrangedSubview(sendIconView)
  1024. sendContentStack.addArrangedSubview(sendLabel)
  1025. findJobsButton.translatesAutoresizingMaskIntoConstraints = false
  1026. findJobsButton.title = ""
  1027. findJobsButton.isBordered = false
  1028. findJobsButton.bezelStyle = .inline
  1029. findJobsButton.wantsLayer = true
  1030. findJobsButton.layer?.backgroundColor = NSColor.clear.cgColor
  1031. findJobsButton.focusRingType = .none
  1032. findJobsButton.pointerCursor = true
  1033. findJobsButton.target = self
  1034. findJobsButton.action = #selector(didSubmitSearch)
  1035. findJobsButton.setAccessibilityLabel("Send")
  1036. findJobsButton.hoverHandler = { [weak self] hovering in
  1037. self?.findJobsCTAPill.layer?.backgroundColor = (hovering ? Theme.brandBlueHover : Theme.brandBlue).cgColor
  1038. }
  1039. findJobsCTAHost.addSubview(findJobsCTAPill)
  1040. findJobsCTAHost.addSubview(sendContentStack)
  1041. findJobsCTAHost.addSubview(findJobsButton)
  1042. findJobsCTAHost.setContentHuggingPriority(.required, for: .horizontal)
  1043. findJobsCTAHost.setContentCompressionResistancePriority(.required, for: .horizontal)
  1044. NSLayoutConstraint.activate([
  1045. findJobsCTAPill.leadingAnchor.constraint(equalTo: findJobsCTAHost.leadingAnchor),
  1046. findJobsCTAPill.trailingAnchor.constraint(equalTo: findJobsCTAHost.trailingAnchor),
  1047. findJobsCTAPill.topAnchor.constraint(equalTo: findJobsCTAHost.topAnchor),
  1048. findJobsCTAPill.bottomAnchor.constraint(equalTo: findJobsCTAHost.bottomAnchor),
  1049. sendContentStack.centerXAnchor.constraint(equalTo: findJobsCTAPill.centerXAnchor),
  1050. sendContentStack.centerYAnchor.constraint(equalTo: findJobsCTAPill.centerYAnchor),
  1051. sendContentStack.leadingAnchor.constraint(greaterThanOrEqualTo: findJobsCTAPill.leadingAnchor, constant: sendContentPadding),
  1052. sendContentStack.trailingAnchor.constraint(lessThanOrEqualTo: findJobsCTAPill.trailingAnchor, constant: -sendContentPadding),
  1053. sendIconView.widthAnchor.constraint(equalToConstant: 14),
  1054. sendIconView.heightAnchor.constraint(equalToConstant: 14),
  1055. findJobsButton.leadingAnchor.constraint(equalTo: findJobsCTAHost.leadingAnchor),
  1056. findJobsButton.trailingAnchor.constraint(equalTo: findJobsCTAHost.trailingAnchor),
  1057. findJobsButton.topAnchor.constraint(equalTo: findJobsCTAHost.topAnchor),
  1058. findJobsButton.bottomAnchor.constraint(equalTo: findJobsCTAHost.bottomAnchor)
  1059. ])
  1060. let keywordsStack = NSStackView(views: [jobSearchIcon, jobKeywordsField])
  1061. keywordsStack.orientation = .horizontal
  1062. keywordsStack.spacing = 10
  1063. keywordsStack.alignment = .centerY
  1064. keywordsStack.translatesAutoresizingMaskIntoConstraints = false
  1065. keywordsStack.edgeInsets = NSEdgeInsets(top: 0, left: 18, bottom: 0, right: 10)
  1066. keywordsStack.setContentHuggingPriority(.defaultLow, for: .horizontal)
  1067. let row = NSStackView(views: [keywordsStack, findJobsCTAHost])
  1068. row.orientation = .horizontal
  1069. row.spacing = 10
  1070. row.alignment = .centerY
  1071. row.distribution = .fill
  1072. row.translatesAutoresizingMaskIntoConstraints = false
  1073. row.edgeInsets = NSEdgeInsets(top: 0, left: 0, bottom: 0, right: 7)
  1074. searchCard.addSubview(row)
  1075. NSLayoutConstraint.activate([
  1076. searchCard.leadingAnchor.constraint(equalTo: searchBarShadowHost.leadingAnchor),
  1077. searchCard.trailingAnchor.constraint(equalTo: searchBarShadowHost.trailingAnchor),
  1078. searchCard.topAnchor.constraint(equalTo: searchBarShadowHost.topAnchor),
  1079. searchCard.bottomAnchor.constraint(equalTo: searchBarShadowHost.bottomAnchor),
  1080. searchBarShadowHost.heightAnchor.constraint(equalToConstant: barHeight),
  1081. row.leadingAnchor.constraint(equalTo: searchCard.leadingAnchor),
  1082. row.trailingAnchor.constraint(equalTo: searchCard.trailingAnchor),
  1083. row.topAnchor.constraint(equalTo: searchCard.topAnchor),
  1084. row.bottomAnchor.constraint(equalTo: searchCard.bottomAnchor),
  1085. jobSearchIcon.widthAnchor.constraint(equalToConstant: 18),
  1086. jobSearchIcon.heightAnchor.constraint(equalToConstant: 18),
  1087. findJobsCTAHost.heightAnchor.constraint(equalToConstant: ctaHeight),
  1088. findJobsCTAHost.widthAnchor.constraint(greaterThanOrEqualTo: sendContentStack.widthAnchor, constant: sendContentPadding * 2)
  1089. ])
  1090. searchCard.hoverHandler = nil
  1091. }
  1092. private func updateFindJobsCTAShadowPath() {
  1093. guard findJobsCTAHost.bounds.width > 0, findJobsCTAHost.bounds.height > 0 else { return }
  1094. let r = findJobsCTAHost.bounds
  1095. let radius = min(r.height / 2, r.width / 2)
  1096. findJobsCTAHost.layer?.shadowPath = CGPath(
  1097. roundedRect: r,
  1098. cornerWidth: radius,
  1099. cornerHeight: radius,
  1100. transform: nil
  1101. )
  1102. }
  1103. private func configureNonHomePlaceholder() {
  1104. nonHomeHost.translatesAutoresizingMaskIntoConstraints = false
  1105. nonHomeHost.wantsLayer = true
  1106. nonHomeHost.layer?.backgroundColor = Theme.mainHostBackground.cgColor
  1107. nonHomeHost.isHidden = true
  1108. nonHomeHost.userInterfaceLayoutDirection = .leftToRight
  1109. nonHomeGenericContainer.translatesAutoresizingMaskIntoConstraints = false
  1110. savedJobsPageContainer.translatesAutoresizingMaskIntoConstraints = false
  1111. settingsPageContainer.translatesAutoresizingMaskIntoConstraints = false
  1112. cvMakerPageContainer.translatesAutoresizingMaskIntoConstraints = false
  1113. profilePageContainer.translatesAutoresizingMaskIntoConstraints = false
  1114. nonHomeHost.addSubview(nonHomeGenericContainer)
  1115. nonHomeHost.addSubview(savedJobsPageContainer)
  1116. nonHomeHost.addSubview(settingsPageContainer)
  1117. nonHomeHost.addSubview(cvMakerPageContainer)
  1118. nonHomeHost.addSubview(profilePageContainer)
  1119. NSLayoutConstraint.activate([
  1120. nonHomeGenericContainer.leadingAnchor.constraint(equalTo: nonHomeHost.leadingAnchor),
  1121. nonHomeGenericContainer.trailingAnchor.constraint(equalTo: nonHomeHost.trailingAnchor),
  1122. nonHomeGenericContainer.topAnchor.constraint(equalTo: nonHomeHost.topAnchor),
  1123. nonHomeGenericContainer.bottomAnchor.constraint(equalTo: nonHomeHost.bottomAnchor),
  1124. savedJobsPageContainer.leadingAnchor.constraint(equalTo: nonHomeHost.leadingAnchor),
  1125. savedJobsPageContainer.trailingAnchor.constraint(equalTo: nonHomeHost.trailingAnchor),
  1126. savedJobsPageContainer.topAnchor.constraint(equalTo: nonHomeHost.topAnchor),
  1127. savedJobsPageContainer.bottomAnchor.constraint(equalTo: nonHomeHost.bottomAnchor),
  1128. settingsPageContainer.leadingAnchor.constraint(equalTo: nonHomeHost.leadingAnchor),
  1129. settingsPageContainer.trailingAnchor.constraint(equalTo: nonHomeHost.trailingAnchor),
  1130. settingsPageContainer.topAnchor.constraint(equalTo: nonHomeHost.topAnchor),
  1131. settingsPageContainer.bottomAnchor.constraint(equalTo: nonHomeHost.bottomAnchor),
  1132. cvMakerPageContainer.leadingAnchor.constraint(equalTo: nonHomeHost.leadingAnchor),
  1133. cvMakerPageContainer.trailingAnchor.constraint(equalTo: nonHomeHost.trailingAnchor),
  1134. cvMakerPageContainer.topAnchor.constraint(equalTo: nonHomeHost.topAnchor),
  1135. cvMakerPageContainer.bottomAnchor.constraint(equalTo: nonHomeHost.bottomAnchor),
  1136. profilePageContainer.leftAnchor.constraint(equalTo: nonHomeHost.leftAnchor),
  1137. profilePageContainer.rightAnchor.constraint(equalTo: nonHomeHost.rightAnchor),
  1138. profilePageContainer.topAnchor.constraint(equalTo: nonHomeHost.topAnchor),
  1139. profilePageContainer.bottomAnchor.constraint(equalTo: nonHomeHost.bottomAnchor)
  1140. ])
  1141. nonHomeTitleLabel.font = .systemFont(ofSize: 22, weight: .bold)
  1142. nonHomeTitleLabel.textColor = Theme.primaryText
  1143. nonHomeTitleLabel.alignment = .center
  1144. nonHomeTitleLabel.maximumNumberOfLines = 1
  1145. nonHomeSubtitleLabel.font = .systemFont(ofSize: 14, weight: .regular)
  1146. nonHomeSubtitleLabel.textColor = Theme.secondaryText
  1147. nonHomeSubtitleLabel.alignment = .center
  1148. nonHomeSubtitleLabel.maximumNumberOfLines = 0
  1149. nonHomeSubtitleLabel.stringValue = "This area is not available in the preview build. Use Home to search jobs."
  1150. let genericStack = NSStackView(views: [nonHomeTitleLabel, nonHomeSubtitleLabel])
  1151. genericStack.orientation = .vertical
  1152. genericStack.spacing = 10
  1153. genericStack.alignment = .centerX
  1154. genericStack.translatesAutoresizingMaskIntoConstraints = false
  1155. nonHomeGenericContainer.addSubview(genericStack)
  1156. NSLayoutConstraint.activate([
  1157. genericStack.centerXAnchor.constraint(equalTo: nonHomeGenericContainer.centerXAnchor),
  1158. genericStack.centerYAnchor.constraint(equalTo: nonHomeGenericContainer.centerYAnchor),
  1159. genericStack.leadingAnchor.constraint(greaterThanOrEqualTo: nonHomeGenericContainer.leadingAnchor, constant: 32),
  1160. genericStack.trailingAnchor.constraint(lessThanOrEqualTo: nonHomeGenericContainer.trailingAnchor, constant: -32),
  1161. nonHomeSubtitleLabel.widthAnchor.constraint(lessThanOrEqualToConstant: 420)
  1162. ])
  1163. savedJobsPageTitleLabel.font = .systemFont(ofSize: 22, weight: .bold)
  1164. savedJobsPageTitleLabel.textColor = Theme.primaryText
  1165. savedJobsPageTitleLabel.alignment = .left
  1166. savedJobsPageTitleLabel.maximumNumberOfLines = 1
  1167. savedJobsPageSubtitleLabel.font = .systemFont(ofSize: 14, weight: .regular)
  1168. savedJobsPageSubtitleLabel.textColor = Theme.secondaryText
  1169. savedJobsPageSubtitleLabel.alignment = .left
  1170. savedJobsPageSubtitleLabel.maximumNumberOfLines = 0
  1171. savedJobsDocumentView.translatesAutoresizingMaskIntoConstraints = false
  1172. savedJobsStack.orientation = .vertical
  1173. savedJobsStack.spacing = 14
  1174. savedJobsStack.alignment = .leading
  1175. savedJobsStack.distribution = .fill
  1176. savedJobsStack.translatesAutoresizingMaskIntoConstraints = false
  1177. savedJobsStack.setContentHuggingPriority(.defaultHigh, for: .vertical)
  1178. savedJobsStack.setHuggingPriority(.defaultLow, for: .horizontal)
  1179. savedJobsDocumentView.addSubview(savedJobsStack)
  1180. NSLayoutConstraint.activate([
  1181. savedJobsStack.leadingAnchor.constraint(equalTo: savedJobsDocumentView.leadingAnchor),
  1182. savedJobsStack.trailingAnchor.constraint(equalTo: savedJobsDocumentView.trailingAnchor),
  1183. savedJobsStack.topAnchor.constraint(equalTo: savedJobsDocumentView.topAnchor),
  1184. savedJobsStack.bottomAnchor.constraint(equalTo: savedJobsDocumentView.bottomAnchor)
  1185. ])
  1186. savedJobsScrollView.translatesAutoresizingMaskIntoConstraints = false
  1187. savedJobsScrollView.hasVerticalScroller = true
  1188. savedJobsScrollView.hasHorizontalScroller = false
  1189. savedJobsScrollView.scrollerStyle = .legacy
  1190. savedJobsScrollView.autohidesScrollers = true
  1191. savedJobsScrollView.drawsBackground = false
  1192. savedJobsScrollView.borderType = .noBorder
  1193. savedJobsScrollView.documentView = savedJobsDocumentView
  1194. savedJobsScrollView.setContentHuggingPriority(.defaultLow, for: .vertical)
  1195. savedJobsScrollView.setContentCompressionResistancePriority(.defaultLow, for: .vertical)
  1196. let savedHeaderStack = NSStackView(views: [savedJobsPageTitleLabel, savedJobsPageSubtitleLabel])
  1197. savedHeaderStack.orientation = .vertical
  1198. savedHeaderStack.spacing = 6
  1199. savedHeaderStack.alignment = .leading
  1200. savedHeaderStack.translatesAutoresizingMaskIntoConstraints = false
  1201. let savedOuterStack = NSStackView(views: [savedHeaderStack, savedJobsScrollView])
  1202. savedOuterStack.orientation = .vertical
  1203. savedOuterStack.spacing = 16
  1204. // Leading alignment plus explicit column width keeps the title and subtitle on the same edge as the cards.
  1205. savedOuterStack.alignment = .leading
  1206. savedOuterStack.translatesAutoresizingMaskIntoConstraints = false
  1207. savedJobsPageContainer.userInterfaceLayoutDirection = .leftToRight
  1208. savedJobsPageContainer.addSubview(savedOuterStack)
  1209. NSLayoutConstraint.activate([
  1210. savedOuterStack.leadingAnchor.constraint(equalTo: savedJobsPageContainer.leadingAnchor, constant: 32),
  1211. savedOuterStack.trailingAnchor.constraint(equalTo: savedJobsPageContainer.trailingAnchor, constant: -32),
  1212. savedOuterStack.topAnchor.constraint(equalTo: savedJobsPageContainer.topAnchor, constant: 8),
  1213. savedOuterStack.bottomAnchor.constraint(equalTo: savedJobsPageContainer.bottomAnchor),
  1214. savedHeaderStack.widthAnchor.constraint(equalTo: savedOuterStack.widthAnchor),
  1215. savedJobsScrollView.widthAnchor.constraint(equalTo: savedOuterStack.widthAnchor),
  1216. savedJobsDocumentView.topAnchor.constraint(equalTo: savedJobsScrollView.contentView.topAnchor),
  1217. savedJobsDocumentView.leadingAnchor.constraint(equalTo: savedJobsScrollView.contentView.leadingAnchor),
  1218. savedJobsDocumentView.widthAnchor.constraint(equalTo: savedJobsScrollView.contentView.widthAnchor)
  1219. ])
  1220. configureSettingsPage()
  1221. configureCVMakerPage()
  1222. configureProfilePage()
  1223. }
  1224. private func configureCVMakerPage() {
  1225. cvMakerPageContainer.wantsLayer = true
  1226. cvMakerPageContainer.layer?.backgroundColor = Theme.mainHostBackground.cgColor
  1227. cvMakerPageContainer.isHidden = true
  1228. cvMakerPageView.translatesAutoresizingMaskIntoConstraints = false
  1229. cvMakerPageContainer.addSubview(cvMakerPageView)
  1230. NSLayoutConstraint.activate([
  1231. cvMakerPageView.leadingAnchor.constraint(equalTo: cvMakerPageContainer.leadingAnchor),
  1232. cvMakerPageView.trailingAnchor.constraint(equalTo: cvMakerPageContainer.trailingAnchor),
  1233. cvMakerPageView.topAnchor.constraint(equalTo: cvMakerPageContainer.topAnchor),
  1234. cvMakerPageView.bottomAnchor.constraint(equalTo: cvMakerPageContainer.bottomAnchor)
  1235. ])
  1236. cvMakerPageView.onContinueToProfileSelection = { [weak self] template in
  1237. guard let self, self.ensureProAccess() else { return }
  1238. self.pendingCVTemplate = template
  1239. self.profilesListPageView.setPendingCVTemplateDisplayName(template.name)
  1240. self.selectProfileSidebarForCVMakerFlow()
  1241. }
  1242. }
  1243. /// Switches the main panel to **Profile** so the user can pick a saved CV profile after choosing a template in CV Maker.
  1244. private func selectProfileSidebarForCVMakerFlow() {
  1245. guard let index = currentSidebarItems.firstIndex(where: { $0.title == "Profile" }) else { return }
  1246. selectSidebarItem(at: index)
  1247. }
  1248. private func configureProfilePage() {
  1249. profilePageContainer.wantsLayer = true
  1250. profilePageContainer.layer?.backgroundColor = Theme.mainHostBackground.cgColor
  1251. profilePageContainer.isHidden = true
  1252. profilePageContainer.userInterfaceLayoutDirection = .leftToRight
  1253. profilesListPageView.translatesAutoresizingMaskIntoConstraints = false
  1254. myProfilePageView.translatesAutoresizingMaskIntoConstraints = false
  1255. cvFilledPreviewPageView.translatesAutoresizingMaskIntoConstraints = false
  1256. profilePageContainer.addSubview(profilesListPageView)
  1257. profilePageContainer.addSubview(myProfilePageView)
  1258. profilePageContainer.addSubview(cvFilledPreviewPageView)
  1259. NSLayoutConstraint.activate([
  1260. profilesListPageView.leftAnchor.constraint(equalTo: profilePageContainer.leftAnchor),
  1261. profilesListPageView.rightAnchor.constraint(equalTo: profilePageContainer.rightAnchor),
  1262. profilesListPageView.topAnchor.constraint(equalTo: profilePageContainer.topAnchor),
  1263. profilesListPageView.bottomAnchor.constraint(equalTo: profilePageContainer.bottomAnchor),
  1264. myProfilePageView.leftAnchor.constraint(equalTo: profilePageContainer.leftAnchor),
  1265. myProfilePageView.rightAnchor.constraint(equalTo: profilePageContainer.rightAnchor),
  1266. myProfilePageView.topAnchor.constraint(equalTo: profilePageContainer.topAnchor),
  1267. myProfilePageView.bottomAnchor.constraint(equalTo: profilePageContainer.bottomAnchor),
  1268. cvFilledPreviewPageView.leftAnchor.constraint(equalTo: profilePageContainer.leftAnchor),
  1269. cvFilledPreviewPageView.rightAnchor.constraint(equalTo: profilePageContainer.rightAnchor),
  1270. cvFilledPreviewPageView.topAnchor.constraint(equalTo: profilePageContainer.topAnchor),
  1271. cvFilledPreviewPageView.bottomAnchor.constraint(equalTo: profilePageContainer.bottomAnchor)
  1272. ])
  1273. profilesListPageView.onAddProfile = { [weak self] in
  1274. self?.presentProfileEditor(existingID: nil)
  1275. }
  1276. profilesListPageView.onEditProfile = { [weak self] id in
  1277. self?.presentProfileEditor(existingID: id)
  1278. }
  1279. profilesListPageView.onDeleteProfile = { [weak self] id in
  1280. self?.confirmDeleteProfile(id: id)
  1281. }
  1282. profilesListPageView.onBuildCVWithProfile = { [weak self] profileID in
  1283. guard let self,
  1284. let template = self.pendingCVTemplate,
  1285. let profile = SavedProfilesStore.profile(id: profileID) else { return }
  1286. self.presentCVDocumentPreview(profile: profile, template: template)
  1287. }
  1288. cvFilledPreviewPageView.onDismiss = { [weak self] in
  1289. self?.dismissCVDocumentPreview()
  1290. }
  1291. myProfilePageView.onDismiss = { [weak self] in
  1292. self?.dismissProfileEditor()
  1293. }
  1294. isProfileEditorPresented = false
  1295. isCVDocumentPreviewPresented = false
  1296. profilesListPageView.isHidden = false
  1297. myProfilePageView.isHidden = true
  1298. cvFilledPreviewPageView.isHidden = true
  1299. profilesListPageView.reloadFromStore()
  1300. }
  1301. private func presentCVDocumentPreview(profile: SavedProfile, template: CVTemplate) {
  1302. guard ensureProAccess() else { return }
  1303. isCVDocumentPreviewPresented = true
  1304. cvFilledPreviewPageView.configure(profile: profile, template: template)
  1305. cvFilledPreviewPageView.isHidden = false
  1306. profilesListPageView.isHidden = true
  1307. myProfilePageView.isHidden = true
  1308. }
  1309. private func dismissCVDocumentPreview() {
  1310. isCVDocumentPreviewPresented = false
  1311. cvFilledPreviewPageView.isHidden = true
  1312. profilesListPageView.reloadFromStore()
  1313. profilesListPageView.isHidden = false
  1314. myProfilePageView.isHidden = true
  1315. }
  1316. private func presentProfileEditor(existingID: UUID?) {
  1317. guard ensureProAccess() else { return }
  1318. if isCVDocumentPreviewPresented {
  1319. dismissCVDocumentPreview()
  1320. }
  1321. isProfileEditorPresented = true
  1322. if let id = existingID, let profile = SavedProfilesStore.profile(id: id) {
  1323. myProfilePageView.loadSavedProfile(profile)
  1324. } else {
  1325. myProfilePageView.prepareNewProfile()
  1326. }
  1327. profilesListPageView.isHidden = true
  1328. myProfilePageView.isHidden = false
  1329. }
  1330. private func dismissProfileEditor() {
  1331. isProfileEditorPresented = false
  1332. profilesListPageView.reloadFromStore()
  1333. profilesListPageView.isHidden = false
  1334. myProfilePageView.isHidden = true
  1335. }
  1336. private func confirmDeleteProfile(id: UUID) {
  1337. let displayName = SavedProfilesStore.profile(id: id)?.profileDisplayName ?? ""
  1338. let alert = NSAlert()
  1339. alert.messageText = "Delete this profile?"
  1340. alert.informativeText = displayName.isEmpty
  1341. ? "This profile will be removed from this Mac."
  1342. : "“\(displayName)” will be removed from this Mac."
  1343. alert.alertStyle = .warning
  1344. alert.addButton(withTitle: "Cancel")
  1345. alert.addButton(withTitle: "Delete")
  1346. guard let window = window else {
  1347. let response = alert.runModal()
  1348. if response == .alertSecondButtonReturn {
  1349. SavedProfilesStore.delete(id: id)
  1350. profilesListPageView.reloadFromStore()
  1351. }
  1352. return
  1353. }
  1354. alert.beginSheetModal(for: window) { [weak self] response in
  1355. guard let self else { return }
  1356. if response == .alertSecondButtonReturn {
  1357. SavedProfilesStore.delete(id: id)
  1358. if self.isProfileEditorPresented {
  1359. self.dismissProfileEditor()
  1360. } else {
  1361. self.profilesListPageView.reloadFromStore()
  1362. }
  1363. }
  1364. }
  1365. }
  1366. private func configureSettingsPage() {
  1367. settingsPageContainer.wantsLayer = true
  1368. settingsPageContainer.layer?.backgroundColor = Theme.settingsPageBackground.cgColor
  1369. settingsPageContainer.isHidden = true
  1370. let contentStack = NSStackView()
  1371. contentStack.orientation = .vertical
  1372. contentStack.spacing = 26
  1373. contentStack.alignment = .leading
  1374. contentStack.translatesAutoresizingMaskIntoConstraints = false
  1375. let appearanceTitle = NSTextField(labelWithString: "Appearance")
  1376. appearanceTitle.font = .systemFont(ofSize: 12, weight: .semibold)
  1377. appearanceTitle.textColor = Theme.secondaryText
  1378. appearanceTitle.alignment = .left
  1379. let themeSegment = makeAppearanceModeSegment()
  1380. appearanceModeSegment = themeSegment
  1381. let appearanceSection = makeSettingsSection(rows: [
  1382. makeSettingsRow(title: "Theme", systemImage: "circle.lefthalf.filled", accessory: themeSegment, tapAction: nil)
  1383. ])
  1384. let appearanceStack = NSStackView(views: [appearanceTitle, appearanceSection])
  1385. appearanceStack.orientation = .vertical
  1386. appearanceStack.spacing = 14
  1387. appearanceStack.alignment = .leading
  1388. appearanceStack.translatesAutoresizingMaskIntoConstraints = false
  1389. let settingsSection = makeSettingsSection(rows: [
  1390. makeSettingsRow(title: "Share App", systemImage: "square.and.arrow.up", accessory: nil, tapAction: #selector(didTapShareApp)),
  1391. makeSettingsRow(title: "More Apps", systemImage: "square.grid.2x2", accessory: nil, tapAction: #selector(didTapMoreApps))
  1392. ])
  1393. let aboutTitle = NSTextField(labelWithString: "About")
  1394. aboutTitle.font = .systemFont(ofSize: 12, weight: .semibold)
  1395. aboutTitle.textColor = Theme.secondaryText
  1396. aboutTitle.alignment = .left
  1397. let aboutSection = makeSettingsSection(rows: [
  1398. makeSettingsRow(title: "Website", systemImage: "globe", accessory: nil, tapAction: #selector(didTapWebsite)),
  1399. makeSettingsRow(title: "Support", systemImage: "questionmark.circle", accessory: nil, tapAction: #selector(didTapSupport)),
  1400. makeSettingsRow(title: "Terms of Use", systemImage: "doc.text", accessory: nil, tapAction: #selector(didTapTermsOfUse)),
  1401. makeSettingsRow(title: "Privacy Policy", systemImage: "shield", accessory: nil, tapAction: #selector(didTapPrivacyPolicy))
  1402. ])
  1403. let aboutStack = NSStackView(views: [aboutTitle, aboutSection])
  1404. aboutStack.orientation = .vertical
  1405. aboutStack.spacing = 14
  1406. aboutStack.alignment = .leading
  1407. aboutStack.translatesAutoresizingMaskIntoConstraints = false
  1408. contentStack.addArrangedSubview(appearanceStack)
  1409. contentStack.addArrangedSubview(settingsSection)
  1410. contentStack.addArrangedSubview(aboutStack)
  1411. settingsPageContainer.addSubview(contentStack)
  1412. NSLayoutConstraint.activate([
  1413. contentStack.leadingAnchor.constraint(equalTo: settingsPageContainer.leadingAnchor, constant: 42),
  1414. contentStack.trailingAnchor.constraint(lessThanOrEqualTo: settingsPageContainer.trailingAnchor, constant: -42),
  1415. contentStack.topAnchor.constraint(equalTo: settingsPageContainer.topAnchor, constant: 48),
  1416. appearanceStack.widthAnchor.constraint(equalTo: contentStack.widthAnchor),
  1417. appearanceSection.widthAnchor.constraint(equalTo: appearanceStack.widthAnchor),
  1418. settingsSection.widthAnchor.constraint(equalTo: contentStack.widthAnchor),
  1419. aboutStack.widthAnchor.constraint(equalTo: contentStack.widthAnchor),
  1420. aboutSection.widthAnchor.constraint(equalTo: aboutStack.widthAnchor),
  1421. contentStack.widthAnchor.constraint(equalTo: settingsPageContainer.widthAnchor, constant: -84)
  1422. ])
  1423. }
  1424. private func makeAppearanceModeSegment() -> NSSegmentedControl {
  1425. let segment = NSSegmentedControl(
  1426. labels: ["System", "Light", "Dark"],
  1427. trackingMode: .selectOne,
  1428. target: self,
  1429. action: #selector(appearanceModeChanged(_:))
  1430. )
  1431. segment.translatesAutoresizingMaskIntoConstraints = false
  1432. segment.segmentStyle = .automatic
  1433. segment.selectedSegment = AppAppearanceManager.shared.mode.segmentIndex
  1434. segment.setContentHuggingPriority(.required, for: .horizontal)
  1435. segment.setContentCompressionResistancePriority(.required, for: .horizontal)
  1436. return segment
  1437. }
  1438. @objc private func appearanceModeChanged(_ sender: NSSegmentedControl) {
  1439. guard let mode = AppAppearanceManager.Mode(segmentIndex: sender.selectedSegment) else { return }
  1440. AppAppearanceManager.shared.mode = mode
  1441. }
  1442. private func makeSettingsSection(rows: [NSView]) -> NSView {
  1443. let section = NSStackView()
  1444. section.orientation = .vertical
  1445. section.spacing = 0
  1446. section.alignment = .leading
  1447. section.translatesAutoresizingMaskIntoConstraints = false
  1448. section.wantsLayer = true
  1449. section.layer?.backgroundColor = Theme.settingsGroupBackground.cgColor
  1450. section.layer?.cornerRadius = 14
  1451. section.layer?.borderWidth = 1
  1452. section.layer?.borderColor = Theme.border.cgColor
  1453. section.layer?.masksToBounds = true
  1454. for (index, row) in rows.enumerated() {
  1455. section.addArrangedSubview(row)
  1456. row.widthAnchor.constraint(equalTo: section.widthAnchor).isActive = true
  1457. if index < rows.count - 1 {
  1458. let divider = NSView()
  1459. divider.translatesAutoresizingMaskIntoConstraints = false
  1460. divider.wantsLayer = true
  1461. divider.layer?.backgroundColor = Theme.settingsDivider.cgColor
  1462. section.addArrangedSubview(divider)
  1463. NSLayoutConstraint.activate([
  1464. divider.heightAnchor.constraint(equalToConstant: 1),
  1465. divider.leadingAnchor.constraint(equalTo: section.leadingAnchor),
  1466. divider.trailingAnchor.constraint(equalTo: section.trailingAnchor)
  1467. ])
  1468. }
  1469. }
  1470. return section
  1471. }
  1472. private func makeSettingsRow(title: String, systemImage: String, accessory: NSView?, tapAction: Selector? = nil) -> NSView {
  1473. let row = NSView()
  1474. row.translatesAutoresizingMaskIntoConstraints = false
  1475. row.wantsLayer = true
  1476. let iconTile = NSView()
  1477. iconTile.translatesAutoresizingMaskIntoConstraints = false
  1478. iconTile.wantsLayer = true
  1479. iconTile.layer?.backgroundColor = Theme.settingsIconBackground.cgColor
  1480. iconTile.layer?.cornerRadius = 9
  1481. let icon = NSImageView()
  1482. icon.translatesAutoresizingMaskIntoConstraints = false
  1483. icon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 13, weight: .medium)
  1484. icon.image = NSImage(systemSymbolName: systemImage, accessibilityDescription: title)
  1485. icon.contentTintColor = Theme.brandBlue
  1486. let titleLabel = NSTextField(labelWithString: title)
  1487. titleLabel.font = .systemFont(ofSize: 14, weight: .medium)
  1488. titleLabel.textColor = Theme.secondaryText
  1489. titleLabel.alignment = .left
  1490. let rowStack = NSStackView()
  1491. rowStack.orientation = .horizontal
  1492. rowStack.spacing = 16
  1493. rowStack.alignment = .centerY
  1494. rowStack.translatesAutoresizingMaskIntoConstraints = false
  1495. let spacer = NSView()
  1496. spacer.translatesAutoresizingMaskIntoConstraints = false
  1497. spacer.setContentHuggingPriority(NSLayoutConstraint.Priority(1), for: .horizontal)
  1498. iconTile.addSubview(icon)
  1499. rowStack.addArrangedSubview(iconTile)
  1500. rowStack.addArrangedSubview(titleLabel)
  1501. rowStack.addArrangedSubview(spacer)
  1502. if let accessory {
  1503. rowStack.addArrangedSubview(accessory)
  1504. }
  1505. row.addSubview(rowStack)
  1506. NSLayoutConstraint.activate([
  1507. row.heightAnchor.constraint(equalToConstant: 68),
  1508. iconTile.widthAnchor.constraint(equalToConstant: 38),
  1509. iconTile.heightAnchor.constraint(equalToConstant: 38),
  1510. icon.centerXAnchor.constraint(equalTo: iconTile.centerXAnchor),
  1511. icon.centerYAnchor.constraint(equalTo: iconTile.centerYAnchor),
  1512. icon.widthAnchor.constraint(equalToConstant: 20),
  1513. icon.heightAnchor.constraint(equalToConstant: 20),
  1514. rowStack.leadingAnchor.constraint(equalTo: row.leadingAnchor, constant: 16),
  1515. rowStack.trailingAnchor.constraint(equalTo: row.trailingAnchor, constant: -16),
  1516. rowStack.topAnchor.constraint(equalTo: row.topAnchor),
  1517. rowStack.bottomAnchor.constraint(equalTo: row.bottomAnchor)
  1518. ])
  1519. if let tapAction {
  1520. let rowButton = NSButton(title: "", target: self, action: tapAction)
  1521. rowButton.translatesAutoresizingMaskIntoConstraints = false
  1522. rowButton.isBordered = false
  1523. rowButton.bezelStyle = .regularSquare
  1524. rowButton.setButtonType(.momentaryChange)
  1525. rowButton.focusRingType = .none
  1526. rowButton.wantsLayer = true
  1527. rowButton.layer?.backgroundColor = .clear
  1528. row.addSubview(rowButton)
  1529. NSLayoutConstraint.activate([
  1530. rowButton.leadingAnchor.constraint(equalTo: row.leadingAnchor),
  1531. rowButton.trailingAnchor.constraint(equalTo: row.trailingAnchor),
  1532. rowButton.topAnchor.constraint(equalTo: row.topAnchor),
  1533. rowButton.bottomAnchor.constraint(equalTo: row.bottomAnchor)
  1534. ])
  1535. }
  1536. return row
  1537. }
  1538. private func reloadSavedJobsListings() {
  1539. savedJobsStack.arrangedSubviews.forEach {
  1540. savedJobsStack.removeArrangedSubview($0)
  1541. $0.removeFromSuperview()
  1542. }
  1543. if savedJobOrder.isEmpty {
  1544. savedJobsPageSubtitleLabel.stringValue = "Save jobs from Home to see them here."
  1545. let empty = NSTextField(wrappingLabelWithString: "No saved jobs yet. Search on Home, then tap Save on a listing.")
  1546. empty.font = .systemFont(ofSize: 14, weight: .regular)
  1547. empty.textColor = Theme.secondaryText
  1548. empty.alignment = .left
  1549. empty.maximumNumberOfLines = 0
  1550. empty.translatesAutoresizingMaskIntoConstraints = false
  1551. savedJobsStack.addArrangedSubview(empty)
  1552. empty.widthAnchor.constraint(equalTo: savedJobsStack.widthAnchor).isActive = true
  1553. return
  1554. }
  1555. savedJobsPageSubtitleLabel.stringValue = "\(savedJobOrder.count) saved \(savedJobOrder.count == 1 ? "position" : "positions")"
  1556. for job in savedJobOrder {
  1557. let card = makeJobListingCard(job, context: .savedJobsPage)
  1558. savedJobsStack.addArrangedSubview(card)
  1559. card.widthAnchor.constraint(equalTo: savedJobsStack.widthAnchor).isActive = true
  1560. }
  1561. }
  1562. private func isSavedJobsSidebarIndex(_ index: Int) -> Bool {
  1563. guard index >= 0, index < currentSidebarItems.count else { return false }
  1564. return currentSidebarItems[index].title == "Saved Jobs"
  1565. }
  1566. private func isHomeSidebarIndex(_ index: Int) -> Bool {
  1567. guard index >= 0, index < currentSidebarItems.count else { return false }
  1568. return currentSidebarItems[index].title == "Home"
  1569. }
  1570. private func isSettingsSidebarIndex(_ index: Int) -> Bool {
  1571. guard index >= 0, index < currentSidebarItems.count else { return false }
  1572. return currentSidebarItems[index].title == "Settings"
  1573. }
  1574. private func isCVMakerSidebarIndex(_ index: Int) -> Bool {
  1575. guard index >= 0, index < currentSidebarItems.count else { return false }
  1576. return currentSidebarItems[index].title == "CV Maker"
  1577. }
  1578. private func isProfileSidebarIndex(_ index: Int) -> Bool {
  1579. guard index >= 0, index < currentSidebarItems.count else { return false }
  1580. return currentSidebarItems[index].title == "Profile"
  1581. }
  1582. private func updateMainContentVisibility() {
  1583. if isIndeedJobBrowserPresented {
  1584. mainOverlay.isHidden = true
  1585. nonHomeHost.isHidden = true
  1586. indeedJobBrowserHost.isHidden = false
  1587. return
  1588. }
  1589. indeedJobBrowserHost.isHidden = true
  1590. let home = isHomeSidebarIndex(selectedSidebarIndex)
  1591. let savedJobs = isSavedJobsSidebarIndex(selectedSidebarIndex)
  1592. let settings = isSettingsSidebarIndex(selectedSidebarIndex)
  1593. let cvMaker = isCVMakerSidebarIndex(selectedSidebarIndex)
  1594. let profile = isProfileSidebarIndex(selectedSidebarIndex)
  1595. mainOverlay.isHidden = !home
  1596. nonHomeHost.isHidden = home
  1597. nonHomeGenericContainer.isHidden = savedJobs || settings || cvMaker || profile
  1598. savedJobsPageContainer.isHidden = !savedJobs
  1599. settingsPageContainer.isHidden = !settings
  1600. cvMakerPageContainer.isHidden = !cvMaker
  1601. profilePageContainer.isHidden = !profile
  1602. if !profile {
  1603. isProfileEditorPresented = false
  1604. isCVDocumentPreviewPresented = false
  1605. pendingCVTemplate = nil
  1606. profilesListPageView.setPendingCVTemplateDisplayName(nil)
  1607. cvFilledPreviewPageView.isHidden = true
  1608. profilesListPageView.isHidden = false
  1609. myProfilePageView.isHidden = true
  1610. }
  1611. if profile, !isProfileEditorPresented {
  1612. if isCVDocumentPreviewPresented {
  1613. profilesListPageView.isHidden = true
  1614. myProfilePageView.isHidden = true
  1615. cvFilledPreviewPageView.isHidden = false
  1616. } else {
  1617. profilesListPageView.reloadFromStore()
  1618. profilesListPageView.isHidden = false
  1619. myProfilePageView.isHidden = true
  1620. cvFilledPreviewPageView.isHidden = true
  1621. }
  1622. }
  1623. if !home, selectedSidebarIndex < currentSidebarItems.count {
  1624. if savedJobs {
  1625. reloadSavedJobsListings()
  1626. } else if settings || cvMaker || profile {
  1627. window?.makeFirstResponder(nil)
  1628. } else {
  1629. nonHomeTitleLabel.stringValue = currentSidebarItems[selectedSidebarIndex].title
  1630. }
  1631. }
  1632. }
  1633. /// Restores the main job-search field when returning to Home; chat history is kept until the user chooses **Clear chat**.
  1634. private func applyHomeState() {
  1635. jobKeywordsField.stringValue = ""
  1636. window?.makeFirstResponder(nil)
  1637. }
  1638. @objc private func didTapClearChat() {
  1639. guard !isAwaitingResponse else { return }
  1640. resetChatState()
  1641. window?.makeFirstResponder(nil)
  1642. }
  1643. private func updateSearchBarShadowPath() {
  1644. guard searchBarShadowHost.bounds.width > 0, searchBarShadowHost.bounds.height > 0 else { return }
  1645. let r = searchBarShadowHost.bounds
  1646. let radius = min(r.height / 2, 27)
  1647. searchBarShadowHost.layer?.shadowPath = CGPath(
  1648. roundedRect: r,
  1649. cornerWidth: radius,
  1650. cornerHeight: radius,
  1651. transform: nil
  1652. )
  1653. }
  1654. @objc private func didSubmitSearch() {
  1655. guard ensureProAccess() else { return }
  1656. let prompt = jobKeywordsField.stringValue.trimmingCharacters(in: .whitespacesAndNewlines)
  1657. guard !prompt.isEmpty, !isAwaitingResponse else { return }
  1658. let isContinuation = isContinuationPrompt(prompt)
  1659. let effectiveQuery = resolvedSearchQuery(for: prompt)
  1660. appendChatBubble(text: prompt, isUser: true)
  1661. chatMessages.append(ChatMessage(role: "user", content: prompt))
  1662. jobKeywordsField.stringValue = ""
  1663. startJobSearchRequest(effectiveQuery: effectiveQuery, isContinuation: isContinuation)
  1664. window?.makeFirstResponder(nil)
  1665. }
  1666. @objc private func didTapFeatureRole() {
  1667. selectFeatureShortcut(.role)
  1668. focusSearchField(seed: "Find roles similar to: ")
  1669. }
  1670. @objc private func didTapFeatureCompany() {
  1671. selectFeatureShortcut(.company)
  1672. focusSearchField(seed: "Find jobs at company: ")
  1673. }
  1674. @objc private func didTapFeatureSkill() {
  1675. selectFeatureShortcut(.skill)
  1676. focusSearchField(seed: "Find jobs that require skill: ")
  1677. }
  1678. private func selectFeatureShortcut(_ shortcut: FeatureShortcut) {
  1679. for (index, view) in featureCardsRow.arrangedSubviews.enumerated() {
  1680. guard let card = view as? FeatureShortcutCardView else { continue }
  1681. card.isSelected = (index == shortcut.rawValue)
  1682. }
  1683. }
  1684. @objc private func didTapShareApp(_ sender: NSButton) {
  1685. presentAppShareMenu(anchoredTo: sender)
  1686. }
  1687. /// Shows the macOS share menu (Mail, Messages, AirDrop, Copy Link, etc.) with the app link.
  1688. private func presentAppShareMenu(anchoredTo sender: NSButton) {
  1689. guard let row = sender.superview else { return }
  1690. let items = AppMarketingLinks.shareItems
  1691. guard !items.isEmpty else { return }
  1692. let picker = NSSharingServicePicker(items: items)
  1693. picker.delegate = self
  1694. appSharePicker = picker
  1695. // Match `makeSettingsRow` layout: 16pt leading inset + 38pt icon tile — anchor the
  1696. // popover beside the share icon, not the horizontal center of the full-width row.
  1697. let iconTileInset: CGFloat = 16
  1698. let iconTileSize: CGFloat = 38
  1699. let anchorRect = NSRect(
  1700. x: iconTileInset,
  1701. y: row.bounds.minY + 6,
  1702. width: iconTileSize,
  1703. height: max(row.bounds.height - 12, 1)
  1704. )
  1705. picker.show(relativeTo: anchorRect, of: row, preferredEdge: .minY)
  1706. }
  1707. func sharingServicePicker(_ sharingServicePicker: NSSharingServicePicker, delegateFor sharingService: NSSharingService) -> Any? {
  1708. self
  1709. }
  1710. func sharingServicePicker(_ sharingServicePicker: NSSharingServicePicker, didChoose sharingService: NSSharingService?) {
  1711. appSharePicker = nil
  1712. }
  1713. func sharingService(_ sharingService: NSSharingService, willShareItems items: [Any]) -> [Any] {
  1714. if sharingService == NSSharingService(named: .composeEmail) {
  1715. sharingService.subject = AppMarketingLinks.shareEmailSubject
  1716. }
  1717. return items
  1718. }
  1719. @objc private func didTapMoreApps() {
  1720. guard let url = AppMarketingLinks.developerAppsURL else {
  1721. presentAppMarketingConfigurationAlert(feature: "More Apps")
  1722. return
  1723. }
  1724. NSWorkspace.shared.open(url)
  1725. }
  1726. private func presentAppMarketingConfigurationAlert(feature: String) {
  1727. let alert = NSAlert()
  1728. alert.messageText = "\(feature) isn’t available yet"
  1729. alert.informativeText = """
  1730. Add your Mac App Store IDs in the target’s build settings:
  1731. • AppStoreAppID — numeric app ID from App Store Connect
  1732. • AppStoreDeveloperID — numeric developer ID (for your other apps page)
  1733. """
  1734. alert.alertStyle = .informational
  1735. alert.addButton(withTitle: "OK")
  1736. if let window {
  1737. alert.beginSheetModal(for: window)
  1738. } else {
  1739. alert.runModal()
  1740. }
  1741. }
  1742. @objc private func didTapWebsite() {
  1743. AppLegalURLs.openInSafari(AppLegalURLs.marketingHome)
  1744. }
  1745. @objc private func didTapSupport() {
  1746. AppLegalURLs.openInSafari(AppLegalURLs.support)
  1747. }
  1748. @objc private func didTapTermsOfUse() {
  1749. AppLegalURLs.openInSafari(AppLegalURLs.termsOfUse)
  1750. }
  1751. @objc private func didTapPrivacyPolicy() {
  1752. AppLegalURLs.openInSafari(AppLegalURLs.privacyPolicy)
  1753. }
  1754. private func focusSearchField(seed: String) {
  1755. guard ensureProAccess() else { return }
  1756. jobKeywordsField.stringValue = seed
  1757. window?.makeFirstResponder(jobKeywordsField)
  1758. if let editor = jobKeywordsField.window?.fieldEditor(true, for: jobKeywordsField) as? NSTextView {
  1759. editor.moveToEndOfDocument(nil)
  1760. }
  1761. }
  1762. @objc private func didTapLoadMoreJobs() {
  1763. guard ensureProAccess() else { return }
  1764. let prompt = "Show more jobs"
  1765. guard !isAwaitingResponse, isContinuationPrompt(prompt) else { return }
  1766. if anchorUserJobQuery(excludingLatestUserMessage: prompt) == nil { return }
  1767. appendChatBubble(text: prompt, isUser: true)
  1768. chatMessages.append(ChatMessage(role: "user", content: prompt))
  1769. let effectiveQuery = resolvedSearchQuery(for: prompt)
  1770. startJobSearchRequest(effectiveQuery: effectiveQuery, isContinuation: true)
  1771. }
  1772. private func startJobSearchRequest(effectiveQuery: String, isContinuation: Bool) {
  1773. isAwaitingResponse = true
  1774. addInlineChatThinkingRow()
  1775. setInputEnabled(false)
  1776. let contextMessages = chatMessages
  1777. let maxJobs = Self.clampedJobsPerRequest()
  1778. jobSearchService.searchJobs(query: effectiveQuery, conversation: contextMessages, maxJobs: maxJobs) { [weak self] result in
  1779. DispatchQueue.main.async {
  1780. guard let self else { return }
  1781. self.removeInlineChatThinkingRow()
  1782. self.isAwaitingResponse = false
  1783. self.setInputEnabled(true)
  1784. switch result {
  1785. case .success(let output):
  1786. let normalizedJobs = self.normalizedJobs(output.jobs)
  1787. let freshJobs: [JobListing]
  1788. if isContinuation {
  1789. // Continuations append only the *new* matches; previous cards already live in their own assistant message above.
  1790. let alreadySeen = Set(self.lastSearchResults)
  1791. freshJobs = normalizedJobs.filter { !alreadySeen.contains($0) }
  1792. } else {
  1793. freshJobs = normalizedJobs
  1794. }
  1795. self.lastSearchResults.append(contentsOf: freshJobs)
  1796. let reply = self.makeAssistantSearchReply(
  1797. query: effectiveQuery,
  1798. newJobsCount: freshJobs.count,
  1799. isContinuation: isContinuation
  1800. )
  1801. self.chatMessages.append(ChatMessage(role: "assistant", content: reply))
  1802. self.appendChatBubble(text: reply, isUser: false, jobs: freshJobs)
  1803. case .failure(let error):
  1804. self.appendChatBubble(text: error.localizedDescription, isUser: false)
  1805. }
  1806. }
  1807. }
  1808. }
  1809. private func normalizedJobs(_ jobs: [JobListing]) -> [JobListing] {
  1810. let trimmed = jobs.map {
  1811. JobListing(
  1812. title: $0.title.trimmingCharacters(in: .whitespacesAndNewlines),
  1813. description: $0.description.trimmingCharacters(in: .whitespacesAndNewlines),
  1814. url: $0.url?.trimmingCharacters(in: .whitespacesAndNewlines)
  1815. )
  1816. }
  1817. return trimmed.filter { !$0.title.isEmpty && !$0.description.isEmpty }
  1818. }
  1819. private func makeAssistantSearchReply(query: String, newJobsCount: Int, isContinuation: Bool) -> String {
  1820. if newJobsCount == 0 {
  1821. if isContinuation {
  1822. return "I couldn't find new matches for \u{201C}\(query)\u{201D}. Try a different angle or a more specific keyword."
  1823. }
  1824. return "No jobs found for \u{201C}\(query)\u{201D}. Try another title, skill, company, or location."
  1825. }
  1826. let plural = newJobsCount == 1 ? "match" : "matches"
  1827. if isContinuation {
  1828. return "Here are \(newJobsCount) more \(plural) for \u{201C}\(query)\u{201D}."
  1829. }
  1830. return "Found \(newJobsCount) \(plural) for \u{201C}\(query)\u{201D}. Tap Apply to open the listing or Save to revisit later."
  1831. }
  1832. private func resolvedSearchQuery(for prompt: String) -> String {
  1833. let anchor = anchorUserJobQuery(excludingLatestUserMessage: prompt)
  1834. if isContinuationPrompt(prompt), !isRefinementPrompt(prompt) {
  1835. if let anchor { return anchor }
  1836. return prompt
  1837. }
  1838. if isRefinementPrompt(prompt), let anchor {
  1839. return "\(anchor). User follow-up (apply on top of the same search topic): \(prompt)"
  1840. }
  1841. return prompt
  1842. }
  1843. /// First prior user message that looks like an original job query (skips short continuations and refinements so follow-ups keep a stable topic anchor).
  1844. private func anchorUserJobQuery(excludingLatestUserMessage latest: String) -> String? {
  1845. let prior = Array(chatMessages.dropLast())
  1846. for message in prior.reversed() where message.role == "user" {
  1847. let candidate = message.content.trimmingCharacters(in: .whitespacesAndNewlines)
  1848. guard !candidate.isEmpty, candidate != latest else { continue }
  1849. if isContinuationPrompt(candidate) { continue }
  1850. if isRefinementPrompt(candidate) { continue }
  1851. return candidate
  1852. }
  1853. return nil
  1854. }
  1855. private func isContinuationPrompt(_ prompt: String) -> Bool {
  1856. let normalized = prompt.trimmingCharacters(in: .whitespacesAndNewlines).lowercased()
  1857. let continuationPhrases: Set<String> = [
  1858. "more",
  1859. "show more",
  1860. "more jobs",
  1861. "more results",
  1862. "do more searches",
  1863. "more searches",
  1864. "search more",
  1865. "continue",
  1866. "next"
  1867. ]
  1868. if continuationPhrases.contains(normalized) {
  1869. return true
  1870. }
  1871. return normalized.contains("more search") || normalized.contains("more job")
  1872. }
  1873. /// Follow-ups that narrow, re-rank, or re-frame results rather than starting a brand-new role search.
  1874. /// Strong phrases always count. Single-word cues (e.g. "remote") only count after we already showed results, so first searches like "Senior iOS remote" stay anchored as primary queries.
  1875. private func isRefinementPrompt(_ prompt: String) -> Bool {
  1876. let n = prompt.trimmingCharacters(in: .whitespacesAndNewlines).lowercased()
  1877. if n.isEmpty { return false }
  1878. let strongPhrases = [
  1879. "higher pay", "high pay", "better pay", "more pay", "top pay", "best pay",
  1880. "higher salary", "better salary", "more salary", "pay rate", "hourly rate",
  1881. "paid more", "paying more", "earn more", "better paid", "paying better",
  1882. "work from home", "in office", "in-office", "on-site only", "remote only",
  1883. "hybrid only", "onsite only", "visa sponsorship", "h1b",
  1884. "entry level", "entry-level", "mid level", "mid-level", "full time", "full-time",
  1885. "part time", "part-time",
  1886. "closer to", "nearer", "different city", "different state", "relocate",
  1887. "filter", "only show", "just show", "exclude", "without", "sort by", "rank by",
  1888. "cheaper", "lower pay", "less travel",
  1889. "better benefits", "equity", "bonus", "overtime",
  1890. "get me the jobs", "show me the jobs", "give me the jobs", "narrow", "refine"
  1891. ]
  1892. if strongPhrases.contains(where: { n.contains($0) }) { return true }
  1893. if n.hasPrefix("only ") || n.hasPrefix("just ") { return true }
  1894. guard !lastSearchResults.isEmpty, n.count <= 52 else { return false }
  1895. let softAfterResults = [
  1896. "remote", "hybrid", "onsite", "on-site", "senior", "junior", "staff", "lead",
  1897. "principal", "intern", "contract", "location"
  1898. ]
  1899. return softAfterResults.contains(where: { n.contains($0) })
  1900. }
  1901. func controlTextDidBeginEditing(_ obj: Notification) {
  1902. applySearchFieldInsertionPoint(obj.object)
  1903. }
  1904. func controlTextDidChange(_ obj: Notification) {
  1905. applySearchFieldInsertionPoint(obj.object)
  1906. }
  1907. func control(_ control: NSControl, textView: NSTextView, doCommandBy commandSelector: Selector) -> Bool {
  1908. guard control === jobKeywordsField, commandSelector == #selector(NSResponder.insertNewline(_:)) else {
  1909. return false
  1910. }
  1911. didSubmitSearch()
  1912. return true
  1913. }
  1914. private func applySearchFieldInsertionPoint(_ object: Any?) {
  1915. guard let field = object as? NSTextField,
  1916. field === jobKeywordsField,
  1917. let textView = field.window?.fieldEditor(true, for: field) as? NSTextView else { return }
  1918. textView.insertionPointColor = Theme.primaryText
  1919. }
  1920. private func resetChatState() {
  1921. removeInlineChatThinkingRow()
  1922. trailingLoadMoreJobsRow = nil
  1923. trailingLoadMoreJobsButton = nil
  1924. chatMessages.removeAll()
  1925. lastSearchResults.removeAll()
  1926. chatStack.arrangedSubviews.forEach {
  1927. chatStack.removeArrangedSubview($0)
  1928. $0.removeFromSuperview()
  1929. }
  1930. let welcome = "Tell me what role you want and I will return job descriptions, key skills, and a quick fit summary."
  1931. chatMessages.append(ChatMessage(role: "assistant", content: welcome))
  1932. appendChatBubble(text: welcome, isUser: false)
  1933. }
  1934. private func appendChatBubble(text: String, isUser: Bool, jobs: [JobListing]? = nil) {
  1935. let host = NSView()
  1936. host.translatesAutoresizingMaskIntoConstraints = false
  1937. if isUser {
  1938. installUserBubble(text: text, into: host)
  1939. } else {
  1940. installAssistantBubble(text: text, jobs: jobs, into: host)
  1941. }
  1942. chatStack.addArrangedSubview(host)
  1943. host.widthAnchor.constraint(equalTo: chatStack.widthAnchor).isActive = true
  1944. if prefersReducedMotion {
  1945. host.alphaValue = 1
  1946. } else {
  1947. host.alphaValue = 0
  1948. }
  1949. DispatchQueue.main.async { [weak self] in
  1950. guard let self else { return }
  1951. let scroll: () -> Void = {
  1952. if isUser {
  1953. self.scrollChatToBottom()
  1954. } else {
  1955. self.scrollChatToShowTopOfView(host)
  1956. }
  1957. }
  1958. if self.prefersReducedMotion {
  1959. self.updateChatBubbleWidths()
  1960. scroll()
  1961. return
  1962. }
  1963. NSAnimationContext.runAnimationGroup { ctx in
  1964. ctx.duration = 0.3
  1965. ctx.timingFunction = CAMediaTimingFunction(name: .easeOut)
  1966. host.animator().alphaValue = 1
  1967. }
  1968. self.updateChatBubbleWidths()
  1969. scroll()
  1970. }
  1971. }
  1972. private func installUserBubble(text: String, into host: NSView) {
  1973. let bubble = makeChatBubbleContainer(text: text, isUser: true)
  1974. host.addSubview(bubble)
  1975. NSLayoutConstraint.activate([
  1976. bubble.topAnchor.constraint(equalTo: host.topAnchor),
  1977. bubble.bottomAnchor.constraint(equalTo: host.bottomAnchor),
  1978. bubble.trailingAnchor.constraint(equalTo: host.trailingAnchor),
  1979. bubble.leadingAnchor.constraint(greaterThanOrEqualTo: host.leadingAnchor, constant: 64),
  1980. bubble.widthAnchor.constraint(lessThanOrEqualTo: host.widthAnchor, multiplier: 0.78)
  1981. ])
  1982. }
  1983. private func installAssistantBubble(text: String, jobs: [JobListing]?, into host: NSView) {
  1984. let avatar = makeAssistantAvatarView()
  1985. let nameLabel = NSTextField(labelWithString: AppMarketingLinks.appDisplayName)
  1986. nameLabel.font = .systemFont(ofSize: 11, weight: .semibold)
  1987. nameLabel.textColor = Theme.secondaryText
  1988. nameLabel.alignment = .left
  1989. nameLabel.translatesAutoresizingMaskIntoConstraints = false
  1990. let bubble = makeChatBubbleContainer(text: text, isUser: false)
  1991. let column = NSStackView(views: [nameLabel, bubble])
  1992. column.orientation = .vertical
  1993. column.spacing = 6
  1994. // Leading keeps the assistant label, text bubble, and job cards hugging the left (after the avatar); `.width` was letting narrow intrinsic widths sit on the trailing side so AI read like a second “user” column.
  1995. column.alignment = .leading
  1996. column.translatesAutoresizingMaskIntoConstraints = false
  1997. if let jobs, !jobs.isEmpty {
  1998. trailingLoadMoreJobsRow?.removeFromSuperview()
  1999. trailingLoadMoreJobsRow = nil
  2000. trailingLoadMoreJobsButton = nil
  2001. let jobsStack = makeChatJobsStackView(jobs: jobs)
  2002. column.addArrangedSubview(jobsStack)
  2003. jobsStack.widthAnchor.constraint(equalTo: column.widthAnchor).isActive = true
  2004. let moreRow = makeLoadMoreJobsRowView()
  2005. column.addArrangedSubview(moreRow)
  2006. moreRow.widthAnchor.constraint(equalTo: column.widthAnchor).isActive = true
  2007. trailingLoadMoreJobsRow = moreRow
  2008. }
  2009. host.addSubview(avatar)
  2010. host.addSubview(column)
  2011. host.userInterfaceLayoutDirection = .leftToRight
  2012. NSLayoutConstraint.activate([
  2013. avatar.leadingAnchor.constraint(equalTo: host.leadingAnchor),
  2014. avatar.topAnchor.constraint(equalTo: host.topAnchor),
  2015. avatar.widthAnchor.constraint(equalToConstant: 36),
  2016. avatar.heightAnchor.constraint(equalToConstant: 36),
  2017. column.leadingAnchor.constraint(equalTo: avatar.trailingAnchor, constant: 12),
  2018. column.trailingAnchor.constraint(equalTo: host.trailingAnchor),
  2019. column.topAnchor.constraint(equalTo: host.topAnchor),
  2020. column.bottomAnchor.constraint(equalTo: host.bottomAnchor),
  2021. bubble.widthAnchor.constraint(lessThanOrEqualTo: host.widthAnchor, multiplier: 0.78)
  2022. ])
  2023. }
  2024. private func makeChatBubbleContainer(text: String, isUser: Bool) -> NSView {
  2025. let container = NSView()
  2026. container.translatesAutoresizingMaskIntoConstraints = false
  2027. container.wantsLayer = true
  2028. container.layer?.cornerRadius = 14
  2029. if #available(macOS 11.0, *) {
  2030. container.layer?.cornerCurve = .continuous
  2031. }
  2032. container.layer?.masksToBounds = true
  2033. if isUser {
  2034. container.layer?.backgroundColor = Theme.brandBlue.cgColor
  2035. } else {
  2036. container.layer?.backgroundColor = Theme.chromeBackground.cgColor
  2037. container.layer?.borderWidth = 1
  2038. container.layer?.borderColor = Theme.border.cgColor
  2039. }
  2040. let label = ChatBubbleLabel(wrappingLabelWithString: text)
  2041. label.font = .systemFont(ofSize: 13.5, weight: .regular)
  2042. label.textColor = isUser ? .white : Theme.primaryText
  2043. label.maximumNumberOfLines = 0
  2044. label.lineBreakMode = .byWordWrapping
  2045. label.alignment = .left
  2046. label.translatesAutoresizingMaskIntoConstraints = false
  2047. label.setContentHuggingPriority(.defaultLow, for: .horizontal)
  2048. label.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  2049. container.addSubview(label)
  2050. NSLayoutConstraint.activate([
  2051. label.leadingAnchor.constraint(equalTo: container.leadingAnchor, constant: 14),
  2052. label.trailingAnchor.constraint(equalTo: container.trailingAnchor, constant: -14),
  2053. label.topAnchor.constraint(equalTo: container.topAnchor, constant: 10),
  2054. label.bottomAnchor.constraint(equalTo: container.bottomAnchor, constant: -10)
  2055. ])
  2056. return container
  2057. }
  2058. private func makeAssistantAvatarView() -> NSView {
  2059. let view = NSView()
  2060. view.translatesAutoresizingMaskIntoConstraints = false
  2061. view.wantsLayer = true
  2062. view.layer?.cornerRadius = 18
  2063. if #available(macOS 11.0, *) {
  2064. view.layer?.cornerCurve = .continuous
  2065. }
  2066. view.layer?.backgroundColor = Theme.settingsIconBackground.cgColor
  2067. view.layer?.borderWidth = 1
  2068. view.layer?.borderColor = Theme.proCardBorder.cgColor
  2069. let icon = NSImageView()
  2070. icon.translatesAutoresizingMaskIntoConstraints = false
  2071. icon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 15, weight: .semibold)
  2072. icon.image = NSImage(systemSymbolName: "sparkles", accessibilityDescription: AppMarketingLinks.appDisplayName)
  2073. icon.contentTintColor = Theme.brandBlue
  2074. view.addSubview(icon)
  2075. NSLayoutConstraint.activate([
  2076. icon.centerXAnchor.constraint(equalTo: view.centerXAnchor),
  2077. icon.centerYAnchor.constraint(equalTo: view.centerYAnchor)
  2078. ])
  2079. return view
  2080. }
  2081. private func makeLoadMoreJobsRowView() -> NSView {
  2082. let row = NSView()
  2083. row.translatesAutoresizingMaskIntoConstraints = false
  2084. let button = HoverableButton()
  2085. button.pointerCursor = true
  2086. button.title = "Show more jobs"
  2087. button.font = .systemFont(ofSize: 12, weight: .semibold)
  2088. button.bezelStyle = .rounded
  2089. button.controlSize = .regular
  2090. button.contentTintColor = Theme.brandBlue
  2091. button.target = self
  2092. button.action = #selector(didTapLoadMoreJobs)
  2093. button.translatesAutoresizingMaskIntoConstraints = false
  2094. trailingLoadMoreJobsButton = button
  2095. row.addSubview(button)
  2096. NSLayoutConstraint.activate([
  2097. button.leadingAnchor.constraint(equalTo: row.leadingAnchor),
  2098. button.topAnchor.constraint(equalTo: row.topAnchor, constant: 2),
  2099. button.bottomAnchor.constraint(equalTo: row.bottomAnchor, constant: -2)
  2100. ])
  2101. return row
  2102. }
  2103. private func makeChatJobsStackView(jobs: [JobListing]) -> ChatJobsStackView {
  2104. let stack = ChatJobsStackView()
  2105. stack.orientation = .vertical
  2106. stack.spacing = 10
  2107. stack.alignment = .width
  2108. stack.distribution = .fill
  2109. stack.translatesAutoresizingMaskIntoConstraints = false
  2110. for job in jobs {
  2111. let card = makeJobListingCard(job, context: .homeSearchResults)
  2112. stack.addArrangedSubview(card)
  2113. card.widthAnchor.constraint(equalTo: stack.widthAnchor).isActive = true
  2114. }
  2115. return stack
  2116. }
  2117. private func scrollChatToBottom() {
  2118. let maxY = max(0, chatDocumentView.bounds.height - chatScrollView.contentView.bounds.height)
  2119. chatScrollView.contentView.scroll(to: NSPoint(x: 0, y: maxY))
  2120. chatScrollView.reflectScrolledClipView(chatScrollView.contentView)
  2121. }
  2122. /// Scrolls so the top of `view` sits at the top of the chat clip (flipped document coords). Long assistant replies stay readable from the first line instead of jumping to the end.
  2123. private func scrollChatToShowTopOfView(_ view: NSView) {
  2124. chatDocumentView.layoutSubtreeIfNeeded()
  2125. view.layoutSubtreeIfNeeded()
  2126. let doc = chatDocumentView
  2127. let visibleHeight = chatScrollView.contentView.bounds.height
  2128. let docHeight = doc.bounds.height
  2129. guard docHeight > 0, visibleHeight > 0 else {
  2130. scrollChatToBottom()
  2131. return
  2132. }
  2133. let rectInDoc = view.convert(view.bounds, to: doc)
  2134. let maxY = max(0, docHeight - visibleHeight)
  2135. let targetY = min(max(0, rectInDoc.minY), maxY)
  2136. chatScrollView.contentView.scroll(to: NSPoint(x: 0, y: targetY))
  2137. chatScrollView.reflectScrolledClipView(chatScrollView.contentView)
  2138. }
  2139. private func addInlineChatThinkingRow() {
  2140. removeInlineChatThinkingRow()
  2141. let host = NSView()
  2142. host.translatesAutoresizingMaskIntoConstraints = false
  2143. let indicator = ChatThinkingIndicatorView(compact: false)
  2144. host.addSubview(indicator)
  2145. NSLayoutConstraint.activate([
  2146. indicator.leadingAnchor.constraint(equalTo: host.leadingAnchor, constant: 8),
  2147. indicator.topAnchor.constraint(equalTo: host.topAnchor),
  2148. indicator.bottomAnchor.constraint(equalTo: host.bottomAnchor, constant: -2)
  2149. ])
  2150. chatThinkingRowHost = host
  2151. chatStack.addArrangedSubview(host)
  2152. host.widthAnchor.constraint(equalTo: chatStack.widthAnchor).isActive = true
  2153. indicator.startAnimatingIfNeeded()
  2154. DispatchQueue.main.async { [weak self] in
  2155. self?.updateChatBubbleWidths()
  2156. self?.scrollChatToBottom()
  2157. }
  2158. }
  2159. private func removeInlineChatThinkingRow() {
  2160. guard let host = chatThinkingRowHost else { return }
  2161. for sub in host.subviews {
  2162. (sub as? ChatThinkingIndicatorView)?.stopAnimating()
  2163. }
  2164. chatStack.removeArrangedSubview(host)
  2165. host.removeFromSuperview()
  2166. chatThinkingRowHost = nil
  2167. }
  2168. private func setInputEnabled(_ enabled: Bool) {
  2169. jobKeywordsField.isEnabled = enabled
  2170. findJobsButton.isEnabled = enabled
  2171. findJobsCTAHost.alphaValue = enabled ? 1 : 0.65
  2172. clearChatButton.isEnabled = enabled
  2173. clearChatButton.alphaValue = enabled ? 1 : 0.65
  2174. trailingLoadMoreJobsButton?.isEnabled = enabled
  2175. trailingLoadMoreJobsButton?.alphaValue = enabled ? 1 : 0.65
  2176. }
  2177. private func addIndeedSidebarLaunchRow() {
  2178. let isSelected = isIndeedSidebarSelected
  2179. let rowHost = SidebarNavRowView { [weak self] in
  2180. self?.selectIndeedSidebar()
  2181. }
  2182. rowHost.translatesAutoresizingMaskIntoConstraints = false
  2183. rowHost.wantsLayer = true
  2184. rowHost.layer?.cornerRadius = 8
  2185. rowHost.restingBackgroundColor = isSelected ? Theme.selectionFill : nil
  2186. rowHost.hoverBackgroundColor = isSelected ? Theme.selectionFillHover : Theme.sidebarRowHoverFill
  2187. rowHost.setAccessibilityLabel("Indeed")
  2188. rowHost.setAccessibilityRole(.button)
  2189. rowHost.setAccessibilitySelected(isSelected)
  2190. rowHost.setAccessibilityHelp("Open Indeed to search and apply for jobs")
  2191. let row = NSStackView()
  2192. row.orientation = .horizontal
  2193. row.spacing = 8
  2194. row.alignment = .centerY
  2195. row.translatesAutoresizingMaskIntoConstraints = false
  2196. let logo = IndeedLogoView(displayHeight: 18, variant: .compact)
  2197. logo.translatesAutoresizingMaskIntoConstraints = false
  2198. let text = NSTextField(labelWithString: "Indeed")
  2199. text.font = .systemFont(ofSize: 14, weight: .medium)
  2200. text.textColor = isSelected ? Theme.brandBlue : Theme.secondaryText
  2201. text.refusesFirstResponder = true
  2202. row.addArrangedSubview(logo)
  2203. row.addArrangedSubview(text)
  2204. rowHost.addSubview(row)
  2205. NSLayoutConstraint.activate([
  2206. row.leadingAnchor.constraint(equalTo: rowHost.leadingAnchor, constant: 10),
  2207. row.trailingAnchor.constraint(equalTo: rowHost.trailingAnchor, constant: -10),
  2208. row.topAnchor.constraint(equalTo: rowHost.topAnchor, constant: 8),
  2209. row.bottomAnchor.constraint(equalTo: rowHost.bottomAnchor, constant: -8)
  2210. ])
  2211. rowHost.setContentHuggingPriority(.defaultLow, for: .horizontal)
  2212. sidebar.addArrangedSubview(rowHost)
  2213. let sidebarHorizontalInset = sidebar.edgeInsets.left + sidebar.edgeInsets.right
  2214. rowHost.widthAnchor.constraint(equalTo: sidebar.widthAnchor, constant: -sidebarHorizontalInset).isActive = true
  2215. sidebar.setCustomSpacing(10, after: rowHost)
  2216. }
  2217. private func configureSidebar() {
  2218. let items = currentSidebarItems
  2219. sidebar.arrangedSubviews.forEach {
  2220. sidebar.removeArrangedSubview($0)
  2221. $0.removeFromSuperview()
  2222. }
  2223. let logo = IndeedLogoView(displayHeight: 34, variant: .compact)
  2224. logo.translatesAutoresizingMaskIntoConstraints = false
  2225. let brand = NSTextField(labelWithString: AppMarketingLinks.appDisplayName)
  2226. brand.font = .systemFont(ofSize: 14, weight: .semibold)
  2227. brand.textColor = Theme.brandBlue
  2228. brand.alignment = .left
  2229. brand.maximumNumberOfLines = 2
  2230. brand.preferredMaxLayoutWidth = 194
  2231. let brandHeader = NSStackView(views: [logo, brand])
  2232. brandHeader.orientation = .vertical
  2233. brandHeader.alignment = .leading
  2234. brandHeader.spacing = 6
  2235. brandHeader.translatesAutoresizingMaskIntoConstraints = false
  2236. sidebar.addArrangedSubview(brandHeader)
  2237. sidebar.setCustomSpacing(22, after: brandHeader)
  2238. items.enumerated().forEach { index, item in
  2239. let isSelected = index == selectedSidebarIndex && !isIndeedSidebarSelected
  2240. let rowHost = SidebarNavRowView { [weak self] in
  2241. self?.selectSidebarItem(at: index)
  2242. }
  2243. rowHost.translatesAutoresizingMaskIntoConstraints = false
  2244. rowHost.wantsLayer = true
  2245. rowHost.layer?.cornerRadius = 8
  2246. rowHost.restingBackgroundColor = isSelected ? Theme.selectionFill : nil
  2247. rowHost.hoverBackgroundColor = isSelected ? Theme.selectionFillHover : Theme.sidebarRowHoverFill
  2248. rowHost.setAccessibilityLabel(item.title)
  2249. rowHost.setAccessibilityRole(.button)
  2250. rowHost.setAccessibilitySelected(isSelected)
  2251. let row = NSStackView()
  2252. row.orientation = .horizontal
  2253. row.spacing = 8
  2254. row.alignment = .centerY
  2255. row.translatesAutoresizingMaskIntoConstraints = false
  2256. let icon = NSImageView()
  2257. icon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 13, weight: .medium)
  2258. icon.image = NSImage(systemSymbolName: item.systemImage, accessibilityDescription: item.title)
  2259. icon.contentTintColor = isSelected ? Theme.brandBlue : Theme.secondaryText
  2260. let text = NSTextField(labelWithString: item.title)
  2261. text.font = .systemFont(ofSize: 14, weight: .medium)
  2262. text.textColor = isSelected ? Theme.brandBlue : Theme.secondaryText
  2263. text.refusesFirstResponder = true
  2264. row.addArrangedSubview(icon)
  2265. row.addArrangedSubview(text)
  2266. if let badge = item.badge {
  2267. let badgeField = NSTextField(labelWithString: badge)
  2268. badgeField.font = .systemFont(ofSize: 11, weight: .semibold)
  2269. badgeField.textColor = Theme.primaryText
  2270. badgeField.wantsLayer = true
  2271. badgeField.layer?.backgroundColor = Theme.toggleBackground.cgColor
  2272. badgeField.layer?.cornerRadius = 8
  2273. badgeField.alignment = .center
  2274. badgeField.maximumNumberOfLines = 1
  2275. badgeField.lineBreakMode = .byClipping
  2276. badgeField.refusesFirstResponder = true
  2277. badgeField.translatesAutoresizingMaskIntoConstraints = false
  2278. badgeField.widthAnchor.constraint(equalToConstant: 42).isActive = true
  2279. row.addArrangedSubview(NSView())
  2280. row.addArrangedSubview(badgeField)
  2281. }
  2282. rowHost.addSubview(row)
  2283. NSLayoutConstraint.activate([
  2284. row.leadingAnchor.constraint(equalTo: rowHost.leadingAnchor, constant: 10),
  2285. row.trailingAnchor.constraint(equalTo: rowHost.trailingAnchor, constant: -10),
  2286. row.topAnchor.constraint(equalTo: rowHost.topAnchor, constant: 8),
  2287. row.bottomAnchor.constraint(equalTo: rowHost.bottomAnchor, constant: -8)
  2288. ])
  2289. rowHost.setContentHuggingPriority(.defaultLow, for: .horizontal)
  2290. sidebar.addArrangedSubview(rowHost)
  2291. let sidebarHorizontalInset = sidebar.edgeInsets.left + sidebar.edgeInsets.right
  2292. rowHost.widthAnchor.constraint(equalTo: sidebar.widthAnchor, constant: -sidebarHorizontalInset).isActive = true
  2293. if item.title == "Home" {
  2294. addIndeedSidebarLaunchRow()
  2295. }
  2296. }
  2297. let sidebarBottomSpacer = NSView()
  2298. sidebarBottomSpacer.translatesAutoresizingMaskIntoConstraints = false
  2299. sidebarBottomSpacer.setContentHuggingPriority(.defaultLow, for: .vertical)
  2300. sidebarBottomSpacer.setContentCompressionResistancePriority(.defaultLow, for: .vertical)
  2301. sidebar.addArrangedSubview(sidebarBottomSpacer)
  2302. let upgradeCard = NSView()
  2303. upgradeCard.translatesAutoresizingMaskIntoConstraints = false
  2304. upgradeCard.wantsLayer = true
  2305. upgradeCard.layer?.backgroundColor = Theme.proCardFill.cgColor
  2306. upgradeCard.layer?.cornerRadius = 14
  2307. upgradeCard.layer?.borderWidth = 1
  2308. upgradeCard.layer?.borderColor = Theme.proCardBorder.cgColor
  2309. upgradeCard.layer?.masksToBounds = true
  2310. let accentBar = NSView()
  2311. accentBar.translatesAutoresizingMaskIntoConstraints = false
  2312. accentBar.wantsLayer = true
  2313. accentBar.layer?.backgroundColor = Theme.proAccent.cgColor
  2314. let inner = NSStackView()
  2315. inner.translatesAutoresizingMaskIntoConstraints = false
  2316. inner.orientation = .vertical
  2317. inner.spacing = 10
  2318. inner.alignment = .centerX
  2319. let proIcon = NSImageView()
  2320. proIcon.translatesAutoresizingMaskIntoConstraints = false
  2321. proIcon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 13, weight: .semibold)
  2322. proIcon.image = NSImage(systemSymbolName: "sparkles", accessibilityDescription: nil)
  2323. proIcon.contentTintColor = Theme.proAccent
  2324. let proEyebrow = NSTextField(labelWithString: "Premium")
  2325. proEyebrow.font = .systemFont(ofSize: 11, weight: .heavy)
  2326. proEyebrow.textColor = Theme.proAccent
  2327. proEyebrow.alignment = .center
  2328. let eyebrowRow = NSStackView(views: [proIcon, proEyebrow])
  2329. eyebrowRow.orientation = .horizontal
  2330. eyebrowRow.spacing = 6
  2331. eyebrowRow.alignment = .centerY
  2332. let headline = NSTextField(labelWithString: "Upgrade to Pro")
  2333. headline.font = .systemFont(ofSize: 16, weight: .bold)
  2334. headline.textColor = Theme.primaryText
  2335. headline.alignment = .center
  2336. let upgradeDescription = NSTextField(wrappingLabelWithString: "Unlimited AI matches, smart alerts, and interview prep—all in one place.")
  2337. upgradeDescription.font = .systemFont(ofSize: 12, weight: .regular)
  2338. upgradeDescription.textColor = Theme.secondaryText
  2339. upgradeDescription.alignment = .center
  2340. // Sidebar content width is the fixed sidebar width minus horizontal edge insets; card must stay within that band.
  2341. let cardWidth: CGFloat = 186
  2342. let innerContentWidth = cardWidth - 28
  2343. upgradeDescription.preferredMaxLayoutWidth = innerContentWidth
  2344. let upgradeButton = HoverableButton(title: "Try Pro", target: self, action: #selector(didTapUpgradeToPro))
  2345. upgradeButton.isBordered = false
  2346. upgradeButton.bezelStyle = .rounded
  2347. upgradeButton.font = .systemFont(ofSize: 13, weight: .bold)
  2348. upgradeButton.contentTintColor = Theme.proCTAText
  2349. upgradeButton.alignment = .center
  2350. upgradeButton.wantsLayer = true
  2351. upgradeButton.layer?.backgroundColor = Theme.proCTABackground.cgColor
  2352. upgradeButton.layer?.cornerRadius = 8
  2353. upgradeButton.translatesAutoresizingMaskIntoConstraints = false
  2354. upgradeButton.heightAnchor.constraint(equalToConstant: 32).isActive = true
  2355. upgradeButton.pointerCursor = true
  2356. upgradeButton.hoverHandler = { [weak upgradeButton] hovering in
  2357. upgradeButton?.layer?.backgroundColor = (hovering ? Theme.brandBlueHover : Theme.proCTABackground).cgColor
  2358. }
  2359. inner.addArrangedSubview(eyebrowRow)
  2360. inner.addArrangedSubview(headline)
  2361. inner.addArrangedSubview(upgradeDescription)
  2362. inner.addArrangedSubview(upgradeButton)
  2363. upgradeCard.addSubview(accentBar)
  2364. upgradeCard.addSubview(inner)
  2365. NSLayoutConstraint.activate([
  2366. upgradeCard.widthAnchor.constraint(equalToConstant: cardWidth),
  2367. accentBar.topAnchor.constraint(equalTo: upgradeCard.topAnchor),
  2368. accentBar.leadingAnchor.constraint(equalTo: upgradeCard.leadingAnchor),
  2369. accentBar.trailingAnchor.constraint(equalTo: upgradeCard.trailingAnchor),
  2370. accentBar.heightAnchor.constraint(equalToConstant: 2),
  2371. inner.leadingAnchor.constraint(equalTo: upgradeCard.leadingAnchor, constant: 14),
  2372. inner.trailingAnchor.constraint(equalTo: upgradeCard.trailingAnchor, constant: -14),
  2373. inner.topAnchor.constraint(equalTo: accentBar.bottomAnchor, constant: 12),
  2374. inner.bottomAnchor.constraint(equalTo: upgradeCard.bottomAnchor, constant: -14),
  2375. upgradeButton.widthAnchor.constraint(equalTo: inner.widthAnchor)
  2376. ])
  2377. let upgradeCardHost = NSView()
  2378. upgradeCardHost.translatesAutoresizingMaskIntoConstraints = false
  2379. upgradeCardHost.addSubview(upgradeCard)
  2380. NSLayoutConstraint.activate([
  2381. upgradeCard.centerXAnchor.constraint(equalTo: upgradeCardHost.centerXAnchor),
  2382. upgradeCard.topAnchor.constraint(equalTo: upgradeCardHost.topAnchor),
  2383. upgradeCard.bottomAnchor.constraint(equalTo: upgradeCardHost.bottomAnchor),
  2384. upgradeCard.leadingAnchor.constraint(greaterThanOrEqualTo: upgradeCardHost.leadingAnchor),
  2385. upgradeCard.trailingAnchor.constraint(lessThanOrEqualTo: upgradeCardHost.trailingAnchor)
  2386. ])
  2387. sidebar.addArrangedSubview(upgradeCardHost)
  2388. let upgradeCardHorizontalInset = sidebar.edgeInsets.left + sidebar.edgeInsets.right
  2389. upgradeCardHost.widthAnchor.constraint(equalTo: sidebar.widthAnchor, constant: -upgradeCardHorizontalInset).isActive = true
  2390. sidebarUpgradeCard = upgradeCard
  2391. sidebarUpgradeHeadline = headline
  2392. sidebarUpgradeDescription = upgradeDescription
  2393. sidebarUpgradeButton = upgradeButton
  2394. applyProSubscriptionToSidebar()
  2395. }
  2396. @objc private func didTapUpgradeToPro() {
  2397. Task { @MainActor in
  2398. await SubscriptionStore.shared.refreshEntitlements(deep: true)
  2399. applyProSubscriptionToSidebar()
  2400. presentPremiumPlansSheet()
  2401. }
  2402. }
  2403. private func selectSidebarItem(at index: Int) {
  2404. let leavingIndeedSelection = isIndeedSidebarSelected
  2405. isIndeedSidebarSelected = false
  2406. dismissIndeedJobBrowserEmbedded()
  2407. guard index >= 0, index < currentSidebarItems.count else { return }
  2408. let selectingHome = isHomeSidebarIndex(index)
  2409. if index == selectedSidebarIndex, !leavingIndeedSelection {
  2410. if selectingHome {
  2411. applyHomeState()
  2412. }
  2413. return
  2414. }
  2415. selectedSidebarIndex = index
  2416. configureSidebar()
  2417. updateMainContentVisibility()
  2418. if selectingHome {
  2419. applyHomeState()
  2420. }
  2421. }
  2422. }
  2423. private struct ChatMessage: Codable {
  2424. let role: String
  2425. let content: String
  2426. }
  2427. private final class OpenAIJobSearchService {
  2428. private let endpoint = URL(string: "https://api.openai.com/v1/responses")!
  2429. private let session = URLSession(configuration: .ephemeral)
  2430. func searchJobs(query: String, conversation: [ChatMessage], maxJobs: Int, completion: @escaping (Result<JobSearchOutput, Error>) -> Void) {
  2431. let jobLimit = max(1, min(maxJobs, 25))
  2432. let apiKey = OpenAIConfiguration.apiKey
  2433. guard OpenAIConfiguration.hasAPIKey else {
  2434. completion(.failure(NSError(
  2435. domain: "OpenAIJobSearchService",
  2436. code: 1,
  2437. userInfo: [NSLocalizedDescriptionKey: "Missing API key. Set OPENAI_API_KEY in Xcode Build Settings."]
  2438. )))
  2439. return
  2440. }
  2441. var request = URLRequest(url: endpoint)
  2442. request.httpMethod = "POST"
  2443. request.setValue("application/json", forHTTPHeaderField: "Content-Type")
  2444. request.setValue("Bearer \(apiKey)", forHTTPHeaderField: "Authorization")
  2445. request.timeoutInterval = 45
  2446. let recentContext = conversation.suffix(8)
  2447. .map { "\($0.role.uppercased()): \($0.content)" }
  2448. .joined(separator: "\n")
  2449. let contextBlock: String
  2450. if recentContext.isEmpty {
  2451. contextBlock = "No prior conversation context."
  2452. } else {
  2453. contextBlock = recentContext
  2454. }
  2455. let developerInstructions = """
  2456. You are the job-search backend for an Indeed-focused desktop app. Always use web search, but only to discover roles that are listed on Indeed (indeed.com and regional Indeed sites such as indeed.co.uk, ca.indeed.com, etc.).
  2457. Do not include jobs sourced only from LinkedIn, Glassdoor, company career pages (unless the same opening is clearly on Indeed with an Indeed URL), Google Jobs aggregates, or other job boards.
  2458. Your final assistant message must be JSON that strictly matches the configured response schema (one object with a "jobs" array). Do not add markdown, code fences, or conversational prose outside that JSON.
  2459. Each job entry needs a title, a single-sentence description, and a "url" string. Prefer a stable Indeed **search results** URL on the correct regional domain (path `/jobs` with a `q=` query built from the listing title, company, and location—e.g. `https://www.indeed.com/jobs?q=…&l=…`). Never invent or guess job keys (`jk=`), click IDs, or viewjob URLs you did not copy exactly from live search results—wrong permalinks show 404 inside the app. Use an empty string for "url" only when you cannot construct any Indeed URL (omit the listing if it is not on Indeed).
  2460. Return at most \(jobLimit) distinct listings. If the conversation context already lists jobs, do not repeat the same titles or URLs when the user asks for more—it is fine to return fewer than \(jobLimit) new results.
  2461. Full sentences such as "looking for an AI developer job" are still job queries: always populate "jobs" from Indeed-oriented web search rather than answering with chatty text alone.
  2462. """
  2463. let userInput = """
  2464. Continue this same job-search conversation. Use prior context when useful. The line "Latest user query" is the primary task; earlier USER/ASSISTANT lines are context (previous role, location, or results).
  2465. Conversation context:
  2466. \(contextBlock)
  2467. Latest user query: "\(query)"
  2468. If the user refines pay, seniority, work location, or similar, run a new Indeed-focused search that applies those constraints to the same career topic as in the context.
  2469. """
  2470. let payload = OpenAIJobSearchAPIRequest.jobSearchPayload(
  2471. model: "gpt-4o-mini",
  2472. instructions: developerInstructions,
  2473. input: userInput,
  2474. tools: [OpenAIResponsesTool(type: "web_search_preview")],
  2475. jobLimit: jobLimit
  2476. )
  2477. do {
  2478. request.httpBody = try JSONEncoder().encode(payload)
  2479. } catch {
  2480. completion(.failure(error))
  2481. return
  2482. }
  2483. session.dataTask(with: request) { data, response, error in
  2484. if let error {
  2485. completion(.failure(error))
  2486. return
  2487. }
  2488. guard let data else {
  2489. completion(.failure(NSError(
  2490. domain: "OpenAIJobSearchService",
  2491. code: 2,
  2492. userInfo: [NSLocalizedDescriptionKey: "API returned an empty response."]
  2493. )))
  2494. return
  2495. }
  2496. if let http = response as? HTTPURLResponse, !(200...299).contains(http.statusCode) {
  2497. if let apiError = try? JSONDecoder().decode(OpenAIAPIErrorResponse.self, from: data) {
  2498. completion(.failure(NSError(
  2499. domain: "OpenAIJobSearchService",
  2500. code: http.statusCode,
  2501. userInfo: [NSLocalizedDescriptionKey: apiError.error.message]
  2502. )))
  2503. } else {
  2504. completion(.failure(NSError(
  2505. domain: "OpenAIJobSearchService",
  2506. code: http.statusCode,
  2507. userInfo: [NSLocalizedDescriptionKey: "Job search request failed with status \(http.statusCode)."]
  2508. )))
  2509. }
  2510. return
  2511. }
  2512. do {
  2513. let modelText = try Self.extractModelTextFromResponsesBody(data)
  2514. let trimmed = modelText.trimmingCharacters(in: .whitespacesAndNewlines)
  2515. guard !trimmed.isEmpty else {
  2516. throw NSError(
  2517. domain: "OpenAIJobSearchService",
  2518. code: 4,
  2519. userInfo: [NSLocalizedDescriptionKey: "The API returned an empty text payload."]
  2520. )
  2521. }
  2522. let jobs = try Self.parseJobListings(fromModelText: trimmed)
  2523. .filter(Self.jobListingUsesIndeedOrEmptyURL)
  2524. .map(Self.normalizedJobListing)
  2525. completion(.success(JobSearchOutput(jobs: jobs)))
  2526. } catch {
  2527. completion(.failure(error))
  2528. }
  2529. }.resume()
  2530. }
  2531. /// Walks the `/v1/responses` JSON without strict Codable so tool calls, refusals, and future output item types do not break decoding. Collects only `output_text` segments from assistant `message` items (and any other output items that expose a `content` array).
  2532. private static func extractModelTextFromResponsesBody(_ data: Data) throws -> String {
  2533. let rootObject: Any
  2534. do {
  2535. rootObject = try JSONSerialization.jsonObject(with: data, options: [])
  2536. } catch {
  2537. throw NSError(
  2538. domain: "OpenAIJobSearchService",
  2539. code: 5,
  2540. userInfo: [NSLocalizedDescriptionKey: "The job search service returned data that was not valid JSON."]
  2541. )
  2542. }
  2543. guard let root = rootObject as? [String: Any] else {
  2544. throw NSError(
  2545. domain: "OpenAIJobSearchService",
  2546. code: 5,
  2547. userInfo: [NSLocalizedDescriptionKey: "Unexpected response shape from the job search service."]
  2548. )
  2549. }
  2550. if let status = root["status"] as? String {
  2551. if status == "failed" {
  2552. let message = (root["error"] as? [String: Any])?["message"] as? String ?? "The search request failed."
  2553. throw NSError(domain: "OpenAIJobSearchService", code: 7, userInfo: [NSLocalizedDescriptionKey: message])
  2554. }
  2555. if status == "incomplete",
  2556. let details = root["incomplete_details"] as? [String: Any],
  2557. let reason = details["reason"] as? String {
  2558. throw NSError(
  2559. domain: "OpenAIJobSearchService",
  2560. code: 8,
  2561. userInfo: [NSLocalizedDescriptionKey: "Search stopped early (\(reason)). Try a simpler query or try again."]
  2562. )
  2563. }
  2564. }
  2565. if let direct = root["output_text"] as? String {
  2566. let trimmed = direct.trimmingCharacters(in: .whitespacesAndNewlines)
  2567. if !trimmed.isEmpty { return trimmed }
  2568. }
  2569. guard let output = root["output"] as? [Any] else {
  2570. throw NSError(
  2571. domain: "OpenAIJobSearchService",
  2572. code: 9,
  2573. userInfo: [NSLocalizedDescriptionKey: "The search service returned no assistant text. Try again in a moment."]
  2574. )
  2575. }
  2576. var segments: [String] = []
  2577. for case let item as [String: Any] in output where (item["type"] as? String) == "message" {
  2578. collectOutputTextSegments(fromOutputItem: item, into: &segments)
  2579. }
  2580. if segments.isEmpty {
  2581. for case let item as [String: Any] in output {
  2582. collectOutputTextSegments(fromOutputItem: item, into: &segments)
  2583. }
  2584. }
  2585. let combined = segments.joined(separator: "\n").trimmingCharacters(in: .whitespacesAndNewlines)
  2586. if !combined.isEmpty {
  2587. return combined
  2588. }
  2589. throw NSError(
  2590. domain: "OpenAIJobSearchService",
  2591. code: 9,
  2592. userInfo: [NSLocalizedDescriptionKey: "The model did not return readable job-search text. Try again."]
  2593. )
  2594. }
  2595. private static func collectOutputTextSegments(fromOutputItem item: [String: Any], into segments: inout [String]) {
  2596. guard let content = item["content"] as? [Any] else { return }
  2597. for case let part as [String: Any] in content {
  2598. guard (part["type"] as? String) == "output_text" else { continue }
  2599. if let s = part["text"] as? String {
  2600. segments.append(s)
  2601. } else if let nested = part["text"] as? [String: Any], let value = nested["value"] as? String {
  2602. segments.append(value)
  2603. }
  2604. }
  2605. }
  2606. private static func normalizedJobListing(_ job: JobListing) -> JobListing {
  2607. let trimmedURL = job.url?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
  2608. if trimmedURL.isEmpty {
  2609. return JobListing(title: job.title, description: job.description, url: nil)
  2610. }
  2611. return JobListing(title: job.title, description: job.description, url: trimmedURL)
  2612. }
  2613. /// Drops listings whose `url` points at non-Indeed sites (e.g. LinkedIn) when the model ignores instructions.
  2614. private static func jobListingUsesIndeedOrEmptyURL(_ job: JobListing) -> Bool {
  2615. let trimmed = job.url?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
  2616. if trimmed.isEmpty { return true }
  2617. return isIndeedJobURL(trimmed)
  2618. }
  2619. /// Host looks like an official Indeed property (`indeed.com`, `www.indeed.co.uk`, `ca.indeed.com`, …), not `notindeed.com`.
  2620. private static func isIndeedJobURL(_ string: String) -> Bool {
  2621. guard let host = URL(string: string)?.host?.lowercased() else { return false }
  2622. if host == "indeed.com" { return true }
  2623. if host.hasPrefix("indeed.") { return true }
  2624. return host.contains(".indeed.")
  2625. }
  2626. private static func parseJobListings(fromModelText text: String) throws -> [JobListing] {
  2627. let stripped = text.trimmingCharacters(in: .whitespacesAndNewlines)
  2628. if stripped.hasPrefix("{"), let directData = stripped.data(using: .utf8),
  2629. let payload = try? JSONDecoder().decode(JobSearchResultsPayload.self, from: directData) {
  2630. return payload.jobs
  2631. }
  2632. let jsonString = extractJobJSONObjectString(from: text) ?? extractJSONObject(from: text)
  2633. let jsonData = Data(jsonString.utf8)
  2634. if let payload = try? JSONDecoder().decode(JobSearchResultsPayload.self, from: jsonData) {
  2635. return payload.jobs
  2636. }
  2637. if let listings = try? JSONDecoder().decode([JobListing].self, from: jsonData) {
  2638. return listings
  2639. }
  2640. if let obj = try? JSONSerialization.jsonObject(with: jsonData, options: []) {
  2641. if let dict = obj as? [String: Any], let jobs = jobListings(fromFlexibleJSONObject: dict) {
  2642. return jobs
  2643. }
  2644. if let arr = obj as? [[String: Any]], let jobs = jobListings(fromFlexibleJobArray: arr) {
  2645. return jobs
  2646. }
  2647. }
  2648. throw NSError(
  2649. domain: "OpenAIJobSearchService",
  2650. code: 10,
  2651. userInfo: [NSLocalizedDescriptionKey: "The assistant reply did not include job listings in the expected JSON format. Try your search again."]
  2652. )
  2653. }
  2654. private static func jobListings(fromFlexibleJSONObject dict: [String: Any]) -> [JobListing]? {
  2655. for (key, value) in dict {
  2656. guard key.caseInsensitiveCompare("jobs") == .orderedSame, let arr = value as? [[String: Any]] else { continue }
  2657. if let jobs = jobListings(fromFlexibleJobArray: arr) { return jobs }
  2658. }
  2659. for wrapKey in ["data", "result", "results", "payload"] {
  2660. if let inner = dict[wrapKey] as? [String: Any], let nested = jobListings(fromFlexibleJSONObject: inner) {
  2661. return nested
  2662. }
  2663. }
  2664. return nil
  2665. }
  2666. private static func jobListings(fromFlexibleJobArray jobs: [[String: Any]]) -> [JobListing]? {
  2667. var out: [JobListing] = []
  2668. for item in jobs {
  2669. guard let title = firstString(valuesForKeys: ["title", "job_title", "name", "position"], in: item)?.trimmingCharacters(in: .whitespacesAndNewlines),
  2670. !title.isEmpty,
  2671. let desc = firstString(valuesForKeys: ["description", "snippet", "summary", "desc"], in: item)?.trimmingCharacters(in: .whitespacesAndNewlines),
  2672. !desc.isEmpty else { continue }
  2673. let urlRaw = firstString(valuesForKeys: ["url", "link", "apply_url", "job_url"], in: item)?.trimmingCharacters(in: .whitespacesAndNewlines)
  2674. let url: String? = (urlRaw?.isEmpty == true) ? nil : urlRaw
  2675. out.append(JobListing(title: title, description: desc, url: url))
  2676. }
  2677. return out.isEmpty ? nil : out
  2678. }
  2679. private static func firstString(valuesForKeys keys: [String], in dict: [String: Any]) -> String? {
  2680. for wanted in keys {
  2681. for (dk, dv) in dict {
  2682. guard dk.caseInsensitiveCompare(wanted) == .orderedSame, let s = dv as? String else { continue }
  2683. return s
  2684. }
  2685. }
  2686. return nil
  2687. }
  2688. private static func stripMarkdownCodeFence(_ text: String) -> String {
  2689. var s = text.trimmingCharacters(in: .whitespacesAndNewlines)
  2690. guard s.hasPrefix("```") else { return s }
  2691. s.removeFirst(3)
  2692. if s.lowercased().hasPrefix("json") {
  2693. s.removeFirst(4)
  2694. }
  2695. s = s.trimmingCharacters(in: .whitespacesAndNewlines)
  2696. if let fence = s.range(of: "```", options: .backwards) {
  2697. s = String(s[..<fence.lowerBound])
  2698. }
  2699. return s.trimmingCharacters(in: .whitespacesAndNewlines)
  2700. }
  2701. private static func balancedJSONObject(from openBrace: String.Index, in s: String) -> String? {
  2702. var depth = 0
  2703. var inString = false
  2704. var escaped = false
  2705. var i = openBrace
  2706. while i < s.endIndex {
  2707. let ch = s[i]
  2708. if inString {
  2709. if escaped {
  2710. escaped = false
  2711. } else if ch == "\\" {
  2712. escaped = true
  2713. } else if ch == "\"" {
  2714. inString = false
  2715. }
  2716. } else {
  2717. switch ch {
  2718. case "\"":
  2719. inString = true
  2720. case "{":
  2721. depth += 1
  2722. case "}":
  2723. depth -= 1
  2724. if depth == 0 {
  2725. return String(s[openBrace...i])
  2726. }
  2727. default:
  2728. break
  2729. }
  2730. }
  2731. i = s.index(after: i)
  2732. }
  2733. return nil
  2734. }
  2735. /// Prefers the JSON object that contains a `"jobs"` key so prose before/after the payload does not confuse the decoder.
  2736. private static func extractJobJSONObjectString(from text: String) -> String? {
  2737. let s = stripMarkdownCodeFence(text)
  2738. guard let jobsRange = s.range(of: "\"jobs\"", options: .caseInsensitive) else { return nil }
  2739. let head = s[..<jobsRange.lowerBound]
  2740. guard let open = head.lastIndex(of: "{") else { return nil }
  2741. return balancedJSONObject(from: open, in: s)
  2742. }
  2743. private static func extractJSONObject(from text: String) -> String {
  2744. if let extracted = extractJobJSONObjectString(from: text) {
  2745. return extracted
  2746. }
  2747. let stripped = stripMarkdownCodeFence(text)
  2748. if let first = stripped.firstIndex(of: "{"), let balanced = balancedJSONObject(from: first, in: stripped) {
  2749. return balanced
  2750. }
  2751. if let range = text.range(of: "\\{[\\s\\S]*\\}", options: .regularExpression) {
  2752. return String(text[range])
  2753. }
  2754. return text
  2755. }
  2756. }
  2757. /// Responses API request with structured JSON output so web-search replies cannot omit the `jobs` schema.
  2758. private struct OpenAIJobSearchAPIRequest: Encodable {
  2759. let model: String
  2760. let instructions: String
  2761. let input: String
  2762. let tools: [OpenAIResponsesTool]
  2763. let text: OpenAITextOutputConfig
  2764. static func jobSearchPayload(
  2765. model: String,
  2766. instructions: String,
  2767. input: String,
  2768. tools: [OpenAIResponsesTool],
  2769. jobLimit: Int
  2770. ) -> OpenAIJobSearchAPIRequest {
  2771. let itemProperties = OpenAIJobSearchJobItemProperties(
  2772. title: OpenAIJSONSchemaStringField(type: "string", description: "Job title as shown on the Indeed listing."),
  2773. description: OpenAIJSONSchemaStringField(type: "string", description: "One concise sentence summarizing the role from the Indeed posting."),
  2774. url: OpenAIJSONSchemaStringField(type: "string", description: "Indeed search URL (https, path /jobs, query q=…) on indeed.com or the correct regional Indeed host; never fabricate viewjob/jk links. Empty string only if no Indeed URL exists—never use LinkedIn, Glassdoor, or other boards.")
  2775. )
  2776. let itemSchema = OpenAIJobSearchJobItemSchema(
  2777. type: "object",
  2778. properties: itemProperties,
  2779. required: ["title", "description", "url"],
  2780. additionalProperties: false
  2781. )
  2782. let jobsProperty = OpenAIJobSearchJobsArrayProperty(
  2783. type: "array",
  2784. description: "Up to \(jobLimit) jobs found on Indeed via web search; use an empty array if none are found. Do not include listings that only exist off Indeed.",
  2785. items: itemSchema
  2786. )
  2787. let rootProperties = OpenAIJobSearchRootProperties(jobs: jobsProperty)
  2788. let rootSchema = OpenAIJobSearchRootSchema(
  2789. type: "object",
  2790. properties: rootProperties,
  2791. required: ["jobs"],
  2792. additionalProperties: false
  2793. )
  2794. let format = OpenAIJobSearchResponseJSONSchemaFormat(
  2795. type: "json_schema",
  2796. name: "job_search_results",
  2797. strict: true,
  2798. schema: rootSchema
  2799. )
  2800. return OpenAIJobSearchAPIRequest(
  2801. model: model,
  2802. instructions: instructions,
  2803. input: input,
  2804. tools: tools,
  2805. text: OpenAITextOutputConfig(format: format)
  2806. )
  2807. }
  2808. }
  2809. private struct OpenAITextOutputConfig: Encodable {
  2810. let format: OpenAIJobSearchResponseJSONSchemaFormat
  2811. }
  2812. private struct OpenAIJobSearchResponseJSONSchemaFormat: Encodable {
  2813. let type: String
  2814. let name: String
  2815. let strict: Bool
  2816. let schema: OpenAIJobSearchRootSchema
  2817. }
  2818. private struct OpenAIJobSearchRootSchema: Encodable {
  2819. let type: String
  2820. let properties: OpenAIJobSearchRootProperties
  2821. let required: [String]
  2822. let additionalProperties: Bool
  2823. }
  2824. private struct OpenAIJobSearchRootProperties: Encodable {
  2825. let jobs: OpenAIJobSearchJobsArrayProperty
  2826. }
  2827. private struct OpenAIJobSearchJobsArrayProperty: Encodable {
  2828. let type: String
  2829. let description: String
  2830. let items: OpenAIJobSearchJobItemSchema
  2831. }
  2832. private struct OpenAIJobSearchJobItemSchema: Encodable {
  2833. let type: String
  2834. let properties: OpenAIJobSearchJobItemProperties
  2835. let required: [String]
  2836. let additionalProperties: Bool
  2837. }
  2838. private struct OpenAIJobSearchJobItemProperties: Encodable {
  2839. let title: OpenAIJSONSchemaStringField
  2840. let description: OpenAIJSONSchemaStringField
  2841. let url: OpenAIJSONSchemaStringField
  2842. }
  2843. private struct OpenAIJSONSchemaStringField: Encodable {
  2844. let type: String
  2845. let description: String
  2846. }
  2847. private struct OpenAIResponsesTool: Codable {
  2848. let type: String
  2849. }
  2850. private struct JobSearchResultsPayload: Codable {
  2851. let jobs: [JobListing]
  2852. }
  2853. private struct JobSearchOutput {
  2854. let jobs: [JobListing]
  2855. }
  2856. private struct OpenAIAPIErrorResponse: Codable {
  2857. let error: APIError
  2858. struct APIError: Codable {
  2859. let message: String
  2860. }
  2861. }
  2862. /// Decorative waves and faint sparkles behind the welcome hero (reference layout).
  2863. private final class WelcomeHeroBackgroundView: NSView {
  2864. /// Stroke color for side waves (pastel blue).
  2865. var waveTint = NSColor(srgbRed: 186 / 255, green: 210 / 255, blue: 253 / 255, alpha: 1)
  2866. override var isFlipped: Bool { true }
  2867. override func draw(_ dirtyRect: NSRect) {
  2868. NSColor.clear.setFill()
  2869. bounds.fill()
  2870. guard bounds.width > 24, bounds.height > 24 else { return }
  2871. drawSideWaves(in: bounds, isLeft: true)
  2872. drawSideWaves(in: bounds, isLeft: false)
  2873. drawAmbientSparkles(in: bounds)
  2874. }
  2875. private func drawSideWaves(in bounds: NSRect, isLeft: Bool) {
  2876. for i in 0..<9 {
  2877. let path = NSBezierPath()
  2878. path.lineWidth = 1
  2879. path.lineCapStyle = .round
  2880. let phase = CGFloat(i) * 0.88
  2881. let base = CGFloat(i + 1) * 11 + 4
  2882. var first = true
  2883. for y in stride(from: CGFloat(0), through: bounds.height, by: 2.8) {
  2884. let wobble = sin(y * 0.048 + phase) * (4 + CGFloat(i % 5))
  2885. let x = isLeft ? (base + wobble) : (bounds.width - base - wobble)
  2886. let point = NSPoint(x: x, y: y)
  2887. if first {
  2888. path.move(to: point)
  2889. first = false
  2890. } else {
  2891. path.line(to: point)
  2892. }
  2893. }
  2894. let fade = 1 - CGFloat(i) / 10
  2895. waveTint.withAlphaComponent((0.09 + CGFloat(i % 3) * 0.022) * fade).setStroke()
  2896. path.stroke()
  2897. }
  2898. }
  2899. private func drawAmbientSparkles(in bounds: NSRect) {
  2900. let accent = NSColor(srgbRed: 0, green: 82 / 255, blue: 204 / 255, alpha: 1)
  2901. let specs: [(CGFloat, CGFloat, CGFloat, CGFloat)] = [
  2902. (0.06, 0.14, 9, 0.28),
  2903. (0.94, 0.12, 12, 0.34),
  2904. (0.18, 0.42, 5, 0.18),
  2905. (0.86, 0.44, 6, 0.2),
  2906. (0.5, 0.06, 7, 0.15)
  2907. ]
  2908. for (nx, ny, size, a) in specs {
  2909. let center = NSPoint(x: bounds.width * nx, y: bounds.height * ny)
  2910. fillFourPointStar(center: center, radius: size, color: accent.withAlphaComponent(a))
  2911. }
  2912. }
  2913. private func fillFourPointStar(center: NSPoint, radius: CGFloat, color: NSColor) {
  2914. let path = NSBezierPath()
  2915. for i in 0..<4 {
  2916. let angle = CGFloat(i) * .pi / 2 - .pi / 2
  2917. let x = center.x + cos(angle) * radius
  2918. let y = center.y + sin(angle) * radius
  2919. let point = NSPoint(x: x, y: y)
  2920. if i == 0 {
  2921. path.move(to: point)
  2922. } else {
  2923. path.line(to: point)
  2924. }
  2925. }
  2926. path.close()
  2927. color.setFill()
  2928. path.fill()
  2929. }
  2930. }
  2931. /// Home welcome row: three tappable shortcuts that seed the main search field (reference: white cards, pastel icon well, arrow at bottom trailing).
  2932. private final class FeatureShortcutCardView: NSView {
  2933. private static let cardCornerRadius: CGFloat = 14
  2934. private static let primaryBlue = NSColor(srgbRed: 0, green: 82 / 255, blue: 204 / 255, alpha: 1)
  2935. private static let defaultBorderColor = NSColor(srgbRed: 237 / 255, green: 242 / 255, blue: 247 / 255, alpha: 1)
  2936. var isSelected = false { didSet { updateSelectionAppearance() } }
  2937. private weak var actionTarget: AnyObject?
  2938. private var actionSelector: Selector
  2939. init(symbolName: String, title: String, subtitle: String, target: AnyObject?, action: Selector) {
  2940. self.actionTarget = target
  2941. self.actionSelector = action
  2942. super.init(frame: .zero)
  2943. translatesAutoresizingMaskIntoConstraints = false
  2944. wantsLayer = true
  2945. layer?.cornerRadius = Self.cardCornerRadius
  2946. if #available(macOS 11.0, *) {
  2947. layer?.cornerCurve = .continuous
  2948. }
  2949. layer?.backgroundColor = NSColor.white.cgColor
  2950. layer?.masksToBounds = false
  2951. layer?.shadowColor = NSColor.black.withAlphaComponent(0.06).cgColor
  2952. layer?.shadowOffset = CGSize(width: 0, height: 2)
  2953. layer?.shadowRadius = 12
  2954. layer?.shadowOpacity = 1
  2955. updateSelectionAppearance()
  2956. let primaryBlue = Self.primaryBlue
  2957. // `#EBF2FF` — circular icon well.
  2958. let iconWellColor = NSColor(srgbRed: 235 / 255, green: 242 / 255, blue: 255 / 255, alpha: 1)
  2959. // `#5D6D7E` — muted description.
  2960. let secondary = NSColor(srgbRed: 93 / 255, green: 109 / 255, blue: 126 / 255, alpha: 1)
  2961. let iconSize: CGFloat = 40
  2962. let iconHost = NSView()
  2963. iconHost.translatesAutoresizingMaskIntoConstraints = false
  2964. iconHost.wantsLayer = true
  2965. iconHost.layer?.backgroundColor = iconWellColor.cgColor
  2966. iconHost.layer?.cornerRadius = iconSize / 2
  2967. let icon = NSImageView()
  2968. icon.translatesAutoresizingMaskIntoConstraints = false
  2969. icon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 16, weight: .regular)
  2970. icon.image = NSImage(systemSymbolName: symbolName, accessibilityDescription: nil)
  2971. icon.contentTintColor = primaryBlue
  2972. iconHost.addSubview(icon)
  2973. let titleField = NSTextField(wrappingLabelWithString: title)
  2974. titleField.font = .systemFont(ofSize: 15, weight: .bold)
  2975. titleField.textColor = primaryBlue
  2976. titleField.maximumNumberOfLines = 1
  2977. titleField.isEditable = false
  2978. titleField.isBordered = false
  2979. titleField.drawsBackground = false
  2980. titleField.alignment = .left
  2981. let subtitleField = NSTextField(wrappingLabelWithString: subtitle)
  2982. subtitleField.font = .systemFont(ofSize: 12, weight: .regular)
  2983. subtitleField.textColor = secondary
  2984. subtitleField.maximumNumberOfLines = 2
  2985. subtitleField.isEditable = false
  2986. subtitleField.isBordered = false
  2987. subtitleField.drawsBackground = false
  2988. subtitleField.alignment = .left
  2989. subtitleField.lineBreakMode = .byWordWrapping
  2990. subtitleField.setContentHuggingPriority(.defaultLow, for: .horizontal)
  2991. subtitleField.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  2992. let chevron = NSImageView()
  2993. chevron.translatesAutoresizingMaskIntoConstraints = false
  2994. chevron.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 12, weight: .semibold)
  2995. chevron.image = NSImage(systemSymbolName: "arrow.right", accessibilityDescription: nil)
  2996. chevron.contentTintColor = primaryBlue
  2997. chevron.setContentHuggingPriority(.required, for: .horizontal)
  2998. chevron.setContentCompressionResistancePriority(.required, for: .horizontal)
  2999. let subtitleRow = NSStackView(views: [subtitleField, chevron])
  3000. subtitleRow.orientation = .horizontal
  3001. subtitleRow.spacing = 10
  3002. subtitleRow.alignment = .bottom
  3003. subtitleRow.distribution = .fill
  3004. subtitleRow.translatesAutoresizingMaskIntoConstraints = false
  3005. let textColumn = NSStackView(views: [titleField, subtitleRow])
  3006. textColumn.orientation = .vertical
  3007. textColumn.spacing = 6
  3008. textColumn.alignment = .leading
  3009. textColumn.translatesAutoresizingMaskIntoConstraints = false
  3010. textColumn.setContentHuggingPriority(.defaultLow, for: .horizontal)
  3011. textColumn.setContentCompressionResistancePriority(.defaultLow, for: .horizontal)
  3012. iconHost.setContentHuggingPriority(.required, for: .horizontal)
  3013. iconHost.setContentCompressionResistancePriority(.required, for: .horizontal)
  3014. let row = NSStackView(views: [iconHost, textColumn])
  3015. row.orientation = .horizontal
  3016. row.spacing = 16
  3017. row.alignment = .centerY
  3018. row.distribution = .fill
  3019. row.translatesAutoresizingMaskIntoConstraints = false
  3020. addSubview(row)
  3021. let inset: CGFloat = 16
  3022. NSLayoutConstraint.activate([
  3023. row.leadingAnchor.constraint(equalTo: leadingAnchor, constant: inset),
  3024. row.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -inset),
  3025. row.topAnchor.constraint(equalTo: topAnchor, constant: inset),
  3026. row.bottomAnchor.constraint(equalTo: bottomAnchor, constant: -inset),
  3027. iconHost.widthAnchor.constraint(equalToConstant: iconSize),
  3028. iconHost.heightAnchor.constraint(equalToConstant: iconSize),
  3029. icon.centerXAnchor.constraint(equalTo: iconHost.centerXAnchor),
  3030. icon.centerYAnchor.constraint(equalTo: iconHost.centerYAnchor)
  3031. ])
  3032. setAccessibilityElement(true)
  3033. setAccessibilityRole(.button)
  3034. setAccessibilityLabel("\(title). \(subtitle)")
  3035. }
  3036. private func updateSelectionAppearance() {
  3037. guard let layer else { return }
  3038. if isSelected {
  3039. layer.borderWidth = 2
  3040. layer.borderColor = Self.primaryBlue.cgColor
  3041. layer.shadowOpacity = 0.1
  3042. } else {
  3043. layer.borderWidth = 1
  3044. layer.borderColor = Self.defaultBorderColor.cgColor
  3045. layer.shadowOpacity = 1
  3046. }
  3047. setAccessibilitySelected(isSelected)
  3048. }
  3049. @available(*, unavailable)
  3050. required init?(coder: NSCoder) {
  3051. fatalError("init(coder:) has not been implemented")
  3052. }
  3053. override func layout() {
  3054. super.layout()
  3055. guard let layer = layer, bounds.width > 0, bounds.height > 0 else { return }
  3056. let r = bounds
  3057. let cr = Self.cardCornerRadius
  3058. layer.shadowPath = CGPath(roundedRect: r, cornerWidth: cr, cornerHeight: cr, transform: nil)
  3059. }
  3060. override func mouseDown(with event: NSEvent) {
  3061. if let target = actionTarget {
  3062. _ = target.perform(actionSelector, with: nil)
  3063. }
  3064. }
  3065. override func resetCursorRects() {
  3066. super.resetCursorRects()
  3067. addCursorRect(bounds, cursor: .pointingHand)
  3068. }
  3069. }
  3070. /// `NSButton` that carries a `JobListing` for card actions (`representedObject` is unavailable on `NSButton` in this target).
  3071. private class JobPayloadButton: HoverableButton {
  3072. var jobPayload: JobListing?
  3073. var cardContext: JobListingCardContext = .homeSearchResults
  3074. }
  3075. /// Insets image + title so the Save pill matches the reference (balanced padding, not flush to the stroke).
  3076. private final class SaveJobButtonCell: NSButtonCell {
  3077. private let horizontalInset: CGFloat = 10
  3078. private let verticalInset: CGFloat = 3
  3079. private let imageTitleGap: CGFloat = 5
  3080. override func imageRect(forBounds rect: NSRect) -> NSRect {
  3081. super.imageRect(forBounds: rect.insetBy(dx: horizontalInset, dy: verticalInset))
  3082. }
  3083. override func titleRect(forBounds rect: NSRect) -> NSRect {
  3084. let padded = rect.insetBy(dx: horizontalInset, dy: verticalInset)
  3085. var t = super.titleRect(forBounds: padded)
  3086. t.origin.x += imageTitleGap
  3087. t.size.width = max(0, t.size.width - imageTitleGap)
  3088. return t
  3089. }
  3090. }
  3091. private final class SaveJobPayloadButton: JobPayloadButton {
  3092. override class var cellClass: AnyClass? {
  3093. get { SaveJobButtonCell.self }
  3094. set { }
  3095. }
  3096. }
  3097. /// `NSButton` with a tracking area that reports hover transitions and (optionally) swaps in a pointing-hand cursor while hovered.
  3098. private class HoverableButton: NSButton {
  3099. var hoverHandler: ((Bool) -> Void)?
  3100. var pointerCursor: Bool = false
  3101. private(set) var isHovering: Bool = false
  3102. private var trackingArea: NSTrackingArea?
  3103. private var didPushCursor: Bool = false
  3104. override func updateTrackingAreas() {
  3105. super.updateTrackingAreas()
  3106. if let area = trackingArea { removeTrackingArea(area) }
  3107. let area = NSTrackingArea(
  3108. rect: bounds,
  3109. options: [.activeInKeyWindow, .mouseEnteredAndExited, .inVisibleRect],
  3110. owner: self,
  3111. userInfo: nil
  3112. )
  3113. addTrackingArea(area)
  3114. trackingArea = area
  3115. }
  3116. override func mouseEntered(with event: NSEvent) {
  3117. super.mouseEntered(with: event)
  3118. isHovering = true
  3119. hoverHandler?(true)
  3120. if pointerCursor, !didPushCursor {
  3121. NSCursor.pointingHand.push()
  3122. didPushCursor = true
  3123. }
  3124. }
  3125. override func mouseExited(with event: NSEvent) {
  3126. super.mouseExited(with: event)
  3127. isHovering = false
  3128. hoverHandler?(false)
  3129. if didPushCursor {
  3130. NSCursor.pop()
  3131. didPushCursor = false
  3132. }
  3133. }
  3134. override func viewWillMove(toWindow newWindow: NSWindow?) {
  3135. super.viewWillMove(toWindow: newWindow)
  3136. // Guard against an unbalanced cursor stack if the button is removed mid-hover (e.g. job card replaced after a search).
  3137. if newWindow == nil, didPushCursor {
  3138. NSCursor.pop()
  3139. didPushCursor = false
  3140. isHovering = false
  3141. }
  3142. }
  3143. }
  3144. /// `NSView` companion to `HoverableButton`: emits hover transitions and can manage a pointing-hand cursor. Used to track hover over composite controls like the gradient "Find jobs" pill.
  3145. private class HoverableView: NSView {
  3146. var hoverHandler: ((Bool) -> Void)?
  3147. var pointerCursor: Bool = false
  3148. private(set) var isHovering: Bool = false
  3149. private var trackingArea: NSTrackingArea?
  3150. private var didPushCursor: Bool = false
  3151. override func updateTrackingAreas() {
  3152. super.updateTrackingAreas()
  3153. if let area = trackingArea { removeTrackingArea(area) }
  3154. let area = NSTrackingArea(
  3155. rect: bounds,
  3156. options: [.activeInKeyWindow, .mouseEnteredAndExited, .inVisibleRect],
  3157. owner: self,
  3158. userInfo: nil
  3159. )
  3160. addTrackingArea(area)
  3161. trackingArea = area
  3162. }
  3163. override func mouseEntered(with event: NSEvent) {
  3164. super.mouseEntered(with: event)
  3165. isHovering = true
  3166. hoverHandler?(true)
  3167. if pointerCursor, !didPushCursor {
  3168. NSCursor.pointingHand.push()
  3169. didPushCursor = true
  3170. }
  3171. }
  3172. override func mouseExited(with event: NSEvent) {
  3173. super.mouseExited(with: event)
  3174. isHovering = false
  3175. hoverHandler?(false)
  3176. if didPushCursor {
  3177. NSCursor.pop()
  3178. didPushCursor = false
  3179. }
  3180. }
  3181. override func viewWillMove(toWindow newWindow: NSWindow?) {
  3182. super.viewWillMove(toWindow: newWindow)
  3183. if newWindow == nil, didPushCursor {
  3184. NSCursor.pop()
  3185. didPushCursor = false
  3186. isHovering = false
  3187. }
  3188. }
  3189. }
  3190. /// Document view for the job list `NSScrollView`; flipped coordinates keep short result sets aligned to the top of the clip (avoids a large empty band above the cards on macOS).
  3191. private final class JobListingsDocumentView: NSView {
  3192. override var isFlipped: Bool { true }
  3193. }
  3194. /// Marker subclass for the per-assistant-message job stack embedded inside a chat bubble. `NSView.tag` is read-only on macOS, so a typed subclass is the cleanest way to identify these stacks during dismiss and layout passes.
  3195. private final class ChatJobsStackView: NSStackView {}
  3196. /// Marker subclass for the wrapping label inside a chat bubble. Lets the layout pass find each bubble label to update its `preferredMaxLayoutWidth` when the chat width changes.
  3197. private final class ChatBubbleLabel: NSTextField {}
  3198. /// Captures clicks for the full sidebar pill so icon, label, and padding behave as one tab. Manages its own hover background so non-selected rows highlight subtly on hover without disturbing the selected-row fill.
  3199. private final class SidebarNavRowView: NSView {
  3200. private let onSelect: () -> Void
  3201. var restingBackgroundColor: NSColor? {
  3202. didSet { applyBackground() }
  3203. }
  3204. var hoverBackgroundColor: NSColor?
  3205. private var isHovering: Bool = false
  3206. private var didPushCursor: Bool = false
  3207. init(onSelect: @escaping () -> Void) {
  3208. self.onSelect = onSelect
  3209. super.init(frame: .zero)
  3210. }
  3211. @available(*, unavailable)
  3212. required init?(coder: NSCoder) {
  3213. fatalError("init(coder:) has not been implemented")
  3214. }
  3215. override func hitTest(_ point: NSPoint) -> NSView? {
  3216. guard let superview else { return super.hitTest(point) }
  3217. let local = convert(point, from: superview)
  3218. return bounds.contains(local) ? self : nil
  3219. }
  3220. override func mouseDown(with event: NSEvent) {
  3221. onSelect()
  3222. }
  3223. override func updateTrackingAreas() {
  3224. super.updateTrackingAreas()
  3225. trackingAreas.forEach { removeTrackingArea($0) }
  3226. addTrackingArea(NSTrackingArea(
  3227. rect: bounds,
  3228. options: [.activeInKeyWindow, .mouseEnteredAndExited, .inVisibleRect],
  3229. owner: self,
  3230. userInfo: nil
  3231. ))
  3232. }
  3233. override func mouseEntered(with event: NSEvent) {
  3234. super.mouseEntered(with: event)
  3235. isHovering = true
  3236. applyBackground()
  3237. if !didPushCursor {
  3238. NSCursor.pointingHand.push()
  3239. didPushCursor = true
  3240. }
  3241. }
  3242. override func mouseExited(with event: NSEvent) {
  3243. super.mouseExited(with: event)
  3244. isHovering = false
  3245. applyBackground()
  3246. if didPushCursor {
  3247. NSCursor.pop()
  3248. didPushCursor = false
  3249. }
  3250. }
  3251. override func viewWillMove(toWindow newWindow: NSWindow?) {
  3252. super.viewWillMove(toWindow: newWindow)
  3253. if newWindow == nil, didPushCursor {
  3254. NSCursor.pop()
  3255. didPushCursor = false
  3256. isHovering = false
  3257. }
  3258. }
  3259. private func applyBackground() {
  3260. let color = isHovering ? (hoverBackgroundColor ?? restingBackgroundColor) : restingBackgroundColor
  3261. layer?.backgroundColor = color?.cgColor
  3262. }
  3263. }