diff --git a/cmd/frontend/internal/app/ui/handlers.go b/cmd/frontend/internal/app/ui/handlers.go
index 00faefbcc13..6bef8bc7a39 100644
--- a/cmd/frontend/internal/app/ui/handlers.go
+++ b/cmd/frontend/internal/app/ui/handlers.go
@@ -215,9 +215,9 @@ func serveHome(w http.ResponseWriter, r *http.Request) error {
}
if envvar.SourcegraphDotComMode() && !actor.FromContext(r.Context()).IsAuthenticated() {
- // The user is not signed in and tried to access our main site at sourcegraph.com.
- // Redirect to sourcegraph.com/start so they see general info.
- http.Redirect(w, r, "/start", http.StatusTemporaryRedirect)
+ // The user is not signed in and tried to access Sourcegraph.com. Redirect to /welcome so
+ // they see the welcome page.
+ http.Redirect(w, r, "/welcome", http.StatusTemporaryRedirect)
return nil
}
// sourcegraph.com (not about) homepage. There is none, redirect them to /search.
@@ -248,7 +248,7 @@ func serveSignIn(w http.ResponseWriter, r *http.Request) error {
return renderTemplate(w, "app.html", common)
}
-func serveStart(w http.ResponseWriter, r *http.Request) error {
+func serveWelcome(w http.ResponseWriter, r *http.Request) error {
common, err := newCommon(w, r, "Sourcegraph", serveError)
if err != nil {
return err
@@ -258,8 +258,7 @@ func serveStart(w http.ResponseWriter, r *http.Request) error {
}
if !envvar.SourcegraphDotComMode() {
- // The user is signed in and tried to access sourcegraph.com/start,
- // this page should be a 404 under that situation.
+ // The welcome page only exists on Sourcegraph.com.
w.WriteHeader(http.StatusNotFound)
}
return renderTemplate(w, "app.html", common)
diff --git a/cmd/frontend/internal/app/ui/router.go b/cmd/frontend/internal/app/ui/router.go
index 510cf898543..eeac91543e0 100644
--- a/cmd/frontend/internal/app/ui/router.go
+++ b/cmd/frontend/internal/app/ui/router.go
@@ -64,6 +64,7 @@ const (
routeExtensions = "extensions"
routeHelp = "help"
routeExplore = "explore"
+ routeWelcome = "welcome"
// Legacy redirects
routeLegacyLogin = "login"
@@ -113,6 +114,7 @@ func newRouter() *mux.Router {
r.Path("/").Methods("GET").Name(routeHome)
r.PathPrefix("/threads").Methods("GET").Name(routeThreads)
r.Path("/start").Methods("GET").Name(routeStart)
+ r.PathPrefix("/welcome").Methods("GET").Name(routeWelcome)
r.Path("/search").Methods("GET").Name(routeSearch)
r.Path("/search/badge").Methods("GET").Name(routeSearchBadge)
r.Path("/search/searches").Methods("GET").Name(routeSearchSearches)
@@ -185,7 +187,8 @@ func initRouter() {
router := newRouter()
uirouter.Router = router // make accessible to other packages
router.Get(routeHome).Handler(handler(serveHome))
- router.Get(routeStart).Handler(handler(serveStart))
+ router.Get(routeStart).Handler(staticRedirectHandler("/welcome", http.StatusMovedPermanently))
+ router.Get(routeWelcome).Handler(handler(serveWelcome))
router.Get(routeThreads).Handler(handler(serveBasicPageString("Threads - Sourcegraph")))
router.Get(uirouter.RouteSignIn).Handler(handler(serveSignIn))
router.Get(uirouter.RouteSignUp).Handler(handler(serveBasicPageString("Sign up - Sourcegraph")))
diff --git a/go.mod b/go.mod
index 8ebc5727171..961288505c3 100644
--- a/go.mod
+++ b/go.mod
@@ -10,11 +10,9 @@ require (
github.com/codahale/hdrhistogram v0.0.0-20161010025455-3a0bb77429bd // indirect
github.com/coreos/go-oidc v0.0.0-20171002155002-a93f71fdfe73
github.com/coreos/go-semver v0.2.0
- github.com/cosiner/argv v0.0.1 // indirect
github.com/crewjam/saml v0.0.0-20180831135026-ebc5f787b786
github.com/davecgh/go-spew v1.1.1
github.com/daviddengcn/go-colortext v0.0.0-20171126034257-17e75f6184bc
- github.com/derekparker/delve v1.1.0 // indirect
github.com/dghubble/gologin v1.0.2-0.20181013174641-0e442dd5bb73
github.com/dgrijalva/jwt-go v3.2.0+incompatible // indirect
github.com/die-net/lrucache v0.0.0-20190123005519-19a39ef22a11
@@ -82,7 +80,6 @@ require (
github.com/neelance/parallel v0.0.0-20160708114440-4de9ce63d14c
github.com/opentracing-contrib/go-stdlib v0.0.0-20190104202730-77df8e8e70b4
github.com/opentracing/opentracing-go v1.0.2
- github.com/peterh/liner v1.1.0 // indirect
github.com/peterhellberg/link v1.0.0
github.com/pkg/errors v0.8.1
github.com/pquerna/cachecontrol v0.0.0-20180517163645-1555304b9b35 // indirect
@@ -118,7 +115,6 @@ require (
github.com/xeonx/timeago v1.0.0-rc3
github.com/zenazn/goji v0.9.0 // indirect
go.uber.org/atomic v1.3.2 // indirect
- golang.org/x/arch v0.0.0-20181203225421-5a4828bb7045 // indirect
golang.org/x/crypto v0.0.0-20190104202753-ff983b9c42bc
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e
golang.org/x/oauth2 v0.0.0-20190201180606-99b60b757ec1
diff --git a/go.sum b/go.sum
index ad9152416c2..75726ae1b0e 100644
--- a/go.sum
+++ b/go.sum
@@ -43,8 +43,6 @@ github.com/coreos/go-oidc v0.0.0-20171002155002-a93f71fdfe73 h1:7CNPV0LWRCa1FNmq
github.com/coreos/go-oidc v0.0.0-20171002155002-a93f71fdfe73/go.mod h1:CgnwVTmzoESiwO9qyAFEMiHoZ1nMCKZlZ9V6mm3/LKc=
github.com/coreos/go-semver v0.2.0 h1:3Jm3tLmsgAYcjC+4Up7hJrFBPr+n7rAqYeSw/SZazuY=
github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
-github.com/cosiner/argv v0.0.1 h1:2iAFN+sWPktbZ4tvxm33Ei8VY66FPCxdOxpncUGpAXE=
-github.com/cosiner/argv v0.0.1/go.mod h1:p/NrK5tF6ICIly4qwEDsf6VDirFiWWz0FenfYBwJaKQ=
github.com/crewjam/saml v0.0.0-20180831135026-ebc5f787b786 h1:8OVABJfT9iJh/uHeYlk1HWugxt7j50JPwW2uLOV9Yqs=
github.com/crewjam/saml v0.0.0-20180831135026-ebc5f787b786/go.mod h1:w5eu+HNtubx+kRpQL6QFT2F3yIFfYVe6+EzOFVU7Hko=
github.com/cznic/b v0.0.0-20180115125044-35e9bbe41f07/go.mod h1:URriBxXwVq5ijiJ12C7iIZqlA69nTlI+LgI6/pwftG8=
@@ -62,8 +60,6 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/daviddengcn/go-colortext v0.0.0-20171126034257-17e75f6184bc h1:nqMZEdowWmtK9ysqvFibHJ56mTprkyE5c/6q8ZHwLc0=
github.com/daviddengcn/go-colortext v0.0.0-20171126034257-17e75f6184bc/go.mod h1:dv4zxwHi5C/8AeI+4gX4dCWOIvNi7I6JCSX0HvlKPgE=
-github.com/derekparker/delve v1.1.0 h1:icd65nMp7s2HiLz6y/6RCVXBdoED3xxYLwX09EMaRCc=
-github.com/derekparker/delve v1.1.0/go.mod h1:pMSZMfp0Nhbm8qdZJkuE/yPGOkLpGXLS1I4poXQpuJU=
github.com/dgrijalva/jwt-go v3.2.0+incompatible h1:7qlOGliEKZXTDg6OTjfoBKDXWrumCAMpl/TFQ4/5kLM=
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
github.com/dhui/dktest v0.3.0 h1:kwX5a7EkLcjo7VpsPQSYJcKGbXBXdjI9FGjuUj1jn6I=
@@ -328,8 +324,6 @@ github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaO
github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
github.com/mattn/go-isatty v0.0.4 h1:bnP0vzxcAdeI1zdubAl5PjU6zsERjGZb7raWodagDYs=
github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
-github.com/mattn/go-runewidth v0.0.3 h1:a+kO+98RDGEfo6asOGMmpodZq4FNtnGP54yps8BzLR4=
-github.com/mattn/go-runewidth v0.0.3/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU=
github.com/mattn/go-sqlite3 v1.9.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
github.com/mattn/go-sqlite3 v1.10.0 h1:jbhqpg7tQe4SupckyijYiy0mJJ/pRyHvXf7JdWK860o=
github.com/mattn/go-sqlite3 v1.10.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
@@ -367,8 +361,6 @@ github.com/openzipkin/zipkin-go v0.1.1/go.mod h1:NtoC/o8u3JlF1lSlyPNswIbeQH9bJTm
github.com/pelletier/go-buffruneio v0.2.0/go.mod h1:JkE26KsDizTr40EUHkXVtNPvgGtbSNq5BcowyYOWdKo=
github.com/pelletier/go-toml v1.1.0 h1:cmiOvKzEunMsAxyhXSzpL5Q1CRKpVv0KQsnAIcSEVYM=
github.com/pelletier/go-toml v1.1.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
-github.com/peterh/liner v1.1.0 h1:f+aAedNJA6uk7+6rXsYBnhdo4Xux7ESLe+kcuVUF5os=
-github.com/peterh/liner v1.1.0/go.mod h1:CRroGNssyjTd/qIG2FyxByd2S8JEAZXBl4qUrZf8GS0=
github.com/peterhellberg/link v1.0.0 h1:mUWkiegowUXEcmlb+ybF75Q/8D2Y0BjZtR8cxoKhaQo=
github.com/peterhellberg/link v1.0.0/go.mod h1:gtSlOT4jmkY8P47hbTc8PTgiDDWpdPbFYl75keYyBB8=
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
@@ -504,8 +496,6 @@ go.opencensus.io v0.18.0 h1:Mk5rgZcggtbvtAun5aJzAtjKKN/t0R3jJPlWILlv938=
go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA=
go.uber.org/atomic v1.3.2 h1:2Oa65PReHzfn29GpvgsYwloV9AVFHPDk8tYxt2c2tr4=
go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
-golang.org/x/arch v0.0.0-20181203225421-5a4828bb7045 h1:Pn8fQdvx+z1avAi7fdM2kRYWQNxGlavNDSyzrQg2SsU=
-golang.org/x/arch v0.0.0-20181203225421-5a4828bb7045/go.mod h1:cYlCBUl1MsqxdiKgmc4uh7TxZfWSFLOGSRR090WDxt8=
golang.org/x/crypto v0.0.0-20180505025534-4ec37c66abab/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
golang.org/x/crypto v0.0.0-20190103213133-ff983b9c42bc/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
diff --git a/package.json b/package.json
index 6c289efe9bf..8aa674524d0 100644
--- a/package.json
+++ b/package.json
@@ -54,6 +54,7 @@
"@types/babel__core": "7.0.4",
"@types/chai": "4.1.7",
"@types/chai-as-promised": "7.1.0",
+ "@types/classnames": "^2.2.7",
"@types/cpy": "5.1.0",
"@types/d3-axis": "1.0.11",
"@types/d3-scale": "2.1.0",
@@ -174,6 +175,7 @@
"@sqs/jsonc-parser": "^1.0.3",
"abortable-rx": "^1.0.9",
"bootstrap": "^4.1.3",
+ "classnames": "^2.2.6",
"copy-to-clipboard": "^3.0.7",
"d3-axis": "^1.0.7",
"d3-scale": "^2.0.0",
diff --git a/shared/src/actions/ActionsContainer.tsx b/shared/src/actions/ActionsContainer.tsx
index 8d420d4f9ff..0ae76213465 100644
--- a/shared/src/actions/ActionsContainer.tsx
+++ b/shared/src/actions/ActionsContainer.tsx
@@ -10,7 +10,9 @@ import { PlatformContextProps } from '../platform/context'
import { ActionItem, ActionItemProps } from './ActionItem'
import { ActionsState } from './actions'
-export interface ActionsProps extends ExtensionsControllerProps, PlatformContextProps {
+export interface ActionsProps
+ extends ExtensionsControllerProps<'executeCommand' | 'services'>,
+ PlatformContextProps<'forceUpdateTooltip'> {
menu: ContributableMenu
scope?: ContributionScope
actionItemClass?: string
diff --git a/shared/src/commandPalette/CommandList.tsx b/shared/src/commandPalette/CommandList.tsx
index d2834a32807..07954521587 100644
--- a/shared/src/commandPalette/CommandList.tsx
+++ b/shared/src/commandPalette/CommandList.tsx
@@ -14,7 +14,9 @@ import { getContributedActionItems } from '../contributions/contributions'
import { ExtensionsControllerProps } from '../extensions/controller'
import { PlatformContextProps } from '../platform/context'
-interface Props extends ExtensionsControllerProps, PlatformContextProps {
+interface Props
+ extends ExtensionsControllerProps<'services' | 'executeCommand'>,
+ PlatformContextProps<'forceUpdateTooltip'> {
/** The menu whose commands to display. */
menu: ContributableMenu
diff --git a/web/src/Layout.tsx b/web/src/Layout.tsx
index 874932ad60d..772696fff80 100644
--- a/web/src/Layout.tsx
+++ b/web/src/Layout.tsx
@@ -64,8 +64,6 @@ export interface LayoutProps
isLightTheme: boolean
onThemeChange: () => void
- onMainPage: (mainPage: boolean) => void
- isMainPage: boolean
navbarSearchQuery: string
onNavbarQueryChange: (query: string) => void
diff --git a/web/src/SourcegraphWebApp.scss b/web/src/SourcegraphWebApp.scss
index 7ff21ea846e..6435f3f5281 100644
--- a/web/src/SourcegraphWebApp.scss
+++ b/web/src/SourcegraphWebApp.scss
@@ -240,7 +240,6 @@ hr {
@import './global/GlobalAlerts';
@import './global/GlobalDebug';
@import './docSite/DocSitePage';
-@import './search/input/MainPage';
@import './search/input/ScopePage';
@import './search/input/SearchPage';
@import './search/results/SearchResults';
diff --git a/web/src/SourcegraphWebApp.tsx b/web/src/SourcegraphWebApp.tsx
index 706579fb321..bd41bea124b 100644
--- a/web/src/SourcegraphWebApp.tsx
+++ b/web/src/SourcegraphWebApp.tsx
@@ -73,11 +73,6 @@ interface SourcegraphWebAppState extends PlatformContextProps, SettingsCascadePr
*/
isLightTheme: boolean
- /**
- * Whether the user is on MainPage and therefore not logged in
- */
- isMainPage: boolean
-
/**
* The current search query in the navbar.
*/
@@ -108,7 +103,6 @@ export class SourcegraphWebApp extends React.Component ({ isLightTheme: !state.isLightTheme }))
}
- private onMainPage = (mainPage: boolean) => {
- this.setState(() => ({ isMainPage: mainPage }))
- }
-
private onNavbarQueryChange = (navbarSearchQuery: string) => {
this.setState({ navbarSearchQuery })
}
diff --git a/web/src/auth.ts b/web/src/auth.ts
index 57888c87dbe..23b33706b82 100644
--- a/web/src/auth.ts
+++ b/web/src/auth.ts
@@ -53,7 +53,7 @@ export function refreshAuthenticatedUser(): Observable {
)
}
-const initialSiteConfigAuthPublic = window.context.critical['auth.public']
+const initialSiteConfigAuthPublic = window.context ? window.context.critical['auth.public'] : false // default to false in tests
/**
* Whether auth is required to perform any action.
@@ -67,7 +67,7 @@ const initialSiteConfigAuthPublic = window.context.critical['auth.public']
export const authRequired = authenticatedUser.pipe(map(user => user === null && !initialSiteConfigAuthPublic))
// Populate authenticatedUser.
-if (window.context.isAuthenticatedUser) {
+if (window.context && window.context.isAuthenticatedUser) {
refreshAuthenticatedUser()
.toPromise()
.then(() => void 0, err => console.error(err))
diff --git a/web/src/enterprise.scss b/web/src/enterprise.scss
index 911bef6dc75..91c2b8df15d 100644
--- a/web/src/enterprise.scss
+++ b/web/src/enterprise.scss
@@ -3,6 +3,7 @@
@import './SourcegraphWebApp.scss'; // Need .scss extension because of https://github.com/webpack-contrib/sass-loader/issues/556
// Enterprise styles
+@import './enterprise/dotcom/welcome/WelcomeArea';
@import './enterprise/extensions/registry/RegistryArea';
@import './enterprise/extensions/registry/RegistryNewExtensionPage';
@import './enterprise/extensions/extension/RegistryExtensionManagePage';
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeArea.scss b/web/src/enterprise/dotcom/welcome/WelcomeArea.scss
new file mode 100644
index 00000000000..eec25d18b01
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeArea.scss
@@ -0,0 +1 @@
+@import './WelcomeMainPage';
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeArea.tsx b/web/src/enterprise/dotcom/welcome/WelcomeArea.tsx
new file mode 100644
index 00000000000..067067a8cc8
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeArea.tsx
@@ -0,0 +1,91 @@
+import { LoadingSpinner } from '@sourcegraph/react-loading-spinner'
+import H from 'history'
+import ChevronLeftIcon from 'mdi-react/ChevronLeftIcon'
+import MapSearchIcon from 'mdi-react/MapSearchIcon'
+import React from 'react'
+import { Route, RouteComponentProps, Switch } from 'react-router'
+import { Link } from 'react-router-dom'
+import { ExtensionsControllerProps } from '../../../../../shared/src/extensions/controller'
+import * as GQL from '../../../../../shared/src/graphql/schema'
+import { PlatformContextProps } from '../../../../../shared/src/platform/context'
+import { ErrorBoundary } from '../../../components/ErrorBoundary'
+import { HeroPage } from '../../../components/HeroPage'
+import { RouteDescriptor } from '../../../util/contributions'
+import { WelcomeAreaFooter } from './WelcomeAreaFooter'
+
+const NotFoundPage = () =>
+
+export interface WelcomeAreaRoute extends RouteDescriptor {}
+
+export interface WelcomeAreaProps extends ExtensionsControllerProps, PlatformContextProps, RouteComponentProps<{}> {
+ authenticatedUser: GQL.IUser | null
+ isLightTheme: boolean
+ routes: ReadonlyArray
+ location: H.Location
+ history: H.History
+}
+
+export interface WelcomeAreaRouteContext extends WelcomeAreaProps {}
+
+/**
+ * The welcome area, which contains general product information.
+ */
+export class WelcomeArea extends React.PureComponent {
+ public render(): JSX.Element | null {
+ if (!window.context.sourcegraphDotComMode) {
+ return (
+
+ Visit{' '}
+
+ sourcegraph.com/welcome
+ {' '}
+ instead.
+
+ }
+ />
+ )
+ }
+
+ const { children, ...context } = this.props
+ return (
+
+ {this.props.location.pathname !== '/welcome' && (
+
+
+
+ Welcome
+
+
+ )}
+
+ }>
+
+ {this.props.routes.map(
+ ({ path, exact, render, condition = () => true }) =>
+ condition(context) && (
+ (
+ <>
+ {render({ ...context, ...routeComponentProps })}
+
+ >
+ )}
+ />
+ )
+ )}
+
+
+
+
+
+ )
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeAreaFooter.tsx b/web/src/enterprise/dotcom/welcome/WelcomeAreaFooter.tsx
new file mode 100644
index 00000000000..cb881ad135a
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeAreaFooter.tsx
@@ -0,0 +1,135 @@
+import React from 'react'
+import { Link } from 'react-router-dom'
+
+export const WelcomeAreaFooter: React.FunctionComponent<{ isLightTheme: boolean }> = ({ isLightTheme }) => (
+ <>
+
+
+
+
Features
+
+
+ Code search
+
+
+ Code intelligence
+
+
+ Integrations
+
+
+
+ Enterprise
+
+
+
+
+
+
+
+
+
+
+ Terms
+ {' '}
+ -{' '}
+
+ Privacy
+ {' '}
+ - Copyright © 2018 Sourcegraph, Inc.
+
+ >
+)
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPage.scss b/web/src/enterprise/dotcom/welcome/WelcomeMainPage.scss
new file mode 100644
index 00000000000..35722d18da6
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPage.scss
@@ -0,0 +1,23 @@
+@import './WelcomeMainPageDemos';
+@import './WelcomeMainPageLogos';
+
+.welcome-main-page {
+ &__logo-mark {
+ width: 1.5rem;
+ height: 1.5rem;
+ position: absolute;
+ top: -1.5rem; // make CTAs flush with header text
+ }
+ &__header {
+ font-size: 24px;
+ }
+ &__demo {
+ min-height: 540px; // reduce jitter while page is loading
+ }
+}
+
+.theme-dark .welcome-main-page {
+ &__sign-up {
+ color: #ffffff; // otherwise the signup CTA link text is indistinguishable from text-muted
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPage.tsx b/web/src/enterprise/dotcom/welcome/WelcomeMainPage.tsx
new file mode 100644
index 00000000000..1cb17e24bef
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPage.tsx
@@ -0,0 +1,157 @@
+import * as H from 'history'
+import CloudUploadIcon from 'mdi-react/CloudUploadIcon'
+import * as React from 'react'
+import { Link } from 'react-router-dom'
+import { ExtensionsControllerProps } from '../../../../../shared/src/extensions/controller'
+import * as GQL from '../../../../../shared/src/graphql/schema'
+import { PlatformContextProps } from '../../../../../shared/src/platform/context'
+import { eventLogger } from '../../../tracking/eventLogger'
+import { WelcomeMainPageDemos } from './WelcomeMainPageDemos'
+import { WelcomeMainPageLogos } from './WelcomeMainPageLogos'
+
+// Lambdas are OK in this component because it is not performance sensitive and using them
+// simplifies the code.
+//
+// tslint:disable:jsx-no-lambda
+
+interface Props extends ExtensionsControllerProps, PlatformContextProps {
+ authenticatedUser: GQL.IUser | null
+ isLightTheme: boolean
+ location: H.Location
+ history: H.History
+}
+
+/**
+ * The welcome main page, which describes Sourcegraph functionality and other general information.
+ */
+export class WelcomeMainPage extends React.Component {
+ public render(): JSX.Element | null {
+ return (
+
+
+
+
+
+
+
+
+ Search, navigate, and review code.
+ {' '}
+ Find answers.
+
+
Sourcegraph is a web-based code search and navigation tool for dev teams.
+
+
+ Code search: fast, cross-repository, on any commit/branch (no
+ indexing delay), with support for regexps, diffs, and{' '}
+
+ filters
+
+
+
+ Code navigation: go-to-definition and find-references for{' '}
+
+ all major languages
+
+
+
+ Deep integrations with GitHub, GitHub Enterprise, GitLab,
+ Bitbucket Server, Phabricator, etc., plus a{' '}
+
+ powerful extension API
+
+
+
+
+ Open-source
+
+ , self-hosted, and free (
+
+ Enterprise
+ {' '}
+ upgrade available)
+
+
+
+
+ See how it's used
+ {' '}
+ to build better software faster at:
+
+
+
+
+ …and thousands of other organizations.
+
+
+
+
+
+
+ {!this.props.authenticatedUser && (
+
+ eventLogger.log('WelcomeSignUpForSourcegraphDotCom')}
+ >
+ Sign up on Sourcegraph.com
+
+
+ A public Sourcegraph instance for public code only.
+
+
+ )}
+
+
+
+
+ VIDEO
+
+
+
+
+ )
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.scss b/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.scss
new file mode 100644
index 00000000000..7dfa11c8e7f
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.scss
@@ -0,0 +1,13 @@
+.welcome-main-page-demos {
+ &__item {
+ touch-action: manipulation; // remove iOS tap delay
+ }
+ &__video {
+ // To avoid jitter during loading/switching, calculate height of video based on page
+ // dimensions and video aspect ratio.
+ min-height: calc((100vw - (12px * 4)) * (1596 / 2535));
+ @media screen and (min-width: 1000px) {
+ min-height: 550px;
+ }
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.tsx b/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.tsx
new file mode 100644
index 00000000000..30c849a5609
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPageDemos.tsx
@@ -0,0 +1,83 @@
+import classnames from 'classnames'
+import H from 'history'
+import PlayCircleOutlineIcon from 'mdi-react/PlayCircleOutlineIcon'
+import React from 'react'
+import { Link } from 'react-router-dom'
+import { eventLogger } from '../../../tracking/eventLogger'
+
+interface Props {
+ className: string
+ location: H.Location
+ history: H.History
+}
+
+const BASE_VIDEO_URL = 'https://storage.googleapis.com/sourcegraph-assets/video/welcome/video'
+
+// The mp4 videos are 2535x1596 at 20fps.
+//
+// To convert videos from mp4 (from kazam) to m4v (for iPhone):
+//
+// ffmpeg -i INPUT_FILE -pix_fmt yuv420p -vf "scale=-2:720:flags=lanczos" -vcodec libx264 -level 3.2 -profile:v main -preset medium -crf 23 -x264-params ref=4 -movflags +faststart OUTPUT_FILE
+//
+// To upload files:
+//
+// gsutil cp -a public-read -r INPUT_FILES gs://sourcegraph-assets/video/welcome/video/
+const VIDEOS: { title: string; hash: string; filename: string }[] = [
+ {
+ title: 'Code navigation',
+ hash: 'code-navigation',
+ filename: 'Welcome-CodeNavigation',
+ },
+ {
+ title: 'Code search',
+ hash: 'code-search',
+ filename: 'Welcome-Search',
+ },
+ {
+ title: 'GitHub integration',
+ hash: 'github-integration',
+ filename: 'Welcome-GitHub',
+ },
+]
+
+export class WelcomeMainPageDemos extends React.PureComponent {
+ public render(): JSX.Element | null {
+ const activeTab = this.props.location.hash.replace(/^#/, '') || VIDEOS[0].hash
+ const video = VIDEOS.find(v => v.hash === activeTab) || VIDEOS[0]
+ return (
+
+
+
+ Demos:
+
+ {VIDEOS.map(({ title, hash }, i) => (
+
+ eventLogger.log('WelcomeMainPageDemosVideo', { hash })}
+ >
+ {title}
+
+
+ ))}
+
+
+
+
+ Demo video playback is not supported on your browser.
+
+
+ )
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.scss b/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.scss
new file mode 100644
index 00000000000..fd6692683f2
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.scss
@@ -0,0 +1,18 @@
+.welcome-main-page-logos {
+ &__logo {
+ flex: 0 0 auto;
+ &-1 {
+ height: 1.5rem;
+ }
+ &-2 {
+ height: 2rem;
+ // stylelint-disable-next-line declaration-property-unit-whitelist
+ margin-top: 5px; // equalize perceived text baseline of customer logos
+ }
+ &-3 {
+ height: 3.5rem;
+ // stylelint-disable-next-line declaration-property-unit-whitelist
+ margin-top: 6px; // equalize perceived text baseline of customer logos
+ }
+ }
+}
diff --git a/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.tsx b/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.tsx
new file mode 100644
index 00000000000..0dd36a86d3e
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/WelcomeMainPageLogos.tsx
@@ -0,0 +1,33 @@
+import { shuffle } from 'lodash'
+import React from 'react'
+import { Logo1, Logo2, Logo3 } from './logos'
+
+// Shuffle logos because we love all of them infinitely. :)
+const LOGOS: {
+ component: React.ComponentType<{ className: string; isLightTheme: boolean }>
+ className: string
+}[] = shuffle([
+ {
+ component: Logo1,
+ className: 'welcome-main-page-logos__logo-1 mr-3',
+ },
+ {
+ component: Logo2,
+ className: 'welcome-main-page-logos__logo-2 mr-3',
+ },
+ {
+ component: Logo3,
+ className: 'welcome-main-page-logos__logo-3 mr-3',
+ },
+])
+
+/**
+ * The logos for the welcome main page.
+ */
+export const WelcomeMainPageLogos: React.FunctionComponent<{ isLightTheme: boolean }> = ({ isLightTheme }) => (
+ <>
+ {LOGOS.map(({ component: C, className }, i) => (
+
+ ))}
+ >
+)
diff --git a/web/src/enterprise/dotcom/welcome/logos.tsx b/web/src/enterprise/dotcom/welcome/logos.tsx
new file mode 100644
index 00000000000..23647d1edee
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/logos.tsx
@@ -0,0 +1,133 @@
+import React from 'react'
+
+interface Props {
+ isLightTheme: boolean
+ className: string
+}
+
+// These components are intentionally not named after the companies, to avoid people reusing them
+// and thinking they are the official SVGs for these company logos.
+
+export const Logo1: React.FunctionComponent = ({ isLightTheme, className }) => {
+ const fill = isLightTheme ? '#010202' : 'white'
+ return (
+
+
+
+
+
+
+
+
+ )
+}
+
+export const Logo2: React.FunctionComponent = ({ isLightTheme, className }) => {
+ const fill = isLightTheme ? '#EA0B8C' : 'white'
+ return (
+
+
+
+
+
+
+
+ )
+}
+
+export const Logo3: React.FunctionComponent = ({ isLightTheme, className }) => {
+ // From https://www.yelp.com/brand: "If you need to display the Yelp logo on a white background
+ // be sure to use the version with the grey stroke."
+ const outlineColor = isLightTheme ? '#cccccc' : 'transparent'
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ )
+}
diff --git a/web/src/enterprise/dotcom/welcome/routes.tsx b/web/src/enterprise/dotcom/welcome/routes.tsx
new file mode 100644
index 00000000000..6aed021482a
--- /dev/null
+++ b/web/src/enterprise/dotcom/welcome/routes.tsx
@@ -0,0 +1,16 @@
+import React from 'react'
+import { WelcomeAreaRoute } from './WelcomeArea'
+const WelcomeMainPage = React.lazy(async () => ({
+ default: (await import('./WelcomeMainPage')).WelcomeMainPage,
+}))
+
+export const welcomeAreaRoutes: ReadonlyArray = [
+ {
+ path: '/',
+ exact: true,
+ // tslint:disable-next-line:jsx-no-lambda
+ render: props => ,
+ },
+ // We will add more pages here soon. The other pages (search, code intel, integrations) were
+ // removed to avoid blocking shipping of the new main welcome page.
+]
diff --git a/web/src/enterprise/routes.tsx b/web/src/enterprise/routes.tsx
index 042f5bf71e8..336ccb19163 100644
--- a/web/src/enterprise/routes.tsx
+++ b/web/src/enterprise/routes.tsx
@@ -1,5 +1,10 @@
import React from 'react'
+import { Redirect } from 'react-router'
import { LayoutRouteProps, routes } from '../routes'
+import { welcomeAreaRoutes } from './dotcom/welcome/routes'
+const WelcomeArea = React.lazy(async () => ({
+ default: (await import('./dotcom/welcome/WelcomeArea')).WelcomeArea,
+}))
const NewProductSubscriptionPageOrRedirectUser = React.lazy(async () => ({
default: (await import('./user/productSubscriptions/NewProductSubscriptionPageOrRedirectUser'))
.NewProductSubscriptionPageOrRedirectUser,
@@ -13,5 +18,14 @@ export const enterpriseRoutes: ReadonlyArray = [
exact: true,
render: props => ,
},
+ {
+ path: '/start',
+ render: () => ,
+ exact: true,
+ },
+ {
+ path: '/welcome',
+ render: props => ,
+ },
...routes,
]
diff --git a/web/src/nav/GlobalNavbar.scss b/web/src/nav/GlobalNavbar.scss
index 433fe42932c..70630b39560 100644
--- a/web/src/nav/GlobalNavbar.scss
+++ b/web/src/nav/GlobalNavbar.scss
@@ -22,19 +22,26 @@
}
&__logo {
- width: 1.5rem;
height: 1.5rem;
- &:hover {
- animation: spin 0.5s ease-in-out 1;
+ &--full {
+ // stylelint-disable-next-line declaration-property-unit-whitelist
+ width: 150px;
+ margin-right: 0.25rem; // full logo image has slightly more spacing on left side
+ }
+ &:not(&--full) {
+ width: 1.5rem;
+ &:hover {
+ animation: spin 0.5s ease-in-out 1;
- @keyframes spin {
- 50% {
- transform: rotate(180deg) scale(1.2);
- }
+ @keyframes spin {
+ 50% {
+ transform: rotate(180deg) scale(1.2);
+ }
- 100% {
- transform: rotate(180deg) scale(1);
+ 100% {
+ transform: rotate(180deg) scale(1);
+ }
}
}
}
diff --git a/web/src/nav/GlobalNavbar.tsx b/web/src/nav/GlobalNavbar.tsx
index addaabe16b8..03f33c02462 100644
--- a/web/src/nav/GlobalNavbar.tsx
+++ b/web/src/nav/GlobalNavbar.tsx
@@ -10,6 +10,7 @@ import { authRequired } from '../auth'
import { KeybindingsProps } from '../keybindings'
import { parseSearchURLQuery } from '../search'
import { SearchNavbarItem } from '../search/input/SearchNavbarItem'
+import { showDotComMarketing } from '../util/features'
import { NavLinks } from './NavLinks'
interface Props extends SettingsCascadeProps, PlatformContextProps, ExtensionsControllerProps, KeybindingsProps {
@@ -70,7 +71,19 @@ export class GlobalNavbar extends React.PureComponent {
}
public render(): JSX.Element | null {
- const logo =
+ let logoSrc: string
+ const showFullLogo = this.props.location.pathname === '/welcome'
+ if (showFullLogo) {
+ logoSrc = this.props.isLightTheme
+ ? '/.assets/img/sourcegraph-light-head-logo.svg'
+ : '/.assets/img/sourcegraph-head-logo.svg'
+ } else {
+ logoSrc = '/.assets/img/sourcegraph-mark.svg'
+ }
+
+ const logo = (
+
+ )
return (
{this.props.lowProfile ? (
@@ -84,8 +97,8 @@ export class GlobalNavbar extends React.PureComponent
{
{logo}
)}
- {!this.state.authRequired && (
-
+ {!this.state.authRequired && this.props.location.pathname !== '/welcome' && (
+
{
)}
>
)}
- {!this.state.authRequired && }
+ {!this.state.authRequired && }
)
}
diff --git a/web/src/nav/NavLinks.test.tsx b/web/src/nav/NavLinks.test.tsx
new file mode 100644
index 00000000000..cc8ace555b9
--- /dev/null
+++ b/web/src/nav/NavLinks.test.tsx
@@ -0,0 +1,84 @@
+import * as H from 'history'
+import { flatten } from 'lodash'
+import React from 'react'
+import { createRenderer } from 'react-test-renderer/shallow'
+import { setLinkComponent } from '../../../shared/src/components/Link'
+import { ExtensionsControllerProps } from '../../../shared/src/extensions/controller'
+import * as GQL from '../../../shared/src/graphql/schema'
+import { SettingsCascadeProps } from '../../../shared/src/settings/settings'
+import { KeybindingsProps } from '../keybindings'
+import { NavLinks } from './NavLinks'
+
+// Renders a human-readable list of the NavLinks' contents so that humans can more easily diff
+// snapshots to see what actually changed.
+const renderShallow = (element: React.ReactElement
): any => {
+ const renderer = createRenderer()
+ renderer.render(element)
+
+ const getDisplayName = (element: React.ReactChild): string | string[] => {
+ if (element === null) {
+ return []
+ } else if (typeof element === 'string' || typeof element === 'number') {
+ return element.toString()
+ } else if (element.type === 'li' && (element.props.children.props.href || element.props.children.props.to)) {
+ return `${element.props.children.props.children} ${element.props.children.props.href ||
+ element.props.children.props.to}`
+ } else if (typeof element.type === 'symbol' || typeof element.type === 'string') {
+ return flatten(React.Children.map(element.props.children, element => getDisplayName(element)))
+ } else {
+ return element.type.displayName || element.type.name || 'Unknown'
+ }
+ }
+
+ return flatten(
+ React.Children.map(renderer.getRenderOutput().props.children, e => getDisplayName(e)).filter(e => !!e)
+ )
+}
+
+describe('NavLinks', () => {
+ setLinkComponent((props: any) => )
+ afterAll(() => setLinkComponent(null as any)) // reset global env for other tests
+ const NOOP_EXTENSIONS_CONTROLLER: ExtensionsControllerProps<
+ 'executeCommand' | 'services'
+ >['extensionsController'] = { executeCommand: async () => void 0, services: {} as any }
+ const NOOP_PLATFORM_CONTEXT = { forceUpdateTooltip: () => void 0 }
+ const KEYBINDINGS: KeybindingsProps['keybindings'] = { commandPalette: [] }
+ const SETTINGS_CASCADE: SettingsCascadeProps['settingsCascade'] = { final: null, subjects: null }
+ // tslint:disable-next-line:no-object-literal-type-assertion
+ const USER = { username: 'u' } as GQL.IUser
+ const history = H.createMemoryHistory({ keyLength: 0 })
+ const commonProps = {
+ extensionsController: NOOP_EXTENSIONS_CONTROLLER,
+ platformContext: NOOP_PLATFORM_CONTEXT,
+ isLightTheme: true,
+ onThemeChange: () => void 0,
+ keybindings: KEYBINDINGS,
+ settingsCascade: SETTINGS_CASCADE,
+ }
+
+ // The 3 main props that affect the desired contents of NavLinks are whether the user is signed
+ // in, whether we're on Sourcegraph.com, and the path. Create snapshots of all permutations.
+ for (const authenticatedUser of [null, USER]) {
+ for (const showDotComMarketing of [false, true]) {
+ for (const path of ['/foo', '/search', '/welcome']) {
+ const name = [
+ authenticatedUser ? 'authed' : 'unauthed',
+ showDotComMarketing ? 'Sourcegraph.com' : 'self-hosted',
+ path,
+ ].join(' ')
+ test(name, () => {
+ expect(
+ renderShallow(
+
+ )
+ ).toMatchSnapshot()
+ })
+ }
+ }
+ }
+})
diff --git a/web/src/nav/NavLinks.tsx b/web/src/nav/NavLinks.tsx
index 5c90d146180..c6a4c3031ba 100644
--- a/web/src/nav/NavLinks.tsx
+++ b/web/src/nav/NavLinks.tsx
@@ -1,27 +1,28 @@
import * as H from 'history'
import * as React from 'react'
-import { Link } from 'react-router-dom'
import { Subscription } from 'rxjs'
import { ActionsNavItems } from '../../../shared/src/actions/ActionsNavItems'
import { ContributableMenu } from '../../../shared/src/api/protocol'
import { CommandListPopoverButton } from '../../../shared/src/commandPalette/CommandList'
+import { Link } from '../../../shared/src/components/Link'
import { ExtensionsControllerProps } from '../../../shared/src/extensions/controller'
import * as GQL from '../../../shared/src/graphql/schema'
import { PlatformContextProps } from '../../../shared/src/platform/context'
import { SettingsCascadeProps } from '../../../shared/src/settings/settings'
import { isDiscussionsEnabled } from '../discussions'
import { KeybindingsProps } from '../keybindings'
-import { eventLogger } from '../tracking/eventLogger'
-import { showDotComMarketing } from '../util/features'
import { UserNavItem } from './UserNavItem'
-interface Props extends SettingsCascadeProps, PlatformContextProps, ExtensionsControllerProps, KeybindingsProps {
+interface Props
+ extends SettingsCascadeProps,
+ KeybindingsProps,
+ ExtensionsControllerProps<'executeCommand' | 'services'>,
+ PlatformContextProps<'forceUpdateTooltip'> {
location: H.Location
- history: H.History
authenticatedUser: GQL.IUser | null
isLightTheme: boolean
onThemeChange: () => void
- isMainPage?: boolean
+ showDotComMarketing: boolean
}
export class NavLinks extends React.PureComponent {
@@ -31,24 +32,28 @@ export class NavLinks extends React.PureComponent {
this.subscriptions.unsubscribe()
}
- private onClickInstall = (): void => {
- eventLogger.log('InstallSourcegraphServerCTAClicked', {
- location_on_page: 'Navbar',
- })
- }
-
public render(): JSX.Element | null {
return (
- {showDotComMarketing && (
+ {/* Show "Search" link on small screens when GlobalNavbar hides the SearchNavbarItem. */}
+ {this.props.location.pathname !== '/search' && this.props.location.pathname !== '/welcome' && (
+
+
+ Search
+
+
+ )}
+ {this.props.showDotComMarketing && this.props.location.pathname !== '/welcome' && (
-
- Install Sourcegraph
+
+ Welcome
+
+
+ )}
+ {this.props.showDotComMarketing && this.props.location.pathname === '/welcome' && (
+
+
+ Docs
)}
@@ -59,18 +64,24 @@ export class NavLinks extends React.PureComponent {
platformContext={this.props.platformContext}
location={this.props.location}
/>
-
-
- Explore
-
-
+ {(!this.props.showDotComMarketing ||
+ !!this.props.authenticatedUser ||
+ this.props.location.pathname !== '/welcome') && (
+
+
+ Explore
+
+
+ )}
{!this.props.authenticatedUser && (
<>
-
-
- Extensions
-
-
+ {this.props.location.pathname !== '/welcome' && (
+
+
+ Extensions
+
+
+ )}
{this.props.location.pathname !== '/sign-in' && (
@@ -78,33 +89,37 @@ export class NavLinks extends React.PureComponent {
)}
- {showDotComMarketing && (
+ {this.props.showDotComMarketing && (
About
)}
-
-
- Help
-
-
+ {this.props.location.pathname !== '/welcome' && (
+
+
+ Help
+
+
+ )}
>
)}
-
+ {this.props.location.pathname !== '/welcome' && (
+
+ )}
{this.props.authenticatedUser && (
diff --git a/web/src/nav/UserNavItem.tsx b/web/src/nav/UserNavItem.tsx
index b253d83b843..528b908f13f 100644
--- a/web/src/nav/UserNavItem.tsx
+++ b/web/src/nav/UserNavItem.tsx
@@ -11,7 +11,6 @@ interface Props {
authenticatedUser: GQL.IUser
isLightTheme: boolean
onThemeChange: () => void
- isMainPage?: boolean
showAbout: boolean
showDiscussions: boolean
}
diff --git a/web/src/nav/__snapshots__/NavLinks.test.tsx.snap b/web/src/nav/__snapshots__/NavLinks.test.tsx.snap
new file mode 100644
index 00000000000..14fa222501f
--- /dev/null
+++ b/web/src/nav/__snapshots__/NavLinks.test.tsx.snap
@@ -0,0 +1,125 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`NavLinks authed Sourcegraph.com /foo 1`] = `
+Array [
+ "Search /search",
+ "Welcome /welcome",
+ "ActionsNavItems",
+ "Explore /explore",
+ "CommandListPopoverButton",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks authed Sourcegraph.com /search 1`] = `
+Array [
+ "Welcome /welcome",
+ "ActionsNavItems",
+ "Explore /explore",
+ "CommandListPopoverButton",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks authed Sourcegraph.com /welcome 1`] = `
+Array [
+ "Docs https://docs.sourcegraph.com",
+ "ActionsNavItems",
+ "Explore /explore",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks authed self-hosted /foo 1`] = `
+Array [
+ "Search /search",
+ "ActionsNavItems",
+ "Explore /explore",
+ "CommandListPopoverButton",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks authed self-hosted /search 1`] = `
+Array [
+ "ActionsNavItems",
+ "Explore /explore",
+ "CommandListPopoverButton",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks authed self-hosted /welcome 1`] = `
+Array [
+ "ActionsNavItems",
+ "Explore /explore",
+ "UserNavItem",
+]
+`;
+
+exports[`NavLinks unauthed Sourcegraph.com /foo 1`] = `
+Array [
+ "Search /search",
+ "Welcome /welcome",
+ "ActionsNavItems",
+ "Explore /explore",
+ "Extensions /extensions",
+ "Sign in /sign-in",
+ "About https://about.sourcegraph.com",
+ "Help /help",
+ "CommandListPopoverButton",
+]
+`;
+
+exports[`NavLinks unauthed Sourcegraph.com /search 1`] = `
+Array [
+ "Welcome /welcome",
+ "ActionsNavItems",
+ "Explore /explore",
+ "Extensions /extensions",
+ "Sign in /sign-in",
+ "About https://about.sourcegraph.com",
+ "Help /help",
+ "CommandListPopoverButton",
+]
+`;
+
+exports[`NavLinks unauthed Sourcegraph.com /welcome 1`] = `
+Array [
+ "Docs https://docs.sourcegraph.com",
+ "ActionsNavItems",
+ "Sign in /sign-in",
+ "About https://about.sourcegraph.com",
+]
+`;
+
+exports[`NavLinks unauthed self-hosted /foo 1`] = `
+Array [
+ "Search /search",
+ "ActionsNavItems",
+ "Explore /explore",
+ "Extensions /extensions",
+ "Sign in /sign-in",
+ "Help /help",
+ "CommandListPopoverButton",
+]
+`;
+
+exports[`NavLinks unauthed self-hosted /search 1`] = `
+Array [
+ "ActionsNavItems",
+ "Explore /explore",
+ "Extensions /extensions",
+ "Sign in /sign-in",
+ "Help /help",
+ "CommandListPopoverButton",
+]
+`;
+
+exports[`NavLinks unauthed self-hosted /welcome 1`] = `
+Array [
+ "ActionsNavItems",
+ "Explore /explore",
+ "Sign in /sign-in",
+]
+`;
diff --git a/web/src/routes.tsx b/web/src/routes.tsx
index dc4c7ca8fdc..364ba5e3c10 100644
--- a/web/src/routes.tsx
+++ b/web/src/routes.tsx
@@ -2,9 +2,6 @@ import * as React from 'react'
import { Redirect, RouteComponentProps } from 'react-router'
import { LayoutProps } from './Layout'
import { parseSearchURLQuery } from './search'
-const MainPage = React.lazy(async () => ({
- default: (await import('./search/input/MainPage')).MainPage,
-}))
const SearchPage = React.lazy(async () => ({
default: (await import('./search/input/SearchPage')).SearchPage,
}))
@@ -78,12 +75,11 @@ export const routes: ReadonlyArray = [
{
path: '/',
render: (props: any) =>
- window.context.sourcegraphDotComMode && !props.user ? : ,
- exact: true,
- },
- {
- path: '/start',
- render: (props: any) => ,
+ window.context.sourcegraphDotComMode && !props.user ? (
+
+ ) : (
+
+ ),
exact: true,
},
{
diff --git a/web/src/search/input/CodeIntellifyBlob.tsx b/web/src/search/input/CodeIntellifyBlob.tsx
deleted file mode 100644
index 8c24255a81f..00000000000
--- a/web/src/search/input/CodeIntellifyBlob.tsx
+++ /dev/null
@@ -1,353 +0,0 @@
-import { createHoverifier, findPositionsFromEvents, HoveredToken, HoverState } from '@sourcegraph/codeintellify'
-import { getTokenAtPosition } from '@sourcegraph/codeintellify/lib/token_position'
-import { Position } from '@sourcegraph/extension-api-types'
-import * as H from 'history'
-import * as React from 'react'
-import { Subject, Subscription } from 'rxjs'
-import { catchError, filter, map, withLatestFrom } from 'rxjs/operators'
-import { ActionItemProps } from '../../../../shared/src/actions/ActionItem'
-import { HoverMerged } from '../../../../shared/src/api/client/types/hover'
-import { ExtensionsControllerProps } from '../../../../shared/src/extensions/controller'
-import * as GQL from '../../../../shared/src/graphql/schema'
-import { getHoverActions } from '../../../../shared/src/hover/actions'
-import { HoverContext, HoverOverlay } from '../../../../shared/src/hover/HoverOverlay'
-import { getModeFromPath } from '../../../../shared/src/languages'
-import { PlatformContextProps } from '../../../../shared/src/platform/context'
-import { ErrorLike, isErrorLike } from '../../../../shared/src/util/errors'
-import { isDefined, propertyIsDefined } from '../../../../shared/src/util/types'
-import { FileSpec, ModeSpec, PositionSpec, RepoSpec, ResolvedRevSpec, RevSpec } from '../../../../shared/src/util/url'
-import { getHover } from '../../backend/features'
-import { fetchBlob } from '../../repo/blob/BlobPage'
-
-interface Props extends ExtensionsControllerProps, PlatformContextProps {
- history: H.History
- location: H.Location
- className: string
- startLine: number
- endLine: number
- parentElement: string
-
- overlayPortal?: HTMLElement
- tooltipClass: string
- defaultHoverPosition: Position
-}
-
-interface State extends HoverState {
- /**
- * The blob data or error that happened.
- * undefined while loading.
- */
- blobOrError?: GQL.IGitBlob | ErrorLike
- target?: EventTarget
-}
-
-const domFunctions = {
- getCodeElementFromTarget: (target: HTMLElement): HTMLTableCellElement | null => {
- // If the target is part of the decoration, return null.
- if (
- target.classList.contains('line-decoration-attachment') ||
- target.classList.contains('line-decoration-attachment__contents')
- ) {
- return null
- }
-
- const row = target.closest('tr')
- if (!row) {
- return null
- }
-
- return row.cells[1]
- },
- getCodeElementFromLineNumber: (codeView: HTMLElement, line: number): HTMLElement | null => {
- const lineNumberElement = codeView.querySelector(`td[data-line="${line}"]`)
-
- if (!lineNumberElement) {
- return null
- }
- return lineNumberElement.nextElementSibling as HTMLElement | null
- },
- getLineNumberFromCodeElement: (codeCell: HTMLElement): number => {
- const row = codeCell.closest('tr')
- if (!row) {
- throw new Error('Could not find closest row for codeCell')
- }
- const numberCell = row.cells[0]
- if (!numberCell || !numberCell.dataset.line) {
- throw new Error('Could not find line number')
- }
- return parseInt(numberCell.dataset.line, 10)
- },
-}
-
-const REPO_NAME = 'github.com/gorilla/mux'
-const COMMIT_ID = '9e1f5955c0d22b55d9e20d6faa28589f83b2faca'
-const REV = undefined
-const FILE_PATH = 'mux.go'
-
-export class CodeIntellifyBlob extends React.Component {
- /** Emits whenever the ref callback for the code element is called */
- private codeViewElements = new Subject()
- private nextCodeViewElement = (element: HTMLElement | null) => this.codeViewElements.next(element)
-
- /** Emits whenever the ref callback for the hover element is called */
- private hoverOverlayElements = new Subject()
- private nextOverlayElement = (element: HTMLElement | null) => this.hoverOverlayElements.next(element)
-
- /** Emits whenever the ref callback for the demo file element is called */
- private codeIntellifyBlobElements = new Subject()
- private nextCodeIntellifyBlobElements = (element: HTMLElement | null) =>
- this.codeIntellifyBlobElements.next(element)
-
- /** Emits when the close button was clicked */
- private closeButtonClicks = new Subject()
- private nextCloseButtonClick = (event: MouseEvent) => this.closeButtonClicks.next(event)
-
- private subscriptions = new Subscription()
-
- private componentUpdates = new Subject()
-
- private target: EventTarget | null = null
-
- constructor(props: Props) {
- super(props)
- this.state = {}
-
- const hoverifier = createHoverifier<
- RepoSpec & RevSpec & FileSpec & ResolvedRevSpec,
- HoverMerged,
- ActionItemProps
- >({
- closeButtonClicks: this.closeButtonClicks,
- hoverOverlayElements: this.hoverOverlayElements,
- hoverOverlayRerenders: this.componentUpdates.pipe(
- withLatestFrom(this.hoverOverlayElements, this.codeIntellifyBlobElements),
- map(([, hoverOverlayElement, codeIntellifyBlobElement]) => ({
- hoverOverlayElement,
- codeIntellifyBlobElement,
- })),
- filter(propertyIsDefined('codeIntellifyBlobElement')),
- map(({ hoverOverlayElement, codeIntellifyBlobElement }) => ({
- hoverOverlayElement,
- relativeElement: codeIntellifyBlobElement.closest(this.props.parentElement) as HTMLElement | null,
- })),
- // Can't reposition HoverOverlay or file weren't rendered
- filter(propertyIsDefined('relativeElement')),
- filter(propertyIsDefined('hoverOverlayElement'))
- ),
- getHover: hoveredToken => getHover(this.getLSPTextDocumentPositionParams(hoveredToken), this.props),
- getActions: context => getHoverActions(this.props, context),
- })
-
- this.subscriptions.add(hoverifier)
- const positionEvents = this.codeViewElements.pipe(
- filter(isDefined),
- findPositionsFromEvents(domFunctions)
- )
-
- const targets = positionEvents.pipe(map(({ event: { target } }) => target))
-
- targets.subscribe(target => (this.target = target))
-
- this.subscriptions.add(
- hoverifier.hoverify({
- positionEvents,
- resolveContext: () => ({
- repoName: REPO_NAME,
- commitID: COMMIT_ID,
- rev: REV || '',
- filePath: FILE_PATH,
- }),
- dom: domFunctions,
- })
- )
-
- this.subscriptions.add(hoverifier.hoverStateUpdates.subscribe(update => this.setState(update)))
-
- this.subscriptions.add(
- this.codeViewElements
- .pipe(
- filter(isDefined),
- map(codeView => getTokenAtPosition(codeView, props.defaultHoverPosition, domFunctions)),
- filter(isDefined)
- )
- .subscribe(token => {
- const showOnHomepage = props.className === 'code-intellify-container' && window.innerWidth >= 1393
- const showOnModal =
- props.className === 'code-intellify-container-modal' && window.innerWidth >= 1275
- if (showOnHomepage || showOnModal) {
- token.click()
- }
- })
- )
- }
-
- public componentDidMount(): void {
- // Fetch repository revision.
- fetchBlob({
- repoName: REPO_NAME,
- commitID: COMMIT_ID,
- filePath: FILE_PATH,
- isLightTheme: false,
- disableTimeout: false,
- })
- .pipe(
- catchError(error => {
- console.error(error)
- return [error]
- })
- )
- .subscribe(blobOrError => this.setState({ blobOrError }), err => console.error(err))
-
- this.componentUpdates.next()
-
- this.subscriptions.add(
- this.props.extensionsController.services.model.model.next({
- ...this.props.extensionsController.services.model.model.value,
- visibleViewComponents: [
- {
- type: 'textEditor',
- item: {
- uri: `git://github.com/gorilla/mux?9e1f5955c0d22b55d9e20d6faa28589f83b2faca#mux.go`,
- languageId: 'go',
- text: '',
- },
- selections: [],
- isActive: true,
- },
- ],
- })
- )
- }
-
- public componentDidUpdate(): void {
- this.componentUpdates.next()
- }
-
- private getLSPTextDocumentPositionParams(
- position: HoveredToken & RepoSpec & RevSpec & FileSpec & ResolvedRevSpec
- ): RepoSpec & RevSpec & ResolvedRevSpec & FileSpec & PositionSpec & ModeSpec {
- return {
- repoName: position.repoName,
- filePath: position.filePath,
- commitID: position.commitID,
- rev: position.rev,
- mode: getModeFromPath(FILE_PATH),
- position,
- }
- }
-
- public render(): JSX.Element {
- if (!this.state.blobOrError) {
- // Render placeholder for layout before content is fetched.
- return Loading...
- }
-
- const hoverOverlayProps = this.adjustHoverOverlayPosition(this.target)
-
- return (
-
-
- {!isErrorLike(this.state.blobOrError) && (
-
- )}
- {this.state.hoverOverlayProps && (
-
- )}
-
- )
- }
- /**
- * This function adjusts the position of the hoverOverlay so that it does not overflow on the right side
- * of the viewport. If a hoverOverlay will exceed the viewport, this function will adjust the position
- * so that it aligns the right side of the hover overlay with the right side of the target element.
- *
- */
- private adjustHoverOverlayPosition(
- target: EventTarget | null
- ): HoverState['hoverOverlayProps'] {
- const viewPortEdge = window.innerWidth
- if (!this.state.hoverOverlayProps) {
- return undefined
- }
- if (!target) {
- return this.state.hoverOverlayProps
- }
- const { overlayPosition, ...rest } = this.state.hoverOverlayProps
-
- const targetBounds = (target as HTMLElement).getBoundingClientRect()
- let newOverlayPosition: { top: number; left: number } = overlayPosition!
-
- if (overlayPosition && viewPortEdge < targetBounds.left + 512 && targetBounds.right - 512 >= 0) {
- const containerWidth = (document.querySelector(
- this.props.parentElement
- ) as HTMLElement).parentElement!.getBoundingClientRect().width
-
- const parentWidth = (document.querySelector(
- this.props.parentElement
- ) as HTMLElement).getBoundingClientRect().width
-
- // One side of the total horizontal margin.
- const halfMarginWidth = (viewPortEdge - containerWidth) / 2
- // The difference between the viewport width and parent width. We need to subtract this because
- // `left` is relative to the parent, whereas targetBounds.right is relative to the viewport.
- const relativeElementDifference = viewPortEdge - parentWidth
-
- newOverlayPosition = {
- top: overlayPosition.top,
- // 512 is the width of a hoverOverlay.
- left: targetBounds.right - 512 - relativeElementDifference + halfMarginWidth,
- }
- }
- return { ...rest, overlayPosition: newOverlayPosition }
- }
-}
-
-/**
- * We can only fetch blobs as an entire file. For demo purposes, we only want to show part of the file.
- * This function trims the HTML string of the file that will be code-intellfied on the homepage to only show
- * the lines that we specify. It makes some assumptions for this specific use case, such as the presence
- * of a single table and tbody element in the html, so be careful when changing.
- */
-function trimHTMLString(html: string, startLine: number, endLine: number): string {
- const domParser = new DOMParser()
- const doc = domParser.parseFromString(html, 'text/html')
- const startToRemove = doc.querySelectorAll(`tr:nth-child(n + 0):nth-child(-n + ${startLine})`)
- const endToRemove = doc.querySelectorAll(`tr:nth-child(n + ${endLine})`)
-
- const elementsToRemove = [...startToRemove, ...endToRemove]
- const tableEl = doc.querySelector('tbody')! // assume a single tbody element will exist in blob HTML
-
- for (const el of elementsToRemove) {
- tableEl.removeChild(el)
- }
-
- const xmlSerializer = new XMLSerializer()
- const tbl = doc.querySelector('table')! // assume a single table element will exist in blob HTML
- const trimmedHTMLString = xmlSerializer.serializeToString(tbl)
-
- return trimmedHTMLString
-}
diff --git a/web/src/search/input/MainPage.scss b/web/src/search/input/MainPage.scss
deleted file mode 100644
index da1d2c4f12b..00000000000
--- a/web/src/search/input/MainPage.scss
+++ /dev/null
@@ -1,1080 +0,0 @@
-// Stylelint rules
-// stylelint-disable value-no-vendor-prefix
-// Webkit prefixes are needed
-// stylelint-disable color-named
-// stylelint-disable color-hex-length
-// RGBA is prefered in some CSS properties and hex length should be consistant.
-
-@import '../../nav/GlobalNavbar';
-@import './SearchFilterChips';
-@import './QueryInput';
-@import './QueryInputForModal';
-@import '../saved-queries/SavedQueries';
-
-body.modal-open {
- overflow: hidden !important;
- max-height: 100%;
-}
-body.main-page {
- .nav-links {
- @media screen and (max-width: $media-md) {
- display: none !important;
- }
- @media screen and (max-height: 420px) {
- display: none !important;
- }
- }
- .query-input2__suggestions,
- .query-input2-for-modal__suggestions {
- overflow-y: auto !important;
- overflow-x: hidden !important;
- ul {
- li {
- overflow: hidden !important;
- }
- }
- }
- .query-input2-for-modal__suggestions {
- .query-input2__loading-notifier {
- top: 1rem;
- }
- }
- .hover-overlay__content.hover-overlay__row {
- p {
- font-size: 14px !important;
- }
- }
-}
-.main-page {
- // Colors to match about.sourcegraph.com
- $light-2: #7a8fb8;
- $light-6: #afbcd4;
- $light-11: #f2f4f8;
- $dark-1: #566e9f;
- $dark-4: #405377;
- $dark-11: #0e121b;
- // Viewport breakpoints to match about.sourcegraph.com
- $media-sm: 576px;
- $media-md: 768px;
- $media-lg: 992px;
- width: 100%;
- display: block;
-
- &__logo {
- flex: 0 0 auto;
- display: flex;
- align-items: center;
- width: 20rem;
- margin-top: 6rem;
- max-width: 90%;
- }
-
- &__container {
- width: 100%;
- margin-top: 4rem;
- position: relative;
- }
-
- &__input-container {
- width: 100%;
- display: flex;
- }
-
- &__input-sub-container {
- width: 100%;
- display: block;
- margin-top: 0.5rem;
- margin-bottom: 2rem;
- }
-
- &__query-container {
- flex: 0 0 auto;
- width: 49rem;
- max-width: 90%;
- position: relative;
- }
- @media screen and (max-width: 500px) {
- .btn-primary,
- .btn-secondary {
- width: 100%;
- }
- }
- .hero-section {
- width: 100%;
- position: sticky;
- position: -webkit-sticky;
- @media screen and (min-height: 650px) and (min-width: 376px) {
- top: 44px;
- }
- z-index: 1;
-
- &__bg {
- background: #08335f;
- background: linear-gradient(20deg, #08335f 0%, #000000 65%);
- position: fixed;
- z-index: -1;
- top: 0;
- width: 100%;
- height: 100%;
- }
- .hero-container {
- width: 100%;
- padding-right: 0.75rem;
- padding-left: 0.75rem;
- margin-right: auto;
- margin-left: auto;
- // stylelint-disable-next-line declaration-property-unit-whitelist
- padding-top: 15vh;
- padding-bottom: 10rem;
- min-height: calc(70vh - 8rem);
- @media screen and (max-width: $media-md) {
- padding-top: 2rem;
- padding-bottom: 4rem;
- min-height: calc(100vh - 8rem);
- .row {
- // stylelint-disable-next-line declaration-property-unit-whitelist
- margin-top: 5vh;
- }
- }
- @media screen and (max-width: 800px) and (max-height: 420px) {
- padding-top: 2rem;
- padding-bottom: 4rem;
- min-height: calc(100vh - 8rem);
- .row {
- // stylelint-disable-next-line declaration-property-unit-whitelist
- margin-top: 10vh;
- }
- }
- h1 {
- font-size: 54px;
- line-height: normal;
- margin-top: -0.75rem;
-
- background: radial-gradient(closest-side, #019fde, #b200f8, #f96216);
- background-size: 600% 600%;
- -webkit-background-clip: text;
- -webkit-text-fill-color: transparent;
- -webkit-animation: bg-animation 90s ease infinite;
- animation: bg-animation 90s ease infinite;
-
- @keyframes bg-animation {
- 0% {
- background-position: 0% 17%;
- }
- 50% {
- background-position: 90% 84%;
- }
- 100% {
- background-position: 0% 17%;
- }
- }
- }
-
- h2 {
- font-size: 22px;
- font-weight: 400;
- margin-bottom: 0;
- }
- p {
- font-size: 16px;
- padding-right: 2rem;
- }
- .btn {
- margin-right: 1rem;
- margin-bottom: 1rem;
- }
- .btn-primary {
- border: none;
- }
- }
- .hover-overlay__actions,
- .hover-overlay__action {
- .btn {
- margin: 0;
- }
- }
- }
- .company-logos {
- background-color: #151c28;
- position: sticky;
- position: -webkit-sticky;
- @media screen and (max-width: 800px) and (max-height: 420px) {
- padding-top: 2rem;
- padding-bottom: 3rem;
- // stylelint-disable-next-line declaration-property-unit-whitelist
- min-height: calc(100vh - 8rem);
- .row {
- // stylelint-disable-next-line declaration-property-unit-whitelist
- margin-top: calc(50vh - 60%);
- }
- }
- z-index: 2;
- box-shadow: rgba(0, 0, 0, 0.35) 0 4rem 8rem 12rem;
- padding-top: 1rem;
- padding-bottom: 1rem;
- width: 100%;
- h2 {
- text-align: center;
- font-size: 24px;
- font-weight: 500;
- margin-bottom: 0;
- }
- p {
- font-size: 16px;
- padding-right: 2rem;
- }
- .row {
- padding-top: 2rem;
- padding-bottom: 0;
- }
-
- .logo {
- padding: 2rem 0;
-
- &-image {
- width: 100%;
- background-position: center;
- background-repeat: no-repeat;
- background-size: contain;
- // stylelint-disable-next-line declaration-property-unit-whitelist
- height: 115px;
- }
- }
- .logo4 {
- background-image: url('');
- }
- .logo3 {
- background-image: url('');
- }
- .logo2 {
- background-image: url('');
- }
- .logo1 {
- background-image: url('');
- }
- }
- .about-section {
- position: sticky;
- position: -webkit-sticky;
- @media screen and (max-width: 800px) and (max-height: 420px) {
- padding-top: 4rem;
- padding-bottom: 4rem;
- // stylelint-disable-next-line declaration-property-unit-whitelist
- min-height: calc(100vh - 8rem);
- .row {
- // stylelint-disable-next-line declaration-property-unit-whitelist
- margin-top: calc(50vh - 60%);
- }
- }
- @media screen and (min-width: $media-md) and (min-height: 420px) {
- top: 0;
- }
- &:nth-of-type(2) {
- h1 {
- padding-bottom: 1.5rem;
- display: inline-block;
- -webkit-border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#00b3f8),
- to(#7042f8),
- color-stop(0%, #00b3f8),
- color-stop(100%, #7042f8)
- )
- 0 0 4 0 repeat repeat;
- border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#00b3f8),
- to(#7042f8),
- color-stop(0%, #00b3f8),
- color-stop(50%, #7042f8)
- )
- 0 0 4 0 repeat repeat;
- border-bottom: 4px solid;
- }
- }
- &:nth-of-type(3) {
- h1 {
- padding-bottom: 1.5rem;
- display: inline-block;
- -webkit-border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#dc3876),
- to(#9121f9),
- color-stop(0%, #dc3876),
- color-stop(100%, #9121f9)
- )
- 0 0 4 0 repeat repeat;
- border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#dc3876),
- to(#9121f9),
- color-stop(0%, #dc3876),
- color-stop(100%, #9121f9)
- )
- 0 0 4 0 repeat repeat;
- border-bottom: 4px solid;
- }
- }
- &:nth-of-type(4) {
- h1 {
- padding-bottom: 1.5rem;
- display: inline-block;
- -webkit-border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#dc3876),
- to(#9121f9),
- color-stop(0%, #dc3876),
- color-stop(100%, #9121f9)
- )
- 0 0 4 0 repeat repeat;
- border-image: -webkit-gradient(
- linear,
- left top,
- right top,
- from(#f85f1b),
- to(#0fd5f8),
- color-stop(0%, #f85f1b),
- color-stop(67%, #d9367b),
- color-stop(84%, #8b27f9),
- color-stop(100%, #0fd5f8)
- )
- 0 0 4 0 repeat repeat;
- border-bottom: 4px solid;
- }
- }
-
- h1 {
- font-size: 54px;
- padding-top: 0;
- @media screen and (max-width: $media-md) {
- line-height: normal;
- }
- }
- h2 {
- font-size: 22px;
- font-weight: 400;
- margin-bottom: 0;
- padding-left: 0.25rem;
- }
- p {
- font-size: 16px;
- padding-right: 2rem;
- }
- .row {
- padding-top: 12rem;
- padding-bottom: 8rem;
- }
- // stylelint-disable-next-line declaration-block-no-duplicate-properties
- position: sticky;
- // stylelint-disable-next-line declaration-block-no-duplicate-properties
- position: -webkit-sticky;
- z-index: 2;
- box-shadow: rgba(0, 0, 0, 0.35) 0 4rem 8rem 12rem;
- // stylelint-disable-next-line no-duplicate-selectors
- &:nth-of-type(2) {
- box-shadow: none;
- }
- padding-top: 2rem;
- padding-bottom: 4rem;
- background-color: #000000;
- width: 100%;
- }
- .up-next-section {
- h1 {
- font-size: 54px;
- padding-top: 0;
- color: #1e232e;
- @media screen and (max-width: $media-md) {
- line-height: normal;
- }
- }
- h2 {
- font-size: 22px;
- font-weight: 400;
- margin-bottom: 0;
- color: #000000;
- }
- p {
- font-size: 16px;
- padding-right: 2rem;
- color: #000000;
- }
- .btn {
- margin-right: 1rem;
- margin-bottom: 1rem;
- }
- .col-md-12 {
- padding-bottom: 2rem;
- @media screen and (min-width: $media-md) {
- padding-bottom: 6rem;
- }
- @media screen and (min-height: 420px) {
- padding-bottom: 6rem;
- }
- }
- position: sticky;
- position: -webkit-sticky;
- z-index: 3;
- padding-top: 2rem;
- padding-bottom: 4rem;
- background: #e6ebf3;
- width: 100%;
- .row {
- padding-top: 12rem;
- padding-bottom: 8rem;
- }
- }
- .blog-callout {
- h1,
- h2,
- p {
- color: #fafcff;
- }
- p a {
- color: #fafcfe;
- text-decoration: underline;
- }
- .row {
- padding-top: 1.5rem;
- padding-bottom: 0.5rem;
- }
- background: #f96216;
- background: -moz-radial-gradient(
- center,
- ellipse cover,
- rgba(249, 98, 22, 1) 0%,
- rgba(178, 0, 248, 1) 51%,
- rgba(1, 159, 222, 1) 100%
- );
- background: -webkit-radial-gradient(
- center,
- ellipse cover,
- rgba(249, 98, 22, 1) 0%,
- rgba(178, 0, 248, 1) 51%,
- rgba(1, 159, 222, 1) 100%
- );
- background: radial-gradient(
- ellipse at center,
- rgba(249, 98, 22, 1) 0%,
- rgba(178, 0, 248, 1) 51%,
- rgba(1, 159, 222, 1) 100%
- );
- background-size: 200% 800%;
- background-position: calc(100vw * 1) calc(100vh * 1.15);
- }
- .modal-search,
- .modal-intelligence,
- .modal-integrations {
- .btn-close-top {
- width: 2.5rem;
- height: 2.5rem;
- margin-right: 1rem;
- border-radius: 50%;
- border: none;
- background-color: #222222;
- color: #fafafa;
- position: absolute;
- right: 0;
- cursor: pointer;
- z-index: 9999;
- padding: 0;
- }
- .btn-close-bottom {
- height: 2.5rem;
- border-radius: 1.5rem;
- border: none;
- background-color: #333333;
- color: #fafafa;
- position: absolute;
- right: calc(50% - 53px);
- bottom: 3rem;
- cursor: pointer;
- z-index: 9999;
- font-size: 16px;
- padding-left: 1.4rem;
- padding-right: 1rem;
- padding-top: 0.25rem;
- svg {
- // stylelint-disable-next-line declaration-property-unit-whitelist
- margin-top: -2px;
- }
- }
- .copy-section {
- h1 {
- font-size: 74px;
- padding-top: 0;
- line-height: 4.15rem;
- }
- h2 {
- font-size: 32px;
- font-weight: 400;
- margin-bottom: 1rem;
- }
- h3 {
- font-size: 28px;
- font-weight: 400;
- margin-bottom: 1rem;
- }
- p {
- font-size: 18px;
- padding-right: 2rem;
- }
- @media screen and (max-width: $media-lg) {
- h1 {
- font-size: 52px;
- padding-top: 0;
- line-height: 3rem;
- }
- h2 {
- font-size: 20px;
- font-weight: 400;
- margin-bottom: 0.5rem;
- }
- h3 {
- font-size: 20px;
- font-weight: 400;
- margin-bottom: 0.5rem;
- }
- p {
- font-size: 16px;
- padding-right: 2rem;
- }
- }
- }
-
- z-index: 990;
- padding-top: 5rem;
- position: fixed;
- top: 0;
- width: 100%;
- height: 100%;
- background-color: #000000;
- overflow-y: scroll !important;
- -webkit-overflow-scrolling: touch;
-
- .container {
- display: block;
- position: relative;
- padding-bottom: 10rem;
- }
-
- .search-row {
- width: 100%;
- @media screen and (max-width: $media-md) {
- input {
- font-size: 16px;
- }
- }
- position: -webkit-sticky;
- top: -5rem;
- z-index: 9999;
- padding-top: 1rem;
- padding-bottom: 0.5rem;
- background-color: #000000;
-
- form {
- margin-top: 0;
- padding-top: 0;
- }
- }
- .btn {
- margin-right: 2rem;
- transition: all 0.325s ease-in-out;
- margin-bottom: 1rem;
- }
- .search-modal-container {
- .modal-copy-row {
- h1,
- h2,
- h3,
- h4,
- h5,
- p {
- transition: all 350ms ease;
- color: #666666 !important;
- }
- }
- .activesec {
- h1,
- h2,
- h3,
- h4,
- h5,
- p {
- transition: all 350ms ease;
-
- color: #ffffff !important;
- }
- }
- }
- .modal-copy-row {
- padding-bottom: 4rem;
- }
-
- .main-page__container {
- width: 100%;
- max-width: 100%;
- font-size: 20px;
- font-weight: 400;
- letter-spacing: 0.05rem;
- .search-button > .search-button__btn {
- padding-left: 2rem;
- padding-right: 2rem;
- margin-right: 0;
- font-size: 20px;
- font-weight: 400;
- letter-spacing: 0.05rem;
- height: 3rem;
- @media screen and (max-width: $media-md) {
- margin-left: 0.25rem;
- padding-left: 0.25rem;
- padding-right: 0.25rem;
- }
- }
- }
- .form-control {
- height: 3rem;
- font-size: 20px;
- font-weight: 400;
- letter-spacing: 0.05rem;
-
- // Fake Focus
- color: var(--input-color);
- background-color: var(--input-bg);
- border-color: #1c7ed6;
- outline: 0;
- box-shadow: 0 0 0.125rem 0.125rem rgba(28, 126, 214, 0.25);
- }
- .active {
- box-shadow: rgba(69, 127, 183, 0.7) 0 0 0.16rem 0.125rem;
- }
-
- .intelligence-row {
- padding-top: 4rem;
- }
-
- .hover-overlay__actions,
- .hover-overlay__action {
- .btn {
- margin: 0;
- }
- }
- .action-row {
- justify-content: center;
- font-size: 18px;
- margin-top: 4rem;
- .action-col {
- padding-bottom: 2rem;
- a {
- color: #1c7ed6;
- }
- p {
- margin-bottom: 0.5rem;
- }
- }
- }
- }
- .modal-integrations {
- .btn-integrations {
- display: inline-flex;
- align-items: center;
- font-size: 20px;
- font-weight: 400;
- .logo-icon {
- height: 1.5rem;
- width: 1.5rem;
- display: inline-block;
- margin-right: 0.5rem;
- background-size: contain;
- background-repeat: no-repeat;
- }
- }
- .btn-github {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/github.png);
- }
- }
- .btn-gitlab {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/gitlab.svg);
- }
- }
- .btn-phabricator {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/phabricator.png);
- }
- }
- .btn-chrome {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/chrome.svg);
- }
- }
- .btn-firefox {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/firefox.svg);
- }
- }
- .btn-atom {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/atom.svg);
- }
- }
- .btn-intellij {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/jetbrains.svg);
- }
- }
- .btn-vscode {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/vscode.svg);
- }
- }
- .btn-sublime {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/sublime.svg);
- }
- }
- .btn-vim {
- .logo-icon {
- background-image: url(https://about.sourcegraph.com/integrations/vim.svg);
- }
- }
- }
- .modal-close {
- display: none;
- opacity: 0;
- }
- .modal-open {
- z-index: 990;
- // Animations
- animation: display 350ms ease-in 0s;
- @keyframes display {
- from {
- opacity: 0;
- }
- to {
- opacity: 1;
- }
- }
-
- .modal-header {
- opacity: 0;
- animation: display 350ms ease-in 250ms forwards;
- @keyframes display {
- from {
- opacity: 0;
- transform: translateY(1.25rem);
- }
- to {
- opacity: 1;
- transform: translateY(1.25rem);
- }
- }
- }
- .search-row-animate {
- animation: display 350ms ease-in 400ms forwards;
- @keyframes display {
- from {
- opacity: 0;
- transform: translateY(1.25rem);
- }
- to {
- opacity: 1;
- transform: translateY(0);
- }
- }
- opacity: 1;
- }
- @for $i from 1 through 8 {
- .modal-copy-row:nth-child(#{$i}) {
- $hold-time: (150 * $i) + 300;
- opacity: 0;
- animation: display 350ms ease-in-out #{$hold-time}ms forwards;
- @keyframes display {
- from {
- opacity: 0;
- transform: translateY(1.25rem);
- }
- to {
- opacity: 1;
- transform: translateY(0);
- }
- }
- }
- }
- // End Animations
- }
- .modal-closing {
- animation: closeModal 350ms ease-in 0s forwards;
- @keyframes closeModal {
- from {
- opacity: 1;
- }
- to {
- opacity: 0;
- }
- }
- }
-
- .query-input2__suggestions {
- z-index: 999;
- }
- .code-intellify-container,
- .code-intellify-container-modal {
- background-color: #10111a;
- .code-header {
- width: 100%;
- background-color: #232532;
- border-top-left-radius: 0.5rem;
- border-top-right-radius: 0.5rem;
- padding-left: 0.5rem;
- padding-right: 0.5rem;
- padding-bottom: 0.25rem;
- padding-top: 0.25rem;
- &__title,
- &__link {
- font-family: SFMono-Regular, SF Mono, Consolas, Menlo, DejaVu Sans Mono, monospace;
- font-size: 12px;
- }
- &__link {
- float: right;
- padding-top: 0.25rem;
- a {
- color: #329af0;
- }
- }
- }
- }
- .footer-section {
- // Designers Note: Most of the footer is pixel-aligned to match about.sourcegraph.com.
-
- // Redefine a REM to match about.sourcegraph.com
- font-size: 16px;
- //
-
- position: sticky;
- position: -webkit-sticky;
- z-index: 4;
- background: #151c28;
- color: #7a8fb8;
- padding: 4rem 0 1rem 0;
- overflow: hidden !important;
-
- .small__contact {
- padding-top: 1.5rem;
- margin-bottom: -2rem;
- @media screen and (min-width: $media-md) {
- display: none;
- }
- }
- .logo__section {
- a {
- color: $light-6;
- }
- .footer__logo {
- padding: 0 0 1.25rem 0;
- margin: 0 0 0 -0.5rem;
- }
- @media screen and (max-width: $media-md) {
- padding-bottom: 0.2rem;
- margin-top: -2.5rem;
-
- .footer__contact {
- display: none;
- }
- }
- }
-
- .footer__extend {
- transition: all 0.3s ease-in-out;
- * {
- transition: all 0.3s ease-in-out;
- }
- h3 {
- cursor: pointer;
- }
- ul {
- overflow: hidden !important;
- padding-left: 0;
- li {
- @media screen and (max-width: $media-md) {
- transition: all 0.3s ease-in-out;
- position: relative;
- top: -60px;
- overflow: hidden !important;
- }
- }
- }
- @media screen and (min-width: $media-md) {
- input,
- .close--icon {
- display: none;
- }
- }
- @media screen and (max-width: $media-md) {
- &:last-of-type {
- border-bottom: solid 1px $light-2;
- }
- padding-top: 0.5rem;
- border-top: solid 1px $light-2;
- &.community,
- &.company,
- &.features,
- &.resources {
- .close--icon {
- height: 1.125rem;
- width: 1.125rem;
- position: absolute;
- top: calc(1rem - 10px);
- right: 1rem;
- transition: all 0.3s ease-in-out;
-
- svg {
- height: 1.125rem;
- width: 1.125rem;
- transition: transform 0.3s ease-in-out;
- transform: rotate(-45deg);
- }
- svg path {
- fill: $light-2;
- }
- }
- li:last-child {
- padding-bottom: 0.5rem;
- }
- input {
- width: 100%;
- height: 2rem;
- position: absolute;
- top: 0;
- opacity: 0;
- cursor: pointer;
- z-index: 9999;
- // stylelint-disable-next-line scss/selector-no-redundant-nesting-selector
- & + ul {
- display: hidden;
- height: 0;
- padding: 0;
- }
- &:checked + ul {
- display: block;
- height: 100%;
- }
- &:checked + ul li {
- transition: all 0.3s ease-in-out;
- display: block;
- top: 0;
- }
- &:checked ~ .close--icon {
- svg {
- transition: transform 0.3s ease-in-out;
- transform: rotate(0deg);
- }
- }
- }
- }
- }
- }
-
- .social__column {
- .icon {
- margin-left: 1rem;
- svg path {
- fill: $dark-1;
- }
- &:hover {
- svg path {
- fill: $dark-11;
- }
- }
- }
- img,
- .btn {
- margin-bottom: 1rem;
- }
- .copyright {
- color: $dark-1;
- }
- }
- h3 {
- text-transform: uppercase;
- font-weight: 400;
- font-size: 16px;
- letter-spacing: 0.1rem;
- }
- ul {
- color: $light-6;
- margin: 0;
- list-style: none;
-
- li {
- margin-bottom: 0.5rem;
-
- a {
- color: $light-6;
- &:hover {
- color: $light-11;
- text-decoration: underline;
- }
- }
- }
- }
- .copyright__container {
- margin: 3rem 0 0 0;
- padding-top: 1rem;
- border-top: 2px solid $dark-4;
- .copyright {
- float: left;
- @media screen and (max-width: $media-md) {
- float: none;
- }
- }
- .terms {
- float: right;
- a {
- margin-left: 2rem;
- }
- @media screen and (max-width: $media-md) {
- float: none;
- a:first-of-type {
- margin-left: 0;
- }
- }
- }
- }
- }
- .align-footer {
- align-content: flex-end !important;
-
- @media screen and (max-width: $media-lg) {
- padding-top: 2rem;
- }
-
- * {
- align-self: center;
- }
- }
-}
-.small-hidden {
- @media screen and (max-width: $media-lg) {
- display: none !important;
- }
-}
-
-.theme-light {
- .main-page {
- &__sign-in {
- color: $color-light-text-4;
- }
- }
-}
diff --git a/web/src/search/input/MainPage.tsx b/web/src/search/input/MainPage.tsx
deleted file mode 100644
index cc78c88fccf..00000000000
--- a/web/src/search/input/MainPage.tsx
+++ /dev/null
@@ -1,1037 +0,0 @@
-import * as H from 'history'
-import ChevronRightIcon from 'mdi-react/ChevronRightIcon'
-import CloseIcon from 'mdi-react/CloseIcon'
-import MapSearchIcon from 'mdi-react/MapSearchIcon'
-import * as React from 'react'
-import { parseSearchURLQuery } from '..'
-import { ExtensionsControllerProps } from '../../../../shared/src/extensions/controller'
-import * as GQL from '../../../../shared/src/graphql/schema'
-import { PlatformContextProps } from '../../../../shared/src/platform/context'
-import { Form } from '../../components/Form'
-import { HeroPage } from '../../components/HeroPage'
-import { PageTitle } from '../../components/PageTitle'
-import { eventLogger } from '../../tracking/eventLogger'
-import { limitString } from '../../util'
-import { queryIndexOfScope, submitSearch } from '../helpers'
-import { CodeIntellifyBlob } from './CodeIntellifyBlob'
-import { QueryInputForModal } from './QueryInputForModal'
-import { SearchButton } from './SearchButton'
-
-interface Props extends ExtensionsControllerProps, PlatformContextProps {
- authenticatedUser: GQL.IUser | null
- location: H.Location
- history: H.History
- isLightTheme: boolean
- onThemeChange: () => void
- onMainPage: (mainPage: boolean) => void
-}
-
-interface State {
- /** The query value entered by the user in the query input */
- userQuery: string
- // modalXXopen sets a state that the modal is open before animations or closed after animation
- // modalXXclosing sets a state that starts the closing process
- modalSearchOpen: boolean
- modalIntelligenceOpen: boolean
- modalSearchClosing: boolean
- modalIntelligenceClosing: boolean
- modalIntegrationsOpen: boolean
- modalIntegrationsClosing: boolean
- // determine what button in modal is active
- activeButton?: string
- // determine what section inside a modal is active
- activesection?: string
- // animateModalXX starts the animation process after opening.
- animateModalSearch: boolean
- animateModalIntelligence: boolean
- animateModalIntegrations: boolean
- // Manual click state is to determine if animation should be stopped
- manualClick?: boolean
- bgScrollStyle: string
-}
-const heroEyebrow = 'Sourcegraph'
-const heroTitle = 'Search, navigate, and review code. Find answers.'
-const heroCopyTop =
- 'Sourcegraph is a free, open-source, self-hosted code search and navigation tool for developers. Use it with any Git code host for teams of any size.'
-const heroCopyBottom = 'Upgraded features available for enterprise users.'
-
-const searchSections = [
- {
- title: 'Powerful, flexible queries',
- paragraph:
- 'Sourcegraph code search performs full-text searches and supports both regular expression and exact queries. By default, Sourcegraph searches across all your repositories. Our search query syntax allows for advanced queries, such as searching over any branch or commit, narrowing searches by programming language or file pattern, and more.',
- buttons: [
- { query: 'repo:^gitlab.com/ ', text: 'Code on GitLab' },
- { query: 'repogroup:goteam file:\\.go$', text: 'Go Code by Go Team' },
- {
- query: 'repogroup:ethereum file:\\.(txt|md)$ file:(test|spec) ',
- text: 'Core Ethereum test files',
- },
- { query: 'repogroup:angular file:\\.JSON$', text: 'Angular JSON Files' },
- ],
- },
- {
- title: 'Commit diff search',
- paragraph:
- 'Search over commit diffs using type:diff to see how your codebase has changed over time. This is often used to find changes to particular functions, classes, or areas of the codebase when debugging. You can also search within commit diffs on multiple branches by specifying the branches in a repo: field after the @ sign.',
- buttons: [
- { query: 'repo:^github\\.com/apple/swift$@master-next type:diff', text: 'Swift diff' },
- { query: 'repo:^github.com/facebook/react$ file:(test|spec) type:diff', text: 'React test file diff' },
- {
- query: 'repo:^github.com/golang/oauth2$ type:diff',
- text: 'Go oauth2 diff',
- },
- {
- query: 'repo:^github.com/kubernetes/kubernetes$ type:diff statefulset',
- text: 'Kubernetes statefulset diff',
- },
- ],
- },
- {
- title: 'Commit message search',
- paragraph:
- 'Searching over commit messages is supported in Sourcegraph by adding type:commit to your search query. Separately, you can also use the message:"any string" token to filter type:diff searches for a given commit message.',
- buttons: [
- { query: 'type:commit repogroup:angular author:google.com>$ ', text: 'Angular commits by Googlers' },
- { query: 'repogroup:npm type:commit security', text: 'NPM commits mentioning security' },
- {
- query: 'repogroup:ethereum type:commit money loss',
- text: "Ethereum commits mentioning 'money loss'",
- },
- { query: 'repo:^github.com/sourcegraph/sourcegraph type:commit', text: 'Sourcegraph commits' },
- ],
- },
- {
- title: 'Symbol Search',
- paragraph:
- 'Searching for symbols makes it easier to find specific functions, variables and more. Use the type:symbol filter to search for symbol results. Symbol results also appear in typeahead suggestions, so you can jump directly to symbols by name.',
- buttons: [
- { query: 'repogroup:goteam type:symbol httpRouter', text: 'Go code with httpRouter' },
- { query: 'repo:^github.com/apple/swift$ type:symbol main', text: "Calls to 'main' in Swift" },
- {
- query: 'repo:golang/go$ type:symbol sprintf',
- text: 'Go sprintf',
- },
- { query: 'repo:golang/go$ type:symbol newDecoder', text: 'Go newDecoder' },
- ],
- },
-]
-const intelligenceSections = [
- {
- title: 'Code browsing',
- paragraph:
- 'View open source code, like gorilla/mux, on sourcegraph.com, or deploy your own instance to see public code alongside your private code. See how your codebase changes over time in by browsing through branches, commits, and diffs.',
- },
- {
- title: 'Advanced code intelligence',
- paragraph:
- 'Code intelligence makes browsing code easier, with IDE-like hovers, go-to-definition, and find-references on your code, powered by language servers based on the open-source Language Server Protocol.',
- },
- {
- title: 'Hover tooltip',
- paragraph:
- 'Use the hover tooltip to discover and understand your code faster. Click on a token and then go to its definition, other references, or implementations. Speed through reviews by understanding new code, changed code, and what it affects.',
- },
- {
- title: '',
- paragraph:
- 'Code intelligence is powered by language servers based on the open-standard Language Server Protocol (published by Microsoft, with participation from Facebook, Google, Sourcegraph, GitHub, RedHat, Twitter, Salesforce, Eclipse, and others). Visit langserver.org to learn more about the Language Server Protocol, find the latest support for your favorite language, and get involved.',
- },
-]
-
-const integrationsSections = [
- {
- title: 'Connect across your development workflow.',
- paragraph:
- 'Sourcegraph has powerful integrations for every step of development. From planning with code discussion, development with Sourcegraph and IDE extensions, to review in PRs and Issues. Use Sourcegraph integrations to get code intelligence at every step of your workflow.',
- buttons: [],
- },
- {
- title: 'Browser extensions',
- paragraph:
- 'Code intelligence makes browsing code easier, with IDE-like hovers, go-to-definition, and find-references on your code, powered by language servers based on the open-source Language Server Protocol.',
- buttons: [
- {
- id: 'btn-chrome',
- text: 'Chrome',
- link: 'https://chrome.google.com/webstore/detail/sourcegraph/dgjhfomjieaadpoljlnidmbgkdffpack',
- },
- ],
- },
-
- {
- title: 'Code host integrations',
- paragraph:
- 'The Sourcegraph browser extension will add go-to-definition, find-references, hover tooltips, and code search to all files and diffs on supported code hosts. The extension will also add code intelligence and code search to public repositories. ',
- buttons: [
- { id: 'btn-gitlab', text: 'GitLab', link: 'https://docs.sourcegraph.com/integration/browser_extension' },
- { id: 'btn-github', text: 'GitHub', link: 'https://docs.sourcegraph.com/integration/browser_extension' },
- {
- id: 'btn-phabricator',
- text: 'Phabricator',
- link: 'https://docs.sourcegraph.com/integration/browser_extension',
- },
- ],
- },
- {
- title: 'Editor extensions',
- paragraph:
- 'Our editor plugins let you quickly jump to files and search code on your Sourcegraph instance from your editor. Seamlessly jump for development to review without missing a step.',
- buttons: [
- { id: 'btn-atom', text: 'Atom', link: 'https://atom.io/packages/sourcegraph' },
- { id: 'btn-intellij', text: 'IntelliJ', link: 'https://plugins.jetbrains.com/plugin/9682-sourcegraph' },
- {
- id: 'btn-sublime',
- text: 'Sublime',
- link: 'https://github.com/sourcegraph/sourcegraph-sublime',
- },
- {
- id: 'btn-vscode',
- text: 'Visual Studio Code',
- link: 'https://marketplace.visualstudio.com/items?itemName=sourcegraph.sourcegraph',
- },
- ],
- },
-]
-
-const inlineStyle = `
- .layout {
- display: block !important;
- }
- * {
- overflow: visible !important;
- }
- .hero-tooltip {
- z-index: 1;
- position: fixed !important;
- transform: translateY(44px);
- }
- .hover-overlay__contents {
- overflow-y: auto !important;
- max-height: 200px;
- }
- .modal-tooltip {
- z-index: 9999 !important;
- opacity: 1 !important;
- visbility: visbily !important;
- }
-`
-// Set the defauly hover token of the hero tooltip
-const defaultTooltipHeroPosition = { line: 244, character: 11 }
-
-// Set the defauly hover token of the hero tooltip
-const defaultTooltipModalPosition = { line: 248, character: 11 }
-
-/**
- * The main page
- */
-export class MainPage extends React.Component {
- private static HIDE_REPOGROUP_SAMPLE_STORAGE_KEY = 'MainPage/hideRepogroupSample'
-
- private overlayPortal: HTMLElement | undefined
-
- constructor(props: Props) {
- super(props)
-
- const query = parseSearchURLQuery(props.location.search)
-
- this.state = {
- userQuery: query || '',
- modalSearchOpen: false,
- modalSearchClosing: false,
- modalIntelligenceOpen: false,
- modalIntelligenceClosing: false,
- modalIntegrationsOpen: false,
- modalIntegrationsClosing: false,
- manualClick: false,
- activesection: 'none',
- animateModalSearch: false,
- animateModalIntelligence: false,
- animateModalIntegrations: false,
- bgScrollStyle: `
- .feature-card {
- transform: translateY(-0px);
- opacity: .5;
- }
- .global-alerts {
- position: sticky;
- position: -webkit-sticky;
- z-index: 99;
- }
- `,
- }
- }
-
- public componentDidMount(): void {
- eventLogger.logViewEvent('Home')
- if (
- window.context.sourcegraphDotComMode &&
- !localStorage.getItem(MainPage.HIDE_REPOGROUP_SAMPLE_STORAGE_KEY) &&
- !this.state.userQuery
- ) {
- this.setState({ userQuery: 'repogroup:sample' })
- }
-
- const portal = document.createElement('div')
- document.body.appendChild(portal)
- this.overlayPortal = portal
-
- // communicate onMainPage to SourcegraphWebApp for dark theme look
- if (window.context.sourcegraphDotComMode) {
- this.props.onMainPage(true)
- }
-
- // Add class to body to prevent global element styles from affecting other pages
- const windowBody = document.body
- windowBody.classList.add('main-page')
- }
-
- public componentWillUnmount(): void {
- this.props.onMainPage(false)
- const windowBody = document.body
- windowBody.classList.remove('main-page')
- windowBody.classList.remove('modal-open')
- }
- public render(): JSX.Element | null {
- if (!window.context.sourcegraphDotComMode) {
- return
- }
- return (
-
-
-
-
-
-
-
-
-
-
-
-
Powering developers at
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
Advanced code search
-
Find. Then replace.
-
- Search in files and diffs on your private code using simple terms, regular
- expressions, and other filters.
-
-
- Syncs repositories with your code host and supports searching any commit/branch,
- with no indexing delay.
-
-
- Explore code search
-
-
-
-
-
-
-
-
-
-
Enhanced code browsing and intelligence
-
Mine your language.
-
- Solve problems before they exist, commit by commit. Code intelligence makes
- browsing code easier, with IDE-like hovers, go-to-definition, and
- find-references on your code, powered by Sourcegraph extensions and language
- servers based on the open-source Language Server Protocol.
-
-
- It even works in code review diffs on GitHub and GitLab with our browser
- extensions.
-
-
- Explore code intelligence
-
-
-
-
-
-
-
-
-
-
Integrations
-
Get it. Together.
-
- Connect your Sourcegraph instance with your existing tools. Get code
- intelligence while browsing code on the web, and code search from your editor.
-
-
- Explore integrations
-
-
-
-
-
-
-
-
-
-
-
Deploy Sourcegraph
-
Free. For all.
-
- The pace at which humans can write code is the only thing that stands between us and
- flying cars, a habitat on Mars, and a cure for cancer. That's why developers can get
- started and deploy Sourcegraph for free, and contribute to our code on GitHub.
-
-
- Deploy Sourcegraph
-
-
- Sourcegraph on GitHub
-
-
-
-
Sourcegraph pricing
-
Size. Up.
-
- When you grow to hundreds or thousands of users and repositories, scale up
- instantly, and protect your uptime with Sourcegraph on Kubernetes, external backups,
- and custom support agreements. Start with Sourcegraph core for free and scale with
- your deployment.
-
-
- Sourcegraph pricing
-
-
-
-
-
-
-
-
-
-
Open. For business.
-
Sourcegraph is open source.
-
- We opened up Sourcegraph to bring code search and intelligence to more developers
- and developer ecosystems—and to help us realize the{' '}
- Sourcegraph master plan . We're also
- excited about what this means for Sourcegraph as a company. All of our customers,
- many with hundreds or thousands of developers using Sourcegraph internally every
- day, started out with a single developer spinning up a Sourcegraph instance and
- sharing it with their team. Being open-source makes it even easier to use
- Sourcegraph.
-
-
- Release announcement
-
-
- Sourcegraph on GitHub
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- Copyright © 2018 Sourcegraph, Inc.
-
-
- Terms
- Privacy
-
-
-
-
-
-
-
-
-
-
-
-
-
Advanced code search
- Find. Then replace.
-
-
-
-
- {searchSections.map(({ title, paragraph, buttons }, i) => (
-
-
-
{title}
-
{paragraph}
- {buttons.map(({ text, query }, j) => (
-
- {text}
-
- ))}
-
-
- ))}
-
-
-
- Get started with Sourcegraph for free, and get get cross-repository code
- intelligence, advanced code search, and extensive integrations.
-
-
-
- Deploy Sourcegraph
-
-
-
-
-
- Explore the power and extensibility of Sourcegraph's search query syntax, and learn
- how to search across all of your public and private code.
-
-
- Search documentation
-
-
-
-
-
- Close
-
-
-
-
-
-
-
-
-
-
-
Enhanced code browsing and intelligence
- Mine your language.
-
-
-
-
- {intelligenceSections.map(({ title, paragraph }, i) => (
-
-
-
{title}
-
{paragraph}
-
-
- ))}
-
-
-
-
-
-
-
-
- Get started with Sourcegraph for free, and get get cross-repository code
- intelligence, advanced code search, and extensive integrations.
-
-
-
- Deploy Sourcegraph
-
-
-
-
-
- Explore how Sourcegraph's code intelligence can augment and add to your workflow,
- prepare you for code review, and speed through development.
-
-
- Code intelligence documentation
-
-
-
-
-
- Close
-
-
-
-
-
-
-
-
-
-
-
Integrations
- Get it. Together.
-
-
-
-
- {integrationsSections.map(({ title, paragraph, buttons }, i) => (
-
-
-
{title}
-
{paragraph}
- {buttons.map(({ text, id, link }, j) => (
-
-
- {text}
-
- ))}
-
-
- ))}
-
-
-
-
-
-
- Get started with Sourcegraph for free, and get get cross-repository code
- intelligence, advanced code search, and extensive integrations.
-
-
-
- Deploy Sourcegraph
-
-
-
-
-
- Explore all of Sourcegraph's integrations and see how you can get cross-repository
- code intelligence on your favorite code host and editor.
-
-
- Integrations documentation
-
-
-
-
-
-
- Close
-
-
-
-
- )
- }
-
- private handleQueryChange = (activeButton: string, passedQ: string) => () => {
- if (passedQ === '') {
- this.setState({ activeButton, manualClick: true, activesection: '99' })
- } else {
- this.setState({ activeButton, userQuery: passedQ, manualClick: true, activesection: '99' })
- }
- }
-
- private activateModal = (section: string) => () => {
- const windowBody = document.body
- windowBody.classList.add('modal-open')
-
- if (section === 'intelligence') {
- this.setState(state => ({ modalIntelligenceOpen: !state.modalIntelligenceOpen }))
- this.setState(state => ({ animateModalIntelligence: !state.animateModalIntelligence }))
- this.setState(state => ({ activesection: '99' }))
- } else if (section === 'integrations') {
- this.setState(state => ({ modalIntegrationsOpen: !state.modalIntegrationsOpen }))
- this.setState(state => ({ animateModalIntegrations: !state.animateModalIntegrations }))
- this.setState(state => ({ activesection: '99' }))
- } else if (section === 'search') {
- this.setState(state => ({ modalSearchOpen: !state.modalSearchOpen }))
- this.setState(state => ({ animateModalSearch: !state.modalSearchOpen }))
- setTimeout(() => {
- if (this.state.animateModalSearch) {
- this.setState(state => ({ animateModalSearch: false }))
- }
- }, 750)
- setTimeout(() => {
- if (this.state.manualClick === false) {
- this.setState(state => ({
- activeButton: '0-1',
- activesection: '0',
- userQuery: 'repogroup:goteam file:\\.go$',
- }))
- }
- }, 1450)
- setTimeout(() => {
- if (this.state.manualClick === false) {
- this.setState(state => ({
- activeButton: '1-0',
- activesection: '1',
-
- userQuery: 'repo:^github\\.com/apple/swift$@master-next type:diff',
- }))
- }
- }, 7450)
- setTimeout(() => {
- if (this.state.manualClick === false) {
- this.setState(state => ({
- activeButton: '2-3',
- activesection: '2',
- userQuery: 'repogroup:goteam file:\\.go$',
- }))
- }
- }, 15550)
- setTimeout(() => {
- if (this.state.manualClick === false) {
- this.setState(state => ({
- activeButton: '3-2',
- activesection: '3',
- userQuery: 'repogroup:goteam file:\\.go$',
- }))
- }
- }, 21450)
- setTimeout(() => {
- if (this.state.manualClick === false) {
- this.setState(state => ({
- activeButton: '0-0',
- activesection: '99',
- userQuery: 'repogroup:goteam file:\\.go$',
- }))
- }
- }, 28450)
- }
- }
-
- private closeModal = (modalName: string) => () => {
- const windowBody = document.body
- windowBody.classList.remove('modal-open')
-
- if (modalName === 'search') {
- this.setState(state => ({ modalSearchClosing: !state.modalSearchClosing, animateModalSearch: false }))
- // RESET DID CLOSE
- setTimeout(() => {
- this.setState(state => ({ modalSearchOpen: !state.modalSearchOpen }))
- this.setState(state => ({ modalSearchClosing: !state.modalSearchClosing }))
- }, 400)
- } else if (modalName === 'intelligence') {
- this.setState(state => ({
- modalIntelligenceClosing: !state.modalIntelligenceClosing,
- animateModalIntelligence: false,
- }))
- // RESET DID CLOSE
- setTimeout(() => {
- this.setState(state => ({
- modalIntelligenceOpen: !state.modalIntelligenceOpen,
- modalIntelligenceClosing: !state.modalIntelligenceClosing,
- }))
- }, 400)
- } else if (modalName === 'integrations') {
- this.setState(state => ({
- modalIntegrationsClosing: !state.modalIntegrationsClosing,
- animateModalIntegrations: false,
- }))
- // RESET DID CLOSE
- setTimeout(() => {
- this.setState(state => ({
- modalIntegrationsOpen: !state.modalIntegrationsOpen,
- modalIntegrationsClosing: !state.modalIntegrationsClosing,
- }))
- }, 400)
- }
- }
-
- private onUserQueryChange = (userQuery: string) => {
- this.setState({ userQuery })
-
- if (window.context.sourcegraphDotComMode) {
- if (queryIndexOfScope(userQuery, 'repogroup:sample') !== -1) {
- localStorage.removeItem(MainPage.HIDE_REPOGROUP_SAMPLE_STORAGE_KEY)
- } else {
- localStorage.setItem(MainPage.HIDE_REPOGROUP_SAMPLE_STORAGE_KEY, 'true')
- }
- }
- }
-
- private onSubmit = (event: React.FormEvent): void => {
- event.preventDefault()
- submitSearch(this.props.history, this.state.userQuery, 'home')
- }
-
- private getPageTitle(): string | undefined {
- const query = parseSearchURLQuery(this.props.location.search)
- if (query) {
- return `${limitString(this.state.userQuery, 25, true)}`
- }
- return undefined
- }
-}
diff --git a/web/src/search/input/QueryInputForModal.scss b/web/src/search/input/QueryInputForModal.scss
deleted file mode 100644
index 8fe5e3d9f14..00000000000
--- a/web/src/search/input/QueryInputForModal.scss
+++ /dev/null
@@ -1,76 +0,0 @@
-@import './Suggestion';
-
-.query-input2-for-modal {
- width: 100%;
- position: relative;
-
- &__input {
- // Right side is flush with SearchButton.
- border-top-right-radius: 0;
- border-bottom-right-radius: 0;
- }
-
- &__suggestions {
- position: absolute;
- width: 100%;
- z-index: 1;
- max-height: 25rem;
- margin: 0;
- overflow-x: hidden;
- overflow-y: auto;
- overflow-y: overlay;
- background-color: #000000;
- border: solid 1px #1b548a;
- border-top: none;
- padding: 0;
-
- .theme-light & {
- background-color: $color-light-bg-2;
- border-top: none;
- }
-
- // Custom scrollbar
- &::-webkit-scrollbar {
- width: 0.5rem;
- height: 0.5rem;
- }
-
- &::-webkit-scrollbar-corner,
- &::-webkit-scrollbar-track {
- background-color: transparent;
- }
-
- &::-webkit-scrollbar-thumb {
- background-color: #2a3a51;
- }
-
- .theme-light &::-webkit-scrollbar-thumb {
- background-color: $color-light-bg-3;
- }
-
- * {
- display: inline-block;
- }
- li {
- display: block;
- position: relative;
- }
- }
-
- &__loading-notifier {
- position: absolute;
- z-index: 1;
- right: 0.5rem;
- padding-left: 0.25rem;
- bottom: 0.375rem;
- background-color: $color-bg-4;
- }
-}
-
-.theme-light {
- .query-input2 {
- &__loading-notifier {
- background-color: $color-light-bg-1;
- }
- }
-}
diff --git a/web/src/search/input/QueryInputForModal.tsx b/web/src/search/input/QueryInputForModal.tsx
deleted file mode 100644
index fecb598686e..00000000000
--- a/web/src/search/input/QueryInputForModal.tsx
+++ /dev/null
@@ -1,425 +0,0 @@
-import { LoadingSpinner } from '@sourcegraph/react-loading-spinner'
-import * as H from 'history'
-import * as React from 'react'
-import { fromEvent, merge, Observable, of, Subject, Subscription } from 'rxjs'
-import {
- catchError,
- debounceTime,
- delay,
- distinctUntilChanged,
- filter,
- map,
- publishReplay,
- refCount,
- repeat,
- startWith,
- switchMap,
- takeUntil,
- tap,
- toArray,
-} from 'rxjs/operators'
-import { Key } from 'ts-key-enum'
-import { eventLogger } from '../../tracking/eventLogger'
-import { scrollIntoView } from '../../util'
-import { fetchSuggestions } from '../backend'
-import { createSuggestion, Suggestion, SuggestionItem } from './Suggestion'
-
-/**
- * The query input field is clobbered and updated to contain this subject's values, as
- * they are received. This is used to trigger an update; the source of truth is still the URL.
- *
- * This file is mostly the same as Queryinput.tsx but differs for the use on the homepage on line
- * 326-335. It also does not need onInputFocus function.
- *
- */
-export const queryUpdates = new Subject()
-
-interface Props {
- location: H.Location
- history: H.History
-
- /** The value of the query input */
- value: string
-
- /** Called when the value changes */
- onChange: (newValue: string) => void
-
- /**
- * A string that is appended to the query input's query before
- * fetching suggestions.
- */
- prependQueryForSuggestions?: string
-
- /** Whether the input should be autofocused (and the behavior thereof) */
- autoFocus?: true | 'cursor-at-end'
-
- /** The input placeholder, if different from the default is desired. */
- placeholder?: string
-
- /**
- * Whether this input should behave like the global query input: (1)
- * pressing the '/' key focuses it and (2) other components contribute a
- * query to it with their context (such as the repository area contributing
- * 'repo:foo@bar' for the current repository and revision).
- *
- * At most one query input per page should have this behavior.
- */
- hasGlobalQueryBehavior?: boolean
-}
-
-interface State {
- /** Whether the query input is focused */
- inputFocused: boolean
-
- /** Whether suggestions are shown or not */
- hideSuggestions: boolean
-
- /** The suggestions shown to the user */
- suggestions: Suggestion[]
-
- /** Index of the currently selected suggestion (-1 if none selected) */
- selectedSuggestion: number
-
- /** Whether suggestions are currently being fetched */
- loading: boolean
-}
-
-export class QueryInputForModal extends React.Component {
- private static SUGGESTIONS_QUERY_MIN_LENGTH = 2
-
- private componentUpdates = new Subject()
-
- /** Subscriptions to unsubscribe from on component unmount */
- private subscriptions = new Subscription()
-
- /** Emits on keydown events in the input field */
- private inputKeyDowns = new Subject>()
-
- /** Emits new input values */
- private inputValues = new Subject()
-
- /** Emits when the input field is clicked */
- private inputFocuses = new Subject()
-
- /** Emits when the suggestions are hidden */
- private suggestionsHidden = new Subject()
-
- /** Only used for selection and focus management */
- private inputElement?: HTMLInputElement
-
- /** Only used for scroll state management */
- private suggestionListElement?: HTMLElement
-
- /** Only used for scroll state management */
- private selectedSuggestionElement?: HTMLElement
-
- /** Only used to keep track if the user has typed a single character into the input field so we can log an event once. */
- private hasLoggedFirstInput = false
-
- constructor(props: Props) {
- super(props)
-
- this.state = {
- hideSuggestions: false,
- inputFocused: false,
- loading: false,
- selectedSuggestion: -1,
- suggestions: [],
- }
-
- this.subscriptions.add(
- // Trigger new suggestions every time the input field is typed into
- this.inputValues
- .pipe(
- tap(query => this.props.onChange(query)),
- distinctUntilChanged(),
- debounceTime(200),
- switchMap(query => {
- if (query.length < QueryInputForModal.SUGGESTIONS_QUERY_MIN_LENGTH) {
- return [{ suggestions: [], selectedSuggestion: -1, loading: false }]
- }
- const fullQuery = [this.props.prependQueryForSuggestions, this.props.value]
- .filter(s => !!s)
- .join(' ')
- const suggestionsFetch = fetchSuggestions(fullQuery).pipe(
- map(createSuggestion),
- toArray(),
- map((suggestions: Suggestion[]) => ({
- suggestions,
- selectedSuggestion: -1,
- hideSuggestions: false,
- loading: false,
- })),
- catchError((err: Error) => {
- console.error(err)
- this.setState({ loading: false })
- // HACK: if we catchError before 100ms, then the loader will display over us.
- // This is not a good fix.
- setTimeout(() => this.setState({ loading: false }), 120)
- return [{}]
- }),
- publishReplay(),
- refCount()
- )
- return merge(
- suggestionsFetch,
- // Show a loader if the fetch takes longer than 100ms
- of({ loading: true }).pipe(
- delay(100),
- takeUntil(suggestionsFetch)
- )
- )
- }),
- // Abort suggestion display on route change or suggestion hiding
- takeUntil(this.suggestionsHidden),
- // But resubscribe afterwards
- repeat()
- )
- .subscribe(
- state => {
- this.setState(state as State)
- },
- err => {
- this.setState({ loading: false })
- console.error(err)
- }
- )
- )
-
- if (this.props.hasGlobalQueryBehavior) {
- // Quick-Open hotkeys
- this.subscriptions.add(
- fromEvent(window, 'keydown')
- .pipe(
- filter(
- event =>
- // Slash shortcut (if no input element is focused)
- (event.key === '/' &&
- !!document.activeElement &&
- !['INPUT', 'TEXTAREA'].includes(document.activeElement.nodeName)) ||
- // Cmd/Ctrl+P shortcut
- ((event.metaKey || event.ctrlKey) && event.key === 'p') ||
- // Cmd/Ctrl+Shift+F shortcut
- ((event.metaKey || event.ctrlKey) && event.shiftKey && event.key === 'f')
- ),
- switchMap(event => {
- event.preventDefault()
- // Use selection as query
- const selection = window.getSelection().toString()
- if (selection) {
- return new Observable(observer =>
- this.setState(
- {
- // query: selection, TODO(sqs): add back this behavior
- suggestions: [],
- selectedSuggestion: -1,
- },
- () => {
- observer.next()
- observer.complete()
- }
- )
- )
- }
- return [undefined]
- })
- )
- .subscribe(() => {
- if (this.inputElement) {
- // Select all input
- this.inputElement.focus()
- this.inputElement.setSelectionRange(0, this.inputElement.value.length)
- }
- })
- )
-
- // Allow other components to update the query (e.g., to be relevant to what the user is
- // currently viewing).
- this.subscriptions.add(
- queryUpdates.pipe(distinctUntilChanged()).subscribe(query => this.props.onChange(query))
- )
-
- /** Whenever the URL query has a "focus" property, remove it and focus the query input. */
- this.subscriptions.add(
- this.componentUpdates.pipe(startWith(props)).subscribe(props => {
- if (this.inputElement) {
- const value = this.inputElement.value
- if (value !== props.value) {
- this.inputElement.value = props.value
- this.focusInputAndPositionCursorAtEnd()
- }
- }
- })
- )
- }
- }
-
- public componentDidMount(): void {
- switch (this.props.autoFocus) {
- case 'cursor-at-end':
- this.focusInputAndPositionCursorAtEnd()
- break
- }
- }
-
- public componentWillReceiveProps(newProps: Props): void {
- this.componentUpdates.next(newProps)
- }
-
- public componentWillUnmount(): void {
- this.subscriptions.unsubscribe()
- }
-
- public componentDidUpdate(prevProps: Props, prevState: State): void {
- // Check if selected suggestion is out of view
- scrollIntoView(this.suggestionListElement, this.selectedSuggestionElement)
- }
-
- public render(): JSX.Element | null {
- const showSuggestions =
- this.props.value.length >= QueryInputForModal.SUGGESTIONS_QUERY_MIN_LENGTH &&
- this.state.inputFocused &&
- !this.state.hideSuggestions &&
- this.state.suggestions.length !== 0
-
- return (
-
-
(this.inputElement = ref!)}
- />
- {this.state.loading &&
}
- {showSuggestions && (
-
- {this.state.suggestions.map((suggestion, i) => {
- const isSelected = this.state.selectedSuggestion === i
- const onRef = (ref: HTMLLIElement | null) => {
- if (isSelected) {
- this.selectedSuggestionElement = ref || undefined
- }
- }
- return (
- this.selectSuggestion(suggestion)}
- liRef={onRef}
- />
- )
- })}
-
- )}
-
- )
- }
-
- // Handle the cursor changing in the input field and
- // allow the user to type after the string changes.
- // Used in modal on homepage.
- private changeCursor: React.FocusEventHandler = e => {
- setTimeout(() => {
- this.inputFocuses.next()
- this.setState({ inputFocused: true })
- }, 100)
- }
- private setSuggestionListElement = (ref: HTMLElement | null): void => {
- this.suggestionListElement = ref || undefined
- }
-
- private selectSuggestion = (suggestion: Suggestion): void => {
- // 🚨 PRIVACY: never provide any private data in { code_search: { suggestion: { type } } }.
- eventLogger.log('SearchSuggestionSelected', {
- code_search: {
- suggestion: {
- type: suggestion.type,
- url: suggestion.url,
- },
- },
- })
-
- this.props.history.push(suggestion.url)
-
- this.suggestionsHidden.next()
- this.setState({ hideSuggestions: true, selectedSuggestion: -1 })
- }
-
- private focusInputAndPositionCursorAtEnd(): void {
- if (this.inputElement) {
- // Focus the input element and set cursor to the end
- this.inputElement.focus()
- this.inputElement.setSelectionRange(this.inputElement.value.length, this.inputElement.value.length)
- }
- }
-
- private onInputChange: React.ChangeEventHandler = event => {
- if (!this.hasLoggedFirstInput) {
- eventLogger.log('SearchInitiated')
- this.hasLoggedFirstInput = true
- }
- this.inputValues.next(event.currentTarget.value)
- }
-
- private onInputBlur: React.FocusEventHandler = event => {
- this.suggestionsHidden.next()
- this.setState({ inputFocused: false, loading: false, hideSuggestions: true })
- }
-
- private onInputKeyDown: React.KeyboardEventHandler = event => {
- event.persist()
- this.inputKeyDowns.next(event)
- switch (event.key) {
- case Key.Escape: {
- this.suggestionsHidden.next()
- this.setState({ loading: false, hideSuggestions: true, selectedSuggestion: -1 })
- break
- }
- case Key.ArrowDown: {
- event.preventDefault()
- this.moveSelection(1)
- break
- }
- case Key.ArrowUp: {
- event.preventDefault()
- this.moveSelection(-1)
- break
- }
- case Key.Enter: {
- if (this.state.selectedSuggestion === -1) {
- // Submit form and hide suggestions
- this.suggestionsHidden.next()
- this.setState({ hideSuggestions: true })
- break
- }
-
- // Select suggestion
- event.preventDefault()
- if (this.state.suggestions.length === 0) {
- break
- }
- this.selectSuggestion(this.state.suggestions[Math.max(this.state.selectedSuggestion, 0)])
- this.setState({ hideSuggestions: true })
- break
- }
- }
- }
-
- private moveSelection(steps: number): void {
- this.setState({
- selectedSuggestion: Math.max(
- Math.min(this.state.selectedSuggestion + steps, this.state.suggestions.length - 1),
- -1
- ),
- })
- }
-}
diff --git a/web/src/search/input/README.md b/web/src/search/input/README.md
deleted file mode 100644
index 3f259071173..00000000000
--- a/web/src/search/input/README.md
+++ /dev/null
@@ -1 +0,0 @@
-Components for areas in the app that expose a search input
diff --git a/web/src/tracking/services/serverAdminWrapper.tsx b/web/src/tracking/services/serverAdminWrapper.tsx
index 02fa3e2ed5a..0352f63fa0b 100644
--- a/web/src/tracking/services/serverAdminWrapper.tsx
+++ b/web/src/tracking/services/serverAdminWrapper.tsx
@@ -10,7 +10,7 @@ class ServerAdminWrapper {
private isAuthenicated = false
constructor() {
- if (!window.context.sourcegraphDotComMode) {
+ if (window.context && !window.context.sourcegraphDotComMode) {
authenticatedUser.subscribe(user => {
if (user) {
this.isAuthenicated = true
diff --git a/web/src/tracking/services/telligentWrapper.tsx b/web/src/tracking/services/telligentWrapper.tsx
index 1f0a2e3fe55..1d055ea0dfc 100644
--- a/web/src/tracking/services/telligentWrapper.tsx
+++ b/web/src/tracking/services/telligentWrapper.tsx
@@ -11,7 +11,7 @@ class TelligentWrapper {
constructor() {
// Never log anything in self-hosted Sourcegraph instances.
- if (!window.context.sourcegraphDotComMode) {
+ if (!window.context || !window.context.sourcegraphDotComMode) {
return
}
diff --git a/yarn.lock b/yarn.lock
index 945f12c5ebe..61dbcf70c21 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -2046,6 +2046,11 @@
"@types/events" "*"
"@types/node" "*"
+"@types/classnames@^2.2.7":
+ version "2.2.7"
+ resolved "https://registry.npmjs.org/@types/classnames/-/classnames-2.2.7.tgz#fb68cc9be8487e6ea5b13700e759bfbab7e0fefd"
+ integrity sha512-rzOhiQ55WzAiFgXRtitP/ZUT8iVNyllEpylJ5zHzR4vArUvMB39GTk+Zon/uAM0JxEFAWnwsxC2gH8s+tZ3Myg==
+
"@types/cli-color@^0.3.29":
version "0.3.29"
resolved "https://registry.npmjs.org/@types/cli-color/-/cli-color-0.3.29.tgz#c83a71fe02c8c7e1ccec048dd6a2458d1f6c96ea"
@@ -4663,7 +4668,7 @@ class-utils@^0.3.5:
isobject "^3.0.0"
static-extend "^0.1.1"
-classnames@^2.2.3, classnames@^2.2.5:
+classnames@^2.2.3, classnames@^2.2.5, classnames@^2.2.6:
version "2.2.6"
resolved "https://registry.npmjs.org/classnames/-/classnames-2.2.6.tgz#43935bffdd291f326dad0a205309b38d00f650ce"
integrity sha512-JR/iSQOSt+LQIWwrwEzJ9uk0xfN3mTVYMwt1Ir5mUcSN6pU+V4zQFFaJsclJbPuAUQH+yfWef6tm7l1quW3C8Q==