Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
A
AloqaIM-Android
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Administrator
AloqaIM-Android
Commits
a346b70e
Commit
a346b70e
authored
May 07, 2019
by
Hussein El Feky
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Added RTL ViewPager
parent
11f67b6e
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
299 additions
and
16 deletions
+299
-16
EmojiKeyboardPopup.kt
...main/java/chat/rocket/android/emoji/EmojiKeyboardPopup.kt
+5
-7
EmojiPickerPopup.kt
...c/main/java/chat/rocket/android/emoji/EmojiPickerPopup.kt
+3
-7
EmojiViewPager.kt
...src/main/java/chat/rocket/android/emoji/EmojiViewPager.kt
+289
-0
emoji_picker.xml
emoji/src/main/res/layout-land/emoji_picker.xml
+1
-1
emoji_picker.xml
emoji/src/main/res/layout/emoji_picker.xml
+1
-1
No files found.
emoji/src/main/java/chat/rocket/android/emoji/EmojiKeyboardPopup.kt
View file @
a346b70e
...
@@ -16,7 +16,6 @@ import androidx.appcompat.app.AppCompatActivity
...
@@ -16,7 +16,6 @@ import androidx.appcompat.app.AppCompatActivity
import
androidx.core.content.ContextCompat
import
androidx.core.content.ContextCompat
import
androidx.core.content.edit
import
androidx.core.content.edit
import
androidx.core.graphics.drawable.DrawableCompat
import
androidx.core.graphics.drawable.DrawableCompat
import
androidx.viewpager.widget.ViewPager
import
chat.rocket.android.emoji.internal.EmojiCategory
import
chat.rocket.android.emoji.internal.EmojiCategory
import
chat.rocket.android.emoji.internal.EmojiPagerAdapter
import
chat.rocket.android.emoji.internal.EmojiPagerAdapter
import
chat.rocket.android.emoji.internal.PREF_EMOJI_SKIN_TONE
import
chat.rocket.android.emoji.internal.PREF_EMOJI_SKIN_TONE
...
@@ -27,7 +26,7 @@ import kotlinx.coroutines.GlobalScope
...
@@ -27,7 +26,7 @@ import kotlinx.coroutines.GlobalScope
import
kotlinx.coroutines.launch
import
kotlinx.coroutines.launch
class
EmojiKeyboardPopup
(
context
:
Context
,
view
:
View
)
:
OverKeyboardPopupWindow
(
context
,
view
)
{
class
EmojiKeyboardPopup
(
context
:
Context
,
view
:
View
)
:
OverKeyboardPopupWindow
(
context
,
view
)
{
private
lateinit
var
viewPager
:
ViewPager
private
lateinit
var
viewPager
:
Emoji
ViewPager
private
lateinit
var
tabLayout
:
TabLayout
private
lateinit
var
tabLayout
:
TabLayout
private
lateinit
var
searchView
:
View
private
lateinit
var
searchView
:
View
private
lateinit
var
backspaceView
:
View
private
lateinit
var
backspaceView
:
View
...
@@ -155,7 +154,7 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
...
@@ -155,7 +154,7 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
val
fragments
=
(
it
as
AppCompatActivity
).
supportFragmentManager
.
fragments
val
fragments
=
(
it
as
AppCompatActivity
).
supportFragmentManager
.
fragments
if
(
fragments
.
size
==
0
||
fragments
[
0
]
!
is
EmojiKeyboardListener
)
{
if
(
fragments
.
size
==
0
||
fragments
[
0
]
!
is
EmojiKeyboardListener
)
{
// Since the app can arrive in an inconsistent state at this point, do not throw
// Since the app can arrive in an inconsistent state at this point, do not throw
//
throw IllegalStateException("activity/fragment should implement Listener interface")
//
throw IllegalStateException("activity/fragment should implement Listener interface")
null
null
}
else
{
}
else
{
fragments
[
0
]
as
EmojiKeyboardListener
fragments
[
0
]
as
EmojiKeyboardListener
...
@@ -177,8 +176,8 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
...
@@ -177,8 +176,8 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
val
tab
=
tabLayout
.
getTabAt
(
category
.
ordinal
)
val
tab
=
tabLayout
.
getTabAt
(
category
.
ordinal
)
val
tabView
=
LayoutInflater
.
from
(
context
).
inflate
(
R
.
layout
.
emoji_picker_tab
,
null
)
val
tabView
=
LayoutInflater
.
from
(
context
).
inflate
(
R
.
layout
.
emoji_picker_tab
,
null
)
tab
?.
customView
=
tabView
tab
?.
customView
=
tabView
val
text
View
=
tabView
.
findViewById
(
R
.
id
.
image_category
)
as
ImageView
val
image
View
=
tabView
.
findViewById
(
R
.
id
.
image_category
)
as
ImageView
text
View
.
setImageResource
(
category
.
resourceIcon
())
image
View
.
setImageResource
(
category
.
resourceIcon
())
}
}
val
currentTab
=
if
(
EmojiRepository
.
getRecents
().
isEmpty
())
{
val
currentTab
=
if
(
EmojiRepository
.
getRecents
().
isEmpty
())
{
...
@@ -186,7 +185,6 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
...
@@ -186,7 +185,6 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
}
else
{
}
else
{
EmojiCategory
.
RECENTS
.
ordinal
EmojiCategory
.
RECENTS
.
ordinal
}
}
viewPager
.
currentItem
=
currentTab
viewPager
.
currentItem
=
currentTab
}
}
}
}
...
@@ -203,7 +201,7 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
...
@@ -203,7 +201,7 @@ class EmojiKeyboardPopup(context: Context, view: View) : OverKeyboardPopupWindow
val
start
=
message
.
getSpanStart
(
span
)
val
start
=
message
.
getSpanStart
(
span
)
val
end
=
message
.
getSpanEnd
(
span
)
val
end
=
message
.
getSpanEnd
(
span
)
// Remove the span
// Remove the span
.
message
.
removeSpan
(
span
)
message
.
removeSpan
(
span
)
// Remove the remaining emoticon text.
// Remove the remaining emoticon text.
...
...
emoji/src/main/java/chat/rocket/android/emoji/EmojiPickerPopup.kt
View file @
a346b70e
...
@@ -7,9 +7,6 @@ import android.view.LayoutInflater
...
@@ -7,9 +7,6 @@ import android.view.LayoutInflater
import
android.view.Window
import
android.view.Window
import
android.view.WindowManager
import
android.view.WindowManager
import
android.widget.ImageView
import
android.widget.ImageView
import
androidx.annotation.ColorInt
import
androidx.core.content.ContextCompat
import
androidx.core.content.edit
import
chat.rocket.android.emoji.internal.EmojiCategory
import
chat.rocket.android.emoji.internal.EmojiCategory
import
chat.rocket.android.emoji.internal.EmojiPagerAdapter
import
chat.rocket.android.emoji.internal.EmojiPagerAdapter
import
chat.rocket.android.emoji.internal.PREF_EMOJI_SKIN_TONE
import
chat.rocket.android.emoji.internal.PREF_EMOJI_SKIN_TONE
...
@@ -18,7 +15,6 @@ import kotlinx.coroutines.Dispatchers
...
@@ -18,7 +15,6 @@ import kotlinx.coroutines.Dispatchers
import
kotlinx.coroutines.GlobalScope
import
kotlinx.coroutines.GlobalScope
import
kotlinx.coroutines.launch
import
kotlinx.coroutines.launch
class
EmojiPickerPopup
(
context
:
Context
)
:
Dialog
(
context
)
{
class
EmojiPickerPopup
(
context
:
Context
)
:
Dialog
(
context
)
{
var
listener
:
EmojiKeyboardListener
?
=
null
var
listener
:
EmojiKeyboardListener
?
=
null
...
@@ -60,15 +56,15 @@ class EmojiPickerPopup(context: Context) : Dialog(context) {
...
@@ -60,15 +56,15 @@ class EmojiPickerPopup(context: Context) : Dialog(context) {
changeSkinTone
(
Fitzpatrick
.
valueOf
(
it
))
changeSkinTone
(
Fitzpatrick
.
valueOf
(
it
))
}
}
pager_categories
.
adapter
=
adapter
pager_categories
.
offscreenPageLimit
=
EmojiCategory
.
values
().
size
pager_categories
.
offscreenPageLimit
=
EmojiCategory
.
values
().
size
pager_categories
.
adapter
=
adapter
for
(
category
in
EmojiCategory
.
values
())
{
for
(
category
in
EmojiCategory
.
values
())
{
val
tab
=
tabs
.
getTabAt
(
category
.
ordinal
)
val
tab
=
tabs
.
getTabAt
(
category
.
ordinal
)
val
tabView
=
LayoutInflater
.
from
(
context
).
inflate
(
R
.
layout
.
emoji_picker_tab
,
null
)
val
tabView
=
LayoutInflater
.
from
(
context
).
inflate
(
R
.
layout
.
emoji_picker_tab
,
null
)
tab
?.
customView
=
tabView
tab
?.
customView
=
tabView
val
text
View
=
tabView
.
findViewById
(
R
.
id
.
image_category
)
as
ImageView
val
image
View
=
tabView
.
findViewById
(
R
.
id
.
image_category
)
as
ImageView
text
View
.
setImageResource
(
category
.
resourceIcon
())
image
View
.
setImageResource
(
category
.
resourceIcon
())
}
}
val
currentTab
=
if
(
EmojiRepository
.
getRecents
().
isEmpty
())
{
val
currentTab
=
if
(
EmojiRepository
.
getRecents
().
isEmpty
())
{
...
...
emoji/src/main/java/chat/rocket/android/emoji/EmojiViewPager.kt
0 → 100644
View file @
a346b70e
package
chat.rocket.android.emoji
import
android.content.Context
import
android.database.DataSetObserver
import
android.os.Parcelable
import
android.util.AttributeSet
import
android.view.View
import
android.view.ViewGroup
import
androidx.core.view.ViewCompat
import
androidx.viewpager.widget.PagerAdapter
import
androidx.viewpager.widget.ViewPager
class
EmojiViewPager
:
ViewPager
{
private
var
mLayoutDirection
=
ViewCompat
.
LAYOUT_DIRECTION_LTR
private
val
mPageChangeListeners
=
hashMapOf
<
OnPageChangeListener
,
ReversingOnPageChangeListener
>()
private
val
isRtl
:
Boolean
get
()
=
mLayoutDirection
==
ViewCompat
.
LAYOUT_DIRECTION_RTL
constructor
(
context
:
Context
)
:
super
(
context
)
constructor
(
context
:
Context
,
attrs
:
AttributeSet
)
:
super
(
context
,
attrs
)
override
fun
onRtlPropertiesChanged
(
layoutDirection
:
Int
)
{
super
.
onRtlPropertiesChanged
(
layoutDirection
)
val
viewCompatLayoutDirection
=
if
(
layoutDirection
==
View
.
LAYOUT_DIRECTION_RTL
)
{
ViewCompat
.
LAYOUT_DIRECTION_RTL
}
else
{
ViewCompat
.
LAYOUT_DIRECTION_LTR
}
if
(
viewCompatLayoutDirection
!=
mLayoutDirection
)
{
val
adapter
=
super
.
getAdapter
()
var
position
=
0
if
(
adapter
!=
null
)
{
position
=
currentItem
}
mLayoutDirection
=
viewCompatLayoutDirection
if
(
adapter
!=
null
)
{
adapter
.
notifyDataSetChanged
()
currentItem
=
position
}
}
}
override
fun
setAdapter
(
adapter
:
PagerAdapter
?)
{
val
adapter
=
if
(
adapter
!=
null
)
{
ReversingAdapter
(
adapter
)
}
else
{
adapter
}
super
.
setAdapter
(
adapter
)
currentItem
=
0
}
override
fun
getAdapter
():
PagerAdapter
?
{
return
super
.
getAdapter
()
as
ReversingAdapter
?
}
override
fun
getCurrentItem
():
Int
{
var
item
=
super
.
getCurrentItem
()
val
adapter
=
super
.
getAdapter
()
if
(
adapter
!=
null
&&
isRtl
)
{
item
=
adapter
.
count
-
item
-
1
}
return
item
}
override
fun
setCurrentItem
(
position
:
Int
,
smoothScroll
:
Boolean
)
{
val
adapter
=
super
.
getAdapter
()
val
position
=
if
(
adapter
!=
null
&&
isRtl
)
{
adapter
.
count
-
position
-
1
}
else
{
position
}
super
.
setCurrentItem
(
position
,
smoothScroll
)
}
override
fun
setCurrentItem
(
position
:
Int
)
{
val
adapter
=
super
.
getAdapter
()
val
position
=
if
(
adapter
!=
null
&&
isRtl
)
{
adapter
.
count
-
position
-
1
}
else
{
position
}
super
.
setCurrentItem
(
position
)
}
override
fun
setOnPageChangeListener
(
listener
:
OnPageChangeListener
)
{
super
.
setOnPageChangeListener
(
ReversingOnPageChangeListener
(
listener
))
}
override
fun
addOnPageChangeListener
(
listener
:
OnPageChangeListener
)
{
val
reversingListener
=
ReversingOnPageChangeListener
(
listener
)
mPageChangeListeners
.
put
(
listener
,
reversingListener
)
super
.
addOnPageChangeListener
(
reversingListener
)
}
override
fun
removeOnPageChangeListener
(
listener
:
OnPageChangeListener
)
{
val
reverseListener
=
mPageChangeListeners
.
remove
(
listener
)
if
(
reverseListener
!=
null
)
{
super
.
removeOnPageChangeListener
(
reverseListener
)
}
}
override
fun
clearOnPageChangeListeners
()
{
super
.
clearOnPageChangeListeners
()
mPageChangeListeners
.
clear
()
}
override
fun
onMeasure
(
widthMeasureSpec
:
Int
,
heightMeasureSpec
:
Int
)
{
val
heightMeasureSpec
=
if
(
MeasureSpec
.
getMode
(
heightMeasureSpec
)
==
MeasureSpec
.
UNSPECIFIED
)
{
var
height
=
0
for
(
i
in
0
until
childCount
)
{
val
child
=
getChildAt
(
i
)
child
.
measure
(
widthMeasureSpec
,
MeasureSpec
.
makeMeasureSpec
(
0
,
MeasureSpec
.
UNSPECIFIED
))
val
h
=
child
.
measuredHeight
if
(
h
>
height
)
{
height
=
h
}
}
MeasureSpec
.
makeMeasureSpec
(
height
,
MeasureSpec
.
EXACTLY
)
}
else
{
heightMeasureSpec
}
super
.
onMeasure
(
widthMeasureSpec
,
heightMeasureSpec
)
}
private
inner
class
ReversingOnPageChangeListener
(
private
val
mListener
:
OnPageChangeListener
)
:
OnPageChangeListener
{
override
fun
onPageScrolled
(
position
:
Int
,
positionOffset
:
Float
,
positionOffsetPixels
:
Int
)
{
var
position
=
position
var
positionOffset
=
positionOffset
var
positionOffsetPixels
=
positionOffsetPixels
val
width
=
width
val
adapter
=
super
@EmojiViewPager
.
getAdapter
()
if
(
adapter
!=
null
&&
isRtl
)
{
val
count
=
adapter
.
count
var
remainingWidth
=
(
width
*
(
1
-
adapter
.
getPageWidth
(
position
))).
toInt
()
+
positionOffsetPixels
while
(
position
<
count
&&
remainingWidth
>
0
)
{
position
+=
1
remainingWidth
-=
(
width
*
adapter
.
getPageWidth
(
position
)).
toInt
()
}
position
=
count
-
position
-
1
positionOffsetPixels
=
-
remainingWidth
positionOffset
=
positionOffsetPixels
/
(
width
*
adapter
.
getPageWidth
(
position
))
}
mListener
.
onPageScrolled
(
position
,
positionOffset
,
positionOffsetPixels
)
}
override
fun
onPageSelected
(
position
:
Int
)
{
val
adapter
=
super
@EmojiViewPager
.
getAdapter
()
val
position
=
if
(
adapter
!=
null
&&
isRtl
)
{
adapter
.
count
-
position
-
1
}
else
{
position
}
mListener
.
onPageSelected
(
position
)
}
override
fun
onPageScrollStateChanged
(
state
:
Int
)
{
mListener
.
onPageScrollStateChanged
(
state
)
}
}
private
inner
class
ReversingAdapter
(
private
val
adapter
:
PagerAdapter
)
:
PagerAdapter
()
{
override
fun
isViewFromObject
(
view
:
View
,
obj
:
Any
):
Boolean
{
return
adapter
.
isViewFromObject
(
view
,
obj
)
}
override
fun
getCount
():
Int
{
return
adapter
.
count
}
override
fun
getItemPosition
(
obj
:
Any
):
Int
{
var
position
=
adapter
.
getItemPosition
(
obj
)
if
(
isRtl
)
{
if
(
position
==
POSITION_UNCHANGED
||
position
==
POSITION_NONE
)
{
position
=
POSITION_NONE
}
else
{
position
=
getCount
()
-
position
-
1
}
}
return
position
}
override
fun
getPageTitle
(
position
:
Int
):
CharSequence
?
{
return
adapter
.
getPageTitle
(
position
)
}
override
fun
getPageWidth
(
position
:
Int
):
Float
{
return
adapter
.
getPageWidth
(
position
)
}
override
fun
instantiateItem
(
container
:
ViewGroup
,
position
:
Int
):
Any
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
return
adapter
.
instantiateItem
(
container
,
position
)
}
override
fun
instantiateItem
(
container
:
View
,
position
:
Int
):
Any
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
return
adapter
.
instantiateItem
(
container
,
position
)
}
override
fun
destroyItem
(
container
:
ViewGroup
,
position
:
Int
,
obj
:
Any
)
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
adapter
.
destroyItem
(
container
,
position
,
obj
)
}
override
fun
destroyItem
(
container
:
View
,
position
:
Int
,
obj
:
Any
)
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
adapter
.
destroyItem
(
container
,
position
,
obj
)
}
override
fun
setPrimaryItem
(
container
:
ViewGroup
,
position
:
Int
,
obj
:
Any
)
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
adapter
.
setPrimaryItem
(
container
,
position
,
obj
)
}
override
fun
setPrimaryItem
(
container
:
View
,
position
:
Int
,
obj
:
Any
)
{
val
position
=
if
(
isRtl
)
{
count
-
position
-
1
}
else
{
position
}
adapter
.
setPrimaryItem
(
container
,
position
,
obj
)
}
override
fun
startUpdate
(
container
:
ViewGroup
)
{
adapter
.
startUpdate
(
container
)
}
override
fun
startUpdate
(
container
:
View
)
{
adapter
.
startUpdate
(
container
)
}
override
fun
finishUpdate
(
container
:
ViewGroup
)
{
adapter
.
finishUpdate
(
container
)
}
override
fun
finishUpdate
(
container
:
View
)
{
adapter
.
finishUpdate
(
container
)
}
override
fun
saveState
():
Parcelable
?
{
return
adapter
.
saveState
()
}
override
fun
restoreState
(
state
:
Parcelable
?,
loader
:
ClassLoader
?)
{
adapter
.
restoreState
(
state
,
loader
)
}
override
fun
notifyDataSetChanged
()
{
adapter
.
notifyDataSetChanged
()
}
override
fun
registerDataSetObserver
(
observer
:
DataSetObserver
)
{
adapter
.
registerDataSetObserver
(
observer
)
}
override
fun
unregisterDataSetObserver
(
observer
:
DataSetObserver
)
{
adapter
.
unregisterDataSetObserver
(
observer
)
}
}
}
emoji/src/main/res/layout-land/emoji_picker.xml
View file @
a346b70e
...
@@ -15,7 +15,7 @@
...
@@ -15,7 +15,7 @@
app:tabGravity=
"fill"
app:tabGravity=
"fill"
app:tabMode=
"scrollable"
/>
app:tabMode=
"scrollable"
/>
<
androidx.viewpager.widget.
ViewPager
<
chat.rocket.android.emoji.Emoji
ViewPager
android:id=
"@+id/pager_categories"
android:id=
"@+id/pager_categories"
android:layout_width=
"match_parent"
android:layout_width=
"match_parent"
android:layout_height=
"match_parent"
android:layout_height=
"match_parent"
...
...
emoji/src/main/res/layout/emoji_picker.xml
View file @
a346b70e
...
@@ -16,7 +16,7 @@
...
@@ -16,7 +16,7 @@
app:tabMaxWidth=
"48dp"
app:tabMaxWidth=
"48dp"
app:tabMode=
"scrollable"
/>
app:tabMode=
"scrollable"
/>
<
androidx.viewpager.widget.
ViewPager
<
chat.rocket.android.emoji.Emoji
ViewPager
android:id=
"@+id/pager_categories"
android:id=
"@+id/pager_categories"
android:layout_width=
"match_parent"
android:layout_width=
"match_parent"
android:layout_height=
"match_parent"
android:layout_height=
"match_parent"
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment