Convert Elixir columns range to JavaScript (#472)

This commit is contained in:
Jonatan Kłosko 2021-07-27 16:50:05 +02:00 committed by GitHub
parent f225ddbdcf
commit afe06517d7
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 74 additions and 21 deletions

View file

@ -225,7 +225,7 @@ defmodule Livebook.Evaluator do
error -> Logger.error(Exception.format(:error, error, __STACKTRACE__)) error -> Logger.error(Exception.format(:error, error, __STACKTRACE__))
end end
send(send_to, {:intellisense_response, ref, response}) send(send_to, {:intellisense_response, ref, request, response})
{:noreply, state} {:noreply, state}
end end

View file

@ -42,11 +42,10 @@ defmodule Livebook.JSInterop do
@doc """ @doc """
Returns a column number in the Elixir string corresponding to Returns a column number in the Elixir string corresponding to
the given column interpreted in terms of UTF-16 code units as the given column interpreted in terms of UTF-16 code units.
JavaScript does.
""" """
@spec convert_column_to_elixir(pos_integer(), String.t()) :: pos_integer() @spec js_column_to_elixir(pos_integer(), String.t()) :: pos_integer()
def convert_column_to_elixir(column, line) do def js_column_to_elixir(column, line) do
line line
|> string_to_utf16_code_units() |> string_to_utf16_code_units()
|> Enum.take(column - 1) |> Enum.take(column - 1)
@ -55,7 +54,23 @@ defmodule Livebook.JSInterop do
|> Kernel.+(1) |> Kernel.+(1)
end end
# --- @doc """
Returns a column represented in terms of UTF-16 code units
corresponding to the given column number in Elixir string.
"""
@spec elixir_column_to_js(pos_integer(), String.t()) :: pos_integer()
def elixir_column_to_js(column, line) do
line
|> string_take(column - 1)
|> string_to_utf16_code_units()
|> length()
|> Kernel.+(1)
end
defp string_take(_string, 0), do: ""
defp string_take(string, n) when n > 0, do: String.slice(string, 0..(n - 1))
# UTF-16 helpers
defp string_to_utf16_code_units(string) do defp string_to_utf16_code_units(string) do
string string

View file

@ -184,7 +184,7 @@ defprotocol Livebook.Runtime do
the text editor. the text editor.
The response is sent to the `send_to` process as The response is sent to the `send_to` process as
`{:intellisense_response, ref, response}`. `{:intellisense_response, ref, request, response}`.
The given `locator` idenfities an evaluation that may be used The given `locator` idenfities an evaluation that may be used
as context when resolving the request (if relevant). as context when resolving the request (if relevant).

View file

@ -223,7 +223,7 @@ defmodule Livebook.Runtime.ErlDist.RuntimeServer do
binding = [] binding = []
env = :elixir.env_for_eval([]) env = :elixir.env_for_eval([])
response = Livebook.Intellisense.handle_request(request, binding, env) response = Livebook.Intellisense.handle_request(request, binding, env)
send(send_to, {:intellisense_response, ref, response}) send(send_to, {:intellisense_response, ref, request, response})
end) end)
end end

View file

@ -8,6 +8,7 @@ defmodule LivebookWeb.SessionLive do
alias LivebookWeb.SidebarHelpers alias LivebookWeb.SidebarHelpers
alias Livebook.{SessionSupervisor, Session, Delta, Notebook, Runtime, LiveMarkdown} alias Livebook.{SessionSupervisor, Session, Delta, Notebook, Runtime, LiveMarkdown}
alias Livebook.Notebook.Cell alias Livebook.Notebook.Cell
alias Livebook.JSInterop
@impl true @impl true
def mount(%{"id" => session_id}, %{"current_user_id" => current_user_id} = session, socket) do def mount(%{"id" => session_id}, %{"current_user_id" => current_user_id} = session, socket) do
@ -633,7 +634,7 @@ defmodule LivebookWeb.SessionLive do
{:completion, hint} {:completion, hint}
%{"type" => "details", "line" => line, "column" => column} -> %{"type" => "details", "line" => line, "column" => column} ->
column = Livebook.JSInterop.convert_column_to_elixir(column, line) column = JSInterop.js_column_to_elixir(column, line)
{:details, line, column} {:details, line, column}
%{"type" => "format", "code" => code} -> %{"type" => "format", "code" => code} ->
@ -758,7 +759,8 @@ defmodule LivebookWeb.SessionLive do
|> push_redirect(to: Routes.home_path(socket, :page))} |> push_redirect(to: Routes.home_path(socket, :page))}
end end
def handle_info({:intellisense_response, ref, response}, socket) do def handle_info({:intellisense_response, ref, request, response}, socket) do
response = process_intellisense_response(response, request)
payload = %{"ref" => inspect(ref), "response" => response} payload = %{"ref" => inspect(ref), "response" => response}
{:noreply, push_event(socket, "intellisense_response", payload)} {:noreply, push_event(socket, "intellisense_response", payload)}
end end
@ -1046,6 +1048,21 @@ defmodule LivebookWeb.SessionLive do
end) end)
end end
defp process_intellisense_response(
%{range: %{from: from, to: to}} = response,
{:details, line, _column}
) do
%{
response
| range: %{
from: JSInterop.elixir_column_to_js(from, line),
to: JSInterop.elixir_column_to_js(to, line)
}
}
end
defp process_intellisense_response(response, _request), do: response
# Builds view-specific structure of data by cherry-picking # Builds view-specific structure of data by cherry-picking
# only the relevant attributes. # only the relevant attributes.
# We then use `@data_view` in the templates and consequently # We then use `@data_view` in the templates and consequently

View file

@ -223,7 +223,9 @@ defmodule Livebook.EvaluatorTest do
test "sends completion response to the given process", %{evaluator: evaluator} do test "sends completion response to the given process", %{evaluator: evaluator} do
request = {:completion, "System.ver"} request = {:completion, "System.ver"}
Evaluator.handle_intellisense(evaluator, self(), :ref, request) Evaluator.handle_intellisense(evaluator, self(), :ref, request)
assert_receive {:intellisense_response, :ref, %{items: [%{label: "version/0"}]}}, 1_000
assert_receive {:intellisense_response, :ref, ^request, %{items: [%{label: "version/0"}]}},
1_000
end end
test "given evaluation reference uses its bindings and env", %{evaluator: evaluator} do test "given evaluation reference uses its bindings and env", %{evaluator: evaluator} do
@ -237,12 +239,15 @@ defmodule Livebook.EvaluatorTest do
request = {:completion, "num"} request = {:completion, "num"}
Evaluator.handle_intellisense(evaluator, self(), :ref, request, :code_1) Evaluator.handle_intellisense(evaluator, self(), :ref, request, :code_1)
assert_receive {:intellisense_response, :ref, %{items: [%{label: "number"}]}}, 1_000
assert_receive {:intellisense_response, :ref, ^request, %{items: [%{label: "number"}]}},
1_000
request = {:completion, "ANSI.brigh"} request = {:completion, "ANSI.brigh"}
Evaluator.handle_intellisense(evaluator, self(), :ref, request, :code_1) Evaluator.handle_intellisense(evaluator, self(), :ref, request, :code_1)
assert_receive {:intellisense_response, :ref, %{items: [%{label: "bright/0"}]}}, 1_000 assert_receive {:intellisense_response, :ref, ^request, %{items: [%{label: "bright/0"}]}},
1_000
end end
end end

View file

@ -44,18 +44,18 @@ defmodule Livebook.JSInteropTest do
end end
end end
describe "convert_column_to_elixir/2" do describe "js_column_to_elixir/2" do
test "keeps the column as is for ASCII characters" do test "keeps the column as is for ASCII characters" do
column = 4 column = 4
line = "String.replace" line = "String.replace"
assert JSInterop.convert_column_to_elixir(column, line) == 4 assert JSInterop.js_column_to_elixir(column, line) == 4
end end
test "shifts the column given characters spanning multiple UTF-16 code units" do test "shifts the column given characters spanning multiple UTF-16 code units" do
# 🚀 consists of 2 UTF-16 code units, so JavaScript assumes "🚀".length is 2 # 🚀 consists of 2 UTF-16 code units, so JavaScript assumes "🚀".length is 2
column = 7 column = 7
line = "🚀🚀 String.replace" line = "🚀🚀 String.replace"
assert JSInterop.convert_column_to_elixir(column, line) == 5 assert JSInterop.js_column_to_elixir(column, line) == 5
end end
test "returns proper column if a middle UTF-16 code unit is given" do test "returns proper column if a middle UTF-16 code unit is given" do
@ -63,7 +63,22 @@ defmodule Livebook.JSInteropTest do
# 3th and 4th code unit correspond to the second 🚀 # 3th and 4th code unit correspond to the second 🚀
column = 3 column = 3
line = "🚀🚀 String.replace" line = "🚀🚀 String.replace"
assert JSInterop.convert_column_to_elixir(column, line) == 2 assert JSInterop.js_column_to_elixir(column, line) == 2
end
end
describe "elixir_column_to_js/2" do
test "keeps the column as is for ASCII characters" do
column = 4
line = "String.replace"
assert JSInterop.elixir_column_to_js(column, line) == 4
end
test "shifts the column given characters spanning multiple UTF-16 code units" do
# 🚀 consists of 2 UTF-16 code units, so JavaScript assumes "🚀".length is 2
column = 5
line = "🚀🚀 String.replace"
assert JSInterop.elixir_column_to_js(column, line) == 7
end end
end end
end end

View file

@ -135,7 +135,7 @@ defmodule Livebook.Runtime.ErlDist.RuntimeServerTest do
request = {:completion, "System.ver"} request = {:completion, "System.ver"}
RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil}) RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil})
assert_receive {:intellisense_response, :ref, %{items: [%{label: "version/0"}]}} assert_receive {:intellisense_response, :ref, ^request, %{items: [%{label: "version/0"}]}}
end end
test "provides extended completion when previous evaluation reference is given", %{pid: pid} do test "provides extended completion when previous evaluation reference is given", %{pid: pid} do
@ -145,7 +145,7 @@ defmodule Livebook.Runtime.ErlDist.RuntimeServerTest do
request = {:completion, "num"} request = {:completion, "num"}
RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, :e1}) RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, :e1})
assert_receive {:intellisense_response, :ref, %{items: [%{label: "number"}]}} assert_receive {:intellisense_response, :ref, ^request, %{items: [%{label: "number"}]}}
end end
end end
@ -154,7 +154,8 @@ defmodule Livebook.Runtime.ErlDist.RuntimeServerTest do
request = {:details, "System.version", 10} request = {:details, "System.version", 10}
RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil}) RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil})
assert_receive {:intellisense_response, :ref, %{range: %{from: 1, to: 15}, contents: [_]}} assert_receive {:intellisense_response, :ref, ^request,
%{range: %{from: 1, to: 15}, contents: [_]}}
end end
end end
@ -163,7 +164,7 @@ defmodule Livebook.Runtime.ErlDist.RuntimeServerTest do
request = {:format, "System.version"} request = {:format, "System.version"}
RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil}) RuntimeServer.handle_intellisense(pid, self(), :ref, request, {:c1, nil})
assert_receive {:intellisense_response, :ref, %{code: "System.version()"}} assert_receive {:intellisense_response, :ref, ^request, %{code: "System.version()"}}
end end
end end