2016-11-20 77 views
5

我有一个TextView其中每个单词是ClickableSpan。单击时,单词变成加粗并且单词的字典定义显示在另一个TextView中。该应用程序正常工作,直到我使TextView中的文本可选。当文本可选时,定义会在点击中显示,但双击时该字只有加粗。双击或长按选择文本(但长按不会使文字变粗)。处理ClickableSpan与可选文本需要双击

我的猜测是这个问题与动作处理过程中的绘制状态更新有关,但我无法找到修复。我试图设置TextViewfocusable="false"但没有任何改变。相关代码如下。

curSpan = new WordSpan(index) { 
    @Override 
    public void onClick(View view) { 
     handleWordClick(index,this); // handles code to display definition 
     setMarking(true); 
     view.invalidate(); 
     tvText.invalidate(); 
    } 
}; 

spannableStringBuilder.setSpan(curSpan, totalLength, totalLength + strWord, Spanned.SPAN_COMPOSING); 

而且WordSpan定义:

class WordSpan extends ClickableSpan 
    { 
     int id; 
     private boolean marking = false; 

     public WordSpan(int id) { 
      this.id = id; 
     } 

     @Override 
     public void updateDrawState(TextPaint ds) { 
      ds.setColor(Color.BLACK); 
      ds.setUnderlineText(false); 
      if (marking) { 
       ds.setTypeface(Typeface.create(myFont,Typeface.BOLD)); 
      } 
     } 

     @Override 
     public void onClick(View v) {} 

     public void setMarking(boolean m) { 
      marking = m; 
     } 
    } 

设置运动方法TextView的:

private MovementMethod createMovementMethod (Context context) { 
     final GestureDetector detector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() { 
       @Override 
       public boolean onSingleTapUp (MotionEvent e) { 
        return true; 
       } 

       @Override 
       public boolean onSingleTapConfirmed (MotionEvent e) { 
        return true; 
       } 
      }); 

     return new ScrollingMovementMethod() { 

      @Override 
      public boolean canSelectArbitrarily() { 
       return true; 
      } 

      @Override 
      public void initialize(TextView widget, Spannable text) { 
       Selection.setSelection(text, text.length()); 
      } 

      @Override 
      public void onTakeFocus(TextView view, Spannable text, int dir) { 
       if ((dir & (View.FOCUS_FORWARD | View.FOCUS_DOWN)) != 0) { 
        if (view.getLayout() == null) { 
         // This shouldn't be null, but do something sensible if it is. 
         Selection.setSelection(text, text.length()); 
        } 
       } else { 
        Selection.setSelection(text, text.length()); 
       } 
      } 

      @Override 
      public boolean onTouchEvent (TextView widget, Spannable buffer, MotionEvent event) { 
       // check if event is a single tab 
       boolean isClickEvent = detector.onTouchEvent(event); 

       // detect span that was clicked 
       if (isClickEvent) { 
        int x = (int) event.getX(); 
        int y = (int) event.getY(); 

        x -= widget.getTotalPaddingLeft(); 
        y -= widget.getTotalPaddingTop(); 

        x += widget.getScrollX(); 
        y += widget.getScrollY(); 

        Layout layout = widget.getLayout(); 
        int line = layout.getLineForVertical(y); 
        int off = layout.getOffsetForHorizontal(line, x); 

        WordSpan[] link = buffer.getSpans(off, off, WordSpan.class); 

        if (link.length != 0) { 
         // execute click only for first clickable span 
         // can be a for each loop to execute every one 
         if (event.getAction() == MotionEvent.ACTION_UP) { 
          link[0].onClick(widget); 
         } else if (event.getAction() == MotionEvent.ACTION_DOWN) { 
          Selection.setSelection(buffer, 
                buffer.getSpanStart(link[0]), 
                buffer.getSpanEnd(link[0])); 
         } 
         return true; 
        } 
       } 

       // let scroll movement handle the touch 
       return super.onTouchEvent(widget, buffer, event); 
      } 
     }; 
    } 

