Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
M
Material
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Dmitriy Stepanets
Material
Commits
57b9f836
Commit
57b9f836
authored
Aug 02, 2017
by
Daniel Dahan
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
added initial ChipBar
parent
d75cb0e8
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
323 additions
and
18 deletions
+323
-18
Material.xcodeproj/project.pbxproj
+12
-0
Sources/iOS/ChipBar.swift
+300
-0
Sources/iOS/TabBar.swift
+0
-0
Sources/iOS/TabsController.swift
+11
-18
No files found.
Material.xcodeproj/project.pbxproj
View file @
57b9f836
...
...
@@ -7,6 +7,7 @@
objects
=
{
/* Begin PBXBuildFile section */
961154CC1F32A7B100A78D74
/* ChipBar.swift in Sources */
=
{
isa
=
PBXBuildFile
;
fileRef
=
961154CB1F32A7B100A78D74
/* ChipBar.swift */
;
};
961409B11E43D15C00E7BA99
/* FABMenu.swift in Headers */
=
{
isa
=
PBXBuildFile
;
fileRef
=
96A183621E0C6CE200083C30
/* FABMenu.swift */
;
settings
=
{
ATTRIBUTES
=
(
Public
,
);
};
};
961409B21E43D15C00E7BA99
/* FABMenuController.swift in Headers */
=
{
isa
=
PBXBuildFile
;
fileRef
=
96A183641E0C6DD400083C30
/* FABMenuController.swift */
;
settings
=
{
ATTRIBUTES
=
(
Public
,
);
};
};
9617B07D1DFCA8CF00410F8F
/* Application.swift in Headers */
=
{
isa
=
PBXBuildFile
;
fileRef
=
961E6BDE1DDA2A95004E6C93
/* Application.swift */
;
settings
=
{
ATTRIBUTES
=
(
Public
,
);
};
};
...
...
@@ -206,6 +207,7 @@
/* End PBXBuildFile section */
/* Begin PBXFileReference section */
961154CB1F32A7B100A78D74
/* ChipBar.swift */
=
{
isa
=
PBXFileReference
;
fileEncoding
=
4
;
lastKnownFileType
=
sourcecode.swift
;
path
=
ChipBar.swift
;
sourceTree
=
"<group>"
;
};
961276621DCD8B1800A7D920
/* CharacterAttribute.swift */
=
{
isa
=
PBXFileReference
;
fileEncoding
=
4
;
lastKnownFileType
=
sourcecode.swift
;
path
=
CharacterAttribute.swift
;
sourceTree
=
"<group>"
;
};
961E6BDE1DDA2A95004E6C93
/* Application.swift */
=
{
isa
=
PBXFileReference
;
fileEncoding
=
4
;
lastKnownFileType
=
sourcecode.swift
;
path
=
Application.swift
;
sourceTree
=
"<group>"
;
};
961E6BE11DDA2AF3004E6C93
/* Screen.swift */
=
{
isa
=
PBXFileReference
;
fileEncoding
=
4
;
lastKnownFileType
=
sourcecode.swift
;
path
=
Screen.swift
;
sourceTree
=
"<group>"
;
};
...
...
@@ -354,6 +356,14 @@
name
=
Text
;
sourceTree
=
"<group>"
;
};
961154CA1F32999000A78D74
/* Chip */
=
{
isa
=
PBXGroup
;
children
=
(
961154CB1F32A7B100A78D74
/* ChipBar.swift */
,
);
name
=
Chip
;
sourceTree
=
"<group>"
;
};
961E6BDD1DDA2A7E004E6C93
/* Application */
=
{
isa
=
PBXGroup
;
children
=
(
...
...
@@ -617,6 +627,7 @@
962DDD081D6FBBD0001C307C
/* BottomTabBar */
,
96BCB8031CB40F4B00C806FE
/* Button */
,
96BCB8021CB40F3B00C806FE
/* Card */
,
961154CA1F32999000A78D74
/* Chip */
,
96BCB8051CB40F9C00C806FE
/* Collection */
,
96BCB8001CB40F0300C806FE
/* Color */
,
96328B9A1E05C135009A4C90
/* Data */
,
...
...
@@ -1058,6 +1069,7 @@
96A1808B1F2E2B6E006BD69D
/* Motion+UIView.swift in Sources */
,
96A180971F2E2B6E006BD69D
/* MotionCoordinateSpace.swift in Sources */
,
96E09DC81F2287E50000B121
/* TabsController.swift in Sources */
,
961154CC1F32A7B100A78D74
/* ChipBar.swift in Sources */
,
965689611F002F4C001C656D
/* CardCollectionViewController.swift in Sources */
,
96A180951F2E2B6E006BD69D
/* MotionContext.swift in Sources */
,
965E80CC1DD4C50600D61E4B
/* Bar.swift in Sources */
,
...
...
Sources/iOS/ChipBar.swift
0 → 100644
View file @
57b9f836
/*
* Copyright (C) 2015 - 2017, Daniel Dahan and CosmicMind, Inc. <http://cosmicmind.com>.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of CosmicMind nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
import
UIKit
open
class
ChipItem
:
FlatButton
{}
@objc(ChipBarDelegate)
public
protocol
ChipBarDelegate
{
/**
A delegation method that is executed when the chipItem will trigger the
animation to the next chip.
- Parameter chipBar: A ChipBar.
- Parameter chipItem: A ChipItem.
*/
@objc
optional
func
chipBar
(
chipBar
:
ChipBar
,
willSelect
chipItem
:
ChipItem
)
/**
A delegation method that is executed when the chipItem did complete the
animation to the next chip.
- Parameter chipBar: A ChipBar.
- Parameter chipItem: A ChipItem.
*/
@objc
optional
func
chipBar
(
chipBar
:
ChipBar
,
didSelect
chipItem
:
ChipItem
)
}
@objc(ChipBarStyle)
public
enum
ChipBarStyle
:
Int
{
case
auto
case
nonScrollable
case
scrollable
}
open
class
ChipBar
:
Bar
{
/// A boolean indicating if the ChipBar line is in an animation state.
open
fileprivate
(
set
)
var
isAnimating
=
false
/// The total width of the chipItems.
fileprivate
var
chipItemsTotalWidth
:
CGFloat
{
var
w
:
CGFloat
=
0
for
v
in
chipItems
{
w
+=
v
.
sizeThatFits
(
CGSize
(
width
:
CGFloat
.
greatestFiniteMagnitude
,
height
:
contentView
.
height
))
.
width
+
interimSpace
}
return
w
}
/// An enum that determines the chip bar style.
open
var
chipBarStyle
=
ChipBarStyle
.
auto
{
didSet
{
layoutSubviews
()
}
}
/// A reference to the scroll view when the chip bar style is scrollable.
open
let
scrollView
=
UIScrollView
()
/// Enables and disables bouncing when swiping.
open
var
isScrollBounceEnabled
:
Bool
{
get
{
return
scrollView
.
bounces
}
set
(
value
)
{
scrollView
.
bounces
=
value
}
}
/// A delegation reference.
open
weak
var
delegate
:
ChipBarDelegate
?
/// The currently selected chipItem.
open
fileprivate
(
set
)
var
selected
:
ChipItem
?
/// Buttons.
open
var
chipItems
=
[
ChipItem
]()
{
didSet
{
for
b
in
oldValue
{
b
.
removeFromSuperview
()
}
prepareChipItems
()
layoutSubviews
()
}
}
open
override
func
layoutSubviews
()
{
super
.
layoutSubviews
()
guard
willLayout
else
{
return
}
var
lc
=
0
var
rc
=
0
grid
.
begin
()
grid
.
views
.
removeAll
()
for
v
in
leftViews
{
if
let
b
=
v
as?
ChipItem
{
b
.
contentEdgeInsets
=
.
zero
b
.
titleEdgeInsets
=
.
zero
}
v
.
width
=
v
.
intrinsicContentSize
.
width
v
.
sizeToFit
()
v
.
grid
.
columns
=
Int
(
ceil
(
v
.
width
/
gridFactor
))
+
2
lc
+=
v
.
grid
.
columns
grid
.
views
.
append
(
v
)
}
grid
.
views
.
append
(
contentView
)
for
v
in
rightViews
{
if
let
b
=
v
as?
ChipItem
{
b
.
contentEdgeInsets
=
.
zero
b
.
titleEdgeInsets
=
.
zero
}
v
.
width
=
v
.
intrinsicContentSize
.
width
v
.
sizeToFit
()
v
.
grid
.
columns
=
Int
(
ceil
(
v
.
width
/
gridFactor
))
+
2
rc
+=
v
.
grid
.
columns
grid
.
views
.
append
(
v
)
}
contentView
.
grid
.
begin
()
contentView
.
grid
.
offset
.
columns
=
0
var
l
:
CGFloat
=
0
var
r
:
CGFloat
=
0
if
.
center
==
contentViewAlignment
{
if
leftViews
.
count
<
rightViews
.
count
{
r
=
CGFloat
(
rightViews
.
count
)
*
interimSpace
l
=
r
}
else
{
l
=
CGFloat
(
leftViews
.
count
)
*
interimSpace
r
=
l
}
}
let
p
=
width
-
l
-
r
-
contentEdgeInsets
.
left
-
contentEdgeInsets
.
right
let
columns
=
Int
(
ceil
(
p
/
gridFactor
))
if
.
center
==
contentViewAlignment
{
if
lc
<
rc
{
contentView
.
grid
.
columns
=
columns
-
2
*
rc
contentView
.
grid
.
offset
.
columns
=
rc
-
lc
}
else
{
contentView
.
grid
.
columns
=
columns
-
2
*
lc
rightViews
.
first
?
.
grid
.
offset
.
columns
=
lc
-
rc
}
}
else
{
contentView
.
grid
.
columns
=
columns
-
lc
-
rc
}
grid
.
axis
.
columns
=
columns
if
.
scrollable
==
chipBarStyle
||
(
.
auto
==
chipBarStyle
&&
chipItemsTotalWidth
>
bounds
.
width
)
{
var
w
:
CGFloat
=
0
for
v
in
chipItems
{
let
x
=
v
.
sizeThatFits
(
CGSize
(
width
:
CGFloat
.
greatestFiniteMagnitude
,
height
:
contentView
.
height
))
.
width
+
interimSpace
scrollView
.
addSubview
(
v
)
v
.
height
=
scrollView
.
height
v
.
width
=
x
v
.
x
=
w
w
+=
x
}
scrollView
.
contentSize
=
CGSize
(
width
:
w
,
height
:
height
)
}
else
{
scrollView
.
grid
.
views
=
chipItems
scrollView
.
grid
.
axis
.
columns
=
chipItems
.
count
scrollView
.
contentSize
=
CGSize
(
width
:
scrollView
.
width
,
height
:
height
)
}
grid
.
commit
()
contentView
.
grid
.
commit
()
layoutDivider
()
}
open
override
func
prepare
()
{
super
.
prepare
()
contentEdgeInsetsPreset
=
.
none
interimSpacePreset
=
.
interimSpace6
prepareContentView
()
prepareScrollView
()
prepareDivider
()
}
}
fileprivate
extension
ChipBar
{
/// Prepares the divider.
func
prepareDivider
()
{
dividerColor
=
Color
.
grey
.
lighten3
}
/// Prepares the chipItems.
func
prepareChipItems
()
{
for
v
in
chipItems
{
v
.
grid
.
columns
=
0
v
.
cornerRadius
=
0
v
.
contentEdgeInsets
=
.
zero
}
}
/// Prepares the contentView.
func
prepareContentView
()
{
contentView
.
zPosition
=
6000
}
/// Prepares the scroll view.
func
prepareScrollView
()
{
scrollView
.
isPagingEnabled
=
false
scrollView
.
showsVerticalScrollIndicator
=
false
scrollView
.
showsHorizontalScrollIndicator
=
false
centerViews
=
[
scrollView
]
}
}
extension
ChipBar
{
/**
Selects a given index from the chipItems array.
- Parameter at index: An Int.
- Paramater completion: An optional completion block.
*/
open
func
select
(
at
index
:
Int
,
completion
:
((
ChipItem
)
->
Void
)?
=
nil
)
{
guard
-
1
<
index
,
index
<
chipItems
.
count
else
{
return
}
animate
(
to
:
chipItems
[
index
],
isTriggeredByUserInteraction
:
false
,
completion
:
completion
)
}
/**
Animates to a given chipItem.
- Parameter to chipItem: A ChipItem.
- Parameter completion: An optional completion block.
*/
open
func
animate
(
to
chipItem
:
ChipItem
,
completion
:
((
ChipItem
)
->
Void
)?
=
nil
)
{
animate
(
to
:
chipItem
,
isTriggeredByUserInteraction
:
false
,
completion
:
completion
)
}
/**
Animates to a given chipItem.
- Parameter to chipItem: A ChipItem.
- Parameter isTriggeredByUserInteraction: A boolean indicating whether the
state was changed by a user interaction, true if yes, false otherwise.
- Parameter completion: An optional completion block.
*/
fileprivate
func
animate
(
to
chipItem
:
ChipItem
,
isTriggeredByUserInteraction
:
Bool
,
completion
:
((
ChipItem
)
->
Void
)?
=
nil
)
{
if
isTriggeredByUserInteraction
{
delegate
?
.
chipBar
?(
chipBar
:
self
,
willSelect
:
chipItem
)
}
selected
=
chipItem
isAnimating
=
true
if
!
scrollView
.
bounds
.
contains
(
chipItem
.
frame
)
{
let
contentOffsetX
=
(
chipItem
.
x
<
scrollView
.
bounds
.
minX
)
?
chipItem
.
x
:
chipItem
.
frame
.
maxX
-
scrollView
.
bounds
.
width
let
normalizedOffsetX
=
min
(
max
(
contentOffsetX
,
0
),
scrollView
.
contentSize
.
width
-
scrollView
.
bounds
.
width
)
scrollView
.
setContentOffset
(
CGPoint
(
x
:
normalizedOffsetX
,
y
:
0
),
animated
:
true
)
}
}
}
Sources/iOS/TabBar.swift
View file @
57b9f836
This diff is collapsed.
Click to expand it.
Sources/iOS/TabsController.swift
View file @
57b9f836
...
...
@@ -32,13 +32,6 @@ import UIKit
fileprivate
var
TabItemKey
:
UInt8
=
0
open
class
TabItem
:
FlatButton
{
open
override
func
prepare
()
{
super
.
prepare
()
pulseAnimation
=
.
none
}
}
@objc(TabBarAlignment)
public
enum
TabBarAlignment
:
Int
{
case
top
...
...
@@ -166,7 +159,7 @@ open class TabsController: UIViewController {
view
.
contentScaleFactor
=
Screen
.
scale
prepareContainer
()
prepareTabBar
()
prepareTabBar
Button
s
()
prepareTabBar
Item
s
()
prepareViewControllers
()
}
}
...
...
@@ -183,18 +176,18 @@ fileprivate extension TabsController {
view
.
addSubview
(
tabBar
)
}
/// Prepares the
tabBar buttons
.
func
prepareTabBar
Button
s
()
{
var
buttons
=
[
UIButton
]()
/// Prepares the
`tabBar.tabItems`
.
func
prepareTabBar
Item
s
()
{
var
tabItems
=
[
TabItem
]()
for
v
in
viewControllers
{
let
b
=
v
.
tabItem
b
.
removeTarget
(
self
,
action
:
#selector(
handle
TabBarButton(button
:)
)
,
for
:
.
touchUpInside
)
b
.
addTarget
(
self
,
action
:
#selector(
handle
TabBarButton(button
:)
)
,
for
:
.
touchUpInside
)
button
s
.
append
(
b
)
b
.
removeTarget
(
self
,
action
:
#selector(
handle
(tabItem
:)
)
,
for
:
.
touchUpInside
)
b
.
addTarget
(
self
,
action
:
#selector(
handle
(tabItem
:)
)
,
for
:
.
touchUpInside
)
tabItem
s
.
append
(
b
)
}
tabBar
.
buttons
=
button
s
tabBar
.
tabItems
=
tabItem
s
}
/// Prepares all the view controllers.
...
...
@@ -307,11 +300,11 @@ fileprivate extension TabsController {
fileprivate
extension
TabsController
{
/**
Handles the tabItem.
- Parameter
button: A UIButton
.
- Parameter
tabItem: A TabItem
.
*/
@objc
func
handle
TabBarButton
(
button
:
UIButton
)
{
guard
let
i
=
tabBar
.
buttons
.
index
(
of
:
button
)
else
{
func
handle
(
tabItem
:
TabItem
)
{
guard
let
i
=
tabBar
.
tabItems
.
index
(
of
:
tabItem
)
else
{
return
}
...
...
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