Display Inputs and Results


In [1]:
from salib import extend, NBImporter
from Nodes import Node
from Members import Member
from Frame2D_Base import Frame2D
import Frame2D_Input


Compiling notebook file 'Frame2D_Input.ipynb' to python.

In [2]:
##test:
f = Frame2D('frame-1')
f.input_all()

Display Nodes


In [3]:
def prhead(txt,ul='='):
    """Print a heading and underline it."""
    print()
    print(txt)
    if ul:
        print(ul*(len(txt)//len(ul)))
    print()

In [4]:
@extend
class Frame2D:

    def print_nodes(self,precision=0,printdof=False):
        prhead('Nodes:')
        print('Node          X         Y  Constraints  DOF #s')
        print('----      -----     -----  -----------  ------')
        for nid,node in self.nodes.items():
            ct = ','.join(sorted(node.constraints,key=lambda t: Node.DIRECTIONS[t]))
            dt = ','.join([str(x) for x in node.dofnums])
            print('{:<5s}{:>10.{precision}f}{:>10.{precision}f}  {:<11s}  {}'\
                  .format(nid,node.x,node.y,ct,dt,precision=precision))
        if not printdof:
            return
        print()
        print('DOF#   Node  Dirn')
        print('----   ----  ----')
        for i in range(len(self.dofdesc)):
            node,dirn = self.dofdesc[i]
            print('{:>4d}   {:<4s}  {}'.format(i,node.id,dirn))

In [5]:
##test:
f.print_nodes(printdof=True)


Nodes:
======

Node          X         Y  Constraints  DOF #s
----      -----     -----  -----------  ------
A             0         0  FX,FY,MZ     7,8,9
B             0      4000               0,1,2
C          8000      4000               3,4,5
D          8000         0  FX,FY        10,11,6

DOF#   Node  Dirn
----   ----  ----
   0   B     FX
   1   B     FY
   2   B     MZ
   3   C     FX
   4   C     FY
   5   C     MZ
   6   D     MZ
   7   A     FX
   8   A     FY
   9   A     MZ
  10   D     FX
  11   D     FY

Display Members


In [6]:
@extend
class Frame2D:
    
    def print_members(self,precision=1):
        prhead('Members:')
        print('Member   Node-J  Node-K    Length       dcx       dcy  Size                Ix           A  Releases')
        print('------   ------  ------    ------   -------   -------  --------      --------       -----  --------')
        for mid,memb in self.members.items():
            nj = memb.nodej
            nk = memb.nodek
            rt = ','.join(sorted(memb.releases,key=lambda t: Member.RELEASES[t]))
            print('{:<7s}  {:<6s}  {:<6s}  {:>8.{precision}f}  {:>8.5f}  {:>8.5f}  {:<10s}  {:>10g}  {:>10g}  {}'\
                  .format(memb.id,nj.id,nk.id,memb.L,memb.dcx,memb.dcy,str(memb.size),memb.Ix,memb.A,rt,precision=precision))

In [7]:
##test:
f.print_members()


Members:
========

Member   Node-J  Node-K    Length       dcx       dcy  Size                Ix           A  Releases
------   ------  ------    ------   -------   -------  --------      --------       -----  --------
AB       A       B         4000.0   0.00000   1.00000  W310x97       2.22e+08       12300  MZK
BC       B       C         8000.0   1.00000   0.00000  W460x106      4.88e+08       13500  
CD       C       D         4000.0   0.00000  -1.00000                2.22e+08       12300  MZJ

Display loads


In [8]:
@extend
class Frame2D:
    
    def print_loads(self,precision=0):
        
        prhead('Node Loads:')
        if self.nodeloads:
            print('Type      Node      FX          FY          MZ')
            print('----      ----  ----------  ----------  ----------')
            for lname,node,load in self.nodeloads:
                print('{:<8s}  {:<4s}  {:>10.{precision}f}  {:>10.{precision}f}  {:>10.{precision}f}'
                      .format(lname,node.id,load.fx,load.fy,load.mz,precision=precision))
        else:
            print(" - - - none - - -")

        prhead('Member Loads:')
        if self.memberloads:
            print('Type      Member  Load')
            print('----      ------  ----------------')
            for lname,memb,load in self.memberloads:
                print("{:<8s}  {:<6s}  {}".format(lname,memb.id,load))
        else:
            print(" - - - none - - -")

        prhead('Support Displacements:')
        if self.nodedeltas:
            print('Type      Node      DX          DY          TZ')
            print('----      ----  ----------  ----------  ----------')
            for lname,node,load in self.nodedeltas:
                print('{:<8s}  {:<4s}  {:>10.{precision}f}  {:>10.{precision}f}  {:>10.{precision}f}'
                      .format(lname,node.id,load.fx,load.fy,load.mz,precision=precision))
        else:
            print(" - - - none - - -")

        prhead("Load Combinations:")
        if self.loadcombinations:
            print('Case   Type      Factor')
            print('-----  ----      ------')
            prev = None
            for cname,lname,f in self.loadcombinations:
                cn = ' '*(len(prev)//2)+'"' if cname == prev else cname
                print("{:<5s}  {:<8s}  {:>6.2f}".format(cn,lname,f))
                prev = cname
        else:
            print(" - - - none - - -")

In [9]:
##test:
f.print_loads()


Node Loads:
===========

Type      Node      FX          FY          MZ
----      ----  ----------  ----------  ----------
wind      B        -200000           0           0

Member Loads:
=============

Type      Member  Load
----      ------  ----------------
live      BC      UDL(L=8000.0,w=-50)
live      BC      PL(L=8000.0,P=-200000,a=5000.0)

Support Displacements:
======================

Type      Node      DX          DY          TZ
----      ----  ----------  ----------  ----------
other     A              0         -10           0

Load Combinations:
==================

Case   Type      Factor
-----  ----      ------
one    live        1.50
 "     wind        1.75
all    wind        1.00
 "     other       1.00
 "     live        1.00

In [10]:
@extend
class Frame2D:
    
    def print_input(self):
        
        prhead('Frame '+str(self.dsname)+':')
        print()
        print('              # of nodal degrees of freedom:',self.ndof)
        print('  # of constrained nodal degrees of freedom:',self.ncons)
        print('# of unconstrained nodal degrees of freedom:',self.nfree,' (= degree of kinematic indeterminacy)')
        m = len(self.members)
        r = self.ncons
        j = len(self.nodes)
        c = len(self.rawdata.releases)
        print()
        print('                               # of members:',m)
        print('                             # of reactions:',r)
        print('                                 # of nodes:',j)
        print('                            # of conditions:',c)
        print('           degree of statical indeterminacy:',(3*m+r)-(3*j+c))
        print('\n')

        self.print_nodes()
        print('\n')
        self.print_members()
        print('\n')
        self.print_loads()

In [11]:
##test:
f.print_input()


Frame frame-1:
==============


              # of nodal degrees of freedom: 12
  # of constrained nodal degrees of freedom: 5
# of unconstrained nodal degrees of freedom: 7  (= degree of kinematic indeterminacy)

                               # of members: 3
                             # of reactions: 5
                                 # of nodes: 4
                            # of conditions: 2
           degree of statical indeterminacy: 0



Nodes:
======

Node          X         Y  Constraints  DOF #s
----      -----     -----  -----------  ------
A             0         0  FX,FY,MZ     7,8,9
B             0      4000               0,1,2
C          8000      4000               3,4,5
D          8000         0  FX,FY        10,11,6



Members:
========

Member   Node-J  Node-K    Length       dcx       dcy  Size                Ix           A  Releases
------   ------  ------    ------   -------   -------  --------      --------       -----  --------
AB       A       B         4000.0   0.00000   1.00000  W310x97       2.22e+08       12300  MZK
BC       B       C         8000.0   1.00000   0.00000  W460x106      4.88e+08       13500  
CD       C       D         4000.0   0.00000  -1.00000                2.22e+08       12300  MZJ



Node Loads:
===========

Type      Node      FX          FY          MZ
----      ----  ----------  ----------  ----------
wind      B        -200000           0           0

Member Loads:
=============

Type      Member  Load
----      ------  ----------------
live      BC      UDL(L=8000.0,w=-50)
live      BC      PL(L=8000.0,P=-200000,a=5000.0)

Support Displacements:
======================

Type      Node      DX          DY          TZ
----      ----  ----------  ----------  ----------
other     A              0         -10           0

Load Combinations:
==================

Case   Type      Factor
-----  ----      ------
one    live        1.50
 "     wind        1.75
all    wind        1.00
 "     other       1.00
 "     live        1.00

In [12]:
@extend
class Frame2D:
    
    COLUMNS_pdelta_forces = ['NODEID','FX','FY','MZ']
    
    def list_pdelta_forces(self,rs,):
        ans = []
        if not rs.pdelta:
            return ans
        pdf = rs.pdelta_forces
        for node in self.nodes.values():
            p = pdf[node.dofnums]
            if p[0,0] or p[1,0] or p[2,0]:
                ans.append((node.id,p[0,0],p[1,0],p[2,0]))
        return ans
    
    def print_pdelta_forces(self,rs,mult=[1e-3,1e-3,1e-6]):
        plist = self.list_pdelta_forces(rs)
        if not plist:
            return
        prhead('P-Delta Node Forces:')
        print('Node        FX         FY         MZ  ')
        print('----     -------    -------    -------')
        for nid,fx,fy,mz in plist:
            print('{:<5s} {:>10.3f} {:>10.3f} {:>11.7f}'.format(nid,fx*mult[0],fy*mult[1],mz*mult[2]))

In [13]:
@extend
class Frame2D:
    
    def print_node_displacements(self,D=None,rs=None):
        if rs is not None and D is None:
            D = rs.node_displacements
        prhead('Node Displacements:')
        print('Node        DX         DY      Rotation')
        print('----      ------     ------   ---------')
        for node in self.nodes.values():
            d = D[node.dofnums]
            print('{:<5s} {:>10.3f} {:>10.3f} {:>11.7f}'.format(node.id,d[0,0],d[1,0],d[2,0]))

In [14]:
##test:
import numpy as np
f.print_node_displacements(D=np.random.rand(f.ndof,1))


Node Displacements:
===================

Node        DX         DY      Rotation
----      ------     ------   ---------
A          0.250      0.989   0.3940273
B          0.297      0.531   0.7260796
C          0.186      0.312   0.4311868
D          0.337      0.047   0.1147584

In [15]:
@extend
class Frame2D:
    
    def print_reactions(self,R=None,rs=None,mult=[1E-3,1E-3,1E-6]):
        if R is None and rs is not None:
            R = rs.reaction_forces
        prhead('Reactions:')
        print('Node        FX         FY         MZ  ')
        print('----     -------    -------    -------')
        for node in self.nodes.values():
            if node.constraints:
                efs = ['--   '] * 3
                for dirn in node.constraints:
                    i = Node.DIRECTIONS[dirn]
                    j = node.dofnums[i]
                    val = R[j-self.nfree,0]
                    efs[i] = '{:>10.3f}'.format(val*mult[i])
                print('{:<5s} {:>10s} {:>10s} {:>10s}'.format(node.id, *efs))

In [16]:
##test:
f.print_reactions(R=np.random.rand(f.ncons,1),mult=[1,1,1])


Reactions:
==========

Node        FX         FY         MZ  
----     -------    -------    -------
A          0.731      0.242      0.026
D          0.848      0.148      --   

In [17]:
@extend
class Frame2D:
    
    def print_mefs(self,rs,mult=[1E-3,1E-3,1E-6],precision=3):
        prhead('Member End Forces:')
        print('          /----- Axial -----/   /----- Shear -----/   /----- Moment ----/')
        print('Member       FX J       FX K       FY J       FY K       MZ J       MZ K')
        print('------     -------    -------    -------    -------    -------    -------')
        for memb in self.members.values():
            mefs = rs.member_efs[memb].fefs
            fs = [mefs[i,0]*mult[i%len(mult)] for i in [0,3,1,4,2,5]]
            s = ['{:>10.{precision}f}'.format(x,precision=precision) for x in fs]
            print('{:<7s} {}'.format(memb.id,' '.join(s)))

In [18]:
@extend
class Frame2D:
    
    def print_results(self,rs,mult=[1E-3,1E-3,1E-6]):
        prhead('Results for load case: {}'.format(rs.loadcase),ul='+')
        self.print_node_displacements(rs=rs)
        self.print_pdelta_forces(rs=rs,mult=mult)
        self.print_reactions(rs=rs,mult=mult)
        self.print_mefs(rs=rs,mult=mult)

In [20]:
##test:
import Frame2D_SolveFirstOrder
f.reset()
f.input_all()
rs = f.solve(pdelta=True)
f.print_results(rs)


Compiling notebook file 'Frame2D_SolveFirstOrder.ipynb' to python.
Compiling notebook file 'Frame2D_Display.ipynb' to python.
iter=1, max D=95.51894236578983, max chg=0.600287975663278, max % chg=0.6284491440079758
iter=2, max D=95.52331189326802, max chg=0.004514041250118339, max % chg=0.004725591230716599


Results for load case: all
++++++++++++++++++++++++++


Node Displacements:
===================

Node        DX         DY      Rotation
----      ------     ------   ---------
A          0.000    -10.000   0.0000000
B        -95.523    -10.447  -0.0167332
C        -95.500     -0.528   0.0204936
D          0.000      0.000   0.0238751

P-Delta Node Forces:
====================

Node        FX         FY         MZ  
----     -------    -------    -------
B         -6.567      0.000   0.0000000
C          7.759      0.000   0.0000000

Reactions:
==========

Node        FX         FY         MZ  
----     -------    -------    -------
A        198.808    275.000   -795.232
D         -0.000    325.000      --   

Member End Forces:
==================

          /----- Axial -----/   /----- Shear -----/   /----- Moment ----/
Member       FX J       FX K       FY J       FY K       MZ J       MZ K
------     -------    -------    -------    -------    -------    -------
AB         275.000   -275.000   -198.808    198.808   -795.232      0.000
BC          -7.759      7.759    275.000    325.000      0.000     -0.000
CD         325.000   -325.000      0.000     -0.000      0.000      0.000

In [21]:
import numpy as np
import matplotlib.pyplot as plt

@extend
class Frame2D:
    
    def show(self):
        """Draw the frame geometry."""
        # reformat the coordinates to that expected by plt.plot()
        # (where each column of x is plotted against the corresponding column of y)
        # Thus each column of x contains the x-coords at either end of member; y similar
        n = len(self.members)
        x = np.zeros((2,n))
        y = np.zeros((2,n))
        for j,m in enumerate(self.members.values()):
            x[0,j] = m.nodej.x
            x[1,j] = m.nodek.x
            y[0,j] = m.nodej.y
            y[1,j] = m.nodek.y
        # size and set the drawing limits s.t. x- and y-scales are roughly the same
        xmin = np.min(x)
        xmax = np.max(x)
        ymin = np.min(y)
        ymax = np.max(y)
        size = max(xmax-xmin,ymax-ymin)*1.05
        xmid = (xmax+xmin)/2.
        ymid = (ymax+ymin)/2.
        
        plt.figure(figsize=(8.,8.))
        plt.axis([xmid-size/2.,xmid+size/2.,ymid-size/2.,ymid+size/2.])
        plt.grid(True)
        plt.plot(x,y,'b-')

In [22]:
##test:
%matplotlib inline
f.show()



In [ ]: