Sen descrición

DashboardView.swift 157KB

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