From 691258a56c8a386dc500116e9a9f107ba3b3b488 Mon Sep 17 00:00:00 2001 From: Kamil Jarosz Date: Fri, 29 Nov 2024 16:26:00 +0100 Subject: [PATCH 1/4] avm2: Make TextField.getCharIndexAtPoint accurate This patch fixes getCharIndexAtPoint() so that it's accurate for all inputs. --- .../src/avm2/globals/flash/text/text_field.rs | 20 ++------- core/src/display_object/edit_text.rs | 43 +++++++++++++++++++ 2 files changed, 47 insertions(+), 16 deletions(-) diff --git a/core/src/avm2/globals/flash/text/text_field.rs b/core/src/avm2/globals/flash/text/text_field.rs index 33e392f3220b..f5bd3d0f1701 100644 --- a/core/src/avm2/globals/flash/text/text_field.rs +++ b/core/src/avm2/globals/flash/text/text_field.rs @@ -10,7 +10,7 @@ use crate::avm2::{ArrayObject, ArrayStorage, Error}; use crate::display_object::{AutoSizeMode, EditText, TDisplayObject, TextSelection}; use crate::html::TextFormat; use crate::string::AvmString; -use crate::{avm2_stub_getter, avm2_stub_method, avm2_stub_setter}; +use crate::{avm2_stub_getter, avm2_stub_setter}; use swf::{Color, Point}; pub fn text_field_allocator<'gc>( @@ -1654,19 +1654,6 @@ pub fn get_char_index_at_point<'gc>( ) -> Result, Error<'gc>> { let this = this.as_object().unwrap(); - // TODO This currently uses screen_position_to_index, which is inaccurate, because: - // 1. getCharIndexAtPoint should return -1 when clicked outside of a character, - // 2. screen_position_to_index returns caret index, not clicked character index. - // Currently, it is difficult to prove accuracy of this method, as at the time - // of writing this comment, text layout behaves differently compared to Flash. - // However, the current implementation is good enough to make some SWFs work. - avm2_stub_method!( - activation, - "flash.text.TextField", - "getCharIndexAtPoint", - "inaccurate char index detection" - ); - let Some(this) = this .as_display_object() .and_then(|this| this.as_edit_text()) @@ -1674,10 +1661,11 @@ pub fn get_char_index_at_point<'gc>( return Ok(Value::Undefined); }; - let x = args.get_f64(activation, 0)?; + // No idea why FP does this weird 1px translation... + let x = args.get_f64(activation, 0)? + 1.0; let y = args.get_f64(activation, 1)?; - if let Some(index) = this.screen_position_to_index(Point::from_pixels(x, y)) { + if let Some(index) = this.char_index_at_point(Point::from_pixels(x, y)) { Ok(index.into()) } else { Ok(Value::Number(-1f64)) diff --git a/core/src/display_object/edit_text.rs b/core/src/display_object/edit_text.rs index 1104d14d60b6..f063ad881330 100644 --- a/core/src/display_object/edit_text.rs +++ b/core/src/display_object/edit_text.rs @@ -2100,6 +2100,49 @@ impl<'gc> EditText<'gc> { ) } + /// Returns the index of the character that is at the given position. + /// + /// It returns `None` when there's no character at the given position. + /// It takes into account various quirks of Flash Player: + /// 1. It will return the index of the newline when `x` + /// is zero and the line is empty. + /// 2. It assumes (exclusive, inclusive) bounds. + /// 3. Positions with `y` below the last line will behave + /// the same way as at the last line. + pub fn char_index_at_point(self, position: Point) -> Option { + let line_index = self.line_index_at_point(position)?; + + let edit_text = self.0.read(); + let line = &edit_text.layout.lines()[line_index]; + + // KJ: It's a bug in FP, it doesn't take into account horizontal + // scroll, but it does take into account vertical scroll. + // See https://github.com/airsdk/Adobe-Runtime-Support/issues/2315 + // I guess we'll have to take scrollH into account here when + // we start supporting Harman runtimes. + let x = position.x - Self::GUTTER; + + // Yes, this will return the index of the newline when the line is empty. + // Yes, that's how Flash Player does it. + if x == Twips::ZERO { + return Some(line.start()); + } + + // TODO Use binary search here when possible + for ch in line.start()..line.end() { + let bounds = line.char_x_bounds(ch); + let Some((a, b)) = bounds else { + continue; + }; + + if a < x && x <= b { + return Some(ch); + } + } + + None + } + pub fn line_index_of_char(self, index: usize) -> Option { self.0.read().layout.find_line_index_by_position(index) } From 8eee08dacee0a649b2b6b6db42db8799587497d9 Mon Sep 17 00:00:00 2001 From: Kamil Jarosz Date: Fri, 29 Nov 2024 16:27:36 +0100 Subject: [PATCH 2/4] tests: Add avm2/edittext_get_char_index_at_point test This test verifies the behavior of getCharIndexAtPoint(). --- .../edittext_get_char_index_at_point/Test.as | 154 ++++++++++++++++++ .../TestFont.ttf | Bin 0 -> 1600 bytes .../output.expected.png | Bin 0 -> 4290 bytes .../output.txt | 4 + .../edittext_get_char_index_at_point/test.swf | Bin 0 -> 3121 bytes .../test.toml | 8 + 6 files changed, 166 insertions(+) create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/Test.as create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/TestFont.ttf create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.expected.png create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.txt create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/test.swf create mode 100644 tests/tests/swfs/avm2/edittext_get_char_index_at_point/test.toml diff --git a/tests/tests/swfs/avm2/edittext_get_char_index_at_point/Test.as b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/Test.as new file mode 100644 index 000000000000..8bae59a39aeb --- /dev/null +++ b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/Test.as @@ -0,0 +1,154 @@ +package { +import flash.display.Sprite; +import flash.display.Bitmap; +import flash.display.BitmapData; +import flash.text.TextField; +import flash.text.TextFormat; +import flash.geom.Rectangle; +import flash.utils.ByteArray; + +[SWF(width="400", height="400")] +public class Test extends Sprite { + [Embed(source="TestFont.ttf", fontName="TestFont", embedAsCFF="false", unicodeRange="U+0061-U+0064")] + private var TestFont:Class; + + private var colors: Array = [ + 0xFFFF0000, + 0xFF00FF00, + 0xFF0000FF, + 0xFFFFFF00, + 0xFF00FFFF, + 0xFFFF00FF, + 0xFFFFFFFF, + ]; + + private var text1: TextField; + + public function Test() { + var t1: TextField = newTextField(); + t1.htmlText = "a\naa\naaa\naa"; + renderMap(t1); + + var t2: TextField = newTextField(t1); + t2.height = 50; + t2.htmlText = ""; + renderMap(t2); + + var t3: TextField = newTextField(t2); + t3.height = 50; + t3.type = "input"; + t3.htmlText = ""; + renderMap(t3); + + var t4: TextField = newTextField(t3); + t4.type = "input"; + t4.htmlText = "aaaaaa"; + renderMap(t4); + + var t5: TextField = newTextField(null, t4); + t5.height = 50; + t5.htmlText = "\n"; + renderMap(t5); + + var t6: TextField = newTextField(t5, t4); + t6.height = 50; + t6.type = "input"; + t6.htmlText = "\n"; + renderMap(t6); + + var t7: TextField = newTextField(t6, t4); + t7.height = 50; + t7.htmlText = "a\naaa\na"; + renderMap(t7); + + var t8: TextField = newTextField(t7, t4); + t8.htmlText = "\n\na\na"; + renderMap(t8); + + var t9: TextField = newTextField(null, t8); + t9.height = 60; + t9.htmlText = "\n\n\n\n\n\n\n\n"; + t9.scrollV = 3; + renderMap(t9); + + var t10: TextField = newTextField(t9, t8); + t10.htmlText = "a\na\na\na\n"; + renderMap(t10); + + var t11: TextField = newTextField(t10, t8); + t11.height = 30; + t11.htmlText = "aaaaaaaaaaaaaaaaaaaaaaaaaa"; + t11.scrollH = 50; + trace("scrollh = " + t11.scrollH); + trace("maxscrollh = " + t11.maxScrollH); + renderMap(t11); + + var t12: TextField = newTextField(t11, t8); + t12.htmlText = "aaaaaaaaaaaaaaaaaaaaaaaaaa\n\naaaaaaaaaaaaaaaaaaaaaaaaaa"; + t12.scrollH = 50; + trace("scrollh = " + t12.scrollH); + trace("maxscrollh = " + t12.maxScrollH); + renderMap(t12); + + var t13: TextField = newTextField(t8, t4); + t13.height = 50; + t13.htmlText = "

a a a aaaa

"; + t13.wordWrap = true; + renderMap(t13); + } + + private function newTextField(lastY: TextField = null, lastX: TextField = null):TextField { + var tf = new TextFormat(); + tf.font = "TestFont"; + tf.size = 20; + tf.leading = 2; + + var field: TextField = new TextField(); + field.x = 10; + field.y = 10; + if (lastX != null) { + field.x = lastX.x + lastX.width + 12; + } + if (lastY != null) { + field.y = lastY.y + lastY.height + 12; + } + field.embedFonts = true; + field.border = true; + field.defaultTextFormat = tf; + field.width = 100; + field.height = 100; + return field; + } + + private function renderMap(field: TextField, resolution: Number = 2.0):void { + addChild(field); + var w = resolution * (field.width + 10); + var h = resolution * (field.height + 10); + var data:BitmapData = new BitmapData(w, h); + var pixels: ByteArray = new ByteArray(); + + for (var y = 0; y < h; ++y) { + for (var x = 0; x < w; ++x) { + var ix = field.getCharIndexAtPoint(x / resolution - 5, y / resolution - 5); + + var color; + if (ix == -1) { + color = 0xFF000000; + } else { + color = colors[ix % colors.length]; + } + pixels.writeUnsignedInt(color); + } + } + + pixels.position = 0; + data.setPixels(new Rectangle(0, 0, w, h), pixels); + var bitmap:Bitmap = new Bitmap(data); + bitmap.scaleX = 1 / resolution; + bitmap.scaleY = 1 / resolution; + bitmap.x = field.x - 5; + bitmap.y = field.y - 5; + addChild(bitmap); + } +} +} diff --git a/tests/tests/swfs/avm2/edittext_get_char_index_at_point/TestFont.ttf b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/TestFont.ttf new file mode 100644 index 0000000000000000000000000000000000000000..761128c1eb3c323d499229f8c34f7523983545de GIT binary patch literal 1600 zcmcgsOK4L;6g~4kX-ma^u!vA)9v1tdO<$5?t7K8s#!nGNLlIHzYnmp3{CG*Fbs;WA zL|loe3lYJc*n%MFJ}zB|t1e1cF6>g=RN|SJX>Gd@x^W)EoO|xQGw03R83+LTaStY@ z6N%)-H_u=80-+wV-Dl5CB+!Kew40TE>FRKF>h#FD9KThYr+ zA+Zr(<`%9;V;xRMZ>-~}BU--KYR0qQbB=LMZ*x#}JS&DcrrcOnoxjUnRuxG^U&QGO ze5wF=Yu|1iki;pC1#56#V}lAGX>6hgs~THq76&!9(I&<<4w0Wx6=JDc9A|G}fp(-c zHn{FxjZK`x6OApj3tM9wM}*WkME){Pql7XR;Uk9(vZz9$k8}2OjKXCf;r*1DWStin zlPA7F>@%;TZt&?+dC|{hvQ^oa>X%VBIx;9Hy+ST8FL-{b!iXf&46P!;lp;qY)0JwX zR3y4WSBATi4`#%bbY?#9`Pd3oDLy08ftoP&NStKfJ!ON9A%Y={vTrJfS|(BQGiezO zyYi&mT$(Tz85$i@djF|#ifj9np9{oGDsZ`qQldUgrTt2-RFtk83A-+d-O9H8EKg0} v)cSiW)YZlDZ{sV(6E|4R6AJO}8@dAcs`9;8mr~6kbk}pNS$!mSzL$RhFdoS* literal 0 HcmV?d00001 diff --git a/tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.expected.png b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.expected.png new file mode 100644 index 0000000000000000000000000000000000000000..2fa5264b5930b2738c96a191f384e797ab3ad4bd GIT binary patch literal 4290 zcmd5-*--n_eF#`J8f|S062&54S53qu7Li-9jHiOxNNrUfE05Zf`m8mhVrQ-G%C%;r#xuc4Q|!N6kyqopz=EaqOtcR5mss0cQTB z%>K687#E{1@Rjna19j6!S{25l&Nu*Ygxz8w5K)=v2oqo&ON0hG$=HRw`&-j9^pg+v z-JNoY{S?$<)KZK@kCqy)O0R(v)IL~1L{d@u6BE!SfRW|F8r6Y2o$Q*vB<}C{vq-3Z zRi@C}_lM7%=Mc?VzP%`mdKc{)56i)?;043|FWpIrdc44X z;S1V68?gEHc{>znOR2=*T`omh3D-vuHf!p=GrJ*LwF>~KF~UHCsU&GasY8b9Nl}W3 zDy8Mrg6sLJ^SlKyS&K}Ut*3H>Ywm|4qk z!NJzbYpO!I_6tNK(u%cwDo>tr1~Nf$QnwEzkP=ArpS0JS9ptR$jl z>ZnL*Y_QTayXCHD8)zLo7w-U2RS;0hsOtw*?=R}fBLn898#fj+-ZZFaOu#rk{fQi) z&0U+&LM(AAN(3PNYJDM~E-v)IQOzoQ-!NW*MaPz2F~ zKuxPR$8V_PX%2u6gSwu!ujAd_gkbyz$Eqh0sx9NpIFq?%cdfoqQtAP*!fI>r*|L=c&GCgVF*ECK}HRfSv`?I+&n%!gH1o{ap?H4_N z*1D!?+LXy2+sK*slJVSG>`}>nc!2Q(WR1Ne?XK5F^}qWK%+w>x%h}zf^vUPGpL(|X UN=4)j>aGXjdm=)rUWqIC7q(h-CIA2c literal 0 HcmV?d00001 diff --git a/tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.txt b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.txt new file mode 100644 index 000000000000..ec394db79555 --- /dev/null +++ b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/output.txt @@ -0,0 +1,4 @@ +scrollh = 50 +maxscrollh = 320 +scrollh = 50 +maxscrollh = 320 diff --git a/tests/tests/swfs/avm2/edittext_get_char_index_at_point/test.swf b/tests/tests/swfs/avm2/edittext_get_char_index_at_point/test.swf new file mode 100644 index 0000000000000000000000000000000000000000..391b45738878f3f3461b9a6dfddaeff46498ceac GIT binary patch literal 3121 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 Date: Thu, 12 Dec 2024 22:39:29 +0100 Subject: [PATCH 3/4] text: Add doc to line_index_at_point --- core/src/display_object/edit_text.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/core/src/display_object/edit_text.rs b/core/src/display_object/edit_text.rs index f063ad881330..970850423ad2 100644 --- a/core/src/display_object/edit_text.rs +++ b/core/src/display_object/edit_text.rs @@ -2081,6 +2081,11 @@ impl<'gc> EditText<'gc> { Some(first_box.start()) } + /// Returns the index of the line that is at the given position. + /// + /// It returns `None` when there's no line at the given position, + /// with the exception that positions below the last line will + /// return the index of the last line. pub fn line_index_at_point(self, position: Point) -> Option { let edit_text = self.0.read(); From 1e6626274d74ba70e4d00beb8f4354ed6b47c712 Mon Sep 17 00:00:00 2001 From: Kamil Jarosz Date: Thu, 12 Dec 2024 22:42:47 +0100 Subject: [PATCH 4/4] text: Add doc to screen_position_to_index --- core/src/display_object/edit_text.rs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/core/src/display_object/edit_text.rs b/core/src/display_object/edit_text.rs index 970850423ad2..b0785b70830e 100644 --- a/core/src/display_object/edit_text.rs +++ b/core/src/display_object/edit_text.rs @@ -1432,6 +1432,11 @@ impl<'gc> EditText<'gc> { self.0.write(context.gc()).max_chars = value; } + /// Map the position on the screen to caret index. + /// + /// This method is used exclusively for placing a caret inside text. + /// It implements the Flash Player's behavior of placing a caret. + /// Characters are divided in half, the last line is extended, etc. pub fn screen_position_to_index(self, position: Point) -> Option { let text = self.0.read(); let position = self.global_to_local(position)?;