From be8ee10391215bdc6f069b865de59e86b752dcc7 Mon Sep 17 00:00:00 2001 From: "Daniel P. Purkhus" Date: Wed, 1 Nov 2023 17:31:18 +0000 Subject: [PATCH] WIP --- src/Components/MSBuild.fs | 1 + src/Components/SolutionExplorer.fs | 5 ++ src/Core/DTO.fs | 3 ++ src/Core/LanguageService.fs | 14 +++-- src/Core/Project.fs | 82 ++++++++++++++++++++---------- 5 files changed, 73 insertions(+), 32 deletions(-) diff --git a/src/Components/MSBuild.fs b/src/Components/MSBuild.fs index 5c685048..77a1cf92 100644 --- a/src/Components/MSBuild.fs +++ b/src/Components/MSBuild.fs @@ -399,6 +399,7 @@ module MSBuild = let buildTaskListForSolution (s: WorkspacePeekFound) : JS.Promise = promise { match s with + | WorkspacePeekFound.Fsproj _ -> return Seq.empty | WorkspacePeekFound.Directory _ -> return Seq.empty | WorkspacePeekFound.Solution({ Path = p }) -> let! dotnet = dotnetBinary () diff --git a/src/Components/SolutionExplorer.fs b/src/Components/SolutionExplorer.fs index 365dc6e8..c3ad7afa 100644 --- a/src/Components/SolutionExplorer.fs +++ b/src/Components/SolutionExplorer.fs @@ -262,6 +262,11 @@ module SolutionExplorer = let result = Workspace items setParentRefs items result result + | WorkspacePeekFound.Fsproj proj -> + let result = getProjItem proj.Fsproj + let root = Workspace [ result ] + setParentRef result root + root let private getSolution () = Project.getLoadedSolution () |> Option.map getSolutionModel diff --git a/src/Core/DTO.fs b/src/Core/DTO.fs index 4be1a17e..aea9a7e1 100644 --- a/src/Core/DTO.fs +++ b/src/Core/DTO.fs @@ -272,8 +272,11 @@ module DTO = type WorkspacePeek = { Found: WorkspacePeekFound[] } and WorkspacePeekFound = + | Fsproj of WorkspacePeekFoundFsproj | Directory of WorkspacePeekFoundDirectory | Solution of WorkspacePeekFoundSolution + and WorkspacePeekFoundFsproj = + { Fsproj: string } and WorkspacePeekFoundDirectory = { Directory: string; Fsprojs: string[] } diff --git a/src/Core/LanguageService.fs b/src/Core/LanguageService.fs index ab5d84a8..b3a09381 100644 --- a/src/Core/LanguageService.fs +++ b/src/Core/LanguageService.fs @@ -516,22 +516,26 @@ Consider: Kind = kind } | None -> None - let mapFound (f: obj) : WorkspacePeekFound option = + let mapFound (f: obj) : WorkspacePeekFound array = let data = f?Data match f?Type |> unbox with - | "directory" -> Some(WorkspacePeekFound.Directory(data |> unbox)) + | "directory" -> + let folderPeek = [|WorkspacePeekFound.Directory(data |> unbox)|] + let projectPeeks = data?Fsprojs |> unbox |> Array.map (fun p -> WorkspacePeekFound.Fsproj({ Fsproj = p })) + + Array.concat [|projectPeeks;folderPeek|] | "solution" -> let sln = { WorkspacePeekFoundSolution.Path = data?Path |> unbox Configurations = data?Configurations |> unbox Items = data?Items |> unbox |> Array.choose mapItem } - Some(WorkspacePeekFound.Solution sln) - | _ -> None + [|WorkspacePeekFound.Solution sln|] + | _ -> [||] let parse (ws: obj) = - { WorkspacePeek.Found = ws?Found |> unbox |> Array.choose mapFound } + { WorkspacePeek.Found = ws?Found |> unbox |> Array.map mapFound |> Array.concat } match client with | None -> Promise.empty diff --git a/src/Core/Project.fs b/src/Core/Project.fs index c9c2e3aa..f9c97b79 100644 --- a/src/Core/Project.fs +++ b/src/Core/Project.fs @@ -107,6 +107,7 @@ module Project = sln.Items |> Array.collect getProjs |> Array.toList | Some(WorkspacePeekFound.Directory dir) -> dir.Fsprojs |> Array.toList + | Some(WorkspacePeekFound.Fsproj fsproj) -> [ fsproj.Fsproj ] let getNotLoaded () = @@ -251,6 +252,7 @@ module Project = match loadedWorkspace with | None -> Array.empty | Some(WorkspacePeekFound.Directory dir) -> dir.Fsprojs + | Some(WorkspacePeekFound.Fsproj fsproj) -> [| fsproj.Fsproj |] | Some(WorkspacePeekFound.Solution sln) -> sln.Items |> Array.collect foldFsproj |> Array.map fst let loadingInProgress p = @@ -308,6 +310,7 @@ module Project = type ConfiguredWorkspace = | Solution of path: string | Directory of path: string + | Fsproj of path: string module private CurrentWorkspaceConfiguration = let private key = "FSharp.workspacePath" @@ -322,6 +325,8 @@ module Project = if value.ToLowerInvariant().EndsWith(".sln") then ConfiguredWorkspace.Solution fullPath + else if value.ToLowerInvariant().EndsWith(".fsproj") then + ConfiguredWorkspace.Fsproj fullPath else ConfiguredWorkspace.Directory fullPath @@ -333,6 +338,12 @@ module Project = not (isNull stats) && (stats.isDirectory ()) with _ -> false + | ConfiguredWorkspace.Fsproj fsproj -> + try + let stats = node.fs.statSync (U2.Case1 fsproj) + not (isNull stats) && (stats.isFile ()) + with _ -> + false | ConfiguredWorkspace.Solution sln -> try let stats = node.fs.statSync (U2.Case1 sln) @@ -373,6 +384,7 @@ module Project = match value with | ConfiguredWorkspace.Solution path -> path | ConfiguredWorkspace.Directory path -> path + | ConfiguredWorkspace.Fsproj path -> path let private isConfiguredInWorkspace () = match getStringFromWorkspaceConfig () with @@ -400,6 +412,7 @@ module Project = let setFromPeek (value: WorkspacePeekFound) = match value with | WorkspacePeekFound.Directory dir -> ConfiguredWorkspace.Directory dir.Directory + | WorkspacePeekFound.Fsproj fsproj -> ConfiguredWorkspace.Fsproj fsproj.Fsproj | WorkspacePeekFound.Solution sln -> ConfiguredWorkspace.Solution sln.Path |> set @@ -419,6 +432,7 @@ module Project = | ConfiguredWorkspace.Directory valueDir, WorkspacePeekFound.Directory peekDir -> removeEndSlash valueDir = removeEndSlash peekDir.Directory | ConfiguredWorkspace.Solution valueSln, WorkspacePeekFound.Solution peekSln -> valueSln = peekSln.Path + | ConfiguredWorkspace.Fsproj valueFsproj, WorkspacePeekFound.Fsproj peekFsproj -> valueFsproj = peekFsproj.Fsproj | _ -> false let tryFind (value: ConfiguredWorkspace option) (found: WorkspacePeekFound list) = @@ -444,6 +458,12 @@ module Project = item.label <- sprintf "%s%s" check dir.Directory item.description <- Some(sprintf "Directory with %i projects" dir.Fsprojs.Length) item + | WorkspacePeekFound.Fsproj fsproj -> + let relative = node.path.relative (workspace.rootPath.Value, fsproj.Fsproj) + let item = createEmpty + item.label <- sprintf "%s%s" check relative + item.description <- Some(sprintf "Single project") + item | WorkspacePeekFound.Solution sln -> let relative = node.path.relative (workspace.rootPath.Value, sln.Path) let item = createEmpty @@ -556,12 +576,12 @@ module Project = return [] else let! ws = LanguageService.workspacePeek workspace.rootPath.Value deepLevel (excluded |> List.ofArray) - return ws.Found |> Array.sortBy (fun x -> match x with | WorkspacePeekFound.Solution sln -> countProjectsInSln sln + | WorkspacePeekFound.Fsproj _ -> 0 | WorkspacePeekFound.Directory _ -> -1) |> Array.rev |> List.ofArray @@ -572,33 +592,39 @@ module Project = let! ws = workspacePeek () let configured = CurrentWorkspaceConfiguration.get () let configuredPeek = CurrentWorkspaceConfiguration.tryFind configured ws - - match configuredPeek with - | Some peek -> - // If a workspace is configured, use it - return Some peek - | None -> + let! choosen = pickFSACWorkspace ws None + return choosen + + // TODO: Rewrite this so that the sln isn't always chosen, maybe add a config? + + // match configuredPeek with + // | Some peek -> + // // If a workspace is configured, use it + // return Some peek + // | None -> + // prefer the sln, load directly the first one, otherwise ask - let slns = - ws - |> List.choose (fun x -> - match x with - | WorkspacePeekFound.Solution _ -> Some x - | _ -> None) - - let! choosen = - match slns with - | [] -> - ws - |> List.tryPick (fun x -> - match x with - | WorkspacePeekFound.Directory _ -> Some x - | _ -> None) - |> Promise.lift - | [ sln ] -> Promise.lift (Some sln) - | _ -> pickFSACWorkspace ws None - - return choosen + // let slns = + // ws + // |> List.choose (fun x -> + // match x with + // | WorkspacePeekFound.Solution _ -> Some x + // | _ -> None) + + // let! choosen = + // match slns with + // | [] -> + // ws + // |> List.tryPick (fun x -> + // match x with + // | WorkspacePeekFound.Directory _ -> Some x + // | _ -> None) + // |> Promise.lift + // | [ sln ] -> Promise.lift (Some sln) + // | _ -> pickFSACWorkspace ws None + + // let! choosen = pickFSACWorkspace ws None + // return choosen } let handleProjectParsedNotification res = @@ -650,10 +676,12 @@ module Project = let projs = match x with | WorkspacePeekFound.Directory dir -> dir.Fsprojs + | WorkspacePeekFound.Fsproj fsproj -> [| fsproj.Fsproj |] | WorkspacePeekFound.Solution sln -> sln.Items |> Array.collect foldFsproj |> Array.map fst match x with | WorkspacePeekFound.Solution _ -> setAnyProjectContext true + | WorkspacePeekFound.Fsproj _ -> setAnyProjectContext false | WorkspacePeekFound.Directory _ when not (projs |> Array.isEmpty) -> setAnyProjectContext true | _ -> ()