编辑: 我刚刚发现了一个新的怪癖,可以解决帮助。如果我双击但单击之间改变单词(点击一个单词&,然后快速选择另一个单词),在第一次敲击时显示该单词的定义,在第二次敲击时,第一个单词显示为粗体,但第二个单词被选中(突出显示)并且第一个字的定义仍然显示。

因此,举例来说,如果我枪王“第一”,然后“第二”,当我点击 “第一”的定义为“第一”将显示,当我触摸 “第二个”一字“第一个“是粗体字,”second“字是突出显示的 ,但定义不变(仍然显示 ”第一个“的定义)。

+0

嘿!我是否正确理解问题。如何处理带有不同可点击文本的Spannable String的TextView双击? – GensaGames

+0

它应该是单击而不是双击。当用户点击单词时,应该加粗单词并显示定义。双击不应该做任何事情。 –

回答

0

用以下替换createMovementMethod。如果你错误面对错误PLZ修复它并编辑这个答案。

private MovementMethod createMovementMethod (final Context context) { 

     return new ScrollingMovementMethod() { 
      public MotionEvent event; 
      public Spannable buffer; 
      public TextView widget; 
      final GestureDetector detector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() { 
       @Override 
       public boolean onSingleTapUp (MotionEvent e) { 

        return true; 
       } 

       @Override 
       public boolean onSingleTapConfirmed (MotionEvent e) { 
        triggerClick(); 
        return true; 
       } 

       @Override 
       public boolean onDoubleTap(MotionEvent e) { 
        triggerClick(); 
        return true; 
       } 

       private boolean triggerClick() { 
        int x = (int) event.getX(); 
        int y = (int) event.getY(); 

        x -= widget.getTotalPaddingLeft(); 
        y -= widget.getTotalPaddingTop(); 

        x += widget.getScrollX(); 
        y += widget.getScrollY(); 

        Layout layout = widget.getLayout(); 
        int line = layout.getLineForVertical(y); 
        int off = layout.getOffsetForHorizontal(line, x); 

        WordSpan[] link = buffer.getSpans(off, off, WordSpan.class); 

        if (link.length != 0) { 
         // execute click only for first clickable span 
         // can be a for each loop to execute every one 
         if (event.getAction() == MotionEvent.ACTION_UP) { 
          link[0].onClick(widget); 
         } else if (event.getAction() == MotionEvent.ACTION_DOWN) { 
          Selection.setSelection(buffer, 
            buffer.getSpanStart(link[0]), 
            buffer.getSpanEnd(link[0])); 
         } 
         return true; 
        } 

        return true; 
       } 
      }); 
      @Override 
      public boolean canSelectArbitrarily() { 
       return true; 
      } 

      @Override 
      public void initialize(TextView widget, Spannable text) { 
       Selection.setSelection(text, text.length()); 
      } 

      @Override 
      public void onTakeFocus(TextView view, Spannable text, int dir) { 
       if ((dir & (View.FOCUS_FORWARD | View.FOCUS_DOWN)) != 0) { 
        if (view.getLayout() == null) { 
         // This shouldn't be null, but do something sensible if it is. 
         Selection.setSelection(text, text.length()); 
        } 
       } else { 
        Selection.setSelection(text, text.length()); 
       } 
      } 

      @Override 
      public boolean onTouchEvent (TextView widget, Spannable buffer, MotionEvent event) { 
       // check if event is a single tab 
       boolean isClickEvent = detector.onTouchEvent(event); 

       //record this for GestureDetector 
       this.widget = widget; 
       this.buffer = buffer; 
       this.event = event; 

       // detect span that was clicked 
       if (isClickEvent) { 
        //ignore click here 
        return true; 
       } 

       // let scroll movement handle the touch 
       return super.onTouchEvent(widget, buffer, event); 
      } 
     }; 
    } 
+0

感谢您的回复,但这不能解决问题,并增加了触摸时选择了错误词汇的其他问题。 –

+0

此外,onDoubleTap()不会由双击触发。它只在双击时才起作用,但是这种方法没有被触发,所以我无法捕捉它,并且像这个答案一样尝试去重定向。 –

+0

是的,我知道,需要类似的东西。它没有测试你需要修复它并编辑答案本身。谢谢 – Qamar