%% -*- erlang -*- %% %% %CopyrightBegin% %% %% Copyright Ericsson AB 2009-2012. All Rights Reserved. %% %% Licensed under the Apache License, Version 2.0 (the "License"); %% you may not use this file except in compliance with the License. %% You may obtain a copy of the License at %% %% http://www.apache.org/licenses/LICENSE-2.0 %% %% Unless required by applicable law or agreed to in writing, software %% distributed under the License is distributed on an "AS IS" BASIS, %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %% See the License for the specific language governing permissions and %% limitations under the License. %% %% %CopyrightEnd% %%%------------------------------------------------------------------- %%% File : test.erl %%% Author : Dan Gudmundsson <dan.gudmundsson@ericsson.com> %%% Description : Test emacs mode indention and font-locking %%% this file is intentionally not indented. %%% Copy the file and indent it and you should end up with test.erl.indented %%% Created : 6 Oct 2009 by Dan Gudmundsson <dan.gudmundsson@ericsson.com> %%%------------------------------------------------------------------- %% Start off with syntax highlighting you have to verify this by looking here %% and see that the code looks alright -module(test). -compile(export_all). %% Used to cause an "Unbalanced parentheses" error. foo(M) -> M#{a :=<<"a">> ,b:=1}. foo() -> #{a =><<"a">> ,b=>1}. %% Module attributes should be highlighted -export([t/1]). -record(record1, {a, b, c }). -record(record2, { a, b }). -record(record3, {a = 8#42423 bor 8#4234, b = 8#5432 bor 2#1010101 c = 123 + 234, d}). -record(record4, { a = 8#42423 bor 8#4234, b = 8#5432 bor 2#1010101 c = 123 + 234, d}). -define(MACRO_1, macro). -define(MACRO_2(_), macro). -spec t(integer()) -> any(). -type ann() :: Var :: integer(). -type ann2() :: Var :: 'return' | 'return_white_spaces' | 'return_comments' | 'text' | ann(). -type paren() :: (ann2()). -type t1() :: atom(). -type t2() :: [t1()]. -type t3(Atom) :: integer(Atom). -type t4() :: t3(foobar). -type t5() :: {t1(), t3(foo)}. -type t6() :: 1 | 2 | 3 | 'foo' | 'bar'. -type t7() :: []. -type t71() :: [_]. -type t8() :: {any(),none(),pid(),port(), reference(),float()}. -type t9() :: [1|2|3|foo|bar] | list(a | b | c) | t71(). -type t10() :: {1|2|3|foo|t9()} | {}. -type t11() :: 1..2. -type t13() :: maybe_improper_list(integer(), t11()). -type t14() :: [erl_scan:foo() | %% Should be highlighted term() | bool() | byte() | char() | non_neg_integer() | nonempty_list() | pos_integer() | neg_integer() | number() | list() | nonempty_improper_list() | nonempty_maybe_improper_list() | maybe_improper_list() | string() | iolist() | byte() | module() | mfa() | node() | timeout() | no_return() | %% Should not be highlighted nonempty_() | nonlist() | erl_scan:bar(34, 92) | t13() | m:f(integer() | <<_:_*16>>)]. -type t15() :: {binary(),<<>>,<<_:34>>,<<_:_*42>>, <<_:3,_:_*14>>,<<>>} | [<<>>|<<_:34>>|<<_:16>>| <<_:3,_:_*1472>>|<<_:19,_:_*14>>| <<_:34>>| <<_:34>>|<<_:34>>|<<_:34>>]. -type t16() :: fun(). -type t17() :: fun((...) -> paren()). -type t18() :: fun(() -> t17() | t16()). -type t19() :: fun((t18()) -> t16()) | fun((nonempty_maybe_improper_list('integer', any())| 1|2|3|a|b|<<_:3,_:_*14>>|integer()) -> nonempty_maybe_improper_list('integer', any())| 1|2|3|a|b|<<_:3,_:_*14>>|integer()). -type t20() :: [t19(), ...]. -type t21() :: tuple(). -type t21(A) :: A. -type t22() :: t21(integer()). -type t23() :: #rec1{}. -type t24() :: #rec2{a :: t23(), b :: [atom()]}. -type t25() :: #rec3{f123 :: [t24() | 1|2|3|4|a|b|c|d| nonempty_maybe_improper_list(integer, any())]}. -type t99() :: {t2(),t4(),t5(),t6(),t7(),t8(),t10(),t14(), t15(),t20(),t21(), t22(),t25()}. -spec t1(FooBar :: t99()) -> t99(); (t2()) -> t2(); (t4()) -> t4() when is_subtype(t4(), t24); (t23()) -> t23() when is_subtype(t23(), atom()), is_subtype(t23(), t14()); (t24()) -> t24() when is_subtype(t24(), atom()), is_subtype(t24(), t14()), is_subtype(t24(), t4()). -spec over(I :: integer()) -> R1 :: foo:typen(); (A :: atom()) -> R2 :: foo:atomen(); (T :: tuple()) -> R3 :: bar:typen(). -spec mod:t2() -> any(). -spec handle_cast(Cast :: {'exchange', node(), [[name(),...]]} | {'del_member', name(), pid()}, #state{}) -> {'noreply', #state{}}. -spec handle_cast(Cast :: {'exchange', node(), [[name(),...]]} | {'del_member', name(), pid()}, #state{}) -> {'noreply', #state{}}. -spec all(fun((T) -> boolean()), List :: [T]) -> boolean() when is_subtype(T, term()). % (*) -spec get_closest_pid(term()) -> Return :: pid() | {'error', {'no_process', term()} | {'no_such_group', term()}}. -opaque attributes_data() :: [{'column', column()} | {'line', info_line()} | {'text', string()}] | {line(),column()}. -record(r,{ f1 :: attributes_data(), f222 = foo:bar(34, #rec3{}, 234234234423, aassdsfsdfsdf, 2234242323) :: [t24() | 1|2|3|4|a|b|c|d| nonempty_maybe_improper_list(integer, any())], f333 :: [t24() | 1|2|3|4|a|b|c|d| nonempty_maybe_improper_list(integer, any())], f3 = x:y(), f4 = x:z() :: t99(), f17 :: 'undefined', f18 :: 1 | 2 | 'undefined', f19 = 3 :: integer()|undefined, f5 = 3 :: undefined|integer()}). -record(state, { sequence_number = 1 :: integer() }). highlighting(X) % Function definitions should be highlighted when is_integer(X) -> % and so should `when' and `is_integer' be %% Highlighting %% Various characters (we keep an `atom' after to see that highlighting ends) $a,atom, % Characters should be marked "string",atom, % and strings 'asdasd',atom, % quote should be atoms?? 'VaV',atom, 'aVa',atom, '\'atom',atom, 'atom\'',atom, 'at\'om',atom, '#1',atom, $", atom, % atom should be ok $', atom, "string$", atom, "string$", atom, % currently buggy I know... "string\$", atom, % workaround for bug above "char $in string", atom, 'atom$', atom, 'atom$', atom, 'atom\$', atom, 'char $in atom', atom, $[, ${, $\\, atom, ?MACRO_1, ?MACRO_2(foo), %% Numerical constants 16#DD, % AD Should not be highlighted 32#dd, % AD Should not be highlighted 32#ddAB, % AD Should not be highlighted 32#101, % AD Should not be highlighted 32#ABTR, % AD Should not be highlighted %% Variables Variables = lists:foo(), _Variables = lists:foo(), % AD AppSpec = Xyz/2, Module42 = Xyz(foo, bar), Module:foo(), _Module:foo(), % AD FooÅÅ = lists:reverse([tl,hd,tl,hd]), % AD Should highlight FooÅÅ _FooÅÅ = 42, % AD Should highlight _FooÅÅ %% Bifs erlang:registered(), registered(), hd(tl(tl(hd([a,b,c])))), erlang:anything(lists), %% Guards is_atom(foo), is_float(2.3), is_integer(32), is_number(4323.3), is_function(Fun), is_pid(self()), not_a_guard:is_list([]), %% Other Types atom, % not (currently) hightlighted 234234, 234.43, [list, are, not, higlighted], {nor, is, tuple}, ok. %%% %%% Indentation %%% %%% Left %% Indented % Right indent_basics(X, Y, Z) when X > 42, Z < 13; Y =:= 4711 -> %% comments % right comments case lists:filter(fun(_, AlongName, B, C) -> true end, [a,v,b]) of [] -> Y = 5 * 43, ok; [_|_] -> Y = 5 * 43, ok end, Y, %% List, tuples and binaries [a, b, c ], [ a, b, c ], [ a, b ], {a, b,c }, { a, b,c }, { a, b }, <<1:8, 2:8 >>, << 1:8, 2:8 >>, << 1:8, 2:8 >>, (a, b, c ), ( a, b, c ), ( a, b, c ), call(2#42423 bor #4234, 2#5432, other_arg), ok; indent_basics(Xlongname, #struct{a=Foo, b=Bar}, [X| Y]) -> testing_next_clause, ok; indent_basics( % AD added clause X, % not sure how this should look Y, Z) when X < 42, Z > 13; Y =:= 4711 -> foo; indent_basics(X, Y, Z) when % AD added clause X < 42, Z > 13; % testing when indentation Y =:= 4711 -> foo; indent_basics(X, Y, Z) % AD added clause when % testing when indentation X < 42, Z > 13; % unsure about this one Y =:= 4711 -> foo. indent_nested() -> [ {foo, 2, "string"}, {bar, 3, "another string"} ]. indent_icr(Z) -> % icr = if case receive %% If if Z >= 0 -> X = 43 div 4, foo(X); Z =< 10 -> X = 43 div 4, foo(X); Z == 5 orelse Z == 7 -> X = 43 div 4, foo(X); true -> if_works end, %% Case case {Z, foo, bar} of {Z,_,_} -> X = 43 div 4, foo(X); {Z,_,_} when Z =:= 42 -> % AD line should be indented as a when X = 43 div 4, foo(X); {Z,_,_} when Z < 10 -> % AD when should be indented X = 43 div 4, foo(X); {Z,_,_} when % AD when should be indented Z < 10 % and the guards should follow when andalso % unsure about how though true -> X = 43 div 4, foo(X) end, %% begin begin sune, X = 74234 + foo(8456) + 345 div 43, ok end, %% receive receive {Z,_,_} -> X = 43 div 4, foo(X); Z -> X = 43 div 4, foo(X) end, receive {Z,_,_} -> X = 43 div 4, foo(X); Z % AD added clause when Z =:= 1 -> % This line should be indented by 2 X = 43 div 4, foo(X); Z when % AD added clause Z =:= 2 -> % This line should be indented by 2 X = 43 div 4, foo(X); Z -> X = 43 div 4, foo(X) after infinity -> foo(X), asd(X), 5*43 end, receive after 10 -> foo(X), asd(X), 5*43 end, ok. indent_fun() -> %% Changed fun to one indention level Var = spawn(fun(X) when X == 2; X > 10 -> hello, case Hello() of true when is_atom(X) -> foo; false -> bar end; (Foo) when is_atom(Foo), is_integer(X) -> X = 6* 45, Y = true andalso kalle end), %% check EEP37 named funs Fn1 = fun Fact(N) when N > 0 -> F = Fact(N-1), N * F; Fact(0) -> 1 end, %% check anonymous funs too Fn2 = fun(0) -> 1; (N) -> N end, ok. indent_try_catch() -> try io:format(stdout, "Parsing file ~s, ", [St0#leex.xfile]), {ok,Line3,REAs,Actions,St3} = parse_rules(Xfile, Line2, Macs, St2) catch exit:{badarg,R} -> foo(R), io:format(stdout, "ERROR reason ~p~n", R); error:R % AD added clause when R =:= 42 -> % when should be indented foo(R); error:R % AD added clause when % when should be indented R =:= 42 -> % but unsure about this (maybe 2 more) foo(R); error:R when % AD added clause R =:= foo -> % line should be 2 indented (works) foo(R); error:R -> foo(R), io:format(stdout, "ERROR reason ~p~n", R) after foo('after'), file:close(Xfile) end; indent_try_catch() -> try foo(bar) of X when true andalso kalle -> io:format(stdout, "Parsing file ~s, ", [St0#leex.xfile]), {ok,Line3,REAs,Actions,St3} = parse_rules(Xfile, Line2, Macs, St2); X % AD added clause when false andalso % when should be 2 indented bengt -> gurka(); X when % AD added clause false andalso % line should be 2 indented not bengt -> gurka(); X -> io:format(stdout, "Parsing file ~s, ", [St0#leex.xfile]), {ok,Line3,REAs,Actions,St3} = parse_rules(Xfile, Line2, Macs, St2) catch exit:{badarg,R} -> foo(R), io:format(stdout, "ERROR reason ~p~n", R); error:R -> foo(R), io:format(stdout, "ERROR reason ~p~n", R) after foo('after'), file:close(Xfile), bar(with_long_arg, with_second_arg) end; indent_try_catch() -> try foo() after foo(), bar(with_long_arg, with_second_arg) end. indent_catch() -> D = B + float(43.1), B = catch oskar(X), A = catch (baz + bax), catch foo(), C = catch B + float(43.1), case catch foo(X) of A -> B end, case catch foo(X) of A -> B end, case foo(X) of A -> catch B, X end, try sune of _ -> foo catch _:_ -> baf end, try sune of _ -> X = 5, (catch foo(X)), X + 10 catch _:_ -> baf end, try (catch sune) of _ -> catch foo() %% BUGBUG can't handle catch inside try without parentheses catch _:_ -> baf end, try (catch exit()) catch _ -> catch baf() end, ok. indent_binary() -> X = lists:foldr(fun(M) -> <<Ma/binary, " ">> end, [], A), A = <<X/binary, 0:8>>, B. indent_comprehensions() -> %% I don't have a good idea how we want to handle this %% but they are here to show how they are indented today. Result1 = [X || #record{a=X} <- lists:seq(1, 10), true = (X rem 2) ], Result2 = [X || <<X:32,_:32>> <= <<0:512>>, true = (X rem 2) ], Binary1 = << <<X:8>> || #record{a=X} <- lists:seq(1, 10), true = (X rem 2) >>, Binary2 = << <<X:8>> || <<X:32,_:32>> <= <<0:512>>, true = (X rem 2) >>, ok. %% This causes an error in earlier erlang-mode versions. foo() -> [#foo{ foo = foo}]. %% Record indentation some_function_with_a_very_long_name() -> #'a-long-record-name-like-it-sometimes-is-with-asn.1-records'{ field1=a, field2=b}, case dummy_function_with_a_very_very_long_name(x) of #'a-long-record-name-like-it-sometimes-is-with-asn.1-records'{ field1=a, field2=b} -> ok; Var = #'a-long-record-name-like-it-sometimes-is-with-asn.1-records'{ field1=a, field2=b} -> Var#'a-long-record-name-like-it-sometimes-is-with-asn.1-records'{ field1=a, field2=b}; #xyz{ a=1, b=2} -> ok end. another_function_with_a_very_very_long_name() -> #rec{ field1=1, field2=1}. some_function_name_xyz(xyzzy, #some_record{ field1=Field1, field2=Field2}) -> SomeVariable = f(#'Some-long-record-name'{ field_a = 1, 'inter-xyz-parameters' = #'Some-other-very-long-record-name'{ field2 = Field1, field2 = Field2}}), {ok, SomeVariable}. commas_first() -> {abc, [ {some_var, 1} , {some_other_var, 2} , {erlang_ftw, 9} , {erlang_cookie, 'cookie'} , {cmds, [ {one, "sudo ls"} , {one, "sudo ls"} , {two, "sudo ls"} , {three, "sudo ls"} , {four, "sudo ls"} , {three, "sudo ls"} ] } , {ssh_username, "yow"} , {cluster, [ {aaaa, [ {"10.198.55.12" , "" } , {"10.198.55.13" , "" } ] } , {bbbb, [ {"10.198.55.151", "" } , {"10.198.55.123", "" } , {"10.198.55.34" , "" } , {"10.198.55.85" , "" } , {"10.198.55.67" , "" } ] } , {cccc, [ {"10.198.55.68" , "" } , {"10.198.55.69" , "" } ] } ] } ] }. %% this used to result in a scan-sexp error [{ }]. %% this used to result in 2x the correct indentation within the function %% body, due to the function name being mistaken for a keyword catcher(N) -> try generate_exception(N) of Val -> {N, normal, Val} catch throw:X -> {N, caught, thrown, X}; exit:X -> {N, caught, exited, X}; error:X -> {N, caught, error, X} end.