edc_editor: add pair of bracket highlighting feature
Summary: when mouse cursor on bracket which are '{', '}', the bracket is highlighted with its pair. Reviewers: Jaehyun_Cho, Hermet Reviewed By: Hermet Subscribers: herb, NikaWhite, Jaehyun_Cho Differential Revision: https://phab.enlightenment.org/D3588
This commit is contained in:
parent
f9edc4363d
commit
ed7e655add
|
@ -175,6 +175,28 @@ error_highlight(edit_data *ed, Evas_Object *tb)
|
||||||
}
|
}
|
||||||
evas_textblock_cursor_free(cur1);
|
evas_textblock_cursor_free(cur1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bracket_highlight(edit_data *ed, Evas_Object *tb)
|
||||||
|
{
|
||||||
|
Evas_Textblock_Cursor *cur1 = evas_object_textblock_cursor_new(tb);
|
||||||
|
|
||||||
|
int left_bracket = parser_left_bracket_pos_get(ed->pd);
|
||||||
|
int right_bracket = parser_right_bracket_pos_get(ed->pd);
|
||||||
|
|
||||||
|
evas_textblock_cursor_pos_set(cur1, left_bracket);
|
||||||
|
evas_object_textblock_text_markup_prepend(cur1, "<hilight><color=#B1B1B1FF>");
|
||||||
|
evas_textblock_cursor_pos_set(cur1, left_bracket+1);
|
||||||
|
evas_object_textblock_text_markup_prepend(cur1, "</color></hilight>");
|
||||||
|
|
||||||
|
evas_textblock_cursor_pos_set(cur1, right_bracket);
|
||||||
|
evas_object_textblock_text_markup_prepend(cur1, "<hilight><color=#B1B1B1FF>");
|
||||||
|
evas_textblock_cursor_pos_set(cur1, right_bracket+1);
|
||||||
|
evas_object_textblock_text_markup_prepend(cur1, "</color></hilight>");
|
||||||
|
|
||||||
|
evas_textblock_cursor_free(cur1);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
syntax_color_apply(edit_data *ed, Eina_Bool partial)
|
syntax_color_apply(edit_data *ed, Eina_Bool partial)
|
||||||
{
|
{
|
||||||
|
@ -202,6 +224,7 @@ syntax_color_apply(edit_data *ed, Eina_Bool partial)
|
||||||
Logically that's unnecessary in this case. */
|
Logically that's unnecessary in this case. */
|
||||||
evas_object_textblock_text_markup_set(tb, translated);
|
evas_object_textblock_text_markup_set(tb, translated);
|
||||||
error_highlight(ed, tb);
|
error_highlight(ed, tb);
|
||||||
|
bracket_highlight(ed, tb);
|
||||||
entry_recover(ed, pos);
|
entry_recover(ed, pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -269,6 +292,50 @@ syntax_color_thread_cancel_cb(void *data, Ecore_Thread *thread EINA_UNUSED)
|
||||||
free(td);
|
free(td);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
bracket_changed_cb(void *data)
|
||||||
|
{
|
||||||
|
edit_data *ed = data;
|
||||||
|
syntax_color_partial_update(ed, SYNTAX_COLOR_SHORT_TIME);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bracket_update(edit_data *ed)
|
||||||
|
{
|
||||||
|
int pos = elm_entry_cursor_pos_get(ed->en_edit);
|
||||||
|
if (pos == 0) return;
|
||||||
|
|
||||||
|
Evas_Object *tb = elm_entry_textblock_get(ed->en_edit);
|
||||||
|
Evas_Textblock_Cursor *cur1 = evas_object_textblock_cursor_get(tb);
|
||||||
|
|
||||||
|
char ch1, ch2;
|
||||||
|
ch1 = -1;
|
||||||
|
ch2 = -1;
|
||||||
|
|
||||||
|
ch1 = evas_textblock_cursor_content_get(cur1)[0];
|
||||||
|
Eina_Bool is_exist = evas_textblock_cursor_char_prev(cur1);
|
||||||
|
if (is_exist)
|
||||||
|
ch2 = evas_textblock_cursor_content_get(cur1)[0];
|
||||||
|
evas_textblock_cursor_char_next(cur1);
|
||||||
|
|
||||||
|
if (ch1 != '{' && ch1 != '}' && ch2 != '{' && ch2 != '}')
|
||||||
|
{
|
||||||
|
int prev_left_bracket = parser_prev_left_bracket_pos_get(ed->pd);
|
||||||
|
int prev_right_bracket = parser_prev_right_bracket_pos_get(ed->pd);
|
||||||
|
if (prev_left_bracket != -1 && prev_right_bracket != -1)
|
||||||
|
{
|
||||||
|
parser_left_bracket_pos_set(ed->pd, -1);
|
||||||
|
parser_right_bracket_pos_set(ed->pd, -1);
|
||||||
|
parser_prev_left_bracket_pos_set(ed->pd, -1);
|
||||||
|
parser_prev_right_bracket_pos_set(ed->pd, -1);
|
||||||
|
syntax_color_partial_update(ed, SYNTAX_COLOR_SHORT_TIME);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
parser_bracket_pair_find(ed->pd, ed->en_edit, bracket_changed_cb, ed);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
edit_changed_cb(void *data, Evas_Object *obj EINA_UNUSED, void *event_info)
|
edit_changed_cb(void *data, Evas_Object *obj EINA_UNUSED, void *event_info)
|
||||||
{
|
{
|
||||||
|
@ -324,8 +391,11 @@ edit_changed_cb(void *data, Evas_Object *obj EINA_UNUSED, void *event_info)
|
||||||
if (info->change.del.content[0] == ' ') return;
|
if (info->change.del.content[0] == ' ') return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!syntax_color) return;
|
if (!syntax_color) return;
|
||||||
syntax_color_partial_update(ed, SYNTAX_COLOR_DEFAULT_TIME);
|
syntax_color_partial_update(ed, SYNTAX_COLOR_DEFAULT_TIME);
|
||||||
|
|
||||||
|
parser_bracket_pair_cancel(ed->pd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -730,6 +800,8 @@ edit_cursor_changed_cb(void *data, Evas_Object *obj EINA_UNUSED,
|
||||||
edit_data *ed = data;
|
edit_data *ed = data;
|
||||||
cur_line_pos_set(ed, EINA_FALSE);
|
cur_line_pos_set(ed, EINA_FALSE);
|
||||||
edit_view_sync(ed);
|
edit_view_sync(ed);
|
||||||
|
|
||||||
|
bracket_update(ed);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -52,6 +52,22 @@ typedef struct type_init_thread_data_s
|
||||||
parser_data *pd;
|
parser_data *pd;
|
||||||
} type_init_td;
|
} type_init_td;
|
||||||
|
|
||||||
|
typedef struct bracket_data_s
|
||||||
|
{
|
||||||
|
int left;
|
||||||
|
int right;
|
||||||
|
int prev_left;
|
||||||
|
int prev_right;
|
||||||
|
} bracket;
|
||||||
|
|
||||||
|
typedef struct bracket_thread_data_s
|
||||||
|
{
|
||||||
|
int pos;
|
||||||
|
const char *text;
|
||||||
|
Bracket_Update_Cb update_cb;
|
||||||
|
void *data;
|
||||||
|
} bracket_data_td;
|
||||||
|
|
||||||
struct parser_s
|
struct parser_s
|
||||||
{
|
{
|
||||||
Eina_Inarray *attrs;
|
Eina_Inarray *attrs;
|
||||||
|
@ -59,10 +75,15 @@ struct parser_s
|
||||||
type_init_td *titd;
|
type_init_td *titd;
|
||||||
Eina_List *macro_list;
|
Eina_List *macro_list;
|
||||||
|
|
||||||
|
bracket_data_td *bracket_td;
|
||||||
|
Ecore_Thread *bracket_thread;
|
||||||
|
bracket bracket_pos;
|
||||||
|
|
||||||
|
Eina_Bool is_bracket_thread_req: 1;
|
||||||
|
|
||||||
Eina_Bool macro_update : 1;
|
Eina_Bool macro_update : 1;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
/*****************************************************************************/
|
/*****************************************************************************/
|
||||||
/* Internal method implementation */
|
/* Internal method implementation */
|
||||||
/*****************************************************************************/
|
/*****************************************************************************/
|
||||||
|
@ -1857,6 +1878,8 @@ parser_init(void)
|
||||||
|
|
||||||
td->pd = pd;
|
td->pd = pd;
|
||||||
pd->titd = td;
|
pd->titd = td;
|
||||||
|
pd->bracket_pos.left = -1;
|
||||||
|
pd->bracket_pos.right = -1;
|
||||||
td->thread = ecore_thread_run(type_init_thread_blocking,
|
td->thread = ecore_thread_run(type_init_thread_blocking,
|
||||||
type_init_thread_end,
|
type_init_thread_end,
|
||||||
type_init_thread_cancel, td);
|
type_init_thread_cancel, td);
|
||||||
|
@ -1949,3 +1972,260 @@ parser_macro_update(parser_data *pd, Eina_Bool macro_update)
|
||||||
{
|
{
|
||||||
pd->macro_update = macro_update;
|
pd->macro_update = macro_update;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Eina_Bool
|
||||||
|
bracket_pair_parse(int pos, const char *text, int *left, int *right)
|
||||||
|
{
|
||||||
|
int left_bracket = -1;
|
||||||
|
int right_bracket = -1;
|
||||||
|
|
||||||
|
int cur_pos = pos;
|
||||||
|
int depth = 0;
|
||||||
|
|
||||||
|
if (cur_pos == 0)
|
||||||
|
return EINA_FALSE;
|
||||||
|
|
||||||
|
const char *utf8 = text;
|
||||||
|
int length = strlen(utf8);
|
||||||
|
|
||||||
|
// left, {
|
||||||
|
if (utf8[cur_pos] == '{')
|
||||||
|
{
|
||||||
|
left_bracket = cur_pos;
|
||||||
|
cur_pos++;
|
||||||
|
while (cur_pos < length)
|
||||||
|
{
|
||||||
|
if (utf8[cur_pos] == '{') depth++;
|
||||||
|
else if (utf8[cur_pos] == '}')
|
||||||
|
{
|
||||||
|
if (depth) depth--;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
right_bracket = cur_pos;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cur_pos++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// left, }
|
||||||
|
else if(utf8[cur_pos] == '}')
|
||||||
|
{
|
||||||
|
right_bracket = cur_pos;
|
||||||
|
cur_pos--;
|
||||||
|
while (cur_pos)
|
||||||
|
{
|
||||||
|
if (utf8[cur_pos] == '}') depth++;
|
||||||
|
else if(utf8[cur_pos] == '{')
|
||||||
|
{
|
||||||
|
if(depth) depth--;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
left_bracket = cur_pos;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cur_pos--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// right, {
|
||||||
|
else if(utf8[cur_pos - 1] == '{')
|
||||||
|
{
|
||||||
|
left_bracket = cur_pos - 1;
|
||||||
|
while (cur_pos < length)
|
||||||
|
{
|
||||||
|
if (utf8[cur_pos] == '{') depth++;
|
||||||
|
else if (utf8[cur_pos] == '}')
|
||||||
|
{
|
||||||
|
if (depth) depth--;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
right_bracket = cur_pos;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cur_pos++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// right, }
|
||||||
|
else if(utf8[cur_pos - 1] == '}')
|
||||||
|
{
|
||||||
|
right_bracket = cur_pos - 1;
|
||||||
|
cur_pos -= 2;
|
||||||
|
while (cur_pos)
|
||||||
|
{
|
||||||
|
if (utf8[cur_pos] == '}') depth++;
|
||||||
|
else if (utf8[cur_pos] == '{')
|
||||||
|
{
|
||||||
|
if(depth) depth--;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
left_bracket = cur_pos;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cur_pos--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (left_bracket == -1 || right_bracket == -1)
|
||||||
|
{
|
||||||
|
left_bracket = -1;
|
||||||
|
right_bracket = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
*left = left_bracket;
|
||||||
|
*right = right_bracket;
|
||||||
|
|
||||||
|
return EINA_TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bracket_thread_cb(void *data, Ecore_Thread *thread EINA_UNUSED);
|
||||||
|
static void bracket_thread_end_cb(void *data, Ecore_Thread *thread EINA_UNUSED);
|
||||||
|
static void bracket_thread_cancel_cb(void *data, Ecore_Thread *thread EINA_UNUSED);
|
||||||
|
|
||||||
|
static void
|
||||||
|
bracket_thread_cb(void *data, Ecore_Thread *thread EINA_UNUSED)
|
||||||
|
{
|
||||||
|
parser_data *pd = data;
|
||||||
|
|
||||||
|
if (bracket_pair_parse(pd->bracket_td->pos, pd->bracket_td->text,
|
||||||
|
&(pd->bracket_pos.left),
|
||||||
|
&(pd->bracket_pos.right)) == EINA_FALSE)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (pd->bracket_pos.left != -1 && pd->bracket_pos.right != -1)
|
||||||
|
{
|
||||||
|
if (pd->bracket_pos.prev_left != pd->bracket_pos.left
|
||||||
|
&& pd->bracket_pos.prev_right != pd->bracket_pos.right)
|
||||||
|
{
|
||||||
|
pd->bracket_td->update_cb(pd->bracket_td->data);
|
||||||
|
pd->bracket_pos.prev_left = pd->bracket_pos.left;
|
||||||
|
pd->bracket_pos.prev_right = pd->bracket_pos.right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if(pd->bracket_pos.prev_left != -1 && pd->bracket_pos.prev_right != -1)
|
||||||
|
{
|
||||||
|
pd->bracket_td->update_cb(pd->bracket_td->data);
|
||||||
|
pd->bracket_pos.prev_left = -1;
|
||||||
|
pd->bracket_pos.prev_right = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bracket_thread_end_cb(void *data, Ecore_Thread *thread EINA_UNUSED)
|
||||||
|
{
|
||||||
|
parser_data *pd = data;
|
||||||
|
|
||||||
|
if (pd->is_bracket_thread_req == EINA_TRUE)
|
||||||
|
{
|
||||||
|
pd->is_bracket_thread_req = EINA_FALSE;
|
||||||
|
pd->bracket_thread = ecore_thread_run(bracket_thread_cb,
|
||||||
|
bracket_thread_end_cb,
|
||||||
|
bracket_thread_cancel_cb,
|
||||||
|
pd);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
free(pd->bracket_td);
|
||||||
|
pd->bracket_td = NULL;
|
||||||
|
pd->bracket_thread = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bracket_thread_cancel_cb(void *data, Ecore_Thread *thread EINA_UNUSED)
|
||||||
|
{
|
||||||
|
parser_data *pd = data;
|
||||||
|
|
||||||
|
if (pd->is_bracket_thread_req == EINA_TRUE)
|
||||||
|
{
|
||||||
|
pd->is_bracket_thread_req = EINA_FALSE;
|
||||||
|
pd->bracket_thread = ecore_thread_run(bracket_thread_cb,
|
||||||
|
bracket_thread_end_cb,
|
||||||
|
bracket_thread_cancel_cb,
|
||||||
|
pd);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
free(pd->bracket_td);
|
||||||
|
pd->bracket_td = NULL;
|
||||||
|
pd->bracket_thread = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
parser_bracket_pair_find(parser_data *pd, Evas_Object *entry,
|
||||||
|
Bracket_Update_Cb func, void *data)
|
||||||
|
{
|
||||||
|
|
||||||
|
ecore_thread_cancel(pd->bracket_thread);
|
||||||
|
|
||||||
|
if (pd->bracket_td == NULL)
|
||||||
|
pd->bracket_td = calloc(1, sizeof(bracket_data_td));
|
||||||
|
|
||||||
|
const char *text = elm_entry_entry_get(entry);
|
||||||
|
char *utf8 = elm_entry_markup_to_utf8(text);
|
||||||
|
int pos = elm_entry_cursor_pos_get(entry);
|
||||||
|
|
||||||
|
pd->bracket_td->pos = pos;
|
||||||
|
pd->bracket_td->text = utf8;
|
||||||
|
pd->bracket_td->update_cb = func;
|
||||||
|
pd->bracket_td->data = data;
|
||||||
|
|
||||||
|
if (pd->bracket_thread == NULL)
|
||||||
|
{
|
||||||
|
pd->is_bracket_thread_req = EINA_FALSE;
|
||||||
|
pd->bracket_thread = ecore_thread_run(bracket_thread_cb,
|
||||||
|
bracket_thread_end_cb,
|
||||||
|
bracket_thread_cancel_cb,
|
||||||
|
pd);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
pd->is_bracket_thread_req = EINA_TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void parser_bracket_pair_cancel(parser_data *pd)
|
||||||
|
{
|
||||||
|
ecore_thread_cancel(pd->bracket_thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
void parser_left_bracket_pos_set(parser_data *pd, int pos)
|
||||||
|
{
|
||||||
|
pd->bracket_pos.left = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
void parser_right_bracket_pos_set(parser_data *pd, int pos)
|
||||||
|
{
|
||||||
|
pd->bracket_pos.right = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
void parser_prev_left_bracket_pos_set(parser_data *pd, int pos)
|
||||||
|
{
|
||||||
|
pd->bracket_pos.prev_left = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
void parser_prev_right_bracket_pos_set(parser_data *pd, int pos)
|
||||||
|
{
|
||||||
|
pd->bracket_pos.prev_right = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parser_left_bracket_pos_get(parser_data *pd)
|
||||||
|
{
|
||||||
|
return pd->bracket_pos.left;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parser_right_bracket_pos_get(parser_data *pd)
|
||||||
|
{
|
||||||
|
return pd->bracket_pos.right;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parser_prev_left_bracket_pos_get(parser_data *pd)
|
||||||
|
{
|
||||||
|
return pd->bracket_pos.prev_left;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parser_prev_right_bracket_pos_get(parser_data *pd)
|
||||||
|
{
|
||||||
|
return pd->bracket_pos.prev_right;
|
||||||
|
}
|
||||||
|
|
|
@ -139,7 +139,17 @@ Eina_Bool parser_state_info_get(Evas_Object *entry, state_info *info);
|
||||||
void parser_macro_list_set(parser_data *pd, const char *text);
|
void parser_macro_list_set(parser_data *pd, const char *text);
|
||||||
Eina_List *parser_macro_list_get(parser_data *pd);
|
Eina_List *parser_macro_list_get(parser_data *pd);
|
||||||
void parser_macro_update(parser_data *pd, Eina_Bool macro_update);
|
void parser_macro_update(parser_data *pd, Eina_Bool macro_update);
|
||||||
|
typedef void (*Bracket_Update_Cb)(void *data);
|
||||||
|
void parser_bracket_pair_find(parser_data *pd, Evas_Object *entry, Bracket_Update_Cb func, void *data);
|
||||||
|
void parser_bracket_pair_cancel(parser_data *pd);
|
||||||
|
void parser_left_bracket_pos_set(parser_data *pd, int pos);
|
||||||
|
void parser_right_bracket_pos_set(parser_data *pd, int pos);
|
||||||
|
void parser_prev_left_bracket_pos_set(parser_data *pd, int pos);
|
||||||
|
void parser_prev_right_bracket_pos_set(parser_data *pd, int pos);
|
||||||
|
int parser_left_bracket_pos_get(parser_data *pd);
|
||||||
|
int parser_right_bracket_pos_get(parser_data *pd);
|
||||||
|
int parser_prev_left_bracket_pos_get(parser_data *pd);
|
||||||
|
int parser_prev_right_bracket_pos_get(parser_data *pd);
|
||||||
|
|
||||||
/* syntax helper */
|
/* syntax helper */
|
||||||
syntax_helper *syntax_init(void);
|
syntax_helper *syntax_init(void);
|
||||||
|
|
Loading…
Reference in New Issue