| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452 |
- //
- // DashboardView.swift
- // App for Indeed
- //
- import Cocoa
- final class DashboardView: NSView {
- private enum Theme {
- static let pageBackground = NSColor(calibratedRed: 0.02, green: 0.02, blue: 0.03, alpha: 1)
- static let chromeBackground = NSColor(calibratedRed: 0.05, green: 0.05, blue: 0.06, alpha: 1)
- static let sidebarBackground = NSColor(calibratedRed: 0.07, green: 0.07, blue: 0.09, alpha: 1)
- static let heroBackground = NSColor(calibratedRed: 0.09, green: 0.09, blue: 0.12, alpha: 1)
- static let sectionBackground = NSColor(calibratedRed: 0.08, green: 0.08, blue: 0.1, alpha: 1)
- static let statCardBackground = NSColor(calibratedRed: 0.1, green: 0.1, blue: 0.12, alpha: 1)
- static let insightCardBackground = NSColor(calibratedRed: 0.09, green: 0.09, blue: 0.11, alpha: 1)
- static let accent = NSColor(calibratedRed: 0.3, green: 0.48, blue: 1.0, alpha: 1)
- static let accentMuted = NSColor(calibratedRed: 0.3, green: 0.48, blue: 1.0, alpha: 0.3)
- static let iconBackground = NSColor(calibratedRed: 0.3, green: 0.48, blue: 1.0, alpha: 0.24)
- static let primaryText = NSColor(calibratedRed: 0.95, green: 0.95, blue: 0.97, alpha: 1)
- static let secondaryText = NSColor(calibratedRed: 0.73, green: 0.74, blue: 0.79, alpha: 1)
- static let tertiaryText = NSColor(calibratedRed: 0.56, green: 0.57, blue: 0.62, alpha: 1)
- static let bubbleText = NSColor(calibratedRed: 0.86, green: 0.89, blue: 0.98, alpha: 1)
- static let bubbleBackground = NSColor(calibratedRed: 0.16, green: 0.2, blue: 0.33, alpha: 0.45)
- }
- private let contentStack = NSStackView()
- private let documentContainer = NSView()
- private let chromeContainer = NSView()
- private let sidebar = NSStackView()
- private let mainColumn = NSStackView()
- private let greetingLabel = NSTextField(labelWithString: "")
- private let subtitleLabel = NSTextField(labelWithString: "Find your perfect job with the power of AI.")
- private let heroCard = NSView()
- private let statGrid = NSGridView(views: [[]])
- private let recommendationsStack = NSStackView()
- private let insightsStack = NSStackView()
- private let scrollView = NSScrollView()
- private var recommendationsWidthConstraint: NSLayoutConstraint?
- private var insightsWidthConstraint: NSLayoutConstraint?
- override init(frame frameRect: NSRect) {
- super.init(frame: frameRect)
- setupLayout()
- }
- required init?(coder: NSCoder) {
- super.init(coder: coder)
- setupLayout()
- }
- override func layout() {
- super.layout()
- updateDocumentLayout()
- }
- func render(_ data: DashboardData) {
- greetingLabel.stringValue = "Welcome back, \(data.greetingName)! 👋"
- configureSidebar(data.sidebarItems)
- configureStats(data.stats)
- configureRecommendations(data.recommendations)
- configureInsights(data.insights)
- updateDocumentLayout()
- }
- private func setupLayout() {
- wantsLayer = true
- layer?.backgroundColor = Theme.pageBackground.cgColor
- scrollView.translatesAutoresizingMaskIntoConstraints = false
- scrollView.hasVerticalScroller = true
- scrollView.drawsBackground = false
- addSubview(scrollView)
- contentStack.orientation = .horizontal
- contentStack.spacing = 20
- contentStack.translatesAutoresizingMaskIntoConstraints = false
- contentStack.alignment = .top
- contentStack.edgeInsets = NSEdgeInsets(top: 24, left: 24, bottom: 24, right: 24)
- documentContainer.translatesAutoresizingMaskIntoConstraints = true
- documentContainer.autoresizingMask = [.width]
- documentContainer.frame = NSRect(x: 0, y: 0, width: 1040, height: 900)
- chromeContainer.translatesAutoresizingMaskIntoConstraints = false
- chromeContainer.wantsLayer = true
- chromeContainer.layer?.backgroundColor = Theme.chromeBackground.cgColor
- chromeContainer.layer?.cornerRadius = 18
- documentContainer.addSubview(chromeContainer)
- chromeContainer.addSubview(contentStack)
- scrollView.documentView = documentContainer
- sidebar.orientation = .vertical
- sidebar.spacing = 10
- sidebar.alignment = .leading
- sidebar.wantsLayer = true
- sidebar.layer?.backgroundColor = Theme.sidebarBackground.cgColor
- sidebar.layer?.cornerRadius = 16
- sidebar.edgeInsets = NSEdgeInsets(top: 18, left: 14, bottom: 18, right: 14)
- sidebar.translatesAutoresizingMaskIntoConstraints = false
- mainColumn.orientation = .vertical
- mainColumn.spacing = 14
- mainColumn.alignment = .leading
- mainColumn.translatesAutoresizingMaskIntoConstraints = false
- greetingLabel.font = .systemFont(ofSize: 30, weight: .bold)
- greetingLabel.textColor = Theme.primaryText
- subtitleLabel.font = .systemFont(ofSize: 14, weight: .regular)
- subtitleLabel.textColor = Theme.secondaryText
- heroCard.wantsLayer = true
- heroCard.layer?.backgroundColor = Theme.heroBackground.cgColor
- heroCard.layer?.cornerRadius = 18
- heroCard.translatesAutoresizingMaskIntoConstraints = false
- let hero = buildHeroContent()
- heroCard.addSubview(hero)
- statGrid.translatesAutoresizingMaskIntoConstraints = false
- statGrid.rowSpacing = 10
- statGrid.columnSpacing = 10
- let lowerSection = NSStackView()
- lowerSection.orientation = .horizontal
- lowerSection.spacing = 12
- lowerSection.alignment = .top
- lowerSection.distribution = .fill
- lowerSection.translatesAutoresizingMaskIntoConstraints = false
- let recommendationsBox = sectionBox(title: "Recommended for You", content: recommendationsStack)
- let insightsBox = sectionBox(title: "AI Insights", content: insightsStack)
- recommendationsBox.translatesAutoresizingMaskIntoConstraints = false
- insightsBox.translatesAutoresizingMaskIntoConstraints = false
- lowerSection.addArrangedSubview(recommendationsBox)
- lowerSection.addArrangedSubview(insightsBox)
- mainColumn.addArrangedSubview(greetingLabel)
- mainColumn.addArrangedSubview(subtitleLabel)
- mainColumn.addArrangedSubview(heroCard)
- mainColumn.addArrangedSubview(statGrid)
- mainColumn.addArrangedSubview(lowerSection)
- contentStack.addArrangedSubview(sidebar)
- contentStack.addArrangedSubview(mainColumn)
- NSLayoutConstraint.activate([
- scrollView.leadingAnchor.constraint(equalTo: leadingAnchor),
- scrollView.trailingAnchor.constraint(equalTo: trailingAnchor),
- scrollView.topAnchor.constraint(equalTo: topAnchor),
- scrollView.bottomAnchor.constraint(equalTo: bottomAnchor),
- chromeContainer.topAnchor.constraint(equalTo: documentContainer.topAnchor, constant: 18),
- chromeContainer.bottomAnchor.constraint(equalTo: documentContainer.bottomAnchor, constant: -18),
- chromeContainer.centerXAnchor.constraint(equalTo: documentContainer.centerXAnchor),
- chromeContainer.widthAnchor.constraint(lessThanOrEqualToConstant: 1240),
- chromeContainer.widthAnchor.constraint(equalTo: documentContainer.widthAnchor, constant: -36),
- contentStack.leadingAnchor.constraint(equalTo: chromeContainer.leadingAnchor),
- contentStack.trailingAnchor.constraint(equalTo: chromeContainer.trailingAnchor),
- contentStack.topAnchor.constraint(equalTo: chromeContainer.topAnchor),
- contentStack.bottomAnchor.constraint(equalTo: chromeContainer.bottomAnchor),
- sidebar.widthAnchor.constraint(equalToConstant: 225),
- mainColumn.widthAnchor.constraint(greaterThanOrEqualToConstant: 720),
- heroCard.widthAnchor.constraint(equalTo: mainColumn.widthAnchor),
- heroCard.heightAnchor.constraint(equalToConstant: 140),
- hero.leadingAnchor.constraint(equalTo: heroCard.leadingAnchor, constant: 22),
- hero.trailingAnchor.constraint(equalTo: heroCard.trailingAnchor, constant: -22),
- hero.topAnchor.constraint(equalTo: heroCard.topAnchor, constant: 18),
- hero.bottomAnchor.constraint(equalTo: heroCard.bottomAnchor, constant: -18),
- lowerSection.widthAnchor.constraint(equalTo: mainColumn.widthAnchor)
- ])
- recommendationsWidthConstraint = recommendationsBox.widthAnchor.constraint(equalTo: mainColumn.widthAnchor, multiplier: 0.68)
- insightsWidthConstraint = insightsBox.widthAnchor.constraint(equalTo: mainColumn.widthAnchor, multiplier: 0.32, constant: -8)
- recommendationsWidthConstraint?.isActive = true
- insightsWidthConstraint?.isActive = true
- }
- private func buildHeroContent() -> NSView {
- let container = NSStackView()
- container.orientation = .horizontal
- container.translatesAutoresizingMaskIntoConstraints = false
- container.distribution = .fillEqually
- let left = NSStackView()
- left.orientation = .vertical
- left.spacing = 8
- left.alignment = .leading
- let title = NSTextField(labelWithString: "AI Job Search Assistant")
- title.font = .systemFont(ofSize: 24, weight: .semibold)
- title.textColor = Theme.primaryText
- let body = NSTextField(labelWithString: "Let AI find the best jobs for you on Indeed based on your preferences.")
- body.font = .systemFont(ofSize: 12, weight: .regular)
- body.textColor = Theme.secondaryText
- let action = NSButton(title: "Find Jobs with AI", target: nil, action: nil)
- action.bezelStyle = .rounded
- action.wantsLayer = true
- action.layer?.backgroundColor = Theme.accent.cgColor
- action.layer?.cornerRadius = 8
- action.contentTintColor = .white
- action.font = .systemFont(ofSize: 13, weight: .semibold)
- left.addArrangedSubview(title)
- left.addArrangedSubview(body)
- left.addArrangedSubview(action)
- let right = NSStackView()
- right.orientation = .vertical
- right.alignment = .trailing
- right.addArrangedSubview(tagBubble("Quick"))
- right.addArrangedSubview(tagBubble("Smart"))
- right.addArrangedSubview(tagBubble("Personalized"))
- container.addArrangedSubview(left)
- container.addArrangedSubview(right)
- return container
- }
- private func configureSidebar(_ items: [SidebarItem]) {
- sidebar.arrangedSubviews.forEach {
- sidebar.removeArrangedSubview($0)
- $0.removeFromSuperview()
- }
- let brand = NSTextField(labelWithString: "Indeed AI\nJob Finder")
- brand.font = .systemFont(ofSize: 18, weight: .bold)
- brand.textColor = Theme.primaryText
- sidebar.addArrangedSubview(brand)
- let spacer = NSView()
- spacer.translatesAutoresizingMaskIntoConstraints = false
- spacer.heightAnchor.constraint(equalToConstant: 8).isActive = true
- sidebar.addArrangedSubview(spacer)
- items.enumerated().forEach { index, item in
- let row = NSStackView()
- row.orientation = .horizontal
- row.spacing = 8
- row.alignment = .centerY
- row.wantsLayer = true
- row.layer?.cornerRadius = 8
- row.edgeInsets = NSEdgeInsets(top: 8, left: 10, bottom: 8, right: 10)
- if index == 0 {
- row.layer?.backgroundColor = Theme.accentMuted.cgColor
- }
- let icon = NSImageView()
- icon.symbolConfiguration = NSImage.SymbolConfiguration(pointSize: 13, weight: .medium)
- icon.image = NSImage(systemSymbolName: item.systemImage, accessibilityDescription: item.title)
- icon.contentTintColor = Theme.primaryText
- let text = NSTextField(labelWithString: item.title)
- text.font = .systemFont(ofSize: 14, weight: .medium)
- text.textColor = Theme.primaryText
- row.addArrangedSubview(icon)
- row.addArrangedSubview(text)
- if let badge = item.badge {
- let badgeField = NSTextField(labelWithString: badge)
- badgeField.font = .systemFont(ofSize: 11, weight: .semibold)
- badgeField.textColor = .white
- badgeField.wantsLayer = true
- badgeField.layer?.backgroundColor = Theme.accent.cgColor
- badgeField.layer?.cornerRadius = 8
- badgeField.alignment = .center
- badgeField.maximumNumberOfLines = 1
- badgeField.lineBreakMode = .byClipping
- badgeField.translatesAutoresizingMaskIntoConstraints = false
- badgeField.widthAnchor.constraint(equalToConstant: 42).isActive = true
- row.addArrangedSubview(NSView())
- row.addArrangedSubview(badgeField)
- }
- sidebar.addArrangedSubview(row)
- }
- }
- private func configureStats(_ stats: [StatCard]) {
- statGrid.removeRow(at: 0)
- let cards = stats.map { stat -> NSView in
- let card = NSStackView()
- card.orientation = .vertical
- card.spacing = 6
- card.edgeInsets = NSEdgeInsets(top: 14, left: 14, bottom: 14, right: 14)
- card.wantsLayer = true
- card.layer?.backgroundColor = Theme.statCardBackground.cgColor
- card.layer?.cornerRadius = 14
- let value = NSTextField(labelWithString: stat.value)
- value.font = .systemFont(ofSize: 30, weight: .bold)
- value.textColor = Theme.primaryText
- let title = NSTextField(labelWithString: stat.title)
- title.font = .systemFont(ofSize: 13, weight: .medium)
- title.textColor = Theme.secondaryText
- let trend = NSTextField(labelWithString: stat.trend)
- trend.font = .systemFont(ofSize: 12, weight: .semibold)
- trend.textColor = NSColor.systemGreen
- card.addArrangedSubview(value)
- card.addArrangedSubview(title)
- card.addArrangedSubview(trend)
- card.translatesAutoresizingMaskIntoConstraints = false
- card.widthAnchor.constraint(equalToConstant: 185).isActive = true
- card.heightAnchor.constraint(equalToConstant: 120).isActive = true
- return card
- }
- statGrid.addRow(with: cards)
- }
- private func configureRecommendations(_ recommendations: [JobRecommendation]) {
- recommendationsStack.orientation = .vertical
- recommendationsStack.spacing = 10
- recommendationsStack.alignment = .leading
- recommendationsStack.arrangedSubviews.forEach {
- recommendationsStack.removeArrangedSubview($0)
- $0.removeFromSuperview()
- }
- recommendations.forEach { recommendation in
- let row = NSStackView()
- row.orientation = .horizontal
- row.spacing = 12
- row.alignment = .centerY
- let icon = NSView()
- icon.wantsLayer = true
- icon.layer?.cornerRadius = 10
- icon.layer?.backgroundColor = Theme.iconBackground.cgColor
- icon.translatesAutoresizingMaskIntoConstraints = false
- icon.widthAnchor.constraint(equalToConstant: 38).isActive = true
- icon.heightAnchor.constraint(equalToConstant: 38).isActive = true
- let textColumn = NSStackView()
- textColumn.orientation = .vertical
- textColumn.spacing = 2
- textColumn.alignment = .leading
- let title = NSTextField(labelWithString: recommendation.title)
- title.font = .systemFont(ofSize: 15, weight: .semibold)
- title.textColor = Theme.primaryText
- let subtitle = NSTextField(labelWithString: "\(recommendation.company) • \(recommendation.location)")
- subtitle.font = .systemFont(ofSize: 12, weight: .regular)
- subtitle.textColor = Theme.secondaryText
- textColumn.addArrangedSubview(title)
- textColumn.addArrangedSubview(subtitle)
- let meta = NSStackView()
- meta.orientation = .vertical
- meta.alignment = .trailing
- let match = NSTextField(labelWithString: recommendation.matchRate)
- match.font = .systemFont(ofSize: 12, weight: .semibold)
- match.textColor = NSColor.systemGreen
- let posted = NSTextField(labelWithString: recommendation.postedAgo)
- posted.font = .systemFont(ofSize: 11, weight: .regular)
- posted.textColor = Theme.tertiaryText
- meta.addArrangedSubview(match)
- meta.addArrangedSubview(posted)
- row.addArrangedSubview(icon)
- row.addArrangedSubview(textColumn)
- row.addArrangedSubview(NSView())
- row.addArrangedSubview(meta)
- recommendationsStack.addArrangedSubview(row)
- }
- }
- private func configureInsights(_ insights: [InsightItem]) {
- insightsStack.orientation = .vertical
- insightsStack.spacing = 12
- insightsStack.alignment = .leading
- insightsStack.arrangedSubviews.forEach {
- insightsStack.removeArrangedSubview($0)
- $0.removeFromSuperview()
- }
- insights.forEach { insight in
- let card = NSStackView()
- card.orientation = .vertical
- card.spacing = 4
- card.edgeInsets = NSEdgeInsets(top: 10, left: 10, bottom: 10, right: 10)
- card.wantsLayer = true
- card.layer?.backgroundColor = Theme.insightCardBackground.cgColor
- card.layer?.cornerRadius = 10
- let title = NSTextField(labelWithString: insight.title)
- title.font = .systemFont(ofSize: 14, weight: .semibold)
- title.textColor = Theme.primaryText
- let body = NSTextField(labelWithString: insight.description)
- body.font = .systemFont(ofSize: 12, weight: .regular)
- body.textColor = Theme.secondaryText
- body.maximumNumberOfLines = 2
- body.lineBreakMode = .byWordWrapping
- card.addArrangedSubview(title)
- card.addArrangedSubview(body)
- insightsStack.addArrangedSubview(card)
- }
- }
- private func sectionBox(title: String, content: NSStackView) -> NSView {
- let box = NSStackView()
- box.orientation = .vertical
- box.spacing = 12
- box.alignment = .leading
- box.edgeInsets = NSEdgeInsets(top: 14, left: 14, bottom: 14, right: 14)
- box.wantsLayer = true
- box.layer?.backgroundColor = Theme.sectionBackground.cgColor
- box.layer?.cornerRadius = 16
- let titleLabel = NSTextField(labelWithString: title)
- titleLabel.font = .systemFont(ofSize: 18, weight: .semibold)
- titleLabel.textColor = Theme.primaryText
- box.addArrangedSubview(titleLabel)
- box.addArrangedSubview(content)
- return box
- }
- private func tagBubble(_ text: String) -> NSView {
- let label = NSTextField(labelWithString: text)
- label.font = .systemFont(ofSize: 11, weight: .medium)
- label.textColor = Theme.bubbleText
- label.wantsLayer = true
- label.layer?.backgroundColor = Theme.bubbleBackground.cgColor
- label.layer?.cornerRadius = 7
- label.alignment = .center
- label.translatesAutoresizingMaskIntoConstraints = false
- label.widthAnchor.constraint(equalToConstant: 90).isActive = true
- return label
- }
- private func updateDocumentLayout() {
- documentContainer.layoutSubtreeIfNeeded()
- let fittingHeight = max(chromeContainer.fittingSize.height + 36, bounds.height)
- documentContainer.frame = NSRect(x: 0, y: 0, width: bounds.width, height: fittingHeight)
- }
- }
|