Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vmj-qt
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
Kulya
vmj-qt
Commits
cc1cec7f
Commit
cc1cec7f
authored
Apr 28, 2010
by
Dan Pascu
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Added drag and drop support for the contact list
parent
3d2240fc
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
418 additions
and
20 deletions
+418
-20
contacts.py
blink/contacts.py
+409
-19
blink.ui
resources/blink.ui
+9
-1
No files found.
blink/contacts.py
View file @
cc1cec7f
...
@@ -3,11 +3,12 @@
...
@@ -3,11 +3,12 @@
from
__future__
import
with_statement
from
__future__
import
with_statement
__all__
=
[
'Contact'
,
'ContactGroup'
,
'ContactDelegate'
,
'ContactModel'
,
'ContactSearchModel'
]
__all__
=
[
'Contact'
,
'ContactGroup'
,
'ContactDelegate'
,
'ContactModel'
,
'ContactSearchModel'
,
'ContactListView'
]
from
PyQt4
import
uic
from
PyQt4
import
uic
from
PyQt4.QtCore
import
Qt
,
QAbstractListModel
,
QEvent
,
QModelIndex
,
QPointF
,
QSize
from
PyQt4.QtCore
import
Qt
,
QAbstractListModel
,
QByteArray
,
QDataStream
,
QEvent
,
QIODevice
,
QMimeData
,
QModelIndex
,
QPointF
,
QRectF
,
QSize
,
QStringList
,
QTimer
,
QVariant
from
PyQt4.QtGui
import
QBrush
,
QColor
,
QKeyEvent
,
QLinearGradient
,
QMouseEvent
,
QPainter
,
QPalette
,
QPen
,
QPixmap
,
QPolygonF
,
QStyle
,
QSortFilterProxyModel
,
QStyledItemDelegate
from
PyQt4.QtGui
import
QBrush
,
QColor
,
QKeyEvent
,
QLinearGradient
,
QListView
,
QMouseEvent
,
QPainter
,
QPainterPath
,
QPalette
,
QPen
,
QPixmap
,
QPolygonF
,
QStyle
from
PyQt4.QtGui
import
QSortFilterProxyModel
,
QStyledItemDelegate
from
application.python.util
import
Null
from
application.python.util
import
Null
from
functools
import
partial
from
functools
import
partial
...
@@ -25,6 +26,7 @@ class ContactGroup(object):
...
@@ -25,6 +26,7 @@ class ContactGroup(object):
obj
=
object
.
__new__
(
cls
)
obj
=
object
.
__new__
(
cls
)
obj
.
name
=
name
obj
.
name
=
name
obj
.
widget
=
Null
obj
.
widget
=
Null
obj
.
saved_state
=
Null
cls
.
instances
[
name
]
=
obj
cls
.
instances
[
name
]
=
obj
return
obj
return
obj
...
@@ -56,6 +58,23 @@ class ContactGroup(object):
...
@@ -56,6 +58,23 @@ class ContactGroup(object):
name
=
property
(
_get_name
,
_set_name
)
name
=
property
(
_get_name
,
_set_name
)
del
_get_name
,
_set_name
del
_get_name
,
_set_name
def
collapse
(
self
):
if
not
self
.
widget
.
collapse_button
.
isChecked
():
self
.
widget
.
collapse_button
.
click
()
def
expand
(
self
):
if
self
.
widget
.
collapse_button
.
isChecked
():
self
.
widget
.
collapse_button
.
click
()
def
save_state
(
self
):
"""Saves the current state of the group (collapsed or not)"""
self
.
saved_state
=
self
.
widget
.
collapse_button
.
isChecked
()
def
restore_state
(
self
):
"""Restores the last saved state of the group (collapsed or not)"""
if
self
.
saved_state
!=
self
.
widget
.
collapse_button
.
isChecked
():
self
.
widget
.
collapse_button
.
click
()
class
ContactIconDescriptor
(
object
):
class
ContactIconDescriptor
(
object
):
def
__init__
(
self
,
filename
):
def
__init__
(
self
,
filename
):
...
@@ -121,6 +140,7 @@ class ContactGroupWidget(base_class, ui_class):
...
@@ -121,6 +140,7 @@ class ContactGroupWidget(base_class, ui_class):
self
.
setupUi
(
self
)
self
.
setupUi
(
self
)
self
.
name
=
name
self
.
name
=
name
self
.
selected
=
False
self
.
selected
=
False
self
.
drop_indicator
=
None
self
.
setFocusProxy
(
parent
)
self
.
setFocusProxy
(
parent
)
self
.
label_widget
.
setFocusProxy
(
self
)
self
.
label_widget
.
setFocusProxy
(
self
)
self
.
name_view
.
setCurrentWidget
(
self
.
label_widget
)
self
.
name_view
.
setCurrentWidget
(
self
.
label_widget
)
...
@@ -154,6 +174,18 @@ class ContactGroupWidget(base_class, ui_class):
...
@@ -154,6 +174,18 @@ class ContactGroupWidget(base_class, ui_class):
selected
=
property
(
_get_selected
,
_set_selected
)
selected
=
property
(
_get_selected
,
_set_selected
)
del
_get_selected
,
_set_selected
del
_get_selected
,
_set_selected
def
_get_drop_indicator
(
self
):
return
self
.
__dict__
[
'drop_indicator'
]
def
_set_drop_indicator
(
self
,
value
):
if
self
.
__dict__
.
get
(
'drop_indicator'
,
Null
)
==
value
:
return
self
.
__dict__
[
'drop_indicator'
]
=
value
self
.
update
()
drop_indicator
=
property
(
_get_drop_indicator
,
_set_drop_indicator
)
del
_get_drop_indicator
,
_set_drop_indicator
def
_start_editing
(
self
):
def
_start_editing
(
self
):
#self.name_editor.setText(self.name_label.text())
#self.name_editor.setText(self.name_label.text())
self
.
name_editor
.
selectAll
()
self
.
name_editor
.
selectAll
()
...
@@ -189,16 +221,37 @@ class ContactGroupWidget(base_class, ui_class):
...
@@ -189,16 +221,37 @@ class ContactGroupWidget(base_class, ui_class):
painter
.
drawLine
(
rect
.
bottomLeft
(),
rect
.
bottomRight
())
painter
.
drawLine
(
rect
.
bottomLeft
(),
rect
.
bottomRight
())
#painter.drawLine(option.rect.topRight(), option.rect.bottomRight())
#painter.drawLine(option.rect.topRight(), option.rect.bottomRight())
painter
.
setRenderHint
(
QPainter
.
Antialiasing
,
True
)
painter
.
setPen
(
QPen
(
QBrush
(
QColor
(
'#dc3169'
)),
2.0
))
if
self
.
drop_indicator
is
ContactListView
.
AboveItem
:
line_rect
=
QRectF
(
rect
.
adjusted
(
18
,
0
,
0
,
5
-
rect
.
height
()))
arc_rect
=
line_rect
.
adjusted
(
-
5
,
-
3
,
-
line_rect
.
width
(),
-
3
)
path
=
QPainterPath
(
line_rect
.
topRight
())
path
.
lineTo
(
line_rect
.
topLeft
())
path
.
arcTo
(
arc_rect
,
0
,
-
180
)
painter
.
drawPath
(
path
)
elif
self
.
drop_indicator
is
ContactListView
.
BelowItem
:
line_rect
=
QRectF
(
rect
.
adjusted
(
18
,
rect
.
height
()
-
5
,
0
,
0
))
arc_rect
=
line_rect
.
adjusted
(
-
5
,
2
,
-
line_rect
.
width
(),
2
)
path
=
QPainterPath
(
line_rect
.
bottomRight
())
path
.
lineTo
(
line_rect
.
bottomLeft
())
path
.
arcTo
(
arc_rect
,
0
,
180
)
painter
.
drawPath
(
path
)
elif
self
.
drop_indicator
is
ContactListView
.
OnItem
:
painter
.
setBrush
(
Qt
.
NoBrush
)
painter
.
drawRoundedRect
(
rect
.
adjusted
(
1
,
1
,
-
1
,
-
1
),
3
,
3
)
if
self
.
collapse_button
.
isChecked
():
if
self
.
collapse_button
.
isChecked
():
arrow
=
QPolygonF
([
QPointF
(
0
,
0
),
QPointF
(
0
,
9
),
QPointF
(
8
,
4.5
)])
arrow
=
QPolygonF
([
QPointF
(
0
,
0
),
QPointF
(
0
,
9
),
QPointF
(
8
,
4.5
)])
arrow
.
translate
(
QPointF
(
5
,
4
))
arrow
.
translate
(
QPointF
(
5
,
4
))
else
:
else
:
arrow
=
QPolygonF
([
QPointF
(
0
,
0
),
QPointF
(
9
,
0
),
QPointF
(
4.5
,
8
)])
arrow
=
QPolygonF
([
QPointF
(
0
,
0
),
QPointF
(
9
,
0
),
QPointF
(
4.5
,
8
)])
arrow
.
translate
(
QPointF
(
5
,
5
))
arrow
.
translate
(
QPointF
(
5
,
5
))
painter
.
setRenderHint
(
QPainter
.
Antialiasing
,
True
)
painter
.
setBrush
(
foreground
)
painter
.
setBrush
(
foreground
)
painter
.
setPen
(
QPen
(
painter
.
brush
(),
0
,
Qt
.
NoPen
))
painter
.
setPen
(
QPen
(
painter
.
brush
(),
0
,
Qt
.
NoPen
))
painter
.
drawPolygon
(
arrow
)
painter
.
drawPolygon
(
arrow
)
painter
.
end
()
painter
.
end
()
def
event
(
self
,
event
):
def
event
(
self
,
event
):
...
@@ -245,8 +298,9 @@ class ContactDelegate(QStyledItemDelegate):
...
@@ -245,8 +298,9 @@ class ContactDelegate(QStyledItemDelegate):
def
createEditor
(
self
,
parent
,
options
,
index
):
def
createEditor
(
self
,
parent
,
options
,
index
):
item
=
index
.
model
()
.
data
(
index
,
Qt
.
DisplayRole
)
item
=
index
.
model
()
.
data
(
index
,
Qt
.
DisplayRole
)
if
type
(
item
)
is
ContactGroup
:
if
type
(
item
)
is
ContactGroup
:
if
item
.
widget
is
Null
:
collapsed
=
item
.
collapsed
# if there was a previous editor widget, preserve its collapsed state
item
.
widget
=
ContactGroupWidget
(
item
.
name
,
parent
)
item
.
widget
=
ContactGroupWidget
(
item
.
name
,
parent
)
item
.
widget
.
collapse_button
.
setChecked
(
collapsed
)
item
.
widget
.
collapse_button
.
toggled
.
connect
(
partial
(
self
.
_update_list_view
,
item
))
item
.
widget
.
collapse_button
.
toggled
.
connect
(
partial
(
self
.
_update_list_view
,
item
))
return
item
.
widget
return
item
.
widget
else
:
else
:
...
@@ -296,11 +350,20 @@ class ContactDelegate(QStyledItemDelegate):
...
@@ -296,11 +350,20 @@ class ContactDelegate(QStyledItemDelegate):
def
paintContactGroup
(
self
,
group
,
painter
,
option
,
index
):
def
paintContactGroup
(
self
,
group
,
painter
,
option
,
index
):
if
group
.
widget
.
size
()
!=
option
.
rect
.
size
():
if
group
.
widget
.
size
()
!=
option
.
rect
.
size
():
# For some reason updateEditorGeometry only receives the peak value
of
# For some reason updateEditorGeometry only receives the peak value
#
the size that the widget ever had, so it will never shrink it. -Dan
#
of the size that the widget ever had, so it will never shrink it.
group
.
widget
.
resize
(
option
.
rect
.
size
())
group
.
widget
.
resize
(
option
.
rect
.
size
())
group
.
widget
.
selected
=
bool
(
option
.
state
&
QStyle
.
State_Selected
)
group
.
widget
.
selected
=
bool
(
option
.
state
&
QStyle
.
State_Selected
)
if
option
.
state
&
QStyle
.
State_Selected
and
not
option
.
state
&
QStyle
.
State_HasFocus
:
# This condition is met when dragging is started on this group.
# We use this to to draw the dragged item image.
painter
.
save
()
pixmap
=
QPixmap
(
option
.
rect
.
size
())
group
.
widget
.
render
(
pixmap
)
painter
.
drawPixmap
(
option
.
rect
,
pixmap
)
painter
.
restore
()
def
paint
(
self
,
painter
,
option
,
index
):
def
paint
(
self
,
painter
,
option
,
index
):
item
=
index
.
model
()
.
data
(
index
,
Qt
.
DisplayRole
)
item
=
index
.
model
()
.
data
(
index
,
Qt
.
DisplayRole
)
handler
=
getattr
(
self
,
'paint
%
s'
%
item
.
__class__
.
__name__
,
Null
)
handler
=
getattr
(
self
,
'paint
%
s'
%
item
.
__class__
.
__name__
,
Null
)
...
@@ -311,6 +374,9 @@ class ContactDelegate(QStyledItemDelegate):
...
@@ -311,6 +374,9 @@ class ContactDelegate(QStyledItemDelegate):
class
ContactModel
(
QAbstractListModel
):
class
ContactModel
(
QAbstractListModel
):
# The MIME types we accept in the order they should be handled
accepted_mime_types
=
[
'application/x-blink-contact-group-list'
,
'application/x-blink-contact-list'
,
'text/uri-list'
]
def
__init__
(
self
,
parent
=
None
):
def
__init__
(
self
,
parent
=
None
):
super
(
ContactModel
,
self
)
.
__init__
(
parent
)
super
(
ContactModel
,
self
)
.
__init__
(
parent
)
self
.
items
=
[]
self
.
items
=
[]
...
@@ -321,9 +387,10 @@ class ContactModel(QAbstractListModel):
...
@@ -321,9 +387,10 @@ class ContactModel(QAbstractListModel):
return
[
item
for
item
in
self
.
items
if
type
(
item
)
is
ContactGroup
]
return
[
item
for
item
in
self
.
items
if
type
(
item
)
is
ContactGroup
]
def
flags
(
self
,
index
):
def
flags
(
self
,
index
):
if
not
index
.
isValid
():
if
index
.
isValid
():
return
QAbstractListModel
.
flags
(
self
,
index
)
return
QAbstractListModel
.
flags
(
self
,
index
)
|
Qt
.
ItemIsDropEnabled
|
Qt
.
ItemIsDragEnabled
|
Qt
.
ItemIsEditable
return
QAbstractListModel
.
flags
(
self
,
index
)
|
Qt
.
ItemIsEditable
else
:
return
QAbstractListModel
.
flags
(
self
,
index
)
|
Qt
.
ItemIsDropEnabled
def
rowCount
(
self
,
parent
=
QModelIndex
()):
def
rowCount
(
self
,
parent
=
QModelIndex
()):
return
len
(
self
.
items
)
return
len
(
self
.
items
)
...
@@ -333,6 +400,136 @@ class ContactModel(QAbstractListModel):
...
@@ -333,6 +400,136 @@ class ContactModel(QAbstractListModel):
return
None
return
None
return
self
.
items
[
index
.
row
()]
return
self
.
items
[
index
.
row
()]
def
supportedDropActions
(
self
):
return
Qt
.
CopyAction
|
Qt
.
MoveAction
def
mimeTypes
(
self
):
return
QStringList
([
'application/x-blink-contact-list'
])
def
mimeData
(
self
,
indexes
):
mime_data
=
QMimeData
()
contact_data
=
QByteArray
()
contact_group_data
=
QByteArray
()
contact_stream
=
QDataStream
(
contact_data
,
QIODevice
.
WriteOnly
)
contact_group_stream
=
QDataStream
(
contact_group_data
,
QIODevice
.
WriteOnly
)
for
index
in
(
index
for
index
in
indexes
if
index
.
isValid
()):
row
=
index
.
row
()
item
=
self
.
items
[
row
]
stream
=
contact_group_stream
if
type
(
item
)
is
ContactGroup
else
contact_stream
stream
.
writeInt32
(
row
)
stream
.
writeQVariant
(
QVariant
(
item
))
if
contact_data
:
mime_data
.
setData
(
'application/x-blink-contact-list'
,
contact_data
)
if
contact_group_data
:
mime_data
.
setData
(
'application/x-blink-contact-group-list'
,
contact_group_data
)
return
mime_data
def
dropMimeData
(
self
,
mime_data
,
action
,
row
,
column
,
parent_index
):
# this is here just to keep the default Qt DnD API happy
# the custom handler is in handleDroppedData
return
False
def
handleDroppedData
(
self
,
mime_data
,
action
,
index
):
if
action
==
Qt
.
IgnoreAction
:
return
True
for
mime_type
in
self
.
accepted_mime_types
:
if
mime_data
.
hasFormat
(
mime_type
):
name
=
mime_type
.
replace
(
'/'
,
' '
)
.
replace
(
'-'
,
' '
)
.
title
()
.
replace
(
' '
,
''
)
handler
=
getattr
(
self
,
'_DH_
%
s'
%
name
)
return
handler
(
mime_data
,
action
,
index
)
else
:
return
False
def
_DH_ApplicationXBlinkContactGroupList
(
self
,
mime_data
,
action
,
index
):
contact_groups
=
self
.
contact_groups
group
=
self
.
items
[
index
.
row
()]
if
index
.
isValid
()
else
contact_groups
[
-
1
]
drop_indicator
=
group
.
widget
.
drop_indicator
if
group
.
widget
.
drop_indicator
is
None
:
return
False
selected_indexes
=
self
.
contact_list
.
selectionModel
()
.
selectedIndexes
()
moved_groups
=
set
(
self
.
items
[
index
.
row
()]
for
index
in
selected_indexes
if
index
.
isValid
())
if
group
is
contact_groups
[
0
]
and
group
in
moved_groups
:
drop_group
=
(
group
for
group
in
contact_groups
if
group
not
in
moved_groups
)
.
next
()
drop_position
=
self
.
contact_list
.
AboveItem
elif
group
is
contact_groups
[
-
1
]
and
group
in
moved_groups
:
drop_group
=
(
group
for
group
in
reversed
(
contact_groups
)
if
group
not
in
moved_groups
)
.
next
()
drop_position
=
self
.
contact_list
.
BelowItem
elif
group
in
moved_groups
:
position
=
contact_groups
.
index
(
group
)
if
drop_indicator
is
self
.
contact_list
.
AboveItem
:
drop_group
=
(
group
for
group
in
reversed
(
contact_groups
[:
position
])
if
group
not
in
moved_groups
)
.
next
()
drop_position
=
self
.
contact_list
.
BelowItem
else
:
drop_group
=
(
group
for
group
in
contact_groups
[
position
:]
if
group
not
in
moved_groups
)
.
next
()
drop_position
=
self
.
contact_list
.
AboveItem
else
:
drop_group
=
group
drop_position
=
drop_indicator
items
=
self
.
popItems
(
selected_indexes
)
contact_groups
=
self
.
contact_groups
# they changed so refresh them
if
drop_position
is
self
.
contact_list
.
AboveItem
:
position
=
self
.
items
.
index
(
drop_group
)
else
:
position
=
len
(
self
.
items
)
if
drop_group
is
contact_groups
[
-
1
]
else
self
.
items
.
index
(
contact_groups
[
contact_groups
.
index
(
drop_group
)
+
1
])
self
.
beginInsertRows
(
QModelIndex
(),
position
,
position
+
len
(
items
)
-
1
)
self
.
items
[
position
:
position
]
=
items
self
.
endInsertRows
()
for
index
,
item
in
enumerate
(
items
):
if
type
(
item
)
is
ContactGroup
:
self
.
contact_list
.
openPersistentEditor
(
self
.
index
(
position
+
index
))
else
:
self
.
contact_list
.
setRowHidden
(
position
+
index
,
item
.
group
.
collapsed
)
return
True
def
_DH_ApplicationXBlinkContactList
(
self
,
mime_data
,
action
,
index
):
group
=
self
.
items
[
index
.
row
()]
if
index
.
isValid
()
else
self
.
contact_groups
[
-
1
]
if
group
.
widget
.
drop_indicator
is
None
:
return
False
for
contact
in
self
.
popItems
(
self
.
contact_list
.
selectionModel
()
.
selectedIndexes
()):
contact
.
group
=
group
self
.
addContact
(
contact
)
return
True
def
_DH_TextUriList
(
self
,
mime_data
,
action
,
index
):
return
False
@
staticmethod
def
item_mime_data_iterator
(
data
):
stream
=
QDataStream
(
data
)
while
not
stream
.
atEnd
():
yield
stream
.
readInt32
(),
stream
.
readQVariant
()
.
toPyObject
()
@
staticmethod
def
range_iterator
(
indexes
):
"""Return contiguous ranges from indexes"""
start
=
last
=
None
for
index
in
sorted
(
indexes
):
if
start
is
None
:
start
=
index
elif
index
-
last
>
1
:
yield
(
start
,
last
)
start
=
index
last
=
index
else
:
if
indexes
:
yield
(
start
,
last
)
@
staticmethod
def
reversed_range_iterator
(
indexes
):
"""Return contiguous ranges from indexes starting from the end"""
end
=
last
=
None
for
index
in
reversed
(
sorted
(
indexes
)):
if
end
is
None
:
end
=
index
elif
last
-
index
>
1
:
yield
(
last
,
end
)
end
=
index
last
=
index
else
:
if
indexes
:
yield
(
last
,
end
)
def
addContact
(
self
,
contact
):
def
addContact
(
self
,
contact
):
if
contact
.
group
in
self
.
items
:
if
contact
.
group
in
self
.
items
:
for
position
in
xrange
(
self
.
items
.
index
(
contact
.
group
)
+
1
,
len
(
self
.
items
)):
for
position
in
xrange
(
self
.
items
.
index
(
contact
.
group
)
+
1
,
len
(
self
.
items
)):
...
@@ -353,11 +550,12 @@ class ContactModel(QAbstractListModel):
...
@@ -353,11 +550,12 @@ class ContactModel(QAbstractListModel):
self
.
contact_list
.
openPersistentEditor
(
self
.
index
(
position
))
self
.
contact_list
.
openPersistentEditor
(
self
.
index
(
position
))
self
.
endInsertRows
()
self
.
endInsertRows
()
def
deleteContacts
(
self
,
indexes
):
def
removeContact
(
self
,
contact
):
rows
=
sorted
(
index
.
row
()
for
index
in
indexes
if
index
.
isValid
())
if
contact
not
in
self
.
items
:
self
.
beginRemoveRows
(
QModelIndex
(),
rows
[
0
],
rows
[
-
1
])
return
for
row
in
reversed
(
rows
):
position
=
self
.
items
.
index
(
contact
)
self
.
items
.
pop
(
row
)
self
.
beginRemoveRows
(
QModelIndex
(),
position
,
position
)
del
self
.
items
[
position
]
self
.
endRemoveRows
()
self
.
endRemoveRows
()
def
addGroup
(
self
,
group
):
def
addGroup
(
self
,
group
):
...
@@ -369,19 +567,52 @@ class ContactModel(QAbstractListModel):
...
@@ -369,19 +567,52 @@ class ContactModel(QAbstractListModel):
self
.
contact_list
.
openPersistentEditor
(
self
.
index
(
position
))
self
.
contact_list
.
openPersistentEditor
(
self
.
index
(
position
))
self
.
endInsertRows
()
self
.
endInsertRows
()
def
removeGroup
(
self
,
group
):
if
group
not
in
self
.
items
:
return
start
=
self
.
items
.
index
(
group
)
end
=
start
+
len
([
item
for
item
in
self
.
items
if
type
(
item
)
is
Contact
and
item
.
group
==
group
])
self
.
beginRemoveRows
(
QModelIndex
(),
start
,
end
)
del
self
.
items
[
start
:
end
+
1
]
self
.
endRemoveRows
()
def
removeItems
(
self
,
indexes
):
rows
=
set
(
index
.
row
()
for
index
in
indexes
if
index
.
isValid
())
removed_groups
=
set
(
self
.
items
[
row
]
for
row
in
rows
if
type
(
self
.
items
[
row
])
is
ContactGroup
)
rows
.
update
(
row
for
row
,
item
in
enumerate
(
self
.
items
)
if
type
(
item
)
is
Contact
and
item
.
group
in
removed_groups
)
for
start
,
end
in
self
.
reversed_range_iterator
(
rows
):
self
.
beginRemoveRows
(
QModelIndex
(),
start
,
end
)
del
self
.
items
[
start
:
end
+
1
]
self
.
endRemoveRows
()
def
popItems
(
self
,
indexes
):
items
=
[]
rows
=
set
(
index
.
row
()
for
index
in
indexes
if
index
.
isValid
())
removed_groups
=
set
(
self
.
items
[
row
]
for
row
in
rows
if
type
(
self
.
items
[
row
])
is
ContactGroup
)
rows
.
update
(
row
for
row
,
item
in
enumerate
(
self
.
items
)
if
type
(
item
)
is
Contact
and
item
.
group
in
removed_groups
)
for
start
,
end
in
self
.
reversed_range_iterator
(
rows
):
self
.
beginRemoveRows
(
QModelIndex
(),
start
,
end
)
items
[
0
:
0
]
=
self
.
items
[
start
:
end
+
1
]
del
self
.
items
[
start
:
end
+
1
]
self
.
endRemoveRows
()
return
items
def
test
(
self
):
def
test
(
self
):
work_group
=
ContactGroup
(
'Work'
)
work_group
=
ContactGroup
(
'Work'
)
test_group
=
ContactGroup
(
'Test'
)
for
contact
in
[
Contact
(
work_group
,
'Dan Pascu'
,
'31208005167@ag-projects.com'
,
'icons/avatar.png'
),
Contact
(
work_group
,
'Lucian Stanescu'
,
'31208005164@ag-projects.com'
),
Contact
(
work_group
,
'Test number'
,
'3333@ag-projects.com'
)]:
for
contact
in
[
Contact
(
work_group
,
'Dan Pascu'
,
'31208005167@ag-projects.com'
,
'icons/avatar.png'
),
Contact
(
work_group
,
'Lucian Stanescu'
,
'31208005164@ag-projects.com'
),
Contact
(
work_group
,
'Test number'
,
'3333@ag-projects.com'
)]:
if
contact
.
uri
.
startswith
(
'3333@'
)
or
contact
.
uri
.
startswith
(
'31208005167@'
):
if
contact
.
uri
.
startswith
(
'3333@'
)
or
contact
.
uri
.
startswith
(
'31208005167@'
):
contact
.
status
=
'online'
contact
.
status
=
'online'
else
:
else
:
contact
.
status
=
'busy'
contact
.
status
=
'busy'
self
.
addContact
(
contact
)
self
.
addContact
(
contact
)
self
.
addGroup
(
test_group
)
self
.
addGroup
(
ContactGroup
(
'Test'
))
self
.
addGroup
(
ContactGroup
(
'Test 2'
))
self
.
addGroup
(
ContactGroup
(
'Test 3'
))
class
ContactSearchModel
(
QSortFilterProxyModel
):
class
ContactSearchModel
(
QSortFilterProxyModel
):
accepted_mime_types
=
[
'text/uri-list'
]
def
__init__
(
self
,
model
,
parent
=
None
):
def
__init__
(
self
,
model
,
parent
=
None
):
super
(
ContactSearchModel
,
self
)
.
__init__
(
parent
)
super
(
ContactSearchModel
,
self
)
.
__init__
(
parent
)
self
.
setSourceModel
(
model
)
self
.
setSourceModel
(
model
)
...
@@ -408,3 +639,162 @@ class ContactSearchModel(QSortFilterProxyModel):
...
@@ -408,3 +639,162 @@ class ContactSearchModel(QSortFilterProxyModel):
return
left_item
.
name
<
right_item
.
name
return
left_item
.
name
<
right_item
.
name
class
ContactListView
(
QListView
):
def
__init__
(
self
,
parent
=
None
):
super
(
ContactListView
,
self
)
.
__init__
(
parent
)
self
.
setDragEnabled
(
True
)
self
.
setAcceptDrops
(
True
)
self
.
setDropIndicatorShown
(
False
)
self
.
restore_timer
=
QTimer
(
self
)
self
.
restore_timer
.
setSingleShot
(
True
)
self
.
restore_timer
.
setInterval
(
1250
)
self
.
restore_timer
.
timeout
.
connect
(
self
.
_restore_groups
)
self
.
needs_restore
=
False
def
_restore_groups
(
self
):
for
group
in
self
.
model
()
.
contact_groups
:
group
.
restore_state
()
self
.
needs_restore
=
False
def
paintEvent
(
self
,
event
):
super
(
ContactListView
,
self
)
.
paintEvent
(
event
)
model
=
self
.
model
()
last_group
=
model
.
contact_groups
[
-
1
]
if
last_group
.
widget
.
drop_indicator
is
self
.
BelowItem
:
# draw the bottom part of the drop indicator for the last group
rect
=
self
.
visualRect
(
model
.
index
(
model
.
items
.
index
(
last_group
)))
line_rect
=
QRectF
(
rect
.
adjusted
(
18
,
rect
.
height
(),
0
,
5
))
arc_rect
=
line_rect
.
adjusted
(
-
5
,
-
3
,
-
line_rect
.
width
(),
-
3
)
path
=
QPainterPath
(
line_rect
.
topRight
())
path
.
lineTo
(
line_rect
.
topLeft
())
path
.
arcTo
(
arc_rect
,
0
,
-
180
)
painter
=
QPainter
(
self
.
viewport
())
painter
.
setRenderHint
(
QPainter
.
Antialiasing
,
True
)
painter
.
setPen
(
QPen
(
QBrush
(
QColor
(
'#dc3169'
)),
2.0
))
painter
.
drawPath
(
path
)
painter
.
end
()
def
dragEnterEvent
(
self
,
event
):
event_source
=
event
.
source
()
accepted_mime_types
=
set
(
self
.
model
()
.
accepted_mime_types
)
provided_mime_types
=
set
(
str
(
x
)
for
x
in
event
.
mimeData
()
.
formats
())
acceptable_mime_types
=
accepted_mime_types
&
provided_mime_types
has_blink_contacts
=
'application/x-blink-contact-list'
in
provided_mime_types
has_blink_groups
=
'application/x-blink-contact-group-list'
in
provided_mime_types
if
not
acceptable_mime_types
:
event
.
ignore
()
# no acceptable mime types found
elif
has_blink_contacts
and
has_blink_groups
:
event
.
ignore
()
# we can't handle drops for both groups and contacts at the same time
elif
event_source
is
not
self
and
(
has_blink_contacts
or
has_blink_groups
):
event
.
ignore
()
# we don't handle drops for blink contacts or groups from other sources
else
:
if
event_source
is
self
:
event
.
setDropAction
(
Qt
.
MoveAction
)
if
has_blink_contacts
or
has_blink_groups
:
if
not
self
.
needs_restore
:
for
group
in
self
.
model
()
.
contact_groups
:
group
.
save_state
()
group
.
collapse
()
self
.
needs_restore
=
True
self
.
restore_timer
.
stop
()
event
.
accept
()
self
.
setState
(
self
.
DraggingState
)
def
dragLeaveEvent
(
self
,
event
):
super
(
ContactListView
,
self
)
.
dragLeaveEvent
(
event
)
for
group
in
self
.
model
()
.
contact_groups
:
group
.
widget
.
drop_indicator
=
None
if
self
.
needs_restore
:
self
.
restore_timer
.
start
()
def
dragMoveEvent
(
self
,
event
):
super
(
ContactListView
,
self
)
.
dragMoveEvent
(
event
)
if
event
.
source
()
is
self
:
event
.
setDropAction
(
Qt
.
MoveAction
)
for
mime_type
in
self
.
model
()
.
accepted_mime_types
:
if
event
.
provides
(
mime_type
):
index
=
self
.
indexAt
(
event
.
pos
())
rect
=
self
.
visualRect
(
index
)
item
=
self
.
model
()
.
data
(
index
)
name
=
mime_type
.
replace
(
'/'
,
' '
)
.
replace
(
'-'
,
' '
)
.
title
()
.
replace
(
' '
,
''
)
handler
=
getattr
(
self
,
'_DH_
%
s'
%
name
)
handler
(
event
,
index
,
rect
,
item
)
break
else
:
event
.
ignore
()
def
_DH_ApplicationXBlinkContactGroupList
(
self
,
event
,
index
,
rect
,
item
):
model
=
self
.
model
()
groups
=
model
.
contact_groups
for
group
in
groups
:
group
.
widget
.
drop_indicator
=
None
if
not
index
.
isValid
():
drop_groups
=
(
groups
[
-
1
],
Null
)
rect
=
self
.
viewport
()
.
rect
()
rect
.
setTop
(
self
.
visualRect
(
model
.
index
(
model
.
items
.
index
(
groups
[
-
1
])))
.
bottom
())
elif
type
(
item
)
is
ContactGroup
:
index
=
groups
.
index
(
item
)
rect
.
setHeight
(
rect
.
height
()
/
2
)
if
rect
.
contains
(
event
.
pos
()):
drop_groups
=
(
groups
[
index
-
1
],
groups
[
index
])
if
index
>
0
else
(
Null
,
groups
[
index
])
else
:
drop_groups
=
(
groups
[
index
],
groups
[
index
+
1
])
if
index
<
len
(
groups
)
-
1
else
(
groups
[
index
],
Null
)
rect
.
translate
(
0
,
rect
.
height
())
selected_rows
=
sorted
(
index
.
row
()
for
index
in
self
.
selectionModel
()
.
selectedIndexes
())
if
selected_rows
:
first
=
groups
.
index
(
model
.
items
[
selected_rows
[
0
]])
last
=
groups
.
index
(
model
.
items
[
selected_rows
[
-
1
]])
contiguous_selection
=
len
(
selected_rows
)
==
last
-
first
+
1
else
:
contiguous_selection
=
False
selected_groups
=
set
(
model
.
items
[
row
]
for
row
in
selected_rows
)
overlapping_groups
=
len
(
selected_groups
.
intersection
(
drop_groups
))
allowed_overlapping
=
0
if
contiguous_selection
else
1
if
event
.
source
()
is
not
self
or
overlapping_groups
<=
allowed_overlapping
:
drop_groups
[
0
]
.
widget
.
drop_indicator
=
self
.
BelowItem
drop_groups
[
1
]
.
widget
.
drop_indicator
=
self
.
AboveItem
if
groups
[
-
1
]
in
drop_groups
:
self
.
viewport
()
.
update
()
event
.
accept
(
rect
)
def
_DH_ApplicationXBlinkContactList
(
self
,
event
,
index
,
rect
,
item
):
model
=
self
.
model
()
groups
=
model
.
contact_groups
for
group
in
groups
:
group
.
widget
.
drop_indicator
=
None
if
not
index
.
isValid
():
group
=
groups
[
-
1
]
rect
=
self
.
viewport
()
.
rect
()
rect
.
setTop
(
self
.
visualRect
(
model
.
index
(
model
.
items
.
index
(
group
)))
.
bottom
())
elif
type
(
item
)
is
ContactGroup
:
group
=
item
selected_contact_groups
=
set
(
model
.
items
[
index
.
row
()]
.
group
for
index
in
self
.
selectionModel
()
.
selectedIndexes
())
if
event
.
source
()
is
not
self
or
len
(
selected_contact_groups
)
>
1
or
group
not
in
selected_contact_groups
:
group
.
widget
.
drop_indicator
=
self
.
OnItem
event
.
accept
(
rect
)
def
_DH_TextUriList
(
self
,
event
,
index
,
rect
,
item
):
model
=
self
.
model
()
if
not
index
.
isValid
():
rect
=
self
.
viewport
()
.
rect
()
rect
.
setTop
(
self
.
visualRect
(
model
.
index
(
len
(
model
.
items
)
-
1
))
.
bottom
())
if
type
(
item
)
is
Contact
:
event
.
accept
(
rect
)
else
:
event
.
ignore
(
rect
)
def
dropEvent
(
self
,
event
):
model
=
self
.
model
()
if
event
.
source
()
is
self
:
event
.
setDropAction
(
Qt
.
MoveAction
)
if
model
.
handleDroppedData
(
event
.
mimeData
(),
event
.
dropAction
(),
self
.
indexAt
(
event
.
pos
())):
event
.
accept
()
for
group
in
model
.
contact_groups
:
group
.
widget
.
drop_indicator
=
None
if
self
.
needs_restore
:
group
.
restore_state
()
self
.
needs_restore
=
False
super
(
ContactListView
,
self
)
.
dropEvent
(
event
)
resources/blink.ui
View file @
cc1cec7f
...
@@ -289,13 +289,16 @@
...
@@ -289,13 +289,16 @@
<number>
0
</number>
<number>
0
</number>
</property>
</property>
<item>
<item>
<widget
class=
"
Q
ListView"
name=
"contact_list"
>
<widget
class=
"
Contact
ListView"
name=
"contact_list"
>
<property
name=
"horizontalScrollBarPolicy"
>
<property
name=
"horizontalScrollBarPolicy"
>
<enum>
Qt::ScrollBarAlwaysOff
</enum>
<enum>
Qt::ScrollBarAlwaysOff
</enum>
</property>
</property>
<property
name=
"alternatingRowColors"
>
<property
name=
"alternatingRowColors"
>
<bool>
true
</bool>
<bool>
true
</bool>
</property>
</property>
<property
name=
"selectionMode"
>
<enum>
QAbstractItemView::ExtendedSelection
</enum>
</property>
</widget>
</widget>
</item>
</item>
</layout>
</layout>
...
@@ -804,6 +807,11 @@ buttons below.</string>
...
@@ -804,6 +807,11 @@ buttons below.</string>
<extends>
QLineEdit
</extends>
<extends>
QLineEdit
</extends>
<header>
blink.widgets.lineedit
</header>
<header>
blink.widgets.lineedit
</header>
</customwidget>
</customwidget>
<customwidget>
<class>
ContactListView
</class>
<extends>
QListView
</extends>
<header>
blink.contacts
</header>
</customwidget>
</customwidgets>
</customwidgets>
<tabstops>
<tabstops>
<tabstop>
search_box
</tabstop>
<tabstop>
search_box
</tabstop>
...
...
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