Create pp::unicode option. The idea is to be able to disable unicode characters, but still be able to use mixfix notation.

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-09-03 10:09:19 -07:00
parent e031d7bc10
commit 544229e5d3
28 changed files with 227 additions and 59 deletions

View file

@ -27,7 +27,7 @@ struct frontend::imp {
// Remark: only named objects are stored in the dictionary.
typedef std::unordered_map<name, operator_info, name_hash, name_eq> operator_table;
typedef std::unordered_map<name, implicit_info, name_hash, name_eq> implicit_table;
typedef std::unordered_map<expr, operator_info, expr_hash, std::equal_to<expr>> expr_to_operator;
typedef std::unordered_map<expr, list<operator_info>, expr_hash, std::equal_to<expr>> expr_to_operators;
typedef std::unordered_map<expr_pair, expr, expr_pair_hash, expr_pair_eq> coercion_map;
typedef std::unordered_set<expr, expr_hash, std::equal_to<expr>> coercion_set;
@ -36,7 +36,7 @@ struct frontend::imp {
environment m_env;
operator_table m_nud; // nud table for Pratt's parser
operator_table m_led; // led table for Pratt's parser
expr_to_operator m_expr_to_operator; // map denotations to operators (this is used for pretty printing)
expr_to_operators m_expr_to_operators; // map denotations to operators (this is used for pretty printing)
implicit_table m_implicit_table; // track the number of implicit arguments for a symbol.
coercion_map m_coercion_map; // mapping from (given_type, expected_type) -> coercion
coercion_set m_coercion_set; // Set of coercions
@ -95,12 +95,18 @@ struct frontend::imp {
}
/** \brief Find the operator that is used as notation for the given expression. */
operator_info find_op_for(expr const & e) const {
auto it = m_expr_to_operator.find(e);
if (it != m_expr_to_operator.end())
return it->second;
else if (has_parent())
return m_parent->find_op_for(e);
operator_info find_op_for(expr const & e, bool unicode) const {
auto it = m_expr_to_operators.find(e);
if (it != m_expr_to_operators.end()) {
auto l = it->second;
for (auto op : l) {
if (unicode || op.is_safe_ascii())
return op;
}
}
if (has_parent())
return m_parent->find_op_for(e, unicode);
else
return operator_info();
}
@ -108,20 +114,26 @@ struct frontend::imp {
/** \brief Remove all internal denotations that are associated with the given operator symbol (aka notation) */
void remove_bindings(operator_info const & op) {
for (expr const & d : op.get_denotations()) {
if (has_parent() && m_parent->find_op_for(d)) {
if (has_parent() && m_parent->find_op_for(d, true)) {
// parent has an association for d... we must hide it.
insert(m_expr_to_operator, d, operator_info());
insert(m_expr_to_operators, d, list<operator_info>(operator_info()));
} else {
m_expr_to_operator.erase(d);
m_expr_to_operators.erase(d);
}
}
}
/** \brief Add a new entry d -> op in the mapping m_expr_to_operators */
void insert_expr_to_operator_entry(expr const & d, operator_info const & op) {
list<operator_info> & l = m_expr_to_operators[d];
l = cons(op, l);
}
/** \brief Register the new operator in the tables for parsing and pretty printing. */
void register_new_op(operator_info new_op, expr const & d, bool led) {
new_op.add_expr(d);
insert_op(new_op, led);
insert(m_expr_to_operator, d, new_op);
insert_expr_to_operator_entry(d, new_op);
}
/**
@ -173,7 +185,7 @@ struct frontend::imp {
// overload
if (defined_here(old_op, led)) {
old_op.add_expr(d);
insert(m_expr_to_operator, d, old_op);
insert_expr_to_operator_entry(d, old_op);
} else {
// we must copy the operator because it was defined in
// a parent frontend.
@ -391,7 +403,7 @@ void frontend::add_mixfixl(unsigned sz, name const * opns, unsigned p, expr cons
void frontend::add_mixfixr(unsigned sz, name const * opns, unsigned p, expr const & d) { m_imp->add_mixfixr(sz, opns, p, d); }
void frontend::add_mixfixc(unsigned sz, name const * opns, unsigned p, expr const & d) { m_imp->add_mixfixc(sz, opns, p, d); }
void frontend::add_mixfixo(unsigned sz, name const * opns, unsigned p, expr const & d) { m_imp->add_mixfixo(sz, opns, p, d); }
operator_info frontend::find_op_for(expr const & n) const { return m_imp->find_op_for(n); }
operator_info frontend::find_op_for(expr const & n, bool unicode) const { return m_imp->find_op_for(n, unicode); }
operator_info frontend::find_nud(name const & n) const { return m_imp->find_nud(n); }
operator_info frontend::find_led(name const & n) const { return m_imp->find_led(n); }

View file

@ -93,8 +93,11 @@ public:
return the null operator.
\remark This is used for pretty printing.
\remark If unicode is false, then only operators containing
safe ASCII chars are considered.
*/
operator_info find_op_for(expr const & e) const;
operator_info find_op_for(expr const & e, bool unicode) const;
/**
\brief Return the operator (if one exists) that can appear at
the beginning of a language construct.

View file

@ -65,6 +65,11 @@ name const & operator_info::get_op_name() const { lean_assert(m_ptr); return car
list<name> const & operator_info::get_op_name_parts() const { lean_assert(m_ptr); return m_ptr->m_op_parts; }
bool operator_info::is_safe_ascii() const {
auto l = get_op_name_parts();
return std::all_of(l.begin(), l.end(), [](name const & p) { return p.is_safe_ascii(); });
}
operator_info operator_info::copy() const { return operator_info(new imp(*m_ptr)); }
bool operator==(operator_info const & op1, operator_info const & op2) {

View file

@ -83,6 +83,9 @@ public:
/** \brief Return the operators parts. */
list<name> const & get_op_name_parts() const;
/** \brief Return true if all parts of the operator use only safe ASCII characters */
bool is_safe_ascii() const;
/** \brief Return a copy of the operator. */
operator_info copy() const;

View file

@ -1137,7 +1137,7 @@ class parser::imp {
regular(m_frontend) << m_frontend << endl;
}
} else if (opt_id == g_options_kwd) {
regular(m_frontend) << m_frontend.get_state().get_options() << endl;
regular(m_frontend) << pp(m_frontend.get_state().get_options()) << endl;
} else {
throw parser_error("invalid Show command, expression, 'Options' or 'Environment' expected", m_last_cmd_pos);
}

View file

@ -145,6 +145,7 @@ class pp_fn {
unsigned m_max_depth;
unsigned m_max_steps;
bool m_implict; //!< if true show implicit arguments
bool m_unicode; //!< if true use unicode chars
bool m_coercion; //!< if true show coercions
bool m_notation; //!< if true use notation
bool m_extra_lets; //!< introduce extra let-expression to cope with sharing.
@ -197,7 +198,7 @@ class pp_fn {
}
result pp_ellipsis() {
return mk_result(m_notation ? g_ellipsis_n_fmt : g_ellipsis_fmt, 1);
return mk_result(m_unicode ? g_ellipsis_n_fmt : g_ellipsis_fmt, 1);
}
result pp_var(expr const & e) {
@ -223,14 +224,14 @@ class pp_fn {
}
result pp_value(expr const & e) {
return mk_result(to_value(e).pp(m_notation), 1);
return mk_result(to_value(e).pp(m_unicode), 1);
}
result pp_type(expr const & e) {
if (e == Type()) {
return mk_result(g_Type_fmt, 1);
} else {
return mk_result(format{g_Type_fmt, space(), ::lean::pp(ty_level(e))}, 1);
return mk_result(format{g_Type_fmt, space(), ::lean::pp(ty_level(e), m_unicode)}, 1);
}
}
@ -312,7 +313,7 @@ class pp_fn {
local_names::mk_scope mk(m_local_names);
expr b = collect_nested_quantifiers(e, is_forall, nested);
format head;
if (m_notation)
if (m_unicode)
head = is_forall ? g_forall_n_fmt : g_exists_n_fmt;
else
head = is_forall ? g_forall_fmt : g_exists_fmt;
@ -369,7 +370,7 @@ class pp_fn {
if (is_constant(e) && m_local_names.find(const_name(e)) != m_local_names.end())
return operator_info();
else
return m_frontend.find_op_for(e);
return m_frontend.find_op_for(e, m_unicode);
}
/**
@ -760,7 +761,7 @@ class pp_fn {
lean_assert(!T);
result p_lhs = pp_child(abst_domain(e), depth);
result p_rhs = pp_arrow_body(abst_body(e), depth);
format r_format = group(format{p_lhs.first, space(), m_notation ? g_arrow_n_fmt : g_arrow_fmt, line(), p_rhs.first});
format r_format = group(format{p_lhs.first, space(), m_unicode ? g_arrow_n_fmt : g_arrow_fmt, line(), p_rhs.first});
return mk_result(r_format, p_lhs.second + p_rhs.second + 1);
} else {
buffer<std::pair<name, expr>> nested;
@ -770,7 +771,7 @@ class pp_fn {
unsigned head_indent = m_indent;
format head;
if (!T && !implicit_args) {
if (m_notation) {
if (m_unicode) {
head = is_lambda(e) ? g_lambda_n_fmt : g_Pi_n_fmt;
head_indent = 2;
} else {
@ -987,6 +988,7 @@ class pp_fn {
m_max_depth = get_pp_max_depth(opts);
m_max_steps = get_pp_max_steps(opts);
m_implict = get_pp_implicit(opts);
m_unicode = get_pp_unicode(opts);
m_coercion = get_pp_coercion(opts);
m_notation = get_pp_notation(opts);
m_extra_lets = get_pp_extra_lets(opts);
@ -1146,7 +1148,8 @@ class pp_formatter_cell : public formatter_cell {
}
format pp_uvar_decl(object const & obj, options const & opts) {
return format{highlight_command(format(obj.keyword())), space(), format(obj.get_name()), space(), format("\u2265"), space(), ::lean::pp(obj.get_cnstr_level())};
bool unicode = get_pp_unicode(opts);
return format{highlight_command(format(obj.keyword())), space(), format(obj.get_name()), space(), format(unicode ? "\u2265" : ">="), space(), ::lean::pp(obj.get_cnstr_level(), unicode)};
}
format pp_postulate(object const & obj, options const & opts) {

View file

@ -33,7 +33,7 @@ static char g_normalized[256];
/** \brief Auxiliary class for initializing global variable \c g_normalized. */
class init_normalized_table {
void set(int i, char v) { g_normalized[static_cast<unsigned>(i)] = v; }
void set(int i, char v) { g_normalized[static_cast<unsigned char>(i)] = v; }
public:
init_normalized_table() {
// by default all characters are in group c,

View file

@ -195,7 +195,7 @@ std::ostream & operator<<(std::ostream & out, level const & l) {
return out;
}
format pp(level const & l) {
format pp(level const & l, bool unicode) {
switch (kind(l)) {
case level_kind::UVar:
if (l.is_bottom())
@ -206,19 +206,30 @@ format pp(level const & l) {
if (lift_of(l).is_bottom())
return format(lift_offset(l));
else
return format{pp(lift_of(l)), format("+"), format(lift_offset(l))};
return format{pp(lift_of(l), unicode), format("+"), format(lift_offset(l))};
case level_kind::Max: {
format r = pp(max_level(l, 0));
if (unicode) {
format r = pp(max_level(l, 0), unicode);
for (unsigned i = 1; i < max_size(l); i++) {
r += format{space(), format("\u2294"), space()};
r += pp(max_level(l, i));
r += format{space(), format("\u2294"), line()};
r += pp(max_level(l, i), unicode);
}
return group(r);
} else {
format r = format("max");
for (unsigned i = 0; i < max_size(l); i++)
r += format{line(), pp(max_level(l, i), unicode)};
return group(nest(1, format{lp(), r, rp()}));
}
return r;
}}
lean_unreachable();
return format();
}
format pp(level const & l, options const & opts) {
return pp(l, get_pp_unicode(opts));
}
level max(std::initializer_list<level> const & l) {
if (l.size() == 1)
return *(l.begin());

View file

@ -8,6 +8,7 @@ Author: Leonardo de Moura
#include <iostream>
#include "name.h"
#include "format.h"
#include "options.h"
namespace lean {
class environment;
@ -65,6 +66,7 @@ inline bool is_max (level const & l) { return kind(l) == level_kind::Max; }
inline level const * max_begin_levels(level const & l) { return &max_level(l, 0); }
inline level const * max_end_levels(level const & l) { return max_begin_levels(l) + max_size(l); }
format pp(level const & l);
format pp(level const & l, bool unicode);
format pp(level const & l, options const & opts = options());
}
void print(lean::level const & l);

View file

@ -108,11 +108,21 @@ static void tst5() {
lean_assert(name(name("foo"), 0u).get_prefix().is_string());
}
static void tst6() {
lean_assert(name({"foo", "bla"}).is_safe_ascii());
lean_assert(name(123u).is_safe_ascii());
lean_assert(name(name(name(230u), "bla"), "foo").is_safe_ascii());
lean_assert(!name({"foo", "b\u2200aaa"}).is_safe_ascii());
lean_assert(!name({"\u2200", "boo"}).is_safe_ascii());
lean_assert(!name(name(name(230u), "bla\u2200"), "foo").is_safe_ascii());
}
int main() {
tst1();
tst2();
tst3();
tst4();
tst5();
tst6();
return has_violations() ? 1 : 0;
}

View file

@ -1,2 +1,2 @@
add_library(util trace.cpp debug.cpp name.cpp exception.cpp
hash.cpp escaped.cpp bit_tricks.cpp safe_arith.cpp)
hash.cpp escaped.cpp bit_tricks.cpp safe_arith.cpp ascii.cpp)

44
src/util/ascii.cpp Normal file
View file

@ -0,0 +1,44 @@
/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#include <initializer_list>
namespace lean {
static char g_safe_ascii[256];
class init_safe_ascii {
void set(int i, bool v) { g_safe_ascii[static_cast<unsigned char>(i)] = v; }
public:
init_safe_ascii() {
for (int i = 0; i <= 255; i++)
set(i, false);
// digits and characters are safe
for (int i = '0'; i <= '9'; i++) set(i, true);
for (int i = 'a'; i <= 'z'; i++) set(i, true);
for (int i = 'A'; i <= 'Z'; i++) set(i, true);
// the following characters are also safe
for (unsigned char b : {'_', ' ', '\t', '\r', '\n', '(', ')', '{', '}', ':', '.', ',', '\"', '\'', '`', '!', '#',
'=', '<', '>', '@', '^', '|', '&', '~', '+', '-', '*', '/', '\\', '$', '%', '?', ';', '[', ']'})
set(b, true);
}
};
static init_safe_ascii g_init_safe_ascii;
bool is_safe_ascii(char c) {
return g_safe_ascii[static_cast<unsigned char>(c)];
}
bool is_safe_ascii(char const * str) {
if (str) {
while (*str != 0) {
if (!is_safe_ascii(*str))
return false;
str++;
}
}
return true;
}
}

19
src/util/ascii.h Normal file
View file

@ -0,0 +1,19 @@
/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#pragma once
namespace lean {
/**
\brief Return true iff \c c is a "safe" ASCII characters.
It is a "keyboard" character.
*/
bool is_safe_ascii(char c);
/**
\brief Return true iff the given string contains only "safe"
ASCII character.
*/
bool is_safe_ascii(char const * str);
}

View file

@ -28,6 +28,7 @@ class list {
public:
list():m_ptr(nullptr) {}
list(T const & h, list const & t):m_ptr(new cell(h, t)) {}
list(T const & h):m_ptr(new cell(h, list())) {}
list(list const & s):m_ptr(s.m_ptr) { if (m_ptr) m_ptr->inc_ref(); }
list(list&& s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
list(std::initializer_list<T> const & l):list() {

View file

@ -14,6 +14,7 @@ Author: Leonardo de Moura
#include "buffer.h"
#include "hash.h"
#include "trace.h"
#include "ascii.h"
namespace lean {
constexpr char const * anonymous_str = "[anonymous]";
@ -302,6 +303,18 @@ unsigned name::hash() const {
return m_ptr ? m_ptr->m_hash : 11;
}
bool name::is_safe_ascii() const {
imp * i = m_ptr;
while (i) {
if (i->m_is_string) {
if (!::lean::is_safe_ascii(i->m_str))
return false;
}
i = i->m_prefix;
}
return true;
}
std::string name::to_string() const {
std::ostringstream s;
imp::display(s, m_ptr);

View file

@ -55,11 +55,11 @@ public:
\brief Convert this hierarchical name into a string.
*/
std::string to_string() const;
/**
\brief Size of the this name (in characters).
*/
/** \brief Size of the this name (in characters). */
size_t size() const;
unsigned hash() const;
/** \brief Return true iff the name contains only safe ASCII chars */
bool is_safe_ascii() const;
friend std::ostream & operator<<(std::ostream & out, name const & n);
};
struct name_hash { unsigned operator()(name const & n) const { return n.hash(); } };

View file

@ -15,6 +15,10 @@
#define LEAN_DEFAULT_PP_INDENTATION 4
#endif
#ifndef LEAN_DEFAULT_PP_UNICODE
#define LEAN_DEFAULT_PP_UNICODE true
#endif
#ifndef LEAN_DEFAULT_PP_WIDTH
#define LEAN_DEFAULT_PP_WIDTH 120
#endif
@ -37,10 +41,12 @@
namespace lean {
static name g_pp_indent{"pp", "indent"};
static name g_pp_unicode{"pp", "unicode"};
static name g_pp_colors{"pp", "colors"};
static name g_pp_width{"pp", "width"};
RegisterUnsignedOption(g_pp_indent, LEAN_DEFAULT_PP_INDENTATION, "(pretty printer) default indentation");
RegisterBoolOption(g_pp_unicode, LEAN_DEFAULT_PP_UNICODE, "(pretty printer) use unicode characters");
RegisterBoolOption(g_pp_colors, LEAN_DEFAULT_PP_COLORS, "(pretty printer) use colors");
RegisterUnsignedOption(g_pp_width, LEAN_DEFAULT_PP_WIDTH, "(pretty printer) line width");
@ -48,6 +54,10 @@ unsigned get_pp_indent(options const & o) {
return o.get_unsigned(g_pp_indent, LEAN_DEFAULT_PP_INDENTATION);
}
bool get_pp_unicode(options const & o) {
return o.get_bool(g_pp_unicode, LEAN_DEFAULT_PP_UNICODE);
}
bool get_pp_colors(options const & o) {
return o.get_bool(g_pp_colors, LEAN_DEFAULT_PP_COLORS);
}

View file

@ -305,6 +305,8 @@ inline format fillwords(std::initializer_list<std::string> const & l) {
class options;
/** \brief Extract indentation from options */
unsigned get_pp_indent(options const & o);
/** \brief Return unicode characters flag */
bool get_pp_unicode(options const & o);
/** \brief Format a hierarchical name */
format pp(name const & n);

View file

@ -117,6 +117,7 @@ char const * options::get_string(char const * n, char const * default_value) con
static char const * g_left_angle_bracket = "\u27E8";
static char const * g_right_angle_bracket = "\u27E9";
static char const * g_arrow = "\u21a6";
static char const * g_assign = ":=";
options options::update(name const & n, sexpr const & v) const {
if (contains(n)) {
@ -141,25 +142,31 @@ options join(options const & opts1, options const & opts2) {
}
format pp(options const & o) {
bool unicode = get_pp_unicode(o);
format r;
bool first = true;
char const * arrow = unicode ? g_arrow : g_assign;
for_each(o.m_value, [&](sexpr const & p) {
if (first) { first = false; } else { r += comma(); r += line(); }
name const & n = to_name(head(p));
unsigned sz = n.size();
r += group(nest(sz+3, format{pp(head(p)), space(), format(g_arrow), space(), pp(tail(p))}));
unsigned indent = unicode ? sz+3 : sz+4;
r += group(nest(indent, format{pp(head(p)), space(), format(arrow), space(), pp(tail(p))}));
});
return group(nest(1, format{format(g_left_angle_bracket), r, format(g_right_angle_bracket)}));
format open = unicode ? format(g_left_angle_bracket) : lp();
format close = unicode ? format(g_right_angle_bracket) : rp();
return group(nest(1, format{open, r, close}));
}
std::ostream & operator<<(std::ostream & out, options const & o) {
out << g_left_angle_bracket;
bool unicode = get_pp_unicode(o);
out << (unicode ? g_left_angle_bracket : "(");
bool first = true;
for_each(o.m_value, [&](sexpr const & p) {
if (first) first = false; else out << ", ";
out << head(p) << " " << g_arrow << " " << tail(p);
out << head(p) << " " << (unicode ? g_arrow : g_assign) << " " << tail(p);
});
out << g_right_angle_bracket;
out << (unicode ? g_right_angle_bracket : ")");
return out;
}
}

View file

@ -70,7 +70,7 @@ struct mk_option_declaration {
#define LEAN_MAKE_OPTION_NAME_CORE(LINE) LEAN_OPTION_ ## LINE
#define LEAN_MAKE_OPTION_NAME(LINE) LEAN_MAKE_OPTION_NAME_CORE(LINE)
#define LEAN_OPTION_UNIQUE_NAME LEAN_MAKE_OPTION_NAME(__LINE__)
#define RegisterOption(N,K,D,DESC) ::lean::mk_option_declaration LEAN_OPTION_UNIQUE_NAME(N,K,D,DESC)
#define RegisterOption(N,K,D,DESC) static ::lean::mk_option_declaration LEAN_OPTION_UNIQUE_NAME(N,K,D,DESC)
#define RegisterOptionCore(N,K,D,DESC) RegisterOption(N,K,#D,DESC)
#define RegisterBoolOption(N,D,DESC) RegisterOptionCore(N, BoolOption, D, DESC);
#define RegisterUnsignedOption(N,D,DESC) RegisterOptionCore(N, UnsignedOption, D, DESC);

View file

@ -4,6 +4,8 @@ Set pp::colors false
Show true
Set lean::pp::notation false
Show true && false
Set pp::unicode false
Show true && false
Variable a : Bool
Variable a : Bool
Variable b : Bool

View file

@ -1,13 +1,15 @@
Set: pp::colors
Set: lean::pp::notation
and
Set: pp::unicode
and true false
Assumed: a
Error (line: 8, pos: 0) invalid object declaration, environment already has an object named 'a'
Error (line: 10, pos: 0) invalid object declaration, environment already has an object named 'a'
Assumed: b
and a b
Assumed: A
Error (line: 12, pos: 11) type mismatch at application
Error (line: 14, pos: 11) type mismatch at application
and a A
Function type:
Bool -> Bool -> Bool
@ -15,8 +17,8 @@ Arguments types:
Bool
Type
Variable A : Type
⟨lean::pp::notation ↦ false, pp::colors ↦ false⟩
Error (line: 15, pos: 4) unknown option 'lean::p::notation', type 'Help Options.' for list of available options
Error (line: 16, pos: 23) invalid option value, given option is not an integer
(pp::unicode := false, lean::pp::notation := false, pp::colors := false)
Error (line: 17, pos: 4) unknown option 'lean::p::notation', type 'Help Options.' for list of available options
Error (line: 18, pos: 23) invalid option value, given option is not an integer
Set: lean::pp::notation
a b
a /\ b

View file

@ -3,8 +3,8 @@
Assumed: g
++ ⊥ ++
Set: lean::pp::notation
f (f true false) true
f (f ⊥)
Assumed: a
Assumed: b
g (g a b) a
f (f true false) false
f (f ⊥) ⊥

View file

@ -1,5 +1,5 @@
Set pp::colors false
Show 1 + true
Variable R : Type
Variable T : Type
Variable r2t : R -> T

View file

@ -1,8 +1,9 @@
Error (line: 1, pos: 10) application type mismatch, none of the overloads can be used
Set: pp::colors
Error (line: 2, pos: 9) application type mismatch, none of the overloads can be used
Candidates:
Real::add : Real → Real → Real
Int::add : Int → Int → Int
Nat::add : Nat → Nat → Nat
Real::add : Real → Real → Real
Int::add : Int → Int → Int
Nat::add : Nat → Nat → Nat
Arguments:
1 : Nat
: Bool
@ -23,8 +24,8 @@ f (r2t b) (t2r a)
f a b
Error (line: 20, pos: 10) ambiguous overloads
Candidates:
g : R -> T -> R
f : T -> R -> T
g : R → T → R
f : T → R → T
Arguments:
b : R
a : T

View file

@ -2,7 +2,7 @@
Notation 10 if _ then _ : implies
if then ⊥
if then (if a then ⊥)
implies true (implies a false)
implies (implies a ⊥)
Notation 100 _ |- _ ; _ : f
f c d e
c |- d ; e

9
tests/lean/unicode.lean Normal file
View file

@ -0,0 +1,9 @@
Set pp::colors false
Show true /\ false
Set pp::unicode false
Show true /\ false
Set pp::unicode true
Set lean::pp::notation false
Show true /\ false
Set pp::unicode false
Show true /\ false

View file

@ -0,0 +1,9 @@
Set: pp::colors
∧ ⊥
Set: pp::unicode
true /\ false
Set: pp::unicode
Set: lean::pp::notation
and
Set: pp::unicode
and true false