2017-08-24 14:41:20 +00:00
|
|
|
mod context;
|
2017-09-11 15:00:51 +00:00
|
|
|
mod itemize;
|
2017-09-12 15:28:01 +00:00
|
|
|
mod model_clip_iterator;
|
2017-08-24 14:41:20 +00:00
|
|
|
|
2017-08-31 15:37:55 +00:00
|
|
|
pub use self::context::Context;
|
2017-09-05 15:23:46 +00:00
|
|
|
pub use self::context::CellMetrics;
|
2017-09-13 09:39:02 +00:00
|
|
|
use self::model_clip_iterator::{RowView, ModelClipIteratorFactory};
|
2017-08-31 15:37:55 +00:00
|
|
|
|
2017-09-07 15:51:12 +00:00
|
|
|
use mode;
|
2017-08-31 15:37:55 +00:00
|
|
|
use color;
|
2017-08-18 15:35:33 +00:00
|
|
|
use sys::pango::*;
|
2017-12-31 09:47:50 +00:00
|
|
|
use sys::pangocairo::*;
|
2017-08-18 15:35:33 +00:00
|
|
|
use pango;
|
|
|
|
use cairo;
|
2017-12-31 09:47:50 +00:00
|
|
|
use pangocairo;
|
|
|
|
|
2017-09-07 15:51:12 +00:00
|
|
|
use cursor;
|
2017-08-24 14:41:20 +00:00
|
|
|
use ui_model;
|
2017-08-18 15:35:33 +00:00
|
|
|
|
2017-08-25 15:32:30 +00:00
|
|
|
pub fn render(
|
|
|
|
ctx: &cairo::Context,
|
2017-09-07 15:51:12 +00:00
|
|
|
cursor: &cursor::Cursor,
|
2017-09-06 14:18:30 +00:00
|
|
|
font_ctx: &context::Context,
|
2017-08-31 15:37:55 +00:00
|
|
|
ui_model: &ui_model::UiModel,
|
|
|
|
color_model: &color::ColorModel,
|
2017-09-07 15:51:12 +00:00
|
|
|
mode: &mode::Mode,
|
2017-08-25 15:32:30 +00:00
|
|
|
) {
|
2017-08-31 15:37:55 +00:00
|
|
|
ctx.set_source_rgb(
|
|
|
|
color_model.bg_color.0,
|
|
|
|
color_model.bg_color.1,
|
|
|
|
color_model.bg_color.2,
|
|
|
|
);
|
|
|
|
ctx.paint();
|
2017-08-25 15:32:30 +00:00
|
|
|
|
2017-09-12 20:27:45 +00:00
|
|
|
let cell_metrics = font_ctx.cell_metrics();
|
2017-09-13 09:39:02 +00:00
|
|
|
let &CellMetrics { char_width, .. } = cell_metrics;
|
2017-09-07 15:51:12 +00:00
|
|
|
let (cursor_row, cursor_col) = ui_model.get_cursor();
|
2017-08-26 20:17:09 +00:00
|
|
|
|
2017-12-12 13:40:13 +00:00
|
|
|
for cell_view in ui_model.get_clip_iterator(ctx, cell_metrics) {
|
|
|
|
let mut line_x = 0.0;
|
|
|
|
|
|
|
|
for (col, cell) in cell_view.line.line.iter().enumerate() {
|
|
|
|
draw_cell_bg(&cell_view, color_model, cell, col, line_x);
|
|
|
|
line_x += char_width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-13 09:39:02 +00:00
|
|
|
for cell_view in ui_model.get_clip_iterator(ctx, cell_metrics) {
|
2017-08-27 19:29:43 +00:00
|
|
|
let mut line_x = 0.0;
|
2017-09-13 09:39:02 +00:00
|
|
|
let RowView { line, row, line_y, .. } = cell_view;
|
2017-08-31 15:37:55 +00:00
|
|
|
|
2017-09-14 15:34:08 +00:00
|
|
|
for (col, cell) in line.line.iter().enumerate() {
|
2017-09-06 14:18:30 +00:00
|
|
|
|
2017-09-13 09:39:02 +00:00
|
|
|
draw_cell(&cell_view, color_model, cell, col, line_x);
|
|
|
|
|
|
|
|
draw_underline(&cell_view, color_model, cell, line_x);
|
|
|
|
|
2017-09-07 15:51:12 +00:00
|
|
|
if row == cursor_row && col == cursor_col {
|
2017-09-13 09:39:02 +00:00
|
|
|
let double_width = line.line.get(col + 1).map_or(
|
|
|
|
false,
|
|
|
|
|c| c.attrs.double_width,
|
|
|
|
);
|
2017-09-07 19:53:32 +00:00
|
|
|
ctx.move_to(line_x, line_y);
|
2017-09-07 15:51:12 +00:00
|
|
|
cursor.draw(
|
|
|
|
ctx,
|
|
|
|
font_ctx,
|
|
|
|
mode,
|
|
|
|
line_y,
|
2017-09-13 09:39:02 +00:00
|
|
|
double_width,
|
2017-09-07 15:51:12 +00:00
|
|
|
color_model.actual_cell_bg(cell),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-08-27 19:29:43 +00:00
|
|
|
line_x += char_width;
|
2017-08-25 15:32:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-13 09:39:02 +00:00
|
|
|
fn draw_underline(
|
|
|
|
cell_view: &RowView,
|
|
|
|
color_model: &color::ColorModel,
|
|
|
|
cell: &ui_model::Cell,
|
|
|
|
line_x: f64,
|
|
|
|
) {
|
|
|
|
|
|
|
|
if cell.attrs.underline || cell.attrs.undercurl {
|
|
|
|
|
|
|
|
let &RowView {
|
|
|
|
ctx,
|
|
|
|
line_y,
|
|
|
|
cell_metrics: &CellMetrics {
|
|
|
|
line_height,
|
|
|
|
char_width,
|
|
|
|
underline_position,
|
|
|
|
underline_thickness,
|
|
|
|
..
|
|
|
|
},
|
|
|
|
..
|
|
|
|
} = cell_view;
|
|
|
|
|
|
|
|
if cell.attrs.undercurl {
|
|
|
|
let sp = color_model.actual_cell_sp(cell);
|
|
|
|
ctx.set_source_rgba(sp.0, sp.1, sp.2, 0.7);
|
|
|
|
|
|
|
|
let max_undercurl_height = (line_height - underline_position) * 2.0;
|
|
|
|
let undercurl_height = (underline_thickness * 4.0).min(max_undercurl_height);
|
|
|
|
let undercurl_y = line_y + underline_position - undercurl_height / 2.0;
|
|
|
|
|
2018-01-05 19:13:17 +00:00
|
|
|
pangocairo::functions::show_error_underline(ctx, line_x, undercurl_y, char_width, undercurl_height);
|
2017-09-13 09:39:02 +00:00
|
|
|
} else if cell.attrs.underline {
|
|
|
|
let fg = color_model.actual_cell_fg(cell);
|
|
|
|
ctx.set_source_rgb(fg.0, fg.1, fg.2);
|
|
|
|
ctx.set_line_width(underline_thickness);
|
|
|
|
ctx.move_to(line_x, line_y + underline_position);
|
|
|
|
ctx.line_to(line_x + char_width, line_y + underline_position);
|
|
|
|
ctx.stroke();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-12 13:40:13 +00:00
|
|
|
fn draw_cell_bg(
|
2017-09-13 09:39:02 +00:00
|
|
|
cell_view: &RowView,
|
2017-09-12 20:27:45 +00:00
|
|
|
color_model: &color::ColorModel,
|
|
|
|
cell: &ui_model::Cell,
|
|
|
|
col: usize,
|
|
|
|
line_x: f64,
|
|
|
|
) {
|
2017-09-13 09:39:02 +00:00
|
|
|
let &RowView {
|
|
|
|
ctx,
|
|
|
|
line,
|
|
|
|
line_y,
|
|
|
|
cell_metrics: &CellMetrics {
|
|
|
|
char_width,
|
|
|
|
line_height,
|
|
|
|
..
|
|
|
|
},
|
2017-09-12 20:27:45 +00:00
|
|
|
..
|
2017-09-13 09:39:02 +00:00
|
|
|
} = cell_view;
|
2017-09-12 20:27:45 +00:00
|
|
|
|
2017-12-12 13:40:13 +00:00
|
|
|
let bg = color_model.cell_bg(cell);
|
2017-09-12 20:27:45 +00:00
|
|
|
|
2017-12-12 13:40:13 +00:00
|
|
|
if let Some(bg) = bg {
|
|
|
|
if !line.is_binded_to_item(col) {
|
|
|
|
if bg != &color_model.bg_color {
|
|
|
|
ctx.set_source_rgb(bg.0, bg.1, bg.2);
|
|
|
|
ctx.rectangle(line_x, line_y, char_width, line_height);
|
|
|
|
ctx.fill();
|
|
|
|
}
|
|
|
|
} else {
|
2017-09-12 20:27:45 +00:00
|
|
|
ctx.set_source_rgb(bg.0, bg.1, bg.2);
|
|
|
|
ctx.rectangle(
|
|
|
|
line_x,
|
|
|
|
line_y,
|
|
|
|
char_width * line.item_len_from_idx(col) as f64,
|
|
|
|
line_height,
|
|
|
|
);
|
|
|
|
ctx.fill();
|
|
|
|
}
|
2017-12-12 13:40:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
fn draw_cell(
|
|
|
|
cell_view: &RowView,
|
|
|
|
color_model: &color::ColorModel,
|
|
|
|
cell: &ui_model::Cell,
|
|
|
|
col: usize,
|
|
|
|
line_x: f64,
|
|
|
|
) {
|
|
|
|
|
|
|
|
let &RowView {
|
|
|
|
ctx,
|
|
|
|
line,
|
|
|
|
line_y,
|
|
|
|
cell_metrics: &CellMetrics {
|
|
|
|
ascent,
|
|
|
|
..
|
|
|
|
},
|
|
|
|
..
|
|
|
|
} = cell_view;
|
2017-09-12 20:27:45 +00:00
|
|
|
|
2017-12-12 13:40:13 +00:00
|
|
|
if let Some(item) = line.item_line[col].as_ref() {
|
2017-09-12 20:27:45 +00:00
|
|
|
if let Some(ref glyphs) = item.glyphs {
|
2017-12-12 13:40:13 +00:00
|
|
|
let fg = color_model.actual_cell_fg(cell);
|
|
|
|
|
2017-09-12 20:27:45 +00:00
|
|
|
ctx.move_to(line_x, line_y + ascent);
|
|
|
|
ctx.set_source_rgb(fg.0, fg.1, fg.2);
|
2017-12-31 09:47:50 +00:00
|
|
|
|
|
|
|
show_glyph_string(ctx, item.font(), glyphs);
|
2017-09-12 20:27:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-04 15:32:12 +00:00
|
|
|
pub fn shape_dirty(
|
|
|
|
ctx: &context::Context,
|
|
|
|
ui_model: &mut ui_model::UiModel,
|
|
|
|
color_model: &color::ColorModel,
|
|
|
|
) {
|
2017-08-25 15:32:30 +00:00
|
|
|
for line in ui_model.model_mut() {
|
|
|
|
if line.dirty_line {
|
2017-09-04 15:32:12 +00:00
|
|
|
let styled_line = ui_model::StyledLine::from(line, color_model);
|
2017-08-25 15:32:30 +00:00
|
|
|
let items = ctx.itemize(&styled_line);
|
|
|
|
line.merge(&styled_line, &items);
|
|
|
|
|
2017-09-14 15:34:08 +00:00
|
|
|
for (col, cell) in line.line.iter_mut().enumerate() {
|
|
|
|
if cell.dirty {
|
|
|
|
if let Some(item) = line.item_line[col].as_mut() {
|
2017-08-27 19:29:43 +00:00
|
|
|
let mut glyphs = pango::GlyphString::new();
|
|
|
|
{
|
|
|
|
let analysis = item.analysis();
|
|
|
|
let (offset, length, _) = item.item.offset();
|
|
|
|
pango_shape(
|
|
|
|
&styled_line.line_str,
|
|
|
|
offset,
|
|
|
|
length,
|
|
|
|
&analysis,
|
|
|
|
&mut glyphs,
|
|
|
|
);
|
|
|
|
}
|
2017-08-25 15:32:30 +00:00
|
|
|
|
2017-09-08 15:26:16 +00:00
|
|
|
item.set_glyphs(ctx, glyphs);
|
2017-08-27 19:29:43 +00:00
|
|
|
}
|
2017-08-25 15:32:30 +00:00
|
|
|
}
|
|
|
|
|
2017-09-14 15:34:08 +00:00
|
|
|
cell.dirty = false;
|
2017-08-25 15:32:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
line.dirty_line = false;
|
|
|
|
}
|
2017-08-23 09:45:56 +00:00
|
|
|
}
|
2017-08-18 15:35:33 +00:00
|
|
|
}
